blob: ef1324522690bd4e939838d35f3015f968064c47 [file] [log] [blame]
developer22c7ab62022-01-24 11:13:32 +08001/* Copyright (C) 2021-2022 Mediatek Inc. */
2
3#include <signal.h>
4#include <sys/select.h>
5#include <sys/wait.h>
6#include "atenl.h"
7
8static const char *progname;
9bool atenl_enable;
10
11void sig_handler(int signum)
12{
13 atenl_enable = false;
14}
15
16void atenl_init_signals()
17{
18 if (signal(SIGINT, sig_handler) == SIG_ERR)
19 goto out;
20 if (signal(SIGTERM, sig_handler) == SIG_ERR)
21 goto out;
22 if (signal(SIGABRT, sig_handler) == SIG_ERR)
23 goto out;
24 if (signal(SIGUSR1, sig_handler) == SIG_ERR)
25 goto out;
26 if (signal(SIGUSR2, sig_handler) == SIG_ERR)
27 goto out;
28
29 return;
30out:
31 perror("signal");
32}
33
34static int phy_lookup_idx(const char *name)
35{
36 char buf[128];
37 FILE *f;
38 size_t len;
39 int ret;
40
41 ret = snprintf(buf, sizeof(buf), "/sys/class/ieee80211/%s/index", name);
42 if (snprintf_error(sizeof(buf), ret))
43 return -1;
44
45 f = fopen(buf, "r");
46 if (!f)
47 return -1;
48
49 len = fread(buf, 1, sizeof(buf) - 1, f);
50 fclose(f);
51
52 if (!len)
53 return -1;
54
55 buf[len] = 0;
56 return atoi(buf);
57}
58
developer543811b2022-10-07 15:54:07 +080059static int get_default_bridge_name(struct atenl *an)
60{
61 char buf[128];
62 FILE *f;
developer543811b2022-10-07 15:54:07 +080063 int ret;
64
65 ret = snprintf(buf, sizeof(buf), "/sbin/procd");
66 if (snprintf_error(sizeof(buf), ret))
67 return -1;
68
69 f = fopen(buf, "r");
70
71 /* This procd is openwrt only */
72 if (f) {
73 an->bridge_name = BRIDGE_NAME_OPENWRT;
74 fclose(f);
75 } else {
76 an->bridge_name = BRIDGE_NAME_RDKB;
77 }
78
79 return 0;
80}
81
developer22c7ab62022-01-24 11:13:32 +080082static void usage(void)
83{
developerf30d4472022-05-30 16:40:23 +080084 printf("Usage:\n");
developer22c7ab62022-01-24 11:13:32 +080085 printf(" %s [-u] [-i phyX]\n", progname);
86 printf("options:\n"
87 " -h = show help text\n"
88 " -i = phy name of driver interface, please use first phy for dbdc\n"
developer543811b2022-10-07 15:54:07 +080089 " -u = use unicast to respond to HQADLL\n"
90 " -b = specify your bridge name\n");
developer22c7ab62022-01-24 11:13:32 +080091 printf("examples:\n"
developer543811b2022-10-07 15:54:07 +080092 " %s -u -i phy0 -b br-lan\n", progname);
developer22c7ab62022-01-24 11:13:32 +080093
94 exit(EXIT_FAILURE);
95}
96
developerf30d4472022-05-30 16:40:23 +080097static void atenl_handler_run(struct atenl *an)
developer22c7ab62022-01-24 11:13:32 +080098{
developerf30d4472022-05-30 16:40:23 +080099 int count, sock_eth = an->sock_eth;
developer22c7ab62022-01-24 11:13:32 +0800100 fd_set readfds;
101
developerf30d4472022-05-30 16:40:23 +0800102 atenl_info("Start atenl HQA command handler\n");
developer22c7ab62022-01-24 11:13:32 +0800103
104 while (atenl_enable) {
105 FD_ZERO(&readfds);
106 FD_SET(sock_eth, &readfds);
107 count = select(sock_eth + 1, &readfds, NULL, NULL, NULL);
108
109 if (count < 0) {
110 atenl_err("%s: select failed, %s\n", __func__, strerror(errno));
developer22c7ab62022-01-24 11:13:32 +0800111 } else if (count == 0) {
112 usleep(1000);
developer22c7ab62022-01-24 11:13:32 +0800113 } else {
developerf30d4472022-05-30 16:40:23 +0800114 if (!FD_ISSET(sock_eth, &readfds))
115 continue;
116 atenl_hqa_proc_cmd(an);
developer22c7ab62022-01-24 11:13:32 +0800117 }
118 }
119
developerf30d4472022-05-30 16:40:23 +0800120 atenl_dbg("HQA command handler end\n");
developer22c7ab62022-01-24 11:13:32 +0800121}
122
developer22c7ab62022-01-24 11:13:32 +0800123int main(int argc, char **argv)
124{
125 int opt, phy_idx, ret = 0;
126 char *phy = "phy0", *cmd = NULL;
127 struct atenl *an;
128
129 progname = argv[0];
130
131 an = calloc(1, sizeof(struct atenl));
132 if (!an)
133 return -ENOMEM;
134
135 while(1) {
developer543811b2022-10-07 15:54:07 +0800136 opt = getopt(argc, argv, "hi:uc:b:");
developer22c7ab62022-01-24 11:13:32 +0800137 if (opt == -1)
138 break;
139
140 switch (opt) {
141 case 'h':
142 usage();
developerf30d4472022-05-30 16:40:23 +0800143 goto out;
developer22c7ab62022-01-24 11:13:32 +0800144 case 'i':
145 phy = optarg;
146 break;
developer543811b2022-10-07 15:54:07 +0800147 case 'b':
148 an->bridge_name = optarg;
149 break;
developer22c7ab62022-01-24 11:13:32 +0800150 case 'u':
151 an->unicast = true;
152 printf("Opt: use unicast to send response\n");
153 break;
154 case 'c':
155 cmd = optarg;
156 break;
157 default:
developerf30d4472022-05-30 16:40:23 +0800158 atenl_err("Not supported option: %c\n", opt);
159 goto out;
developer22c7ab62022-01-24 11:13:32 +0800160 }
161 }
162
163 phy_idx = phy_lookup_idx(phy);
164 if (phy_idx < 0 || phy_idx > UCHAR_MAX) {
developerf30d4472022-05-30 16:40:23 +0800165 atenl_err("Could not find phy '%s'\n", phy);
166 goto out;
developer22c7ab62022-01-24 11:13:32 +0800167 }
168
169 if (cmd) {
170 atenl_eeprom_cmd_handler(an, phy_idx, cmd);
171 goto out;
172 }
173
174 atenl_enable = true;
175 atenl_init_signals();
176
developer543811b2022-10-07 15:54:07 +0800177 if (!an->bridge_name) {
178 ret = get_default_bridge_name(an);
179 if (ret) {
180 atenl_err("Get default bridge name failed\n");
181 goto out;
182 }
183
184 atenl_info("Bridge name is not specified, use default bridge name: %s\n", an->bridge_name);
185 } else {
186 atenl_info("Currently using bridge name: %s\n", an->bridge_name);
187 }
188
developer22c7ab62022-01-24 11:13:32 +0800189 /* background ourself */
190 if (!fork()) {
developer22c7ab62022-01-24 11:13:32 +0800191 ret = atenl_eeprom_init(an, phy_idx);
192 if (ret)
193 goto out;
194
195 ret = atenl_eth_init(an);
196 if (ret)
197 goto out;
198
developerf30d4472022-05-30 16:40:23 +0800199 atenl_handler_run(an);
developer22c7ab62022-01-24 11:13:32 +0800200 } else {
201 usleep(800000);
202 }
203
204 ret = 0;
205out:
206 if (an->sock_eth)
207 close(an->sock_eth);
208 if (an->eeprom_fd || an->eeprom_data)
209 atenl_eeprom_close(an);
210 free(an);
211
212 return ret;
213}