cansend.c 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  1. /*
  2. * cansend.c - simple command line tool to send CAN-frames via CAN_RAW sockets
  3. *
  4. * Copyright (c) 2002-2007 Volkswagen Group Electronic Research
  5. * All rights reserved.
  6. *
  7. * Redistribution and use in source and binary forms, with or without
  8. * modification, are permitted provided that the following conditions
  9. * are met:
  10. * 1. Redistributions of source code must retain the above copyright
  11. * notice, this list of conditions and the following disclaimer.
  12. * 2. Redistributions in binary form must reproduce the above copyright
  13. * notice, this list of conditions and the following disclaimer in the
  14. * documentation and/or other materials provided with the distribution.
  15. * 3. Neither the name of Volkswagen nor the names of its contributors
  16. * may be used to endorse or promote products derived from this software
  17. * without specific prior written permission.
  18. *
  19. * Alternatively, provided that this notice is retained in full, this
  20. * software may be distributed under the terms of the GNU General
  21. * Public License ("GPL") version 2, in which case the provisions of the
  22. * GPL apply INSTEAD OF those given above.
  23. *
  24. * The provided data structures and external interfaces from this code
  25. * are not restricted to be used by modules with a GPL compatible license.
  26. *
  27. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  28. * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  29. * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
  30. * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
  31. * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  32. * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  33. * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
  34. * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
  35. * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  36. * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  37. * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
  38. * DAMAGE.
  39. *
  40. * Send feedback to <linux-can@vger.kernel.org>
  41. *
  42. */
  43. #include <stdio.h>
  44. #include <stdlib.h>
  45. #include <string.h>
  46. #include <unistd.h>
  47. #include <net/if.h>
  48. #include <sys/ioctl.h>
  49. #include <sys/socket.h>
  50. #include <linux/can.h>
  51. #include <linux/can/raw.h>
  52. #include "lib.h"
  53. int main(int argc, char **argv)
  54. {
  55. int s; /* can raw socket */
  56. int required_mtu;
  57. int mtu;
  58. int enable_canfd = 1;
  59. struct sockaddr_can addr;
  60. struct canfd_frame frame;
  61. struct ifreq ifr;
  62. /* check command line options */
  63. if (argc != 3) {
  64. fprintf(stderr, "Usage: %s <device> <can_frame>.\n", argv[0]);
  65. return 1;
  66. }
  67. /* parse CAN frame */
  68. required_mtu = parse_canframe(argv[2], &frame);
  69. if (!required_mtu){
  70. fprintf(stderr, "\nWrong CAN-frame format! Try:\n\n");
  71. fprintf(stderr, " <can_id>#{R|data} for CAN 2.0 frames\n");
  72. fprintf(stderr, " <can_id>##<flags>{data} for CAN FD frames\n\n");
  73. fprintf(stderr, "<can_id> can have 3 (SFF) or 8 (EFF) hex chars\n");
  74. fprintf(stderr, "{data} has 0..8 (0..64 CAN FD) ASCII hex-values (optionally");
  75. fprintf(stderr, " separated by '.')\n");
  76. fprintf(stderr, "<flags> a single ASCII Hex value (0 .. F) which defines");
  77. fprintf(stderr, " canfd_frame.flags\n\n");
  78. fprintf(stderr, "e.g. 5A1#11.2233.44556677.88 / 123#DEADBEEF / 5AA# / ");
  79. fprintf(stderr, "123##1 / 213##311\n 1F334455#1122334455667788 / 123#R ");
  80. fprintf(stderr, "for remote transmission request.\n\n");
  81. return 1;
  82. }
  83. /* open socket */
  84. if ((s = socket(PF_CAN, SOCK_RAW, CAN_RAW)) < 0) {
  85. perror("socket");
  86. return 1;
  87. }
  88. strncpy(ifr.ifr_name, argv[1], IFNAMSIZ - 1);
  89. ifr.ifr_name[IFNAMSIZ - 1] = '\0';
  90. ifr.ifr_ifindex = if_nametoindex(ifr.ifr_name);
  91. if (!ifr.ifr_ifindex) {
  92. perror("if_nametoindex");
  93. return 1;
  94. }
  95. addr.can_family = AF_CAN;
  96. addr.can_ifindex = ifr.ifr_ifindex;
  97. if (required_mtu > CAN_MTU) {
  98. /* check if the frame fits into the CAN netdevice */
  99. if (ioctl(s, SIOCGIFMTU, &ifr) < 0) {
  100. perror("SIOCGIFMTU");
  101. return 1;
  102. }
  103. mtu = ifr.ifr_mtu;
  104. if (mtu != CANFD_MTU) {
  105. printf("CAN interface is not CAN FD capable - sorry.\n");
  106. return 1;
  107. }
  108. /* interface is ok - try to switch the socket into CAN FD mode */
  109. if (setsockopt(s, SOL_CAN_RAW, CAN_RAW_FD_FRAMES,
  110. &enable_canfd, sizeof(enable_canfd))){
  111. printf("error when enabling CAN FD support\n");
  112. return 1;
  113. }
  114. /* ensure discrete CAN FD length values 0..8, 12, 16, 20, 24, 32, 64 */
  115. frame.len = can_dlc2len(can_len2dlc(frame.len));
  116. }
  117. /* disable default receive filter on this RAW socket */
  118. /* This is obsolete as we do not read from the socket at all, but for */
  119. /* this reason we can remove the receive list in the Kernel to save a */
  120. /* little (really a very little!) CPU usage. */
  121. setsockopt(s, SOL_CAN_RAW, CAN_RAW_FILTER, NULL, 0);
  122. if (bind(s, (struct sockaddr *)&addr, sizeof(addr)) < 0) {
  123. perror("bind");
  124. return 1;
  125. }
  126. /* send frame */
  127. if (write(s, &frame, required_mtu) != required_mtu) {
  128. perror("write");
  129. return 1;
  130. }
  131. close(s);
  132. return 0;
  133. }