Controls in the cockpit can be manipulated by sending newline-terminated plain text commands to UDP port 7778. The syntax is described in the The DCS-BIOS Import Protocol section.
The current state of cockpit indicators and controls is exported from DCS using a binary protocol (see The DCS-BIOS Export Protocol). This data is sent via UDP to multicast address 239.255.50.10, port 5010 on the loopback interface (127.0.0.1).
Warning
|
When opening a UDP socket to listen on 239.255.50.10:5010, you must specify the allowreuseaddr socket option.
Otherwise Windows will prevent other programs from listening to the same data stream.
|
Note
|
The IPv4 multicast address block 239.255.0.0/16 is defined in RFC 2365 as the "IPv4 Local Scope". It is part of the "Administratively Scoped IPv4 Multicast Space", which is to multicast as the private address ranges 10.0.0.0/8, 192.168.0.0/16 and 172.16.0.0/12 are to unicast Ipv4. |
If you cannot or do not want to deal with multicast UDP, you can also make DCS-BIOS send a copy of the export data stream to a unicast address by editing BIOSConfig.lua
.
You can also use the socat
utility to connect a serial port (and just about anything else that can be read or written) to the DCS-BIOS import and export streams.
For an example, take a look at the batch files that come with DCS-BIOS.
Another way is to open a TCP connection to port 7778. You can use this connection to send commands to DCS-BIOS and receive the export data. If you can, use UDP instead of TCP. Each TCP connection slightly increases the amount of work that is performed inside of DCS (blocking the rest of the simulation).
Commands are sent to DCS-BIOS as lines of plain text.
A command consists of a control identifier, a space, an argument and a newline character.
Note
|
The import protocol does not use binary encodings. Handling binary data in Lua with only the libraries that come with DCS is cumbersome. Also, a plain text protocol is easier to explain, understand and debug. In the input direction, the added verbosity of a plain text protocol is no problem because a human pilot will not flip 20 switches 30 times a second. |
To find out the arguments that a certain control accepts, refer to the reference documentation and look up input interfaces in the User Guide.
The export stream protocol is designed to be as space-efficient as possible to enable the export of the complete cockpit state over a 115200 bps serial connection.
Each piece of cockpit state is encoded as an integer or a string and is assigned an address within a 16-bit address space.
The location of an integer value is defined by a 16-bit word address which specifies the start of the 16-bit word in which the data is located, a 16-bit mask which specifies which bits in that word belong to the value, and a shift value (which can also be deduced from the mask).
Given the start address, mask and shift values, the following pseudo C code decodes a value:
char[] state;
unsigned int value = (((uint16_t*)state)[start_address/2] & mask) >> shift
Changes to the state data are encoded as write accesses to this address space and written to the export data stream in the following format:
<start address (16 bit)>
<data length (16 bit)>
data
Both the start address and the data length will always be even (internally, DCS-BIOS treats the data as a list of 16-bit integers). This ensures that no write access partially updates an integer value (an integer may occupy no more than 16 bit).
All integer values are written in litte-endian byte order.
The following byte sequence is an example of a write access of four bytes starting at address 0x1000:
0x00 0x10 0x04 0x00 0x41 0x2d 0x31 0x30
The location of a string is defined by its 16-bit start address and the length in bytes (all string values have a fixed maximum length and start on a 16-bit aligned address).
A string value with a maximum length greater than two characters can be updated partially. To avoid inconsistent intermediate states, an application that receives DCS-BIOS export data should apply changes to string values at the next frame synchronization sequence (see next section).
DCS-BIOS tries to send 30 updates per second.
DCS-BIOS will send the four bytes 0x55 0x55 0x55 0x55
at the start of each update.
These do not specify a 21845-byte write access to the address 0x5555.
Instead, they are used by consumers of the data stream to find the start of a write access to synchronize to.
DCS-BIOS ensures that this byte sequence cannot appear in the normal data stream.
In the event that the byte sequence 0x55 0x55 0x55 0x55
would appear in a single write access, DCS-BIOS will avoid that by splitting it up into two separate writes.
The DcsBios::PollingInput
class is the base class of all input classes such as Switch2Pos
or RotaryEncoder
.
If your class inherits from PollingInput
, its pollInput()
method will be called whenever DcsBios::PollingInput::pollInputs()
is called from the main loop.
To make this happen, the constructor of PollingInput
maintains a global singly-linked list of all PollingInput
instances.
If your class inherits from ExportStreamListener
, its onDcsBiosWrite(unsigned int address, unsigned int data)
method will be called every time a ProtocolParser
finishes receiving new export stream data.
Its onDcsBiosFrameSync()
method will be called every time the synchronization sequence (0x55 0x55 0x55 0x55
) is received.
The StringBuffer
class uses this to avoid calling your code with an inconsistent string mid-update.
If you feed the export stream data you receive from DCS-BIOS to the processChar
method of a ProtocolParser
instance, it will interpret the data and ensure that the global onDcsBiosWrite
function as well as every ExportStreamListener’s `onDcsBiosWrite
and onDcsBiosFrameSync
methods are called with the results.
DCS-BIOS is loaded by executing the BIOS.lua
file.
That file loads all other DCS-BIOS code. It also defines the hook functions for DCS export (LuaExportStart
and friends), which mostly call functions in Protocol.lua
.
The following is an overview of the other files and their purpose:
- BIOSConfig.lua
-
This file is loaded last, so it can override any settings that are defined in global variables. It has to set the variable
BIOS.protocol.io_connections
to a list of suitable connection objects to define where the export data gets sent to and how DCS-BIOS listens for commands. - lib/ProtocolIO.lua
-
This file contains classes that handle the actual data transmission, i.e. creating and using sockets.
- lib/Protocol.lua
-
The code in this file manages the list of known aircraft modules. Every frame, the
BIOS.protocol.step()
function in this file is called. That function checks what aircraft is currently being used and ensures that the correct data is exported. - lib/Util.lua
-
This file includes a lot of utility classes and functions. Most of them implement the
MemoryMap
class and related classes or provide shortcuts to quickly define controls in aircraft modules. - lib/A-10C.lua, lib/UH-1H.lua, etc
-
Each aircraft module has its own file where all of its controls are defined.
This section will describe how to add support for another aircraft module.
DCS-BIOS consists of several export modules (they are what you select in the "module" drop-down field in the control reference documentation). Each export module is assigned to one or multiple aircraft and several export modules can be active at the same time.
The MetadataStart
and MetadataEnd
modules are special: they are always active, even if there is no active aircraft (e.g. in spectator mode in a multiplayer game). The CommonData
module is always active when any of the aircraft in AircraftList.lua
is active. It exports generic information like altitude, position and heading.
-
Each export module is defined in its own file in the
lib
subdirectory. -
Each export module is loaded by a
dofile(…)
line inBIOS.lua
. -
Each export module needs a
<script>
tag incontrol-reference.html
to show up in the control reference documentation.
First, find out the exact name of your aircraft in DCS: World.
To do this, open the interactive control reference documentation while in your aircraft and look at the _ACFT_NAME value in the MetadataStart
module.
Open AircraftList.lua
. If your aircraft has a clickable cockpit, add a("Your Aircraft Name", true)
. If your aircraft does not have a clickable cockpit, add a("Your Aircraft Name", false)
. This will populate the constants BIOS.ALL_PLAYABLE_AIRCRAFT
, BIOS.CLICKABLE_COCKPIT_AIRCRAFT
and BIOS.FLAMING_CLIFFS_AIRCRAFT
accordingly. After this, the CommonData
export module will be active for your aircraft.
Create a new Lua file with your aircraft name in the lib
subfolder.
The basic structure of an export module looks like this:
BIOS.protocol.beginModule("Your Export Module Name", 0x1234)
BIOS.protocol.setExportModuleAircrafts({"Your Aircraft Name"})
BIOS.protocol.endModule()
The call to BIOS.protocol.beginModule
starts your new export module. The name does not have to be the same as the name of your aircraft, although in most cases it will be. It must be a valid filename.
Replace 0x1234
with a base address for your module. A base address is the address in the DCS-BIOS export address space where the data from your export module starts. Choose it in a way so the address space occupied by your module does not overlap with any other export module that is active at the same time. Ideally, choose it so you do not have an overlap with any other export module. As a rule of thumb, take the highest base address of an existing export module (except MetadataEnd
) and add 1024 (0x400). 1 KiB of address space should be more than enough for most aircraft.
The call to BIOS.protocol.setExportModuleAircrafts
specifies what aircraft you want your export module to be active in. In most cases, you will pass a list with a single entry (the name of your aircraft).
After creating a file for your export module, add a dofile(…)
call in BIOS.lua
and a <script>
tag in control-reference.html
(you will see what to do from the existing entries).
Between the calls to beginModule
and endModule
, you have access to the global table moduleBeingDefined
.
This table has the following entries:
- inputProcessors
-
A table that maps control identifiers to functions. When a message with the given identifier is received, the function will be called with the message argument.
- memoryMap
-
This object manages your export module’s address space. You can ask it to "allocate memory" (reserve address space) for integer or string values.
- exportHooks
-
a list of functions that will be called before sending out an update. Each function will typically get some state information (e.g. the status of an indicator light) from DCS and call
setValue
on a previously createdMemoryAllocation
object. - documentation
-
This is a Lua table that will be serialized and written to
YourModuleName.json
. This will become the machine-readable reference documentation for your export module. It has to follow the format expected by the control reference documentation.
BIOS.protocol.beginModule("ExampleModule", 0x200)
BIOS.protocol.setExportModuleAircrafts({"A-10C"})
local document = BIOS.util.document
local batterySwitchState = moduleBeingDefined.memoryMap:allocateInt{ maxValue = 1 }
moduleBeingDefined.exportHooks[#moduleBeingDefined.exportHooks+1] = function(dev0)
batterySwitchState:setValue(dev0:get_argument_value(246))
end
moduleBeingDefined.inputProcessors["BATTERY_POWER"] = function(value)
if value == "0" then
GetDevice(1):performClickableAction(3006, 0)
elseif value == "1" then
GetDevice(1):performClickableAction(3006, 1)
end
end
document {
identifier = "BATTERY_POWER",
category = "Electrical Power Panel",
description = "Battery Power Switch",
inputs = { interface = "set_state", max_value = 1, description = "set switch state (1=on, 0=off)"},
outputs = {
["type"] = "integer",
suffix = "",
address = tacanTestLEDState.address,
mask = tacanTestLEDState.mask,
shift_by = tacanTestLEDState.shiftBy,
max_value = 1,
description = "1 if light is on, 0 if light is off"
}
}
BIOS.protocol.endModule()
This example shows how to add an export module for the battery power switch in the A-10C.
Once you understand this example, you should be able to read and understand the other export modules and the code in Util.lua
.
Using functions from Util.lua
, we can write this a lot shorter:
BIOS.protocol.beginModule("ExampleModule", 0x200)
BIOS.protocol.setExportModuleAircrafts({"A-10C"})
local document = BIOS.util.document
local defineToggleSwitch = BIOS.util.defineToggleSwitch
defineToggleSwitch("BATTERY_POWER", 1, 3006, 246, "Electrical Power Panel", "Battery Power")
BIOS.protocol.endModule()
Take a look at the functions in Util.lua
and how they are used in the other export modules to get an idea of what to use in which situation. After a while, you should be able to recognize most patterns in clickabledata.lua
and translate them to a line of code for your export module relatively quickly. Some controls will require trial and error in the DCS Witchcraft Lua Console and some custom code, though — each aircraft module seems to do things slightly differently and there is always that one panel that does not want to behave…