]> Git Repo - linux.git/blobdiff - net/can/isotp.c
mm: Clear page->private when splitting or migrating a page
[linux.git] / net / can / isotp.c
index 1e7c6a460ef9a39d2c313c8cbb964b73585f2fbf..43a27d19cdacfddeace6d7a740b0531ab4c31b96 100644 (file)
@@ -104,6 +104,7 @@ MODULE_ALIAS("can-proto-6");
 #define FC_CONTENT_SZ 3        /* flow control content size in byte (FS/BS/STmin) */
 
 #define ISOTP_CHECK_PADDING (CAN_ISOTP_CHK_PAD_LEN | CAN_ISOTP_CHK_PAD_DATA)
+#define ISOTP_ALL_BC_FLAGS (CAN_ISOTP_SF_BROADCAST | CAN_ISOTP_CF_BROADCAST)
 
 /* Flow Status given in FC frame */
 #define ISOTP_FC_CTS 0         /* clear to send */
@@ -159,6 +160,23 @@ static inline struct isotp_sock *isotp_sk(const struct sock *sk)
        return (struct isotp_sock *)sk;
 }
 
+static u32 isotp_bc_flags(struct isotp_sock *so)
+{
+       return so->opt.flags & ISOTP_ALL_BC_FLAGS;
+}
+
+static bool isotp_register_rxid(struct isotp_sock *so)
+{
+       /* no broadcast modes => register rx_id for FC frame reception */
+       return (isotp_bc_flags(so) == 0);
+}
+
+static bool isotp_register_txecho(struct isotp_sock *so)
+{
+       /* all modes but SF_BROADCAST register for tx echo skbs */
+       return (isotp_bc_flags(so) != CAN_ISOTP_SF_BROADCAST);
+}
+
 static enum hrtimer_restart isotp_rx_timer_handler(struct hrtimer *hrtimer)
 {
        struct isotp_sock *so = container_of(hrtimer, struct isotp_sock,
@@ -803,7 +821,6 @@ static void isotp_create_fframe(struct canfd_frame *cf, struct isotp_sock *so,
                cf->data[i] = so->tx.buf[so->tx.idx++];
 
        so->tx.sn = 1;
-       so->tx.state = ISOTP_WAIT_FIRST_FC;
 }
 
 static void isotp_rcv_echo(struct sk_buff *skb, void *data)
@@ -936,7 +953,7 @@ static int isotp_sendmsg(struct socket *sock, struct msghdr *msg, size_t size)
        off = (so->tx.ll_dl > CAN_MAX_DLEN) ? 1 : 0;
 
        /* does the given data fit into a single frame for SF_BROADCAST? */
-       if ((so->opt.flags & CAN_ISOTP_SF_BROADCAST) &&
+       if ((isotp_bc_flags(so) == CAN_ISOTP_SF_BROADCAST) &&
            (size > so->tx.ll_dl - SF_PCI_SZ4 - ae - off)) {
                err = -EINVAL;
                goto err_out_drop;
@@ -1000,12 +1017,41 @@ static int isotp_sendmsg(struct socket *sock, struct msghdr *msg, size_t size)
                /* don't enable wait queue for a single frame transmission */
                wait_tx_done = 0;
        } else {
-               /* send first frame and wait for FC */
+               /* send first frame */
 
                isotp_create_fframe(cf, so, ae);
 
-               /* start timeout for FC */
-               hrtimer_sec = 1;
+               if (isotp_bc_flags(so) == CAN_ISOTP_CF_BROADCAST) {
+                       /* set timer for FC-less operation (STmin = 0) */
+                       if (so->opt.flags & CAN_ISOTP_FORCE_TXSTMIN)
+                               so->tx_gap = ktime_set(0, so->force_tx_stmin);
+                       else
+                               so->tx_gap = ktime_set(0, so->frame_txtime);
+
+                       /* disable wait for FCs due to activated block size */
+                       so->txfc.bs = 0;
+
+                       /* cfecho should have been zero'ed by init */
+                       if (so->cfecho)
+                               pr_notice_once("can-isotp: no fc cfecho %08X\n",
+                                              so->cfecho);
+
+                       /* set consecutive frame echo tag */
+                       so->cfecho = *(u32 *)cf->data;
+
+                       /* switch directly to ISOTP_SENDING state */
+                       so->tx.state = ISOTP_SENDING;
+
+                       /* start timeout for unlikely lost echo skb */
+                       hrtimer_sec = 2;
+               } else {
+                       /* standard flow control check */
+                       so->tx.state = ISOTP_WAIT_FIRST_FC;
+
+                       /* start timeout for FC */
+                       hrtimer_sec = 1;
+               }
+
                hrtimer_start(&so->txtimer, ktime_set(hrtimer_sec, 0),
                              HRTIMER_MODE_REL_SOFT);
        }
@@ -1025,6 +1071,9 @@ static int isotp_sendmsg(struct socket *sock, struct msghdr *msg, size_t size)
                if (hrtimer_sec)
                        hrtimer_cancel(&so->txtimer);
 
+               /* reset consecutive frame echo tag */
+               so->cfecho = 0;
+
                goto err_out_drop;
        }
 
@@ -1055,7 +1104,6 @@ static int isotp_recvmsg(struct socket *sock, struct msghdr *msg, size_t size,
        struct sock *sk = sock->sk;
        struct sk_buff *skb;
        struct isotp_sock *so = isotp_sk(sk);
-       int noblock = flags & MSG_DONTWAIT;
        int ret = 0;
 
        if (flags & ~(MSG_DONTWAIT | MSG_TRUNC | MSG_PEEK))
@@ -1064,8 +1112,7 @@ static int isotp_recvmsg(struct socket *sock, struct msghdr *msg, size_t size,
        if (!so->bound)
                return -EADDRNOTAVAIL;
 
-       flags &= ~MSG_DONTWAIT;
-       skb = skb_recv_datagram(sk, flags, noblock, &ret);
+       skb = skb_recv_datagram(sk, flags, &ret);
        if (!skb)
                return ret;
 
@@ -1122,15 +1169,17 @@ static int isotp_release(struct socket *sock)
        lock_sock(sk);
 
        /* remove current filters & unregister */
-       if (so->bound && (!(so->opt.flags & CAN_ISOTP_SF_BROADCAST))) {
+       if (so->bound && isotp_register_txecho(so)) {
                if (so->ifindex) {
                        struct net_device *dev;
 
                        dev = dev_get_by_index(net, so->ifindex);
                        if (dev) {
-                               can_rx_unregister(net, dev, so->rxid,
-                                                 SINGLE_MASK(so->rxid),
-                                                 isotp_rcv, sk);
+                               if (isotp_register_rxid(so))
+                                       can_rx_unregister(net, dev, so->rxid,
+                                                         SINGLE_MASK(so->rxid),
+                                                         isotp_rcv, sk);
+
                                can_rx_unregister(net, dev, so->txid,
                                                  SINGLE_MASK(so->txid),
                                                  isotp_rcv_echo, sk);
@@ -1163,26 +1212,35 @@ static int isotp_bind(struct socket *sock, struct sockaddr *uaddr, int len)
        struct net *net = sock_net(sk);
        int ifindex;
        struct net_device *dev;
-       canid_t tx_id, rx_id;
+       canid_t tx_id = addr->can_addr.tp.tx_id;
+       canid_t rx_id = addr->can_addr.tp.rx_id;
        int err = 0;
        int notify_enetdown = 0;
-       int do_rx_reg = 1;
 
        if (len < ISOTP_MIN_NAMELEN)
                return -EINVAL;
 
-       /* sanitize tx/rx CAN identifiers */
-       tx_id = addr->can_addr.tp.tx_id;
+       /* sanitize tx CAN identifier */
        if (tx_id & CAN_EFF_FLAG)
                tx_id &= (CAN_EFF_FLAG | CAN_EFF_MASK);
        else
                tx_id &= CAN_SFF_MASK;
 
-       rx_id = addr->can_addr.tp.rx_id;
-       if (rx_id & CAN_EFF_FLAG)
-               rx_id &= (CAN_EFF_FLAG | CAN_EFF_MASK);
-       else
-               rx_id &= CAN_SFF_MASK;
+       /* give feedback on wrong CAN-ID value */
+       if (tx_id != addr->can_addr.tp.tx_id)
+               return -EINVAL;
+
+       /* sanitize rx CAN identifier (if needed) */
+       if (isotp_register_rxid(so)) {
+               if (rx_id & CAN_EFF_FLAG)
+                       rx_id &= (CAN_EFF_FLAG | CAN_EFF_MASK);
+               else
+                       rx_id &= CAN_SFF_MASK;
+
+               /* give feedback on wrong CAN-ID value */
+               if (rx_id != addr->can_addr.tp.rx_id)
+                       return -EINVAL;
+       }
 
        if (!addr->can_ifindex)
                return -ENODEV;
@@ -1194,12 +1252,8 @@ static int isotp_bind(struct socket *sock, struct sockaddr *uaddr, int len)
                goto out;
        }
 
-       /* do not register frame reception for functional addressing */
-       if (so->opt.flags & CAN_ISOTP_SF_BROADCAST)
-               do_rx_reg = 0;
-
-       /* do not validate rx address for functional addressing */
-       if (do_rx_reg && rx_id == tx_id) {
+       /* ensure different CAN IDs when the rx_id is to be registered */
+       if (isotp_register_rxid(so) && rx_id == tx_id) {
                err = -EADDRNOTAVAIL;
                goto out;
        }
@@ -1224,10 +1278,11 @@ static int isotp_bind(struct socket *sock, struct sockaddr *uaddr, int len)
 
        ifindex = dev->ifindex;
 
-       if (do_rx_reg) {
+       if (isotp_register_rxid(so))
                can_rx_register(net, dev, rx_id, SINGLE_MASK(rx_id),
                                isotp_rcv, sk, "isotp", sk);
 
+       if (isotp_register_txecho(so)) {
                /* no consecutive frame echo skb in flight */
                so->cfecho = 0;
 
@@ -1296,6 +1351,15 @@ static int isotp_setsockopt_locked(struct socket *sock, int level, int optname,
                if (!(so->opt.flags & CAN_ISOTP_RX_EXT_ADDR))
                        so->opt.rx_ext_address = so->opt.ext_address;
 
+               /* these broadcast flags are not allowed together */
+               if (isotp_bc_flags(so) == ISOTP_ALL_BC_FLAGS) {
+                       /* CAN_ISOTP_SF_BROADCAST is prioritized */
+                       so->opt.flags &= ~CAN_ISOTP_CF_BROADCAST;
+
+                       /* give user feedback on wrong config attempt */
+                       ret = -EINVAL;
+               }
+
                /* check for frame_txtime changes (0 => no changes) */
                if (so->opt.frame_txtime) {
                        if (so->opt.frame_txtime == CAN_ISOTP_FRAME_TXTIME_ZERO)
@@ -1446,10 +1510,12 @@ static void isotp_notify(struct isotp_sock *so, unsigned long msg,
        case NETDEV_UNREGISTER:
                lock_sock(sk);
                /* remove current filters & unregister */
-               if (so->bound && (!(so->opt.flags & CAN_ISOTP_SF_BROADCAST))) {
-                       can_rx_unregister(dev_net(dev), dev, so->rxid,
-                                         SINGLE_MASK(so->rxid),
-                                         isotp_rcv, sk);
+               if (so->bound && isotp_register_txecho(so)) {
+                       if (isotp_register_rxid(so))
+                               can_rx_unregister(dev_net(dev), dev, so->rxid,
+                                                 SINGLE_MASK(so->rxid),
+                                                 isotp_rcv, sk);
+
                        can_rx_unregister(dev_net(dev), dev, so->txid,
                                          SINGLE_MASK(so->txid),
                                          isotp_rcv_echo, sk);
This page took 0.042556 seconds and 4 git commands to generate.