liyujie
2025-08-28 786ff4f4ca2374bdd9177f2e24b503d43e7a3b93
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
/*
 * iplink_vxcan.c    vxcan device support (Virtual CAN Tunnel)
 *
 *        This program is free software; you can redistribute it and/or
 *        modify it under the terms of the GNU General Public License
 *        as published by the Free Software Foundation; either version
 *        2 of the License, or (at your option) any later version.
 *
 * Author:    Oliver Hartkopp <socketcan@hartkopp.net>
 * Based on:    link_veth.c from Pavel Emelianov <xemul@openvz.org>
 */
 
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <net/if.h>
#include <linux/can/vxcan.h>
 
#include "utils.h"
#include "ip_common.h"
 
static void print_usage(FILE *f)
{
   printf("Usage: ip link <options> type vxcan [peer <options>]\n"
          "To get <options> type 'ip link add help'\n");
}
 
static void usage(void)
{
   print_usage(stderr);
}
 
static int vxcan_parse_opt(struct link_util *lu, int argc, char **argv,
             struct nlmsghdr *hdr)
{
   char *dev = NULL;
   char *name = NULL;
   char *link = NULL;
   char *type = NULL;
   int index = 0;
   int err, len;
   struct rtattr *data;
   int group;
   struct ifinfomsg *ifm, *peer_ifm;
   unsigned int ifi_flags, ifi_change;
 
   if (strcmp(argv[0], "peer") != 0) {
       usage();
       return -1;
   }
 
   ifm = NLMSG_DATA(hdr);
   ifi_flags = ifm->ifi_flags;
   ifi_change = ifm->ifi_change;
   ifm->ifi_flags = 0;
   ifm->ifi_change = 0;
 
   data = NLMSG_TAIL(hdr);
   addattr_l(hdr, 1024, VXCAN_INFO_PEER, NULL, 0);
 
   hdr->nlmsg_len += sizeof(struct ifinfomsg);
 
   err = iplink_parse(argc - 1, argv + 1, (struct iplink_req *)hdr,
              &name, &type, &link, &dev, &group, &index);
   if (err < 0)
       return err;
 
   if (name) {
       len = strlen(name) + 1;
       if (len > IFNAMSIZ)
           invarg("\"name\" too long\n", *argv);
       addattr_l(hdr, 1024, IFLA_IFNAME, name, len);
   }
 
   peer_ifm = RTA_DATA(data);
   peer_ifm->ifi_index = index;
   peer_ifm->ifi_flags = ifm->ifi_flags;
   peer_ifm->ifi_change = ifm->ifi_change;
   ifm->ifi_flags = ifi_flags;
   ifm->ifi_change = ifi_change;
 
   if (group != -1)
       addattr32(hdr, 1024, IFLA_GROUP, group);
 
   data->rta_len = (void *)NLMSG_TAIL(hdr) - (void *)data;
   return argc - 1 - err;
}
 
static void vxcan_print_help(struct link_util *lu, int argc, char **argv,
   FILE *f)
{
   print_usage(f);
}
 
struct link_util vxcan_link_util = {
   .id = "vxcan",
   .parse_opt = vxcan_parse_opt,
   .print_help = vxcan_print_help,
};