/fscc-linux

FSCC Linux Driver

Primary LanguageCMIT LicenseMIT

fscc-linux

This README file is best viewed online.

Installing Driver

Downloading Driver Source Code

The source code for the FSCC driver is hosted on Github code hosting. To check out the latest code you will need Git and to run the following in a terminal.

git clone --recursive git://github.com/commtech/fscc-linux.git fscc

You can also download driver packages directly from our website.

Switch To Stable Version

Now that you have the latest code checked out, switch to the latest stable version v2.2.1 is only listed here as an example.

git tag
git checkout v2.2.1
Build Source Code

Run the make command from within the source code directory to build the driver.

cd fscc/
make
cd serialfc/
make

If you would like to enable debug prints within the driver you need to add the DEBUG option while building the driver.

make DEBUG=1

Once debugging is enabled you will find extra kernel prints in the /var/log/messages and /var/log/debug log files.

If the kernel header files you would like to build against are not in the default location /lib/modules/$(shell uname -r)/build then you can specify the location with the KDIR option while building the driver.

make KDIR="/location/to/kernel_headers/"
Loading Driver

Assuming the driver has been successfully built in the previous step you are now ready to load the driver so you can begin using it. To do this you insert the driver's kernel object file (fscc.ko) into the kernel.

cd fscc/
insmod fscc.ko
cd serialfc/
insmod serialfc.ko

You will more than likely need administrator privileges for this and the following commands.

If no cards are present you will see the following message.

insmod fscc.ko
insmod: error inserting 'fscc.ko': -1 No such device

All driver load time options can be set in your modprobe.conf file for using upon system boot

Installing Driver

If you would like the driver to automatically load at boot use the included installer.

cd fscc/
make install
cd serialfc/
make install

This will also install the header (.h) files.

To uninstall, use the included uninstaller.

cd fscc/
make uninstall
cd serialfc/
make uninstall

Quick Start Guide

There is documentation for each specific function listed below, but lets get started with a quick programming example for fun.

This tutorial has already been set up for you at fscc/examples/tutorial.c.

Create a new C file (named tutorial.c) with the following code.

#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>

int main(void)
{
    int fd = 0;
    char odata[] = "Hello world!";
    char idata[20];

    /* Open port 0 */
    fd = open("/dev/fscc0", O_RDWR);

    if (fd == -1) {
        perror("open");
        return EXIT_FAILURE;
    }

    /* Send "Hello world!" text */
    write(fd, odata, sizeof(odata));

    /* Read the data back in (with our loopback connector) */
    read(fd, idata, sizeof(idata));

    fprintf(stdout, "%s\n", idata);

    close(fd);

    return EXIT_SUCCESS;
}

For this example I will use the gcc compiler, but you can use your compiler of choice.

# gcc -I ..\lib\raw\ tutorial.c

Now attach the included loopback connector.

# ./a.out
Hello world!

You have now transmitted and received an HDLC frame!

API Reference

There are likely other configuration options you will need to set up for your own program. All of these options are described on their respective documentation page.

There are also multiple code libraries to make development easier.

Asynchronous Communication

By default the FSCC driver boots up in synchronous communication mode. To switch to the asynchronous mode you must modify the FSCC card's FCR register to allow for asynchronous communication. There are multiple ways of doing this. Possibly the simplest method is using sysfs and the command line.

echo 03000000 > /sys/class/fscc/fscc0/registers/fcr

More information about using the UART's is available in the SerialFC driver README file.

FAQ

Can I use the old legacy utilties with the new driver?

No. The old utilities use a different API and will not work with the new driver.

Migrating From 1.x to 2.x

There are multiple benefits of using the 2.x driver: accurate posix error codes, intuitive ioctl calls, backend support for multiple languages (Python, C#) and dynamic memory management are some.

The 1.x driver and the 2.x driver are very similar so porting from one to the other should be rather painless.

All ioctl values have changed even if their new names match their old names. This means even if you use an ioctl with an identical name, it will not work correctly.

Setting register values was split into two different ioctl's in the 1.x driver, setting all the registers at once and one at a time. In the 2.x driver these two scenarios have been combined into one ioctl.

Change the following ioctl's to the current ioctl FSCC_SET_REGISTERS (see section V).

FSCC_WRITE_REGISTER (setting a single register at a time) FSCC_INIT (setting all registers at a time)

Getting register values was limited to one at a time in the 1.x driver. In the 2.x driver it has been made more convenient to read multiple register values.

Change the following ioctl to the current ioctl FSCC_GET_REGISTERS (see section VI).

FSCC_READ_REGISTER (reading a single register at a time)

Purging transmit and receive data has not changed. Continue using FSCC_PURGE_TX and FSCC_PURGE_RX.

Getting the frame status has now been designed to be configurable. In the 1.x driver you would always have the frame status appended to your data on a read. In the 2.x driver this can be toggled, and defaults to not appending the status to the data.

Changing the clock frequency is basically the same but the data structure and ioctl name are different.

Change the following ioctl to the current ioctl FSCC_SET_CLOCK_BITS (see section VIII).

FSCC_SET_FREQ (setting the clock frequency)

In the 1.x driver you passed in a structure composed of both the desired frequency and the clock bits that represent the frequency. In the 2.x driver this has been simplified down to just the clock bits.

Why are the /dev/fscc* ports not created even though the driver has loaded?

There are a couple of possibilities but you should first check /var/log/messages for any helpful information. If that doesn't help you out then continue reading.

One possibility is that there is another driver loaded that has claimed our cards. For example if your kernel is patched to use our card for asynchronous transmission the linux serial driver has more than likely claimed the card and in turn we won't receive the appropriate 'probe' notifications to load our card.

Another possibility is that you have accidentally tried insmod'ing with the 'hot_plug' option enabled and your cards are not actually present. Double check that your card shows up in the output of 'lspci' and make sure to use hot_plug=0.

What does poll() and select() base their information on?

Whether or not you can read data will be based on if there is at least 1 byte of data available to be read in your current mode of operation. For example, if there is streaming data it will not be considered when in a frame based mode.

Whether or not you can write data will be based on if you have hit your output memory cap.

Why does executing a purge without a clock put the card in a broken state?

When executing a purge on either the transmitter or receiver there is a TRES or RRES (command from the CMDR register) happening behind the scenes. If there is no clock available the command will stall until a clock is available. This should work in theory but doesn't in practice. So whenever you need to execute a purge without a clock, first put it into clock mode 7, execute your purge then return to your other clock mode.

Why am I receiving the error message 'Couldn't register serial port' when loading the driver.

When loading, the driver will attempt to register the board's UARTs with the built in serial driver. The serial driver statically defines how many UARTs can be registered and will report this error if there isn't enough room.

There are multiple ways of allowing more available UART room which can be found in section VII.

Build Dependencies

  • Kernel Build Tools (GCC, make, kernel headers, etc)

Run-time Dependencies

  • OS: Linux
  • Base Installation: >= 2.6.16 (might work with a lower version)
  • Sysfs Support: >= 2.6.25

API Compatibility

We follow Semantic Versioning when creating releases.

License

Copyright (C) 2019 Commtech, Inc.

Licensed under the MIT license.