Protocol Functions

All protocol function calls pass a single parameter – the protocol table.

This protocol table is an object that you can call all the source function methods on.

Additionally, it contains all of the parameters (as defined by the user), and a field “.n” that specifies the channel number.

Action on setup

Sometimes the protocol requires a particular action when setup.
This call is made when a channel starts firing up the protocol.

function protocols.myprotocol.setup(p)
 -- do something
 --return true --reload the configuration
 return false --just continue
end

It is illegal to attempt to read or write to the source – it is not connected yet.

However, you can choose to initialise variables that are required by the protocol.
Additionally the protocol's setup function can make changes to the source parameters before continuing.

For example, a protocol may choose to force the TCP port number for the channel:

protocols.myprotocol.setup(p)
 if c.chnl[p.n].src.tcp.port ~= 23
 then
  c.chnl[p.n].src.tcp.port = 23
  return true
 end
 return false
end

The function will check the port number and force it to 23 if different. Note that the change will be stored and will effectively overwrite the user's input.

Action on Connect

Like the .setup function, the connect function can be used to setup various working variables for the protocol (like timeout values and clearing buffers).
It is called whenever the source enters the “connected” state.

Note

COM and UDP sources will “connect” immediately. A TCP source will call this when the socket connects and the match/send have completed. An FTP Server source will call this when the FTP client logs in with the username/password that match the source channel.

Unlike the .setup function hook you can read and write to the source.

function protocols.myprotocol.connect(p)
 p:write('Hello') --send a string
 p:readline(1000) --wait 1second for a response
end

Additionally, the protocol can choose to close the source and issue an error:

function protocols.myprotocol.connect(p)
 p:write('Hello') --send a string
 s,t = p:readline(1000) --wait 1second for a response
 if t
 then
  p:error('No response')
  p:close()
 end
 p:write('Now we are running')
end

This functionality is required for the Avaya RSP, for example, where on connection there is an initial conversation to establish the link.

Action every second

Many protocols require timeouts to be kept, and other actions.
The Lua protocol framework allows for a function hook that is called every second, while there is no incoming data.
You can perform any action within the second event hook.

function protocols.ascii.second(p)
 if (p.xontime>=30)
 then
  if (p.xon == 1) then p:writebytes(0x11) end
  p.xontime=0
 end
 p.xontime=(p.xontime or 0) + 1
end

The above function is from the supplied “ASCII lines” protocol. It keeps track of the idle count with the p.xontime counter.
When that exceeds 30, the protocol will send an XON character – only if the user has configured it and the p.xon variable is therefore 1.

Some protocols may perform keep-alive checks with the other end, and decide to close the connection if there is a serious timeout (as is the case with the Avaya RSP protocol).

Handling incoming data

Whenever data arrives at the source, the Lua protocol framework will make a call to the .data function hook of the protocol. It is the responsibility of this function to read data from the source, respond if necessary, and store any data that is considered valid.

As a very simple example:

function protocols.ascii.data(p)
 local s=p:readline()
 p:store(s,"ascii")
end

This function simply calls the source.readline function, pulling the data into a local variable s.
That string is then stored with the tag “ascii” - which involves the optional
time stamping.

Warning

local keyword: it is vitally important that you use the local keyword when defining variables that are used within the scope of a single function.
If you do not, then the variables will be global and can be read, modified, and corrupted by other channels that are using the same protocol or variable name.

Some protocols will require assembling a “frame” of data and perhaps isolating the data.

The NEC NEAX2400 serial protocol is an example that is worth analysing:

function protocols.nec.data(p)
 local r,s
 p._buff = (p._buff or "")..p:read()
 p.time=0
 if string.len(p._buff)>2048 then p._buff=string.sub(p._buff,-
 2048,-1) end
 while true
 do
  _,r,s=string.find(p._buff,"\002(.-)\003")
  if not (s) then break end
  if (s) then p:store(s.."\r\n","cdr nec") end
  p._buff=string.sub(p._buff,r+1,-1)
 end
end

This code builds a buffer that is local to the protocol (stored in p._buff).

Note

Any variables that begin with an underscore will not appear in the diagnostic dump output.

The buffer length is then limited to 2k. Finally, in a “while true” loop the Lua script searches for pattern captures between the 0x02 and 0x03 characters.
Any matching captures are stored in the “s” local variable and are then stored with a CR/LF suffix to the flash.

(There is some other code that uses the .second function to clear the buffer – just in case the protocol is selected and we are being sent non-NEC data!)

On disconnect

#Firmware/v320
To correspond with the protocol's .connect function, .disconnect is called when closing down - just after x.chnl[#].src.onclosing is called (see x.chnl.src - Source Hooks).

function protocols.mine.disconnect(p)
 -- Do other stuff
 log.write('mine', 'disconnecting')
end