1*de1e4e89SAndroid Build Coastguard Worker /*
2*de1e4e89SAndroid Build Coastguard Worker * iplink_hsr.c HSR device support
3*de1e4e89SAndroid Build Coastguard Worker *
4*de1e4e89SAndroid Build Coastguard Worker * This program is free software; you can redistribute it and/or
5*de1e4e89SAndroid Build Coastguard Worker * modify it under the terms of the GNU General Public License
6*de1e4e89SAndroid Build Coastguard Worker * as published by the Free Software Foundation; either version
7*de1e4e89SAndroid Build Coastguard Worker * 2 of the License, or (at your option) any later version.
8*de1e4e89SAndroid Build Coastguard Worker *
9*de1e4e89SAndroid Build Coastguard Worker * Authors: Arvid Brodin <[email protected]>
10*de1e4e89SAndroid Build Coastguard Worker *
11*de1e4e89SAndroid Build Coastguard Worker * Based on iplink_vlan.c by Patrick McHardy <[email protected]>
12*de1e4e89SAndroid Build Coastguard Worker */
13*de1e4e89SAndroid Build Coastguard Worker
14*de1e4e89SAndroid Build Coastguard Worker #include <stdio.h>
15*de1e4e89SAndroid Build Coastguard Worker #include <stdlib.h>
16*de1e4e89SAndroid Build Coastguard Worker #include <string.h>
17*de1e4e89SAndroid Build Coastguard Worker #include <sys/socket.h> /* Needed by linux/if.h for some reason */
18*de1e4e89SAndroid Build Coastguard Worker #include <linux/if.h>
19*de1e4e89SAndroid Build Coastguard Worker #include <linux/if_arp.h>
20*de1e4e89SAndroid Build Coastguard Worker #include "rt_names.h"
21*de1e4e89SAndroid Build Coastguard Worker #include "utils.h"
22*de1e4e89SAndroid Build Coastguard Worker #include "ip_common.h"
23*de1e4e89SAndroid Build Coastguard Worker
print_usage(FILE * f)24*de1e4e89SAndroid Build Coastguard Worker static void print_usage(FILE *f)
25*de1e4e89SAndroid Build Coastguard Worker {
26*de1e4e89SAndroid Build Coastguard Worker fprintf(f,
27*de1e4e89SAndroid Build Coastguard Worker "Usage:\tip link add name NAME type hsr slave1 SLAVE1-IF slave2 SLAVE2-IF\n"
28*de1e4e89SAndroid Build Coastguard Worker "\t[ supervision ADDR-BYTE ] [version VERSION]\n"
29*de1e4e89SAndroid Build Coastguard Worker "\n"
30*de1e4e89SAndroid Build Coastguard Worker "NAME\n"
31*de1e4e89SAndroid Build Coastguard Worker " name of new hsr device (e.g. hsr0)\n"
32*de1e4e89SAndroid Build Coastguard Worker "SLAVE1-IF, SLAVE2-IF\n"
33*de1e4e89SAndroid Build Coastguard Worker " the two slave devices bound to the HSR device\n"
34*de1e4e89SAndroid Build Coastguard Worker "ADDR-BYTE\n"
35*de1e4e89SAndroid Build Coastguard Worker " 0-255; the last byte of the multicast address used for HSR supervision\n"
36*de1e4e89SAndroid Build Coastguard Worker " frames (default = 0)\n"
37*de1e4e89SAndroid Build Coastguard Worker "VERSION\n"
38*de1e4e89SAndroid Build Coastguard Worker " 0,1; the protocol version to be used. (default = 0)\n");
39*de1e4e89SAndroid Build Coastguard Worker }
40*de1e4e89SAndroid Build Coastguard Worker
usage(void)41*de1e4e89SAndroid Build Coastguard Worker static void usage(void)
42*de1e4e89SAndroid Build Coastguard Worker {
43*de1e4e89SAndroid Build Coastguard Worker print_usage(stderr);
44*de1e4e89SAndroid Build Coastguard Worker }
45*de1e4e89SAndroid Build Coastguard Worker
hsr_parse_opt(struct link_util * lu,int argc,char ** argv,struct nlmsghdr * n)46*de1e4e89SAndroid Build Coastguard Worker static int hsr_parse_opt(struct link_util *lu, int argc, char **argv,
47*de1e4e89SAndroid Build Coastguard Worker struct nlmsghdr *n)
48*de1e4e89SAndroid Build Coastguard Worker {
49*de1e4e89SAndroid Build Coastguard Worker int ifindex;
50*de1e4e89SAndroid Build Coastguard Worker unsigned char multicast_spec;
51*de1e4e89SAndroid Build Coastguard Worker unsigned char protocol_version;
52*de1e4e89SAndroid Build Coastguard Worker
53*de1e4e89SAndroid Build Coastguard Worker while (argc > 0) {
54*de1e4e89SAndroid Build Coastguard Worker if (matches(*argv, "supervision") == 0) {
55*de1e4e89SAndroid Build Coastguard Worker NEXT_ARG();
56*de1e4e89SAndroid Build Coastguard Worker if (get_u8(&multicast_spec, *argv, 0))
57*de1e4e89SAndroid Build Coastguard Worker invarg("ADDR-BYTE is invalid", *argv);
58*de1e4e89SAndroid Build Coastguard Worker addattr_l(n, 1024, IFLA_HSR_MULTICAST_SPEC,
59*de1e4e89SAndroid Build Coastguard Worker &multicast_spec, 1);
60*de1e4e89SAndroid Build Coastguard Worker } else if (matches(*argv, "version") == 0) {
61*de1e4e89SAndroid Build Coastguard Worker NEXT_ARG();
62*de1e4e89SAndroid Build Coastguard Worker if (!(get_u8(&protocol_version, *argv, 0) == 0 ||
63*de1e4e89SAndroid Build Coastguard Worker get_u8(&protocol_version, *argv, 0) == 1))
64*de1e4e89SAndroid Build Coastguard Worker invarg("version is invalid", *argv);
65*de1e4e89SAndroid Build Coastguard Worker addattr_l(n, 1024, IFLA_HSR_VERSION,
66*de1e4e89SAndroid Build Coastguard Worker &protocol_version, 1);
67*de1e4e89SAndroid Build Coastguard Worker } else if (matches(*argv, "slave1") == 0) {
68*de1e4e89SAndroid Build Coastguard Worker NEXT_ARG();
69*de1e4e89SAndroid Build Coastguard Worker ifindex = ll_name_to_index(*argv);
70*de1e4e89SAndroid Build Coastguard Worker if (ifindex == 0)
71*de1e4e89SAndroid Build Coastguard Worker invarg("No such interface", *argv);
72*de1e4e89SAndroid Build Coastguard Worker addattr_l(n, 1024, IFLA_HSR_SLAVE1, &ifindex, 4);
73*de1e4e89SAndroid Build Coastguard Worker } else if (matches(*argv, "slave2") == 0) {
74*de1e4e89SAndroid Build Coastguard Worker NEXT_ARG();
75*de1e4e89SAndroid Build Coastguard Worker ifindex = ll_name_to_index(*argv);
76*de1e4e89SAndroid Build Coastguard Worker if (ifindex == 0)
77*de1e4e89SAndroid Build Coastguard Worker invarg("No such interface", *argv);
78*de1e4e89SAndroid Build Coastguard Worker addattr_l(n, 1024, IFLA_HSR_SLAVE2, &ifindex, 4);
79*de1e4e89SAndroid Build Coastguard Worker } else if (matches(*argv, "help") == 0) {
80*de1e4e89SAndroid Build Coastguard Worker usage();
81*de1e4e89SAndroid Build Coastguard Worker return -1;
82*de1e4e89SAndroid Build Coastguard Worker } else {
83*de1e4e89SAndroid Build Coastguard Worker fprintf(stderr, "hsr: what is \"%s\"?\n", *argv);
84*de1e4e89SAndroid Build Coastguard Worker usage();
85*de1e4e89SAndroid Build Coastguard Worker return -1;
86*de1e4e89SAndroid Build Coastguard Worker }
87*de1e4e89SAndroid Build Coastguard Worker argc--, argv++;
88*de1e4e89SAndroid Build Coastguard Worker }
89*de1e4e89SAndroid Build Coastguard Worker
90*de1e4e89SAndroid Build Coastguard Worker return 0;
91*de1e4e89SAndroid Build Coastguard Worker }
92*de1e4e89SAndroid Build Coastguard Worker
hsr_print_opt(struct link_util * lu,FILE * f,struct rtattr * tb[])93*de1e4e89SAndroid Build Coastguard Worker static void hsr_print_opt(struct link_util *lu, FILE *f, struct rtattr *tb[])
94*de1e4e89SAndroid Build Coastguard Worker {
95*de1e4e89SAndroid Build Coastguard Worker SPRINT_BUF(b1);
96*de1e4e89SAndroid Build Coastguard Worker
97*de1e4e89SAndroid Build Coastguard Worker if (!tb)
98*de1e4e89SAndroid Build Coastguard Worker return;
99*de1e4e89SAndroid Build Coastguard Worker
100*de1e4e89SAndroid Build Coastguard Worker if (tb[IFLA_HSR_SLAVE1] &&
101*de1e4e89SAndroid Build Coastguard Worker RTA_PAYLOAD(tb[IFLA_HSR_SLAVE1]) < sizeof(__u32))
102*de1e4e89SAndroid Build Coastguard Worker return;
103*de1e4e89SAndroid Build Coastguard Worker if (tb[IFLA_HSR_SLAVE2] &&
104*de1e4e89SAndroid Build Coastguard Worker RTA_PAYLOAD(tb[IFLA_HSR_SLAVE2]) < sizeof(__u32))
105*de1e4e89SAndroid Build Coastguard Worker return;
106*de1e4e89SAndroid Build Coastguard Worker if (tb[IFLA_HSR_SEQ_NR] &&
107*de1e4e89SAndroid Build Coastguard Worker RTA_PAYLOAD(tb[IFLA_HSR_SEQ_NR]) < sizeof(__u16))
108*de1e4e89SAndroid Build Coastguard Worker return;
109*de1e4e89SAndroid Build Coastguard Worker if (tb[IFLA_HSR_SUPERVISION_ADDR] &&
110*de1e4e89SAndroid Build Coastguard Worker RTA_PAYLOAD(tb[IFLA_HSR_SUPERVISION_ADDR]) < ETH_ALEN)
111*de1e4e89SAndroid Build Coastguard Worker return;
112*de1e4e89SAndroid Build Coastguard Worker
113*de1e4e89SAndroid Build Coastguard Worker if (tb[IFLA_HSR_SLAVE1])
114*de1e4e89SAndroid Build Coastguard Worker print_string(PRINT_ANY,
115*de1e4e89SAndroid Build Coastguard Worker "slave1",
116*de1e4e89SAndroid Build Coastguard Worker "slave1 %s ",
117*de1e4e89SAndroid Build Coastguard Worker ll_index_to_name(rta_getattr_u32(tb[IFLA_HSR_SLAVE1])));
118*de1e4e89SAndroid Build Coastguard Worker else
119*de1e4e89SAndroid Build Coastguard Worker print_null(PRINT_ANY, "slave1", "slave1 %s ", "<none>");
120*de1e4e89SAndroid Build Coastguard Worker
121*de1e4e89SAndroid Build Coastguard Worker if (tb[IFLA_HSR_SLAVE2])
122*de1e4e89SAndroid Build Coastguard Worker print_string(PRINT_ANY,
123*de1e4e89SAndroid Build Coastguard Worker "slave2",
124*de1e4e89SAndroid Build Coastguard Worker "slave2 %s ",
125*de1e4e89SAndroid Build Coastguard Worker ll_index_to_name(rta_getattr_u32(tb[IFLA_HSR_SLAVE2])));
126*de1e4e89SAndroid Build Coastguard Worker else
127*de1e4e89SAndroid Build Coastguard Worker print_null(PRINT_ANY, "slave2", "slave2 %s ", "<none>");
128*de1e4e89SAndroid Build Coastguard Worker
129*de1e4e89SAndroid Build Coastguard Worker if (tb[IFLA_HSR_SEQ_NR])
130*de1e4e89SAndroid Build Coastguard Worker print_int(PRINT_ANY,
131*de1e4e89SAndroid Build Coastguard Worker "seq_nr",
132*de1e4e89SAndroid Build Coastguard Worker "sequence %d ",
133*de1e4e89SAndroid Build Coastguard Worker rta_getattr_u16(tb[IFLA_HSR_SEQ_NR]));
134*de1e4e89SAndroid Build Coastguard Worker
135*de1e4e89SAndroid Build Coastguard Worker if (tb[IFLA_HSR_SUPERVISION_ADDR])
136*de1e4e89SAndroid Build Coastguard Worker print_string(PRINT_ANY,
137*de1e4e89SAndroid Build Coastguard Worker "supervision_addr",
138*de1e4e89SAndroid Build Coastguard Worker "supervision %s ",
139*de1e4e89SAndroid Build Coastguard Worker ll_addr_n2a(RTA_DATA(tb[IFLA_HSR_SUPERVISION_ADDR]),
140*de1e4e89SAndroid Build Coastguard Worker RTA_PAYLOAD(tb[IFLA_HSR_SUPERVISION_ADDR]),
141*de1e4e89SAndroid Build Coastguard Worker ARPHRD_VOID,
142*de1e4e89SAndroid Build Coastguard Worker b1, sizeof(b1)));
143*de1e4e89SAndroid Build Coastguard Worker }
144*de1e4e89SAndroid Build Coastguard Worker
hsr_print_help(struct link_util * lu,int argc,char ** argv,FILE * f)145*de1e4e89SAndroid Build Coastguard Worker static void hsr_print_help(struct link_util *lu, int argc, char **argv,
146*de1e4e89SAndroid Build Coastguard Worker FILE *f)
147*de1e4e89SAndroid Build Coastguard Worker {
148*de1e4e89SAndroid Build Coastguard Worker print_usage(f);
149*de1e4e89SAndroid Build Coastguard Worker }
150*de1e4e89SAndroid Build Coastguard Worker
151*de1e4e89SAndroid Build Coastguard Worker struct link_util hsr_link_util = {
152*de1e4e89SAndroid Build Coastguard Worker .id = "hsr",
153*de1e4e89SAndroid Build Coastguard Worker .maxattr = IFLA_HSR_MAX,
154*de1e4e89SAndroid Build Coastguard Worker .parse_opt = hsr_parse_opt,
155*de1e4e89SAndroid Build Coastguard Worker .print_opt = hsr_print_opt,
156*de1e4e89SAndroid Build Coastguard Worker .print_help = hsr_print_help,
157*de1e4e89SAndroid Build Coastguard Worker };
158