====================
Nodes on a RS485 Bus
====================

    :Author: Erich Wälde
    :Contact: amforth-devel@lists.sourceforge.net
    :Date: 2015-04-19

.. contents::

1 Abstract
----------

The cookbook style recipes presented below are interconnected by the
desire to create a solution connecting several controller nodes together
by rs485 network for periodic data collection. A few decisions have been
taken upfront:

* The rs485-bus is wired up as "simplex" (not duplex).
* There is exactly one node on the rs485 network acting as bus-master (the
  data collecting Linux system) and initiating any data transfer.
* Every node has to exlicitly switch its bus transceiver to "send" mode
  when writing data onto the bus.
* There must be a means to address one node whereas all the other nodes
  must remain silent.

Substantial parts of this solution were inspired or recycled from a project
published by Lubos Pekny.

The author is running a set of currently 5 data collecting controller
nodes on one rs485 bus for several years now. The code provided stable
operation so far.

For this article the author decided to "redo everything" in plain
amForth, well almost everything, as it turned out.



2 Motivation: The Project "collector"
-------------------------------------

When starting with microcontrollers years ago I needed to do something
with them, that looked useful at least to me. So very soon I started to
deal with sensors to measure temperature, pressure, and humidity of
air. The classical "weather station" project. This worked, but soon I
wanted to have the values not only displayed but rather collected ---
some day nice graphs would be interesting, or so I thought. So I
connected a small single-board-computer [1] to the only one controller
with a serial cable. This worked for a long time.

Later I had the idea to collect temperatures at other points in the
house as well. I could have added more sensors and long cables to the
only one controller, but instead I decided to add another controller at
the end of a long cable. Now I needed a way to talk to two (or more)
controllers connected "somehow" the the single board computer. I could
have added more serial interfaces, one for each controller, but I
didn't like the idea. Instead I decided to connect two (or more)
controllers via one long cable using the RS485-Bus specification.

The RS485-Bus is an electrical specification. A (possibly twisted) pair
of wires is used to connect two nodes. The signal is differential, the
difference in potential between the two wires is used as information.
That way the whole setup is fairly immune against noise, as this
normally adds the same shift in potential to both wires. The standard
is good for a distance up to 1200 m, but should work much longer
distances with slow data rates and proper termination.


In order to exchange information on the RS485-bus the same timing and
encoding is used as on a normal serial interface (RS232). The idle
levels are interpreted as 1, the first bit is the start bit and always
0, then 8 data bits follow, and then one or more stop bits are sent.
The stop bits are also 1 and correspond to the idle levels. So there is
nothing new here. 

However, if several nodes are on the bus, to which one am I talking?
And if it is the wrong one, how do I "address" the correct one? In
order to distinguish addresses (or control bytes) from ordinary data,
some agreement has to be established, for example

* Bytes with the most significant bit set are treated as addresses or
  control bytes

* Bytes with the most significant bit cleared are treated as ordinary
  data

If the bytes ``0x00 .. 0x7f`` (the lower half of the ascii table) are
sufficient for all data exchange, then 8 data bits are still good. If
not, 9 data bits can be used in many cases. AVR controllers provide
that possibility. There is a project using 9bit communication on Linux
as well (uLan), Links section.

In my case I decided to transfer all information as ascii strings, e.g.

.. code-block:: none

   7F01:8,22.40,22.87,23.24

where ``7F`` is the station address (stationID) in hex, ``01`` is the
sensor number on that node (also in hex), and after the colon a list of
4 numerical values, their precise meaning being entirely irrelevant at
this point. The main advantage is that I can just read everything on
the bus in clear with little technical overhead.

Currently I run a set of 5 controllers with a variety of sensors:

* temperature and humidity (indoors and outdoors), pressure of air
* voltage of an accumulator providing power to a remote system
* distance (ultrasonic range finder) which translates to the amount of
  water in a tank
* counts of meters (electricity, water, natural gas)

The "collector" is a perl script running on the single board computer
and collecting the data from the controllers every 2 or 10 minutes.
This program acts as the bus master, the other nodes do not initiate
any data exchange. The data is then accumulated in a sqlite database.
A "viewer" perl script will then produce graphs of selected values over
time. Other options are possible, of course.


3 Hardware Requirements
-----------------------

All controller nodes need to have a RS485-transceiver. The transceiver
needs 3 connections to the controller:

1. TX --> Data Out

2. RX <-- Data In

3. Port D7 --> Write/Read select (idle = read = low)

.. figure:: p_rs485_bus.png


