| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386 |
- .. SPDX-License-Identifier: (GPL-2.0 OR BSD-3-Clause)
- ====================
- ISO 15765-2 (ISO-TP)
- ====================
- Overview
- ========
- ISO 15765-2, also known as ISO-TP, is a transport protocol specifically defined
- for diagnostic communication on CAN. It is widely used in the automotive
- industry, for example as the transport protocol for UDSonCAN (ISO 14229-3) or
- emission-related diagnostic services (ISO 15031-5).
- ISO-TP can be used both on CAN CC (aka Classical CAN) and CAN FD (CAN with
- Flexible Datarate) based networks. It is also designed to be compatible with a
- CAN network using SAE J1939 as data link layer (however, this is not a
- requirement).
- Specifications used
- -------------------
- * ISO 15765-2:2024 : Road vehicles - Diagnostic communication over Controller
- Area Network (DoCAN). Part 2: Transport protocol and network layer services.
- Addressing
- ----------
- In its simplest form, ISO-TP is based on two kinds of addressing modes for the
- nodes connected to the same network:
- * physical addressing is implemented by two node-specific addresses and is used
- in 1-to-1 communication.
- * functional addressing is implemented by one node-specific address and is used
- in 1-to-N communication.
- Three different addressing formats can be employed:
- * "normal" : each address is represented simply by a CAN ID.
- * "extended": each address is represented by a CAN ID plus the first byte of
- the CAN payload; both the CAN ID and the byte inside the payload shall be
- different between two addresses.
- * "mixed": each address is represented by a CAN ID plus the first byte of
- the CAN payload; the CAN ID is different between two addresses, but the
- additional byte is the same.
- Transport protocol and associated frame types
- ---------------------------------------------
- When transmitting data using the ISO-TP protocol, the payload can either fit
- inside one single CAN message or not, also considering the overhead the protocol
- is generating and the optional extended addressing. In the first case, the data
- is transmitted at once using a so-called Single Frame (SF). In the second case,
- ISO-TP defines a multi-frame protocol, in which the sender provides (through a
- First Frame - FF) the PDU length which is to be transmitted and also asks for a
- Flow Control (FC) frame, which provides the maximum supported size of a macro
- data block (``blocksize``) and the minimum time between the single CAN messages
- composing such block (``stmin``). Once this information has been received, the
- sender starts to send frames containing fragments of the data payload (called
- Consecutive Frames - CF), stopping after every ``blocksize``-sized block to wait
- confirmation from the receiver which should then send another Flow Control
- frame to inform the sender about its availability to receive more data.
- How to Use ISO-TP
- =================
- As with others CAN protocols, the ISO-TP stack support is built into the
- Linux network subsystem for the CAN bus, aka. Linux-CAN or SocketCAN, and
- thus follows the same socket API.
- Creation and basic usage of an ISO-TP socket
- --------------------------------------------
- To use the ISO-TP stack, ``#include <linux/can/isotp.h>`` shall be used. A
- socket can then be created using the ``PF_CAN`` protocol family, the
- ``SOCK_DGRAM`` type (as the underlying protocol is datagram-based by design)
- and the ``CAN_ISOTP`` protocol:
- .. code-block:: C
- s = socket(PF_CAN, SOCK_DGRAM, CAN_ISOTP);
- After the socket has been successfully created, ``bind(2)`` shall be called to
- bind the socket to the desired CAN interface; to do so:
- * a TX CAN ID shall be specified as part of the sockaddr supplied to the call
- itself.
- * a RX CAN ID shall also be specified, unless broadcast flags have been set
- through socket option (explained below).
- Once bound to an interface, the socket can be read from and written to using
- the usual ``read(2)`` and ``write(2)`` system calls, as well as ``send(2)``,
- ``sendmsg(2)``, ``recv(2)`` and ``recvmsg(2)``.
- Unlike the CAN_RAW socket API, only the ISO-TP data field (the actual payload)
- is sent and received by the userspace application using these calls. The address
- information and the protocol information are automatically filled by the ISO-TP
- stack using the configuration supplied during socket creation. In the same way,
- the stack will use the transport mechanism when required (i.e., when the size
- of the data payload exceeds the MTU of the underlying CAN bus).
- The sockaddr structure used for SocketCAN has extensions for use with ISO-TP,
- as specified below:
- .. code-block:: C
- struct sockaddr_can {
- sa_family_t can_family;
- int can_ifindex;
- union {
- struct { canid_t rx_id, tx_id; } tp;
- ...
- } can_addr;
- }
- * ``can_family`` and ``can_ifindex`` serve the same purpose as for other
- SocketCAN sockets.
- * ``can_addr.tp.rx_id`` specifies the receive (RX) CAN ID and will be used as
- a RX filter.
- * ``can_addr.tp.tx_id`` specifies the transmit (TX) CAN ID
- ISO-TP socket options
- ---------------------
- When creating an ISO-TP socket, reasonable defaults are set. Some options can
- be modified with ``setsockopt(2)`` and/or read back with ``getsockopt(2)``.
- General options
- ~~~~~~~~~~~~~~~
- General socket options can be passed using the ``CAN_ISOTP_OPTS`` optname:
- .. code-block:: C
- struct can_isotp_options opts;
- ret = setsockopt(s, SOL_CAN_ISOTP, CAN_ISOTP_OPTS, &opts, sizeof(opts))
- where the ``can_isotp_options`` structure has the following contents:
- .. code-block:: C
- struct can_isotp_options {
- u32 flags;
- u32 frame_txtime;
- u8 ext_address;
- u8 txpad_content;
- u8 rxpad_content;
- u8 rx_ext_address;
- };
- * ``flags``: modifiers to be applied to the default behaviour of the ISO-TP
- stack. Following flags are available:
- * ``CAN_ISOTP_LISTEN_MODE``: listen only (do not send FC frames); normally
- used as a testing feature.
- * ``CAN_ISOTP_EXTEND_ADDR``: use the byte specified in ``ext_address`` as an
- additional address component. This enables the "mixed" addressing format if
- used alone, or the "extended" addressing format if used in conjunction with
- ``CAN_ISOTP_RX_EXT_ADDR``.
- * ``CAN_ISOTP_TX_PADDING``: enable padding for transmitted frames, using
- ``txpad_content`` as value for the padding bytes.
- * ``CAN_ISOTP_RX_PADDING``: enable padding for the received frames, using
- ``rxpad_content`` as value for the padding bytes.
- * ``CAN_ISOTP_CHK_PAD_LEN``: check for correct padding length on the received
- frames.
- * ``CAN_ISOTP_CHK_PAD_DATA``: check padding bytes on the received frames
- against ``rxpad_content``; if ``CAN_ISOTP_RX_PADDING`` is not specified,
- this flag is ignored.
- * ``CAN_ISOTP_HALF_DUPLEX``: force ISO-TP socket in half duplex mode
- (that is, transport mechanism can only be incoming or outgoing at the same
- time, not both).
- * ``CAN_ISOTP_FORCE_TXSTMIN``: ignore stmin from received FC; normally
- used as a testing feature.
- * ``CAN_ISOTP_FORCE_RXSTMIN``: ignore CFs depending on rx stmin; normally
- used as a testing feature.
- * ``CAN_ISOTP_RX_EXT_ADDR``: use ``rx_ext_address`` instead of ``ext_address``
- as extended addressing byte on the reception path. If used in conjunction
- with ``CAN_ISOTP_EXTEND_ADDR``, this flag effectively enables the "extended"
- addressing format.
- * ``CAN_ISOTP_WAIT_TX_DONE``: wait until the frame is sent before returning
- from ``write(2)`` and ``send(2)`` calls (i.e., blocking write operations).
- * ``CAN_ISOTP_SF_BROADCAST``: use 1-to-N functional addressing (cannot be
- specified alongside ``CAN_ISOTP_CF_BROADCAST``).
- * ``CAN_ISOTP_CF_BROADCAST``: use 1-to-N transmission without flow control
- (cannot be specified alongside ``CAN_ISOTP_SF_BROADCAST``).
- NOTE: this is not covered by the ISO 15765-2 standard.
- * ``CAN_ISOTP_DYN_FC_PARMS``: enable dynamic update of flow control
- parameters.
- * ``frame_txtime``: frame transmission time (defined as N_As/N_Ar inside the
- ISO standard); if ``0``, the default (or the last set value) is used.
- To set the transmission time to ``0``, the ``CAN_ISOTP_FRAME_TXTIME_ZERO``
- macro (equal to 0xFFFFFFFF) shall be used.
- * ``ext_address``: extended addressing byte, used if the
- ``CAN_ISOTP_EXTEND_ADDR`` flag is specified.
- * ``txpad_content``: byte used as padding value for transmitted frames.
- * ``rxpad_content``: byte used as padding value for received frames.
- * ``rx_ext_address``: extended addressing byte for the reception path, used if
- the ``CAN_ISOTP_RX_EXT_ADDR`` flag is specified.
- Flow Control options
- ~~~~~~~~~~~~~~~~~~~~
- Flow Control (FC) options can be passed using the ``CAN_ISOTP_RECV_FC`` optname
- to provide the communication parameters for receiving ISO-TP PDUs.
- .. code-block:: C
- struct can_isotp_fc_options fc_opts;
- ret = setsockopt(s, SOL_CAN_ISOTP, CAN_ISOTP_RECV_FC, &fc_opts, sizeof(fc_opts));
- where the ``can_isotp_fc_options`` structure has the following contents:
- .. code-block:: C
- struct can_isotp_options {
- u8 bs;
- u8 stmin;
- u8 wftmax;
- };
- * ``bs``: blocksize provided in flow control frames.
- * ``stmin``: minimum separation time provided in flow control frames; can
- have the following values (others are reserved):
- * 0x00 - 0x7F : 0 - 127 ms
- * 0xF1 - 0xF9 : 100 us - 900 us
- * ``wftmax``: maximum number of wait frames provided in flow control frames.
- Link Layer options
- ~~~~~~~~~~~~~~~~~~
- Link Layer (LL) options can be passed using the ``CAN_ISOTP_LL_OPTS`` optname:
- .. code-block:: C
- struct can_isotp_ll_options ll_opts;
- ret = setsockopt(s, SOL_CAN_ISOTP, CAN_ISOTP_LL_OPTS, &ll_opts, sizeof(ll_opts));
- where the ``can_isotp_ll_options`` structure has the following contents:
- .. code-block:: C
- struct can_isotp_ll_options {
- u8 mtu;
- u8 tx_dl;
- u8 tx_flags;
- };
- * ``mtu``: generated and accepted CAN frame type, can be equal to ``CAN_MTU``
- for classical CAN frames or ``CANFD_MTU`` for CAN FD frames.
- * ``tx_dl``: maximum payload length for transmitted frames, can have one value
- among: 8, 12, 16, 20, 24, 32, 48, 64. Values above 8 only apply to CAN FD
- traffic (i.e.: ``mtu = CANFD_MTU``).
- * ``tx_flags``: flags set into ``struct canfd_frame.flags`` at frame creation.
- Only applies to CAN FD traffic (i.e.: ``mtu = CANFD_MTU``).
- Transmission stmin
- ~~~~~~~~~~~~~~~~~~
- The transmission minimum separation time (stmin) can be forced using the
- ``CAN_ISOTP_TX_STMIN`` optname and providing an stmin value in microseconds as
- a 32bit unsigned integer; this will overwrite the value sent by the receiver in
- flow control frames:
- .. code-block:: C
- uint32_t stmin;
- ret = setsockopt(s, SOL_CAN_ISOTP, CAN_ISOTP_TX_STMIN, &stmin, sizeof(stmin));
- Reception stmin
- ~~~~~~~~~~~~~~~
- The reception minimum separation time (stmin) can be forced using the
- ``CAN_ISOTP_RX_STMIN`` optname and providing an stmin value in microseconds as
- a 32bit unsigned integer; received Consecutive Frames (CF) which timestamps
- differ less than this value will be ignored:
- .. code-block:: C
- uint32_t stmin;
- ret = setsockopt(s, SOL_CAN_ISOTP, CAN_ISOTP_RX_STMIN, &stmin, sizeof(stmin));
- Multi-frame transport support
- -----------------------------
- The ISO-TP stack contained inside the Linux kernel supports the multi-frame
- transport mechanism defined by the standard, with the following constraints:
- * the maximum size of a PDU is defined by a module parameter, with an hard
- limit imposed at build time.
- * when a transmission is in progress, subsequent calls to ``write(2)`` will
- block, while calls to ``send(2)`` will either block or fail depending on the
- presence of the ``MSG_DONTWAIT`` flag.
- * no support is present for sending "wait frames": whether a PDU can be fully
- received or not is decided when the First Frame is received.
- Errors
- ------
- Following errors are reported to userspace:
- RX path errors
- ~~~~~~~~~~~~~~
- ============ ===============================================================
- -ETIMEDOUT timeout of data reception
- -EILSEQ sequence number mismatch during a multi-frame reception
- -EBADMSG data reception with wrong padding
- ============ ===============================================================
- TX path errors
- ~~~~~~~~~~~~~~
- ========== =================================================================
- -ECOMM flow control reception timeout
- -EMSGSIZE flow control reception overflow
- -EBADMSG flow control reception with wrong layout/padding
- ========== =================================================================
- Examples
- ========
- Basic node example
- ------------------
- Following example implements a node using "normal" physical addressing, with
- RX ID equal to 0x18DAF142 and a TX ID equal to 0x18DA42F1. All options are left
- to their default.
- .. code-block:: C
- int s;
- struct sockaddr_can addr;
- int ret;
- s = socket(PF_CAN, SOCK_DGRAM, CAN_ISOTP);
- if (s < 0)
- exit(1);
- addr.can_family = AF_CAN;
- addr.can_ifindex = if_nametoindex("can0");
- addr.can_addr.tp.tx_id = 0x18DA42F1 | CAN_EFF_FLAG;
- addr.can_addr.tp.rx_id = 0x18DAF142 | CAN_EFF_FLAG;
- ret = bind(s, (struct sockaddr *)&addr, sizeof(addr));
- if (ret < 0)
- exit(1);
- /* Data can now be received using read(s, ...) and sent using write(s, ...) */
- Additional examples
- -------------------
- More complete (and complex) examples can be found inside the ``isotp*`` userland
- tools, distributed as part of the ``can-utils`` utilities at:
- https://github.com/linux-can/can-utils
|