mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2024-12-12 21:44:06 +08:00
72206cc730
n_gsm is based on the 3GPP 07.010 and its newer version is the 3GPP 27.010. See https://portal.3gpp.org/desktopmodules/Specifications/SpecificationDetails.aspx?specificationId=1516 The changes from 07.010 to 27.010 are non-functional. Therefore, I refer to the newer 27.010 here. Chapters 5.4.6.3.4 and 5.1.8.1.3 describe the test command which can be used to test the mux connection between both sides. Currently, no algorithm is implemented to make use of this command. This requires that each multiplexed upper layer protocol supervises the underlying muxer connection to handle possible connection losses. Introduce ioctl commands and functions to optionally enable keep alive handling via the test command as described in chapter 5.4.6.3.4. A single incrementing octet "ka_num" is being used for unique identification of each single keep alive packet. Retries will use the same "ka_num" value as the original packet. Retry count and interval are taken from the general parameters N2 and T2. Add usage description and basic example for the new ioctl to the n_gsm documentation. Note that support for the test command is mandatory and already present in the muxer implementation since the very first version. Also note that the previous ioctl structure gsm_config cannot be extended due to missing checks against zero of the field "unused". Signed-off-by: Daniel Starke <daniel.starke@siemens.com> Link: https://lore.kernel.org/r/20230214122737.1976-1-daniel.starke@siemens.com Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
173 lines
5.3 KiB
ReStructuredText
173 lines
5.3 KiB
ReStructuredText
==============================
|
|
GSM 0710 tty multiplexor HOWTO
|
|
==============================
|
|
|
|
.. contents:: :local:
|
|
|
|
This line discipline implements the GSM 07.10 multiplexing protocol
|
|
detailed in the following 3GPP document:
|
|
|
|
https://www.3gpp.org/ftp/Specs/archive/07_series/07.10/0710-720.zip
|
|
|
|
This document give some hints on how to use this driver with GPRS and 3G
|
|
modems connected to a physical serial port.
|
|
|
|
How to use it
|
|
=============
|
|
|
|
Config Initiator
|
|
----------------
|
|
|
|
#. Initialize the modem in 0710 mux mode (usually ``AT+CMUX=`` command) through
|
|
its serial port. Depending on the modem used, you can pass more or less
|
|
parameters to this command.
|
|
|
|
#. Switch the serial line to using the n_gsm line discipline by using
|
|
``TIOCSETD`` ioctl.
|
|
|
|
#. Configure the mux using ``GSMIOC_GETCONF_EXT``/``GSMIOC_SETCONF_EXT`` ioctl if needed.
|
|
|
|
#. Configure the mux using ``GSMIOC_GETCONF``/``GSMIOC_SETCONF`` ioctl.
|
|
|
|
#. Obtain base gsmtty number for the used serial port.
|
|
|
|
Major parts of the initialization program
|
|
(a good starting point is util-linux-ng/sys-utils/ldattach.c)::
|
|
|
|
#include <stdio.h>
|
|
#include <stdint.h>
|
|
#include <linux/gsmmux.h>
|
|
#include <linux/tty.h>
|
|
|
|
#define DEFAULT_SPEED B115200
|
|
#define SERIAL_PORT /dev/ttyS0
|
|
|
|
int ldisc = N_GSM0710;
|
|
struct gsm_config c;
|
|
struct gsm_config_ext ce;
|
|
struct termios configuration;
|
|
uint32_t first;
|
|
|
|
/* open the serial port connected to the modem */
|
|
fd = open(SERIAL_PORT, O_RDWR | O_NOCTTY | O_NDELAY);
|
|
|
|
/* configure the serial port : speed, flow control ... */
|
|
|
|
/* send the AT commands to switch the modem to CMUX mode
|
|
and check that it's successful (should return OK) */
|
|
write(fd, "AT+CMUX=0\r", 10);
|
|
|
|
/* experience showed that some modems need some time before
|
|
being able to answer to the first MUX packet so a delay
|
|
may be needed here in some case */
|
|
sleep(3);
|
|
|
|
/* use n_gsm line discipline */
|
|
ioctl(fd, TIOCSETD, &ldisc);
|
|
|
|
/* get n_gsm extended configuration */
|
|
ioctl(fd, GSMIOC_GETCONF_EXT, &ce);
|
|
/* use keep-alive once every 5s for modem connection supervision */
|
|
ce.keep_alive = 500;
|
|
/* set the new extended configuration */
|
|
ioctl(fd, GSMIOC_SETCONF_EXT, &ce);
|
|
/* get n_gsm configuration */
|
|
ioctl(fd, GSMIOC_GETCONF, &c);
|
|
/* we are initiator and need encoding 0 (basic) */
|
|
c.initiator = 1;
|
|
c.encapsulation = 0;
|
|
/* our modem defaults to a maximum size of 127 bytes */
|
|
c.mru = 127;
|
|
c.mtu = 127;
|
|
/* set the new configuration */
|
|
ioctl(fd, GSMIOC_SETCONF, &c);
|
|
/* get first gsmtty device node */
|
|
ioctl(fd, GSMIOC_GETFIRST, &first);
|
|
printf("first muxed line: /dev/gsmtty%i\n", first);
|
|
|
|
/* and wait for ever to keep the line discipline enabled */
|
|
daemon(0,0);
|
|
pause();
|
|
|
|
#. Use these devices as plain serial ports.
|
|
|
|
For example, it's possible:
|
|
|
|
- to use *gnokii* to send / receive SMS on ``ttygsm1``
|
|
- to use *ppp* to establish a datalink on ``ttygsm2``
|
|
|
|
#. First close all virtual ports before closing the physical port.
|
|
|
|
Note that after closing the physical port the modem is still in multiplexing
|
|
mode. This may prevent a successful re-opening of the port later. To avoid
|
|
this situation either reset the modem if your hardware allows that or send
|
|
a disconnect command frame manually before initializing the multiplexing mode
|
|
for the second time. The byte sequence for the disconnect command frame is::
|
|
|
|
0xf9, 0x03, 0xef, 0x03, 0xc3, 0x16, 0xf9
|
|
|
|
Config Requester
|
|
----------------
|
|
|
|
#. Receive ``AT+CMUX=`` command through its serial port, initialize mux mode
|
|
config.
|
|
|
|
#. Switch the serial line to using the *n_gsm* line discipline by using
|
|
``TIOCSETD`` ioctl.
|
|
|
|
#. Configure the mux using ``GSMIOC_GETCONF_EXT``/``GSMIOC_SETCONF_EXT``
|
|
ioctl if needed.
|
|
|
|
#. Configure the mux using ``GSMIOC_GETCONF``/``GSMIOC_SETCONF`` ioctl.
|
|
|
|
#. Obtain base gsmtty number for the used serial port::
|
|
|
|
#include <stdio.h>
|
|
#include <stdint.h>
|
|
#include <linux/gsmmux.h>
|
|
#include <linux/tty.h>
|
|
#define DEFAULT_SPEED B115200
|
|
#define SERIAL_PORT /dev/ttyS0
|
|
|
|
int ldisc = N_GSM0710;
|
|
struct gsm_config c;
|
|
struct gsm_config_ext ce;
|
|
struct termios configuration;
|
|
uint32_t first;
|
|
|
|
/* open the serial port */
|
|
fd = open(SERIAL_PORT, O_RDWR | O_NOCTTY | O_NDELAY);
|
|
|
|
/* configure the serial port : speed, flow control ... */
|
|
|
|
/* get serial data and check "AT+CMUX=command" parameter ... */
|
|
|
|
/* use n_gsm line discipline */
|
|
ioctl(fd, TIOCSETD, &ldisc);
|
|
|
|
/* get n_gsm extended configuration */
|
|
ioctl(fd, GSMIOC_GETCONF_EXT, &ce);
|
|
/* use keep-alive once every 5s for peer connection supervision */
|
|
ce.keep_alive = 500;
|
|
/* set the new extended configuration */
|
|
ioctl(fd, GSMIOC_SETCONF_EXT, &ce);
|
|
/* get n_gsm configuration */
|
|
ioctl(fd, GSMIOC_GETCONF, &c);
|
|
/* we are requester and need encoding 0 (basic) */
|
|
c.initiator = 0;
|
|
c.encapsulation = 0;
|
|
/* our modem defaults to a maximum size of 127 bytes */
|
|
c.mru = 127;
|
|
c.mtu = 127;
|
|
/* set the new configuration */
|
|
ioctl(fd, GSMIOC_SETCONF, &c);
|
|
/* get first gsmtty device node */
|
|
ioctl(fd, GSMIOC_GETFIRST, &first);
|
|
printf("first muxed line: /dev/gsmtty%i\n", first);
|
|
|
|
/* and wait for ever to keep the line discipline enabled */
|
|
daemon(0,0);
|
|
pause();
|
|
|
|
11-03-08 - Eric Bénard - <eric@eukrea.com>
|