An RS232 -- RS485 converter of some sort is needed to connect the
serial interface of the collector computer to the bus. USB -- RS485
dongles are available as well. Please note, that the connector should
not produce a local echo of the bytes transmitted, or software needs to
take care of the double echo. The controllers are sending an echo
character as well, this serves as handshake when uploading forth code,
too.

Power supply can be local to any node, but serving power on 2 more
wires is also possible. When having long cables +12V supply voltage and
step down converters on every board seem like a good idea.

4 Software Requirements
-----------------------

1. Any node should be quiet and not write anything to the bus unless
   explicitly requested to do so. This explicit request translates into
   some sort of addressing. Preventing any output is most easily
   achieved by changing ``emit``.

2. Bytes ``0x00 .. 0x7f`` are considered *normal* data, bytes
   ``0x80 .. 0xff`` are considered adresses (or control bytes).

3. every node needs to have an address or stationID assigned

4. The controllers make use of the so called multi processor
   communication (MPC) mode to ignore traffic between other nodes
   already in hardware.

5. a write operation to the bus must assert the W/R pin to write before
   transmission.

6. Upon completion of the (asynchronous) data transmit the W/R pin has
   to be released, e.g. automatically by using the transmit complete
   interrupt.

7. The implementation should be in Forth entirely. A few exceptions
   showed up during implementation.

8. If a node is power cycled, nothing weird should happen on power up.
   Especially nothing should be written to the bus at all.

9. a modified prompt shall include the address (stationID) of the node
   at the other end of the communication. This is solely to provide
   immediate visible feedback, it is not needed for proper
   communication.

10. While experimenting it turned out that sometimes more than one
    controller is in normal mode. They will produce what I call "echo
    loops". The output (mostly error messages) of one node will trigger
    more output (error messages) of the other node. I decided to
    implement harsh measures: whenever command line will produce an
    error message, then instead put the node back to quiet mode.



5 Implementation Plan
---------------------

In order to achieve the above goals, a set of mutually independant
things were implemented.

1. :command:`stationID`, prompts

   This is to satisfy requirements 3 and 9. The code is fairly simple,
   even short.

   :command:`stationID` is an eeprom backed value with a cache place in RAM.

   The prompt itself is produced by :command:`.ready`. This is a deferred
   word and therefore can be overridden easily by another
   function. So we implement a new function :command:`p_id_rd` which will 
   then be registered into the deferred function :command:`.ready`.

2. :command:`-emit` / :command:`+emit`

   In order to prevent **any** output from the controller, I chose to make
   :command:`emit` point to :command:`drop` rather than :command:`tx`.
   :command:`-emit` will take care of this. This word will be called in the
   next part at startup time.

3. rs485 read/write pin handling

   One pin on the controller has to be selected to drive the read/write-pin of
   the transceiver. The idle state should be low (`0`) or `read`, which is
   achieved by a pull-down resistor.

   - provide constants to declare the selected pin

   - switch the pin to output on startup

   - set the pin to write (`1`) before transmission

   - release the pin to read (`0`) upon completion. Use the transmit
     complete interrupt to achieve this.

   We can now write to and read from the RS485 bus. We can handle only
   one controller so far.

4. mpc --- multi processor communication

   This is the most complex part.

   - set quiet mode
     set the serial interface to `7N2` (7 data bits, no parity bit, 2
     stop bits), set the `MPCM0` bit in register `UCSR0C`

   - when receiving a byte with the most significant bit set, inspect the
     byte and decide whether this is the local address or not

   - if not, remain in quiet mode

   - if yes, then switch the serial interface to normal mode (`8N1`) and
     handle all incoming data

   - set normal mode:
     set the serial interface to `8N1`,
     clear the `MPCM0` bit in register `UCSR0C`

   It turned out that the function :command:`usart_rx_isr` is implemented
   in assembly and registered as a *low* level interrupt service routine
   (ISR). This prevents overriding the registered interrupt service routine
   with another function. I decided to change this and make
   :command:`usart_rx_isr` a forth level routine (assembly change 2) and
   register it as a *high* level interrupt. This way the ISR can be
   replaced by some other function.

   Unsurprisingly replacing the ISR needs explicit access to the ring
   buffer that the original ISR is using. It is accessed by :command:`key`
   as well and should not change. So I added forth level headers to make
   the space in RAM available as forth constants/variables (assembly
   change 3)

5. recognizer: go quiet if command not found

   amForth provides recognizers. There is a list of them, which can be
   changed. The first in the list is :command:`rec:word`, it will try to
   find the token in the word list. If it fails, the next one in the list
   is called: :command:`rec:num`. It will try to parse the input token as a
   number. If it fails the list is exhausted and the final
   :command:`r:fail` will be called to issue an error message and do some
   cleanup.

   I decided to add a third recognizer to the end of the list named
   :command:`rec:quiet`. It will not parse the input token again, but clean
   up the arguments. Then it will set the controller to quiet mode (call
   :command:`-emit` :command:`+mpc7`) and signal success rather than error.
   This way the pointer in :command:`r:fail` is not called.

