287 lines
8.8 KiB
Plaintext
287 lines
8.8 KiB
Plaintext
|
/*
|
||
|
* passToHCs.sqf
|
||
|
*
|
||
|
* In the mission editor, name the Headless Clients "HC", "HC2", "HC3" without the quotes
|
||
|
*
|
||
|
* In the mission init.sqf, call passToHCs.sqf with:
|
||
|
* execVM "passToHCs.sqf";
|
||
|
*
|
||
|
* It seems that the dedicated server and headless client processes never use more than 20-22% CPU each.
|
||
|
* With a dedicated server and 3 headless clients, that's about 88% CPU with 10-12% left over. Far more efficient use of your processing power.
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
PassToHC_ReceiveMessage = compileFinal "
|
||
|
if (hasInterface && (serverCommandAvailable '#kick' || isServer)) then {
|
||
|
player globalChat (_this select 0);
|
||
|
};
|
||
|
";
|
||
|
|
||
|
if (!isServer) exitWith {};
|
||
|
|
||
|
waitUntil{!isNil "f_param_headlessClient"};
|
||
|
|
||
|
if (f_param_headlessClient == 0) exitWith {};
|
||
|
|
||
|
private ["_HC_ID","_HC2_ID","_HC3_ID","_rebalanceTimer","_cleanUpThreshold","_maxWait","_loadBalance","_currentHC","_numTransfered","_swap","_rc","_numHC","_numHC2","_numHC3","_numDeleted"];
|
||
|
|
||
|
PassToHC_SendMessage = compileFinal "
|
||
|
diag_log text _this;
|
||
|
if (isServer && hasInterface) then {
|
||
|
[_this] call PassToHC_ReceiveMessage;
|
||
|
} else {
|
||
|
[[_this], 'PassToHC_ReceiveMessage', true, false] call BIS_fnc_MP;
|
||
|
};
|
||
|
";
|
||
|
|
||
|
"passToHCs: Started" call PassToHC_SendMessage;
|
||
|
|
||
|
waitUntil {!isNil "HC"};
|
||
|
waitUntil {!isNull HC};
|
||
|
|
||
|
_HC_ID = -1; // Will become the Client ID of HC
|
||
|
_HC2_ID = -1; // Will become the Client ID of HC2
|
||
|
_HC3_ID = -1; // Will become the Client ID of HC3
|
||
|
_rebalanceTimer = 60; // Rebalance sleep timer in seconds
|
||
|
_cleanUpThreshold = 50; // Threshold of number of dead bodies + destroyed vehicles before forcing a clean up
|
||
|
|
||
|
PassToHC_NumTransfered = 0;
|
||
|
|
||
|
"passToHCs: Waiting for init scripts to settle before starting first pass..." call PassToHC_SendMessage;
|
||
|
|
||
|
sleep 15;
|
||
|
|
||
|
// If DAC is initializing after start delay wait until it finishes or timeout
|
||
|
if (!isNil "DAC_Basic_Value") then {
|
||
|
_maxWait = time + 30;
|
||
|
waituntil {sleep 1; (DAC_Basic_Value > 0) || time > _maxWait};
|
||
|
};
|
||
|
|
||
|
// If UPSMON is initializing after start delay wait until it finishes or timeout
|
||
|
if (!isNil "UPSMON_INIT") then {
|
||
|
_maxWait = time + 30;
|
||
|
waituntil {sleep 1; (UPSMON_INIT > 0) || time > _maxWait};
|
||
|
};
|
||
|
|
||
|
// Wait a bit more just in-case they scripts have not settled/synced yet
|
||
|
sleep 3;
|
||
|
|
||
|
format["passToHCs: First pass beginning now..."] call PassToHC_SendMessage;
|
||
|
|
||
|
while {true} do {
|
||
|
// Do not enable load balancing unless more than one HC is present
|
||
|
// Leave this variable false, we'll enable it automatically under the right conditions
|
||
|
_loadBalance = false;
|
||
|
|
||
|
// Get HC Client ID else set variables to null
|
||
|
try {
|
||
|
_HC_ID = owner HC;
|
||
|
|
||
|
if (_HC_ID > 2) then {
|
||
|
diag_log format ["passToHCs: Found HC with Client ID %1", _HC_ID];
|
||
|
} else {
|
||
|
diag_log "passToHCs: [WARN] HC disconnected";
|
||
|
|
||
|
HC = objNull;
|
||
|
_HC_ID = -1;
|
||
|
};
|
||
|
} catch { diag_log format ["passToHCs: [ERROR] [HC] %1", _exception]; HC = objNull; _HC_ID = -1; };
|
||
|
|
||
|
// Get HC2 Client ID else set variables to null
|
||
|
if (!isNil "HC2") then {
|
||
|
try {
|
||
|
_HC2_ID = owner HC2;
|
||
|
|
||
|
if (_HC2_ID > 2) then {
|
||
|
diag_log format ["passToHCs: Found HC2 with Client ID %1", _HC2_ID];
|
||
|
} else {
|
||
|
diag_log "passToHCs: [WARN] HC2 disconnected";
|
||
|
|
||
|
HC2 = objNull;
|
||
|
_HC2_ID = -1;
|
||
|
};
|
||
|
} catch { diag_log format ["passToHCs: [ERROR] [HC2] %1", _exception]; HC2 = objNull; _HC2_ID = -1; };
|
||
|
} else {
|
||
|
HC2 = objNull;
|
||
|
};
|
||
|
|
||
|
// Get HC3 Client ID else set variables to null
|
||
|
if (!isNil "HC3") then {
|
||
|
try {
|
||
|
_HC3_ID = owner HC3;
|
||
|
|
||
|
if (_HC3_ID > 2) then {
|
||
|
diag_log format ["passToHCs: Found HC2 with Client ID %1", _HC3_ID];
|
||
|
} else {
|
||
|
diag_log "passToHCs: [WARN] HC3 disconnected";
|
||
|
|
||
|
HC3 = objNull;
|
||
|
_HC3_ID = -1;
|
||
|
};
|
||
|
} catch { diag_log format ["passToHCs: [ERROR] [HC3] %1", _exception]; HC3 = objNull; _HC3_ID = -1; };
|
||
|
} else {
|
||
|
HC3 = objNull;
|
||
|
};
|
||
|
|
||
|
// If no HCs present, wait for HC to rejoin
|
||
|
if ( (isNull HC) && (isNull HC2) && (isNull HC3) ) then { waitUntil {!isNull HC}; };
|
||
|
|
||
|
// Check to auto enable Round-Robin load balancing strategy
|
||
|
if ( (!isNull HC && !isNull HC2) || (!isNull HC && !isNull HC3) || (!isNull HC2 && !isNull HC3) ) then { _loadBalance = true; };
|
||
|
|
||
|
if ( _loadBalance ) then {
|
||
|
diag_log "passToHCs: Starting load-balanced transfer of AI groups to HCs";
|
||
|
} else {
|
||
|
// No load balancing
|
||
|
diag_log "passToHCs: Starting transfer of AI groups to HC";
|
||
|
};
|
||
|
|
||
|
// Determine first HC to start with
|
||
|
_currentHC = 0;
|
||
|
|
||
|
if (!isNull HC) then { _currentHC = 1; } else {
|
||
|
if (!isNull HC2) then { _currentHC = 2; } else { _currentHC = 3; };
|
||
|
};
|
||
|
|
||
|
// Pass the AI
|
||
|
_numTransfered = 0;
|
||
|
{
|
||
|
_syncGroup = _x;
|
||
|
_swap = true;
|
||
|
|
||
|
_ownerID = _HC_ID;
|
||
|
|
||
|
if ( _loadBalance ) then {
|
||
|
|
||
|
_ownerID = switch (_currentHC) do {
|
||
|
case 1: { if (!isNull HC2) then { _currentHC = 2; } else { _currentHC = 3; }; _HC_ID };
|
||
|
case 2: { if (!isNull HC3) then { _currentHC = 3; } else { _currentHC = 1; }; _HC2_ID };
|
||
|
case 3: { if (!isNull HC) then { _currentHC = 1; } else { _currentHC = 2; }; _HC3_ID };
|
||
|
default {-1};
|
||
|
};
|
||
|
|
||
|
} else {
|
||
|
|
||
|
_ownerID = switch (_currentHC) do {
|
||
|
case 1: {_HC_ID};
|
||
|
case 2: {_HC2_ID};
|
||
|
case 3: {_HC3_ID};
|
||
|
default {-1};
|
||
|
};
|
||
|
|
||
|
};
|
||
|
|
||
|
// Check if group has already been transfered
|
||
|
if (_syncGroup getVariable ["hc_transfered", false]) then {
|
||
|
_swap = false;
|
||
|
} else {
|
||
|
|
||
|
if (groupOwner _syncGroup == _ownerID) then {
|
||
|
_x setVariable ["hc_transfered", true];
|
||
|
_swap = false;
|
||
|
};
|
||
|
|
||
|
};
|
||
|
|
||
|
// Check if group is blacklisted
|
||
|
if (_syncGroup getVariable ["hc_blacklist", false]) then {
|
||
|
_swap = false;
|
||
|
};
|
||
|
|
||
|
if ( _swap ) then {
|
||
|
{
|
||
|
// If a player is in this group, don't swap to an HC
|
||
|
if (isPlayer _x) exitWith { _swap = false; };
|
||
|
|
||
|
// If a unit has 'hc_blacklist' set to true and is in this group, don't swap to an HC.
|
||
|
if (_x getVariable ["hc_blacklist", false]) exitWith { _swap = false; };
|
||
|
|
||
|
// If unit is in a vehicle check if vehicle or crew is blacklisted
|
||
|
if (vehicle _x != _x) then {
|
||
|
if ((vehicle _x) getVariable ["hc_blacklist", false]) exitWith { _swap = false; };
|
||
|
};
|
||
|
|
||
|
} forEach (units _syncGroup);
|
||
|
};
|
||
|
|
||
|
// Check if group has any waypoints synced to triggers and auto blacklist
|
||
|
if ( _swap ) then {
|
||
|
{
|
||
|
if (count (synchronizedTriggers _x) > 0) exitWith {
|
||
|
_syncGroup setVariable ["hc_blacklist", true];
|
||
|
_swap = false;
|
||
|
};
|
||
|
} forEach (waypoints _syncGroup);
|
||
|
};
|
||
|
|
||
|
// If load balance enabled, round robin between the HCs - else pass all to HC
|
||
|
if ( _swap ) then {
|
||
|
_rc = false;
|
||
|
|
||
|
_syncTrigArray = [];
|
||
|
_syncWayArray = [];
|
||
|
{
|
||
|
_wayNum = _forEachIndex;
|
||
|
_syncedTrigs = synchronizedTriggers _x;
|
||
|
_syncTrigArray set [_wayNum,_syncedTrigs];
|
||
|
|
||
|
_syncedWays = synchronizedWaypoints _x;
|
||
|
_syncWayArray set [_wayNum,_syncedWays];
|
||
|
} forEach waypoints _x;
|
||
|
|
||
|
if (_ownerID >= 0) then {
|
||
|
_rc = _x setGroupOwner _ownerID;
|
||
|
} else {
|
||
|
diag_log format["passToHCs: [ERROR] No Valid HC to pass to. _currentHC = %1", _currentHC];
|
||
|
};
|
||
|
|
||
|
|
||
|
// If the transfer was successful, count it for accounting and diagnostic information
|
||
|
if ( _rc ) then {
|
||
|
_x setVariable ["hc_transfered", true];
|
||
|
PassToHC_NumTransfered = PassToHC_NumTransfered + 1;
|
||
|
};
|
||
|
};
|
||
|
} forEach (allGroups);
|
||
|
|
||
|
if (PassToHC_NumTransfered > 0) then {
|
||
|
// More accounting and diagnostic information
|
||
|
|
||
|
diag_log format ["passToHCs: Transfered %1 AI groups to HC(s)", PassToHC_NumTransfered];
|
||
|
|
||
|
_numHC = 0;
|
||
|
_numHC2 = 0;
|
||
|
_numHC3 = 0;
|
||
|
|
||
|
{
|
||
|
switch (owner ((units _x) select 0)) do {
|
||
|
case _HC_ID: { _numHC = _numHC + 1; };
|
||
|
case _HC2_ID: { _numHC2 = _numHC2 + 1; };
|
||
|
case _HC3_ID: { _numHC3 = _numHC3+ 1; };
|
||
|
};
|
||
|
} forEach (allGroups);
|
||
|
|
||
|
diag_log if (_numHC > 0) then { format ["passToHCs: %1 AI groups currently on HC", _numHC]; };
|
||
|
diag_log if (_numHC2 > 0) then { format ["passToHCs: %1 AI groups currently on HC2", _numHC2]; };
|
||
|
diag_log if (_numHC3 > 0) then { format ["passToHCs: %1 AI groups currently on HC3", _numHC3]; };
|
||
|
|
||
|
diag_log format ["passToHCs: %1 AI groups total across all HC(s)", (_numHC + _numHC2 + _numHC3)];
|
||
|
} else {
|
||
|
diag_log "passToHCs: No rebalance or transfers required this round";
|
||
|
};
|
||
|
|
||
|
// Force clean up dead bodies and destroyed vehicles
|
||
|
if (count allDead > _cleanUpThreshold) then {
|
||
|
_numDeleted = 0;
|
||
|
{
|
||
|
deleteVehicle _x;
|
||
|
|
||
|
_numDeleted = _numDeleted + 1;
|
||
|
} forEach allDead;
|
||
|
|
||
|
diag_log format ["passToHCs: Cleaned up %1 dead bodies/destroyed vehicles", _numDeleted];
|
||
|
};
|
||
|
|
||
|
// Rebalance every rebalanceTimer seconds to avoid hammering the server
|
||
|
sleep _rebalanceTimer;
|
||
|
};
|