Dear Kurt, On Tue, 25 Jun 2019 19:31:37 +0200 Kurt Van Dijck <dev.kurt@xxxxxxxxxxxxxxxxxxxxxx> wrote: > On di, 25 jun 2019 10:54:55 +0200, Oleksij Rempel wrote: > > On 25.06.19 10:43, David Jander wrote: > > >On Tue, 25 Jun 2019 09:30:09 +0200 > > >Oleksij Rempel <o.rempel@xxxxxxxxxxxxxx> wrote: > > > > > >>Hello all, > > >> > > >>We already had a discussion about the J1939 use case for server > > >>implementation. Short description of the challenge will looks as follow: > > >>- main socket is listening on DST address and PGN. > > >>- as soon as connection was requested from peer the server will > > >> create new connect()ed socket with SRC, DST addresses and PGN. > > >> > > >>With current stack implementation both sockets (main and connected) will > > >>receive same packages. At least with huge packages it will start to be > > >>not really good :). > > >> > > >>To solve this issue we have following variants: > > >>variant a: > > >> - all sockets can receive everything (as currently implemented) > > >>variant b: > > >> - only one socket will receive specific tuple. In this case kernel > > >> should calculate RX "priority". Only highest priority will RX packet. > > >> - all sockets with same priority will receive the matching packet > > >> - socket option promisc == same priority as exact match > > > > > >How is this "priority" determined? > > >Something like this: > > > > > > for each socket: > > > prio = 0 > > > listening on same DST or PGN ==> prio++ > > > listening on same DST and PGN ==> prio++ > > > connect()ed to same SRC ==> prio++ > > > deliver frame to socket(s) with highest prio. > > > > > >Is that what you mean? > > > > ACK. > > I don't like any of these. > > The problem you try to solve is 'huge packet duplication where it is > probably not required'. > Your proposed solution puts a policy in the kernel that goes in serious > conflict with a multiuser system. It is driven by a typical > implementation, but did not address the problem you try to solve. > > In order to avoid receiving huge packets where we suspect it is not > really wanted, we should not try to guess what 'a' program wants, nor > implement rules that apply to 1 specific case. > Instead, we should protect sockets from receiving huge packets. > > Why not add a socket option, that implements a ceiling on the > size of received packets. > If that defaults to, let's say, 1785 bytes, so anyone will out of the > box receive all TP sessions, but no ETP session, then the user will not > be really supprised, and we need to make only 1 clear decision during delivery. > > I honestly think that my proprosal puts way less decision policy in the > kernel code, and effectively addresses the problem you tried to solve, > without adding unnecessary multi-user restrictions. > > What's your thought? Thanks for your feedback. I understand it may sound counter-intuitive, but it really isn't. What we are trying to accomplish is for SocketCAN with J1939 to behave much like a network adapter with TCP/IP. The solution you propose is not enough. The problem really is not restricted to "big" messages. If there are a lot of small frames coming from one client, you really do not want all the unnecessary interruptions to other sockets, but only the one that is connected to that client. That is actually one of the main reasons to have J1939 in the kernel as opposed to a user-space implementation... besides of course the obvious gains related to (E)TP. The underlying problem here is that we are trying to have the same sort of "connection oriented" sockets as in TCP/IP, but on a J1939-CAN network there is no way of initiating nor terminating a connection, and bigger data "streams" are artificially encapsulated in these (E)TP sessions. The semantics of J1939 and ISObus nevertheless are very similar to TCP/IP: There are servers that bind() to an address and port number (NAME and PGN), and clients that connect() to a server from a source address and -port number (client-NAME and client-PGN). The problem is that in the "server" case, just as in TCP/IP you would want a new socket to be created (as is done with the accept() system call in TCP/IP) for each new client that connect()s. For TCP/IP there is a defined sequence of TCP messages (SYN/ACK/FIN/...) that initiates and terminates a "connection". Such a stateful protocol inherently requires time-outs on the connection level to work. Probably one of the reasons why J1939 is much simpler and stateless, due to it's real-time requirements. Anyway, since the notion of a "connection" is a lot more vague in J1939, there is some cooperation needed from user-space in order to decide when a connection is established, and when it is closed. We cannot have an accept() system call for J1939 unfortunately. Instead of that, the user-space application needs to open a new socket that does bind() and then connect() to the client's NAME and PGN. At that point (same as with an accept()ed TCP/IP connection) all traffic coming from that specific NAME and PGN to the "server" should be delivered only on that socket. And exactly this is what we are trying to accomplish now, while we only have a bunch of sockets owned by one application and no stateful information of the connection state of each of those sockets. Imagine a J1939 "server" that has 20 "clients" connected to it. It will thus have 21 open sockets (one for the server itself (the bind()ed socket) and one for each of the "clients" (the bind()ed and connect()ed sockets). Now imagine the trouble of having to deal with the fact that every single message from client A is received on all 21 sockets duplicated! You don't want that. Not for big messages, nor for small ones. Best regards, -- David Jander Protonic Holland.