6. startup / turnkey

   In the end all of the above things need to be put together to ensure
   correct startup and initialization of all parts involved. Pay
   attention to turnkey and power cycle.


6 Code
------

This code was re-developed and tested on an atmega644p running amForth
5.5.

6.1 start
~~~~~~~~~

The remainder of this article assumes that we have a working setup
derived from the

.. code-block:: none

  amforth/releases/5.5/appl/template


directory. Set appropriate values for the controller type, crystal
frequency, and baud rate to appropriate values for your board.

- Makefile

  .. code-block:: none

    MCU=atmega644p

- main.asm

  .. code-block:: none

    .equ F_CPU = 11059200
    .set BAUD=115200

Now we are at the point where the controller should talk to us on the
serial interface using a terminal program, e.g. minicom:

.. code-block:: none

  Welcome to minicom 2.7
  
  OPTIONS: I18n 
  Compiled on Jan  1 2014, 09:30:18.
  Port /dev/ttyUSB1, 16:46:00
  
  Press CTRL-A Z for help on special keys
  
  amforth 5.5 ATmega644P
  > 

6.2 changing the prompts
~~~~~~~~~~~~~~~~~~~~~~~~

With the release 6.3 and newer the appearance of the prompt can be changed if we
so desire:

.. code-block:: forth

   amforth 6.3 ATmega644P
   > variable (p_rd)
   ok
   > ' .ready defer@ (p_rd) !
   ok
   > : new_p_rd  cr ." --new> " ; 
    ok
   > ' new_p_rd is .ready
    ok
   --new> 1 3 + .
   4  ok
   --new> (p_rd) @ is .ready
    ok
   > 


This will be used in the next step to display the content represented by
:command:`stationID` in the ready-prompt.

6.3 stationID
~~~~~~~~~~~~~

:command:`StationID` is a value, permanently stored in EEPROM and copied
to a RAM location on startup. So we need to load the appropriate
word :command:`Evalue`:

.. code-block:: forth

  include lib/forth2012/core/value.frt


After that we are able to create a value, the content of which is
backed in EEPROM:

.. code-block:: forth

  $007f Evalue stationID


With this in place we are now in a position to create a new function
implementing a new prompt. In order to make it always look the same (two
digits, leading zeros) we add the word :command:`u0.r` to the dictionary in
``dict_appl.inc`` (please note the leading dot and the quotes, since this
is assembly syntax):

.. code-block:: none

  .include "words/uzerodotr.asm"

reassemble and reflash amForth. Then define the new word :command:`p_id_rd`

.. code-block:: forth

  : p_id_rd
    cr
    base @
    hex
    [char] ~ emit
    stationID 2 u0.r
    [char] > emit
    space
    base !
  ;


We should also take care to save and restore the content of
:command:`base`, since I decided to print out the value of
:command:`stationID` in hexadecimal. Using the new things should work like
this:

.. code-block:: none

  amforth 6.3 ATmega644P ok
  > stationID decimal .
  127  ok
  > p_id_rd
  
  ~7F>  ok
  > ' p_id_rd is .ready
   ok
  ~7F> $42 to stationID
   ok
  ~42> 



The value ``$007F`` is the highest address available for the above
mentioned 7-bit addressing scheme, so I chose it as the default. The exact
value can be changed here or overwritten later when loading the code with
something like

.. code-block:: forth

   $42 to stationID


6.4 emit-on-off
~~~~~~~~~~~~~~~

In order to prevent the controller from writing to the rs485 bus **unless**
explicitly requested, I decided to defer :command:`emit` to :command:`drop`
just to make sure. This requires two fairly simple words

