[PATCH can-gw] add support for counter byte

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Some devices that we communicate with require that a byte be incremented every time that a message is sent.  The patches below implement this functionality, with modifications made to both the can-gw kernel module and the cangw can-utils application.  The user specifies which byte of the message should be incremented and the maximum value that the counter should go up to before resetting and beginning to count again from 0.


can-gw (https://github.com/DV-Electronics/linux/tree/fbd509a7c1373157ade87b145823bfa4a0662cf6)

--- net/can/gw.c.orig   2018-08-29 10:07:59.255121361 -0400
+++ net/can/gw.c        2018-08-29 10:10:30.631121361 -0400
@@ -101,6 +101,14 @@ struct cf_mod {
        void (*modfunc[MAX_MODFUNCTIONS])(struct can_frame *cf,
                                          struct cf_mod *mod);

+       /* CAN frame counter increment after CAN frame modifications */
+       struct {
+               struct cgw_counter msgcounter;
+       } counter;
+       struct {
+               void (*msgcounter)(struct can_frame *cf, struct cgw_counter *msgcounter);
+       } counterfunc;
+
        /* CAN frame checksum calculation after CAN frame modifications */
        struct {
                struct cgw_csum_xor xor;
@@ -201,6 +209,23 @@ static int cgw_chk_csum_parms(s8 fr, s8
                return -EINVAL;
 }

+static int cgw_chk_result_idx_parm(s8 re)
+{
+       /*
+        * absolute dlc values 0 .. 7 => 0 .. 7, e.g. data [0]
+        * relative to received dlc -1 .. -8 :
+        * e.g. for received dlc = 8
+        * -1 => index = 7 (data[7])
+        * -3 => index = 5 (data[5])
+        * -8 => index = 0 (data[0])
+        */
+
+       if (re > -9 && re < 8)
+               return 0;
+       else
+               return -EINVAL;
+}
+
 static inline int calc_idx(int idx, int rx_dlc)
 {
        if (idx < 0)
@@ -346,6 +371,18 @@ static void cgw_csum_crc8_neg(struct can
        cf->data[crc8->result_idx] = crc^crc8->final_xor_val;
 }

+static void cgw_count(struct can_frame *cf, struct cgw_counter *msgcounter)
+{
+       u8 count = msgcounter->value + 1;
+
+       if(count > msgcounter->max_count)
+               count = 0; // roll over the counter
+
+       msgcounter->value = count; // increment for next time
+
+       cf->data[msgcounter->result_idx] = count;
+}
+
 /* the receive & process & send function */
 static void can_can_gw_rcv(struct sk_buff *skb, void *data)
 {
@@ -418,8 +455,12 @@ static void can_can_gw_rcv(struct sk_buf
        while (modidx < MAX_MODFUNCTIONS && gwj->mod.modfunc[modidx])
                (*gwj->mod.modfunc[modidx++])(cf, &gwj->mod);

-       /* check for checksum updates when the CAN frame has been modified */
        if (modidx) {
+               /* check for counter updates when the CAN frame has been modified */
+               if (gwj->mod.counterfunc.msgcounter)
+                       (*gwj->mod.counterfunc.msgcounter)(cf, &gwj->mod.counter.msgcounter);
+
+               /* check for checksum updates when the CAN frame has been modified */
                if (gwj->mod.csumfunc.crc8)
                        (*gwj->mod.csumfunc.crc8)(cf, &gwj->mod.csum.crc8);

@@ -554,6 +595,12 @@ static int cgw_put_job(struct sk_buff *s
                        goto cancel;
        }

+       if (gwj->mod.counterfunc.msgcounter) {
+               if (nla_put(skb, CGW_COUNTER, CGW_COUNTER_LEN,
+                           &gwj->mod.counter.msgcounter) < 0)
+                       goto cancel;
+       }
+
        if (gwj->mod.csumfunc.crc8) {
                if (nla_put(skb, CGW_CS_CRC8, CGW_CS_CRC8_LEN,
                            &gwj->mod.csum.crc8) < 0)
@@ -619,6 +666,7 @@ static const struct nla_policy cgw_polic
        [CGW_MOD_OR]    = { .len = sizeof(struct cgw_frame_mod) },
        [CGW_MOD_XOR]   = { .len = sizeof(struct cgw_frame_mod) },
        [CGW_MOD_SET]   = { .len = sizeof(struct cgw_frame_mod) },
+       [CGW_COUNTER]   = { .len = sizeof(struct cgw_counter) },
        [CGW_CS_XOR]    = { .len = sizeof(struct cgw_csum_xor) },
        [CGW_CS_CRC8]   = { .len = sizeof(struct cgw_csum_crc8) },
        [CGW_SRC_IF]    = { .type = NLA_U32 },
@@ -718,9 +766,22 @@ static int cgw_parse_attr(struct nlmsghd
                        mod->modfunc[modidx++] = mod_set_data;
        }

-       /* check for checksum operations after CAN frame modifications */
        if (modidx) {
+               /* check for counter operations after CAN frame modifications */
+               if (tb[CGW_COUNTER]) {
+                       struct cgw_counter *c = nla_data(tb[CGW_COUNTER]);
+
+                       err = cgw_chk_result_idx_parm(c->result_idx);
+                       if (err)
+                               return err;

+                       nla_memcpy(&mod->counter.msgcounter, tb[CGW_COUNTER],
+                                  CGW_COUNTER_LEN);
+
+                       mod->counterfunc.msgcounter = cgw_count;
+               }
+
+               /* check for checksum operations after CAN frame modifications */
                if (tb[CGW_CS_CRC8]) {
                        struct cgw_csum_crc8 *c = nla_data(tb[CGW_CS_CRC8]);



--- include/uapi/linux/can/gw.h.orig    2018-08-29 10:06:35.919121361 -0400
+++ include/uapi/linux/can/gw.h 2018-08-29 10:09:53.571121361 -0400
@@ -69,6 +69,7 @@ enum {
        CGW_MOD_OR,     /* CAN frame modification binary OR */
        CGW_MOD_XOR,    /* CAN frame modification binary XOR */
        CGW_MOD_SET,    /* CAN frame modification set alternate values */
+       CGW_COUNTER,    /* set message counter into data[index] */
        CGW_CS_XOR,     /* set data[] XOR checksum into data[index] */
        CGW_CS_CRC8,    /* set data[] CRC8 checksum into data[index] */
        CGW_HANDLED,    /* number of handled CAN frames */
@@ -106,6 +107,16 @@ struct cgw_frame_mod {

 #define CGW_MODATTR_LEN sizeof(struct cgw_frame_mod)

+/* message counter modifiers */
+struct cgw_counter {
+       __s8 result_idx;// byte index of the counter
+       __u8 max_count; // roll over counter when you hit this maximum
+       __u8 value;     // incremented every time that a message is sent
+} __attribute__((packed));
+/* length of message counter parameters. idx = index in CAN frame data[] */
+#define CGW_COUNTER_LEN  sizeof(struct cgw_counter)
+
+/* checksum modifiers */
 struct cgw_csum_xor {
        __s8 from_idx;
        __s8 to_idx;
@@ -161,7 +172,7 @@ enum {
  * Limit the number of hops of this specific rule. Usually the received CAN
  * frame can be processed as much as 'max_hops' times (which is given at module
  * load time of the can-gw module). This value is used to reduce the number of
- * possible hops for this gateway rule to a value smaller then max_hops.
+ * possible hops for this gateway rule to a value smaller than max_hops.
  *
  * CGW_MOD_UID (length 4 bytes):
  * Optional non-zero user defined routing job identifier to alter existing




can-utils (https://github.com/DV-Electronics/can-utils/tree/45c38854b984996abc0230aa5f556e7a734ad802)

--- cangw.c.orig        2018-08-29 10:26:08.679121361 -0400
+++ cangw.c     2018-08-29 10:29:01.663121361 -0400
@@ -184,6 +184,13 @@ void print_cs_crc8(struct cgw_csum_crc8
                print_cs_crc8_profile(cs_crc8);
 }

+void print_counter(struct cgw_counter *msgcounter)
+{
+       printf("-n %d:%d:%d ",
+              msgcounter->result_idx, msgcounter->max_count,
+               msgcounter->value);
+}
+
 void print_usage(char *prg)
 {
        fprintf(stderr, "\nUsage: %s [options]\n\n", prg);
@@ -200,6 +207,7 @@ void print_usage(char *prg)
        fprintf(stderr, "           -l <hops> (limit the number of frame hops / routings)\n");
        fprintf(stderr, "           -f <filter> (set CAN filter)\n");
        fprintf(stderr, "           -m <mod> (set frame modifications)\n");
+       fprintf(stderr, "           -n <result_idx>:<max_count>:<value> (message counter)\n");
        fprintf(stderr, "           -x <from_idx>:<to_idx>:<result_idx>:<init_xor_val> (XOR checksum)\n");
        fprintf(stderr, "           -c <from>:<to>:<result>:<init_val>:<xor_val>:<crctab[256]> (CRC8 cs)\n");
        fprintf(stderr, "           -p <profile>:[<profile_data>] (CRC8 checksum profile & parameters)\n");
@@ -421,6 +429,7 @@ int parse_rtlist(char *prgname, unsigned
                        case CGW_MOD_SET:
                        case CGW_MOD_UID:
                        case CGW_LIM_HOPS:
+                       case CGW_COUNTER:
                        case CGW_CS_XOR:
                        case CGW_CS_CRC8:
                                break;
@@ -501,6 +510,10 @@ int parse_rtlist(char *prgname, unsigned
                                printf("-l %d ", *(__u8 *)RTA_DATA(rta));
                                break;

+                       case CGW_COUNTER:
+                               print_counter((struct cgw_counter *)RTA_DATA(rta));
+                               break;
+
                        case CGW_CS_XOR:
                                print_cs_xor((struct cgw_csum_xor *)RTA_DATA(rta));
                                break;
@@ -542,6 +555,7 @@ int main(int argc, char **argv)

        int cmd = UNSPEC;
        int have_filter = 0;
+       int have_counter = 0;
        int have_cs_xor = 0;
        int have_cs_crc8 = 0;

@@ -565,6 +579,7 @@ int main(int argc, char **argv)
        struct can_filter filter;
        struct sockaddr_nl nladdr;

+       struct cgw_counter msgcounter;
        struct cgw_csum_xor cs_xor;
        struct cgw_csum_crc8 cs_crc8;
        char crc8tab[513] = {0};
@@ -574,10 +589,11 @@ int main(int argc, char **argv)
        int i;

        memset(&req, 0, sizeof(req));
+       memset(&msgcounter, 0, sizeof(msgcounter));
        memset(&cs_xor, 0, sizeof(cs_xor));
        memset(&cs_crc8, 0, sizeof(cs_crc8));

-       while ((opt = getopt(argc, argv, "ADFLs:d:teiu:l:f:c:p:x:m:?")) != -1) {
+       while ((opt = getopt(argc, argv, "ADFLs:d:teiu:l:f:c:p:n:x:m:?")) != -1) {
                switch (opt) {

                case 'A':
@@ -690,6 +706,17 @@ int main(int argc, char **argv)
                        exit(0);
                        break;

+               case 'n':       // increment result_idx by one each time that a message is sent, rolling over at max
+                       if ((sscanf(optarg, "%hhd:%hhd:%hhd",
+                                   &msgcounter.result_idx, &msgcounter.max_count,
+                                   &msgcounter.value) == 3)) {
+                               have_counter = 1;
+                       } else {
+                               printf("Bad counter definition '%s'.\n", optarg);
+                               exit(1);
+                       }
+                       break;
+
                default:
                        fprintf(stderr, "Unknown option %c\n", opt);
                        print_usage(basename(argv[0]));
@@ -709,8 +736,8 @@ int main(int argc, char **argv)
                exit(1);
        }

-       if (!modidx && (have_cs_crc8 || have_cs_xor)) {
-               printf("-c or -x can only be used in conjunction with -m\n");
+       if (!modidx && (have_counter || have_cs_crc8 || have_cs_xor)) {
+               printf("-n or -c or -x can only be used in conjunction with -m\n");
                exit(1);
        }

@@ -762,6 +789,9 @@ int main(int argc, char **argv)
        if (have_filter)
                addattr_l(&req.nh, sizeof(req), CGW_FILTER, &filter, sizeof(filter));

+       if (have_counter)
+               addattr_l(&req.nh, sizeof(req), CGW_COUNTER, &msgcounter, sizeof(msgcounter));
+
        if (have_cs_crc8)
                addattr_l(&req.nh, sizeof(req), CGW_CS_CRC8, &cs_crc8, sizeof(cs_crc8));



--- gw.h.orig   2018-08-29 10:32:19.951121361 -0400
+++ gw.h        2018-08-29 10:32:56.199121361 -0400
@@ -70,6 +70,7 @@ enum {
        CGW_MOD_OR,     /* CAN frame modification binary OR */
        CGW_MOD_XOR,    /* CAN frame modification binary XOR */
        CGW_MOD_SET,    /* CAN frame modification set alternate values */
+       CGW_COUNTER,    /* set message counter into data[index] */
        CGW_CS_XOR,     /* set data[] XOR checksum into data[index] */
        CGW_CS_CRC8,    /* set data[] CRC8 checksum into data[index] */
        CGW_HANDLED,    /* number of handled CAN frames */
@@ -107,6 +108,15 @@ struct cgw_frame_mod {

 #define CGW_MODATTR_LEN sizeof(struct cgw_frame_mod)

+/* message counter modifiers */
+struct cgw_counter {
+       __s8 result_idx;// byte index of the counter
+       __u8 max_count; // roll over counter when you hit this maximum
+       __u8 value;     // incremented every time that a message is sent
+} __attribute__((packed));
+/* length of message counter parameters. idx = index in CAN frame data[] */
+#define CGW_COUNTER_LEN  sizeof(struct cgw_counter)
+
 struct cgw_csum_xor {
        __s8 from_idx;
        __s8 to_idx;
@@ -162,7 +172,7 @@ enum {
  * Limit the number of hops of this specific rule. Usually the received CAN
  * frame can be processed as much as 'max_hops' times (which is given at module
  * load time of the can-gw module). This value is used to reduce the number of
- * possible hops for this gateway rule to a value smaller then max_hops.
+ * possible hops for this gateway rule to a value smaller than max_hops.
  *
  * CGW_MOD_UID (length 4 bytes):
  * Optional non-zero user defined routing job identifier to alter existing




Signed-off by: Brad Drehmer <b.drehmer@xxxxxxxxxxxxxxxxx>


TESTING THE FUTURE for OVER 20 YEARS


NOTICE: This e-mail and any attachments may contain confidential, privileged and proprietary information of D & V Electronics Ltd. and all such material is subject to copyright protection in favor of D & V Electronics Ltd. Any unauthorized disclosure or other use of this e-mail or the information contained herein or in any documents attached hereto may be unlawful and is strictly prohibited. If you have received this e-mail in error, please notify the sender immediately and delete this e-mail without reading, printing, copying or forwarding it to anyone. This email was sent from D & V Electronics Ltd. To receive no further email communications, you can send an email to unsubscribe@xxxxxxxxxxxxxxxxx. D & V Electronics is located at 130 Zenway Blvd. Woodbridge, Ontario, Canada.




[Index of Archives]     [Automotive Discussions]     [Linux ARM Kernel]     [Linux ARM]     [Linux Omap]     [Fedora ARM]     [IETF Annouce]     [Security]     [Bugtraq]     [Linux]     [Linux OMAP]     [Linux MIPS]     [eCos]     [Asterisk Internet PBX]     [Linux API]     [CAN Bus]

  Powered by Linux