.. code-block:: forth

  variable old-emit
  ' emit defer@ old-emit !
  : -emit
    ['] emit defer@ old-emit !
    ['] drop is emit
  ;
  : +emit
    old-emit @ is emit
  ;


After loading the code we can test this:

.. code-block:: forth

  amforth 6.3 ATmega644P ok
  ~42> 
  ~42> : hi ." howdy, mate!" cr ;
   ok
  ~42> hi
  howdy, mate!
   ok
  ~42> -emit hi +emit
   ok


6.5 adding rs485 r/w handling
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

In order to drive the rs485 transceiver, we need to implement the
following things:

1. select W/R pin

This pin needs to be selected, initialized as output and set to
low.

.. code-block:: forth
 
     $2B constant RS485_PORT         \ memory mapped
     $2A constant RS485_DDR          \ .
     $80 constant RS485_PIN_MASK
     : rs485-pin-output
       RS485_DDR c@ RS485_PIN_MASK or RS485_DDR c!
     ;

Of course the functions in ``lib/bitnames.frt`` could be used as well,
but for the argument of smaller dependencies, I decided to implement
this directly.


2. set W/R pin high (write) or low (read)

Two simple functions will do this:

.. code-block:: forth

     : rs485-write
       RS485_PORT c@ RS485_PIN_MASK or RS485_PORT c!
     ;
     : rs485-read
       RS485_PORT c@ RS485_PIN_MASK invert and RS485_PORT c!
     ;


3. set W/R pin to `write` (`1`) before sending a byte

   Before sending any byte, we need to set the W/R pin high. So we
   reimplement :command:`tx-poll`, the function that transfers one byte to
   the serial interface.

   .. code-block:: forth

     $C6 constant UDR0 \ usart0 data register
     : rs485-tx-poll ( c -- )
       begin  tx?-poll  until
       rs485-write
       UDR0 c!
     ;
   

4. release W/R pin upon transfer completion

   After sending the byte, the W/R pin should be released to zero.
   This happens `some` time after initiating a transfer. However, the
   Atmel engineers have anticipated this problem and provided the
   `transfer complete` interrupt for our convenience.

   .. code-block:: forth

     : tx-complete-isr
       RS485_PORT c@ RS485_PIN_MASK invert and RS485_PORT c!
     ;
     
     $2C  constant USART0__TXAddr \ USART0, Tx Complete
     $40  constant UCSR0B_TXCIE0
     $C1  constant UCSR0B
     : +rs485
       rs485-pin-output
       rs485-read
       ['] tx-complete-isr USART0__TXAddr int!
       ['] rs485-tx-poll is emit
       UCSR0B c@ UCSR0B_TXCIE0 or UCSR0B c!
     ;
     : -rs485
       ['] noop USART0__TXAddr int!
       ['] tx-poll is emit
       UCSR0B c@ UCSR0B_TXCIE0 invert and UCSR0B c!
     ;


   The functions :command:`+rs485` and :command:`-rs485` enable and disable
   the whole rs485 bus connection. Apart from changing the deferred word
   :command:`emit` and registering the interrupt service routine to the
   `transfer complete interrupt`, the interrupt itself must be enabled in
   the register ``UCSR0B``.


At this point we have everything in place to connect to the controller via
the rs485 bus. :command:`+rs485`` needs to be called during startup, which
is the only missing piece at this point.


6.6 mpc 1: making `usart_rx_isr` a forth level word
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

While working on this particular implemention of my code, namely
reimplementation in as much Forth code as possible, I came across a
subtle feature of the amForth implementation (as of version 5.5).
Interrupt handling in amForth is twofold: the low level part (written
in assembly) is basically doing the bookkeeping, clearing the
interrupt and then calling into a amForth table of registered
functions. This provides the possibility to write interrupt service
routines (ISR) in `high level` Forth rather than assembly. Registering
your own ISR is a matter of one line:

.. code-block:: forth

  ' your-own-isr  Interrupt-Vector-Addr  int!


6.6.1 replace `usart_tx_isr` with a Forth word
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

This part is not particularly difficult, because a Forth equivalent
is found already as a comment in the asm file.

.. code-block:: forth

   ; forth code:
   ; : rx-isr USART_DATA c@
   ;    usart_rx_data usart_rx_in c@ dup >r
   ;    + !
   ;    r> 1+ usart_rx_mask and usart_rx_in c!
   ; ;
   ; setup with
   ; ' rx-isr URXCaddr int!


I kept the name, but please note that it does not refer to the asm label
any more --- :command:`usart_rx_isr` is now a proper Forth word.


6.6.2 register the new function as ISR
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

The new function must be registered somewhere in the startup of amForth,
because otherwise there will be no access to the command loop via the
serial interface. So in function :command:`applturnkey` we add the
equivalent of

.. code-block:: forth

  ' usart_rx_isr  USART0__RXAddr  int!


just before globally enabling interrupts.

.. code-block:: asm

    ; ( -- ) System
    ; R( -- )
    ; application specific turnkey action
    VE_APPLTURNKEY:
        .dw $ff0b
        .db "applturnkey",0
        .dw VE_HEAD
        .set VE_HEAD = VE_APPLTURNKEY
    XT_APPLTURNKEY:
        .dw DO_COLON
    PFA_APPLTURNKEY:
        .dw XT_USART

        ; register usart_rx_isr
        .dw XT_DOLITERAL            ; ' usart_rx_isr URXCaddr int!
        .dw XT_USART_RX_ISR
        .dw XT_DOLITERAL
        .dw URXCaddr
        .dw XT_INTSTORE

        .dw XT_INTON
        .dw XT_VER
        .dw XT_EXIT


Assembling amForth and programming the controller with these changes
must result in an equally usable system as it was before.


6.7 mpc 2: adding mpc after all
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

Entering MPC mode in this case means configuring the serial interface
to `7N2` (7 data bits, no parity bit, 2 stop bits) and setting the
``MPCM0`` bit in register ``USCR0A``.

In that mode, if a data frame is received with the most significant
bit cleared (0), the the data frame is silently ignored.

In that mode, if a data frame is received with the most significant bit set
(1), then the data frame shows up in register ``UDR0``, the data register
of the serial interface. An interrupt is generated and the corresponding
ISR is called.

All nodes on the bus will inspect the just arrived address byte. If the
value of the address byte is the same as the configured node address (also
known as :command:`stationID`), only then the serial interface is
reconfigured to `8N1` and the ``MPCM0`` bit is cleared. This node is then
*awake* from a communication point of view. It will receive all following
data frames and is expected to act on them.

All other nodes on the bus will keep the `7N2` mode of the serial
interface and remain *silent* from a communication point of view.

The *awake* state will not end and must be changed explicitly.

Things that need to be done are

1. provide a few definitions for readability (recycled from
   ``devices/$(MCU)/$(MCU).frt`` --- make sure to load the correct file
   for your controller!)

   .. code-block:: forth

     $2C  constant USART0__TXAddr \ USART0, Tx Complete
     $28  constant USART0__RXAddr \ USART0, Rx Complete
     $40  constant UCSR0B_TXCIE0
     $C0  constant UCSRA \ UCSR0A, really
     $10  constant UCSRA_FE0          \ frame error
     $08  constant UCSRA_DOR0         \ data over run
     $04  constant UCSRA_UPE0         \ parity error
     $01  constant UCSRA_MPCM0        \ mpc mode enabled
     $C1  constant UCSR0B
     $C2  constant UCSRC
     $C6  constant UDR0


2. waiting for the currently active transfer to complete (reusing
   definitions from the rs485 section above)

   .. code-block:: forth

     : txc  begin  RS485_PORT c@  RS485_PIN_MASK  and 0= until ;


   This is needed whenever we want to switch to mpc mode. Without
   waiting we will destroy any ongoing transmit.

3. enabling MPC mode (`7N2`)

   .. code-block:: forth

     : +mpc7
       txc
       $0C UCSRC c!                      \ 7N2
       UCSRA c@ $01 or UCSRA c!          \ MPCM0=1
     ;


4. disabling MPC mode (`8N1`)

   .. code-block:: forth

     : -mpc7 (  --  )
       UCSRA c@ $01 invert and UCSRA c!  \ MPCM=0
       $06 UCSRC c!                      \ 8N1
     ;


5. access to the RX data ring buffer

   Handling incoming data unfortunately requires access to the
   variables of the rx ring buffer, which are not readily available in
   forth. In a local copy of ``drivers/usart-isr-rx.asm`` we add
   appropriate provisions. The existing declaration of the used RAM
   space and sizes

   .. code-block:: asm

     ; sizes have to be powers of 2!
     .equ usart_rx_size = $10
     .equ usart_rx_mask = usart_rx_size - 1
     .dseg
     usart_rx_data: .byte usart_rx_size+2
     usart_rx_in:   .byte 2
     usart_rx_out:  .byte 2
     .cseg


   will be made available as amForth constants and variables.

   .. code-block:: forth

     \ variable USART_RX_DATA N allot \ &buffer[0]
     \ variable USART_RX_IN           \ index
     \ N 1- constant USART_RX_MASK    \ length-1, length=2^n
     
     ; ( -- value ) constant USART_RX_DATA
     VE_USART_RX_DATA:
         .dw $FF0D
         .db "USART_RX_DATA",$00
         .dw VE_HEAD
         .set VE_HEAD = VE_USART_RX_DATA
     XT_USART_RX_DATA:
         .dw PFA_DOVARIABLE
     PFA_USART_RX_DATA:
         .dw usart_rx_data
     
     ; ( -- addr ) variable USART_RX_IN
     VE_USART_RX_IN:
         .dw $ff0b
         .db "USART_RX_IN",$00
         .dw VE_HEAD
         .set VE_HEAD = VE_USART_RX_IN
     XT_USART_RX_IN:
         .dw PFA_DOVARIABLE
     PFA_USART_RX_IN:
         .dw usart_rx_in
     
     ; ( -- value ) constant USART_RX_MASK
     VE_USART_RX_MASK:
         .dw $FF0D
         .db "USART_RX_MASK",$00
         .dw VE_HEAD
         .set VE_HEAD = VE_USART_RX_MASK
     XT_USART_RX_MASK:
         .dw PFA_DOVARIABLE
     PFA_USART_RX_MASK:
         .dw usart_rx_mask


   This provides the words :command:`USART_RX_DATA` :command:`USART_RX_IN`
   :command:`USART_RX_MASK` for our usage. Alternately we could setup our
   own variables and replace :command:`rx-isr` with a version looking at
   them.

6. handling an incoming byte according to MPC-mode

   .. code-block:: forth

       UCSRA_FE0
       UCSRA_DOR0  or
       UCSRA_UPE0  or constant UCSRA_RX_ERR
       : mpc?     UCSRA c@  UCSRA_MPCM0 and ;
       : rx-err?  UCSRA c@  UCSRA_RX_ERR and ;
       : rx-store ( udata -- )
         USART_RX_DATA USART_RX_IN c@  dup >r
         + !
         r> 1+ USART_RX_MASK and USART_RX_IN c!
       ;
       : mpc-rx-isr
         rx-err? 0= if
           UDR0 c@             \ -- udata
           mpc? if
             stationID = if
               -mpc7
             then
           else
             rx-store
           then
         then
       ;


   The new word command:`mpc-rx-isr` will inspect incoming data according
   to whether we are in MPC mode or not. It requires the node address in
   the value :command:`stationID` as defined before.

7. string everything together

   In order to use all of the above we basically need to switch it on
   (and off):

   .. code-block:: forth

     : +rs485.mpc
       ['] prompt_rd is p_rd         \ overwrite p_rd
       +rs485
       ['] mpc-rx-isr USART0__RXAddr int! \ overwrite usart_rx_isr
       -emit
       +mpc7
     ;
     
     : -rs485.mpc
       ['] (p_rd) is p_rd
       ['] usart_rx_isr USART0__RXAddr int!
       -rs485
       -mpc7
       +emit
     ;


   When using this in a turnkey word, make sure to disable :command:`emit`
   before calling the original word :command:`applturnkey`, because
   otherwise the output of :command:`ver` will be written to the bus.

   .. code-block:: forth

     : run-turnkey
       -emit
       applturnkey
       +rs485.mpc
     
       \ more initialization here
     
     \ begin
     \   your periodic work goes here
     \ again
     ;


We are all ready to go. Please note that you need some means to send
``0x80 | 0xStationAddress`` to the bus to address the desired node. Once
connected you need to issue :command:`+emit`, and only after that the
ok-prompt will show up.


6.8 going to quiet mode on unparsable input
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

After everything worked thus far I found out, that sometimes more than
one controller on the bus will be *awake* receiving data and acting on
it. Most of the time this would result in error messages being sent to
the bus, which in turn will create another round of error messages. I
called this *the echo loop*. I did not find out, what really caused
this behaviour, but instead I decided: whenever a node receives
*illegible* input that cannot be handled properly, the node shall
return to mpc *quiet* mode and not write any error messages at all.

The desired behaviour is a fairly fundamental change to the command
loop, however, it is easy to install thanks to the availability of
recognizers.

Any input will be parsed by a list of recognizers, the first to
*understand* the input will trigger the corresponding work. The last in the
list will be the one to possibly issue an error message. So we create a new
recognizer and insert it into the list of recognizers before the one
issueing error messages.

First we need to load the word :command:`recognizer:`

.. code-block:: forth

  include lib/recognizer.frt


After that we create a table holding 3 execution tokens. The first is
to be called at runtime, the second at compile time, and the third
during a postpone operation.

.. code-block:: forth

  :noname -emit +mpc7 ;    \ at runtime call the equivalent of ~end
  ' noop                   \ nothing to do at compile time
  :noname type -48 throw ; \ postpone would be an error
  recognizer: r:quiet


The parsing word does basically nothing. If this recognizer is called,
:command:`rec:word` and :command:`rec:num` have not been able to handle the
input. So we simply drop the references to the unhandled input before the
call into an entry of the newly created table :command:`r:quiet`.

.. code-block:: forth

  : rec:quiet  ( addr length -- t/f ) drop drop r:quiet ;


Registering and deregistering the new recognizer is a little involved,
because we want to place it at the last position --- if the last
recognizer fails, the content of r:fail is called. After some
fiddling, I decided to compare the last value with the one to be
inserted or removed, such that repeated calls to "+rec:quiet" or
"-rec:quiet" will not cause a problem.

.. code-block:: forth

  : +rec:quiet
    ['] rec:quiet         \ -- r0
    forth-recognizer get-recognizers       \ -- r0 r1 r2 2
    dup pick              \ -- r0 r1 r2 2 r1
    ['] rec:quiet <> if   \ -- r0 r1 r2 2
      1+
      forth-recognizer set-recognizers
    else
      drop
    then
  ;

  : -rec:quiet
    forth-recognizer get-recognizers       \ -- r0 r1 r2 3
    dup pick 
    ['] rec:quiet = if
      1- forth-recognizer set-recognizers drop
    else
      0 ?do drop loop
    then
  
  ;


:command:`+rec:quiet` needs to be called in :command:`+rs485.mpc` and
similar for :command:`-rec:quiet`.

.. code-block:: forth

  $28 constant USART0__RXAddr
  : +rs485.mpc
    ['] prompt_rd is p_rd         \ overwrite p_rd
    +rs485
    ['] mpc-rx-isr USART0__RXAddr int! \ overwrite usart_rx_isr
    +rec:quiet
    -emit
    +mpc7
  ;
  
  : -rs485.mpc
    ['] (p_rd) is p_rd
    ['] usart_rx_isr USART0__RXAddr int!
    -rec:quiet
    -rs485
    -mpc7
    +emit
  ;


6.9 turnkey
~~~~~~~~~~~

We are done. We can now put this together in a function to be called at
system boot. The controller will immediately switch off any output and go
to *quiet* mpc mode. As such the controller will behave well on a bus
with possibly other nodes.

.. code-block:: forth

  : run-turnkey
    -emit
    applturnkey
    +rs485.mpc
  ;


Please note that :command:`-emit` must be called before
:command:`applturnkey`, because the later does call :command:`ver`
producing the well known output

.. code-block:: none

  amforth 6.3 ATmega644P ok

or similar. But we do **not** want to write anything on the bus unless
explicitly asked to do so.



7 creating a node
-----------------

While the above implementation is *complete*, it may not be obvious,
how to create a sensor node with all the required bits around it. So at
least the description of a working example seems needed.

In my case the ``collector`` is a perl script, which will periodically
address a list of nodes and for each of these

- write the address byte ``0x80 | addr`` to the bus

- write :command:`+emit` after that (no echo characters expected)

- wait for the ok prompt

- write :command:`~data` to the bus (waiting for each echo character, since
  those are coming from the controller now)

- read all the characters which come as an answer, e.g.

  .. code-block:: none

    __Q 42:0005 4200:0  4201:0  4202:0  4203:0 C-- ok

  or

  .. code-block:: none

    __Q 7F:0005 7F01:3,+19.50,+19.50,+19.50  7F02:3,514,516,518 C-- ok


- write :command:`~end` to the bus (again waiting for each echo character)

The answer string is then parsed into pieces, and individual
measurements are then inserted into a database table.

The ``__Q`` and ``C--`` tokens were inserted only to make parsing simpler.
The second token consists of ``stationID:softwareVersion``, both as a hex
number. Tokens after that are either ``sensorID:Counter`` or
``sensorID:N,xlow,xmean,xhigh`` collections. Other formats are certainly
possible, this is just my choice based on the decision *its all plain ascii*.


This represents the *high level* view of the node as seen from the
*network* (aka bus).
So the words

- :command:`+emit`

- :command:`~data`

- :command:`~end`

must be available on the node.

So there are at least two ways to make :command:`~data` report meaningful output.

1. interrupt only sensors

   If all sensors can be handled by appropriate interrupt service routines,
   those shall fill the variables with meaningful values. :command:`~data`
   will then only read those values and report them over the bus.

   This setup is used for counters or *meters*. In my case the
   electricity meter has a so called `S0` interface with two pins ``+``
   and ``-``. ``+`` must be pulled high by a pullup resistor and connected
   to a controller pin. The meter will short the ``+`` to the ``-`` pin for
   a few milliseconds thus reporting one *count*. If the pin at the
   controller either can react on such a pulse by issueing an interrupt
   (external or pin change interrrupt) or if the pin is connected to a
   counter register, that's all there needs to be done. Every low pulse
   will increment the value reported by :command:`~data`.

.. figure:: p_S0_interface.png

2. using the multitasker to do the work in the background

   If there is more work to be done, either on event or periodically,
   then using the multitasker is an option. There are only two tasks
   involved: the task serving the command line and the task
   periodically collecting sensor readouts into variables. The handling
   of sensors or events could be spread over more tasks, if needed for
   some reason.

   If :command:`~data` is called on the command line, it will report the
   stored values and optionally reset the variables.


7.1 counter sensor
~~~~~~~~~~~~~~~~~~

As an example I will outline the needed bits for a counter node. It
will count *active-low* pulses on one of 4 pins. The controller is an
atmega168, which features *pin-change-interrupts* on all port pins. The
pulses are produced by an electricity meter with a so called `S0`
interface. This particular electricity meter will produce 1000 counts
per kWh consumed, each count consists of pulling pin ``+`` down for
90 milliseconds.


The pin change interrupt will trigger on falling and rising edges.
There is only one interrupt for a group of eight pins (one port). So
the interrupt service routine needs to find out, which pin exaclty
triggered the interrupt, and whether a falling or rising edge did
occur. On the falling edge we need to increment the associated counter
for this pin.

.. code-block:: forth

    \ --- data handling -------------------------
    variable Count 4 cells allot
    variable Pins_old
    : pci1_isr
      ledsensor high
      PINC c@ $0F and       \ -- pins 
      Pins_old c@           \ -- pins alt
      over                  \ -- pins alt pins
      xor                   \ -- pins diff
      dup if                \ -- pins diff

        4 0 do                          \ for each (consequtive input) pin
          dup  1 i lshift  and if       \ . bit one changed?
            over  1 i lshift  and 0= if \ . leading edge?
              1  Count i cells +  +!    \ . increment
            then
          then
        loop

      then
      ( diff ) drop
      ( pins ) Pins_old c!
      ledsensor low
    ;
    : +pci1
      $0F PCMSK1 c!  \ pcint 8..11 active
      $02 PCICR  c!  \ pci1 active
      $02 PCIFR  c!  \ clear PCI1, just in case
      PINC c@ $0F and  Pins_old !
      ['] pci1_isr PCINT1Addr int!
    ;
    : -pci1
      $00 PCICR  c!
      $02 PCIFR  c!  \ clear PCI1, just in case
     ;

The function :command:`~data` will then read the counter and report the value
found as a plain ascii string on the serial interface. No provisions
are taken to implement any access locking, reading the two bytes of
the counter might result in inconsistent values.

.. code-block:: forth

    \ counters are expected signed. A rollover can then be detected
    \ and distinguished from restart of the controller.
    \ therefore '.' not 'u.' in data.ls
    : data.ls
      4 0 do
        space stationID @ >< i + &4 hex u0.r colon  Count i cells + @ decimal .
      loop
    ;  
    : ~data
      leddata high     \ fixme: leddata

      ." __Q"       \ datagram start
      .id+ver       \ stationID + swVersion
      data.ls
      ." C--"       \ datagram end 
      leddata low
    ;

Since data is updated using an interrrupt service routine only, the
available command loop is available to service any requests from the
rs485--serial connection. If work has to be done outside the interrupt
service routine, a multitasker can be used to run two tasks: one to
read and process sensor data, and another one to run the command loop.



8 The *somtimes-not-so-obvious* things
--------------------------------------

As one of my lecturers kept saying: `**Afterwards** everything is obvious`.
However, the path to obviousness can be long and windy at times. 

1. Always provide a jumper to optionally disconnect the RX pin of the
   transceiver, IF you want to keep the existing RS232 transceiver
   working.

2. Consider adding jumpers to disconnect the bus. This is occasionally
   useful.


There is probably more to be said ...



9 Finally
---------

Like always this work would not have been possible without substantial
help from others. Special thanks go to Matthias Trute for amForth, for
providing valuable feedback and picking up suggestions; Lubos Pekny for
proving, that it can be done; the members of the amforth-devel mailing
list, the weekly IRC round and of the German "Forth Gesellschaft e.V.";
countless authors of documentation, code, or processes for all the
countless pieces of software that comprise my workstation setup, e.g.
bash, emacs and perl to name just three.


10 References
-------------

1. net4801 single board computer running the collector

   - `http://www.soekris.com <http://www.soekris.com>`_

2. Lubos Peknys `mFC` project using rs485 and mpc mode highly inspired this
   code and project

   - `http://www.forth.cz <http://www.forth.cz>`_

3. Pavel Pisa, implementing a 9-bit microLAN

   - `http://cmp.felk.cvut.cz/~pisa/papers/pi-ulan-prot.pdf <http://cmp.felk.cvut.cz/~pisa/papers/pi-ulan-prot.pdf>`_
   - `http://ulan.sourceforge.net/ <http://ulan.sourceforge.net/>`_