2019-05-27 14:55:05 +08:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2006-08-25 15:52:06 +08:00
|
|
|
/*
|
|
|
|
* ucon.c
|
|
|
|
*
|
2009-07-22 03:43:51 +08:00
|
|
|
* Copyright (c) 2004+ Evgeniy Polyakov <zbr@ioremap.net>
|
2006-08-25 15:52:06 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <asm/types.h>
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <sys/poll.h>
|
|
|
|
|
|
|
|
#include <linux/netlink.h>
|
|
|
|
#include <linux/rtnetlink.h>
|
|
|
|
|
|
|
|
#include <arpa/inet.h>
|
|
|
|
|
2009-07-18 01:14:26 +08:00
|
|
|
#include <stdbool.h>
|
2006-08-25 15:52:06 +08:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <time.h>
|
2009-07-18 01:14:26 +08:00
|
|
|
#include <getopt.h>
|
2006-08-25 15:52:06 +08:00
|
|
|
|
|
|
|
#include <linux/connector.h>
|
|
|
|
|
|
|
|
#define DEBUG
|
|
|
|
#define NETLINK_CONNECTOR 11
|
|
|
|
|
2009-07-18 01:14:26 +08:00
|
|
|
/* Hopefully your userspace connector.h matches this kernel */
|
|
|
|
#define CN_TEST_IDX CN_NETLINK_USERS + 3
|
|
|
|
#define CN_TEST_VAL 0x456
|
|
|
|
|
2006-08-25 15:52:06 +08:00
|
|
|
#ifdef DEBUG
|
|
|
|
#define ulog(f, a...) fprintf(stdout, f, ##a)
|
|
|
|
#else
|
|
|
|
#define ulog(f, a...) do {} while (0)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static int need_exit;
|
|
|
|
static __u32 seq;
|
|
|
|
|
|
|
|
static int netlink_send(int s, struct cn_msg *msg)
|
|
|
|
{
|
|
|
|
struct nlmsghdr *nlh;
|
|
|
|
unsigned int size;
|
|
|
|
int err;
|
|
|
|
char buf[128];
|
|
|
|
struct cn_msg *m;
|
|
|
|
|
|
|
|
size = NLMSG_SPACE(sizeof(struct cn_msg) + msg->len);
|
|
|
|
|
|
|
|
nlh = (struct nlmsghdr *)buf;
|
|
|
|
nlh->nlmsg_seq = seq++;
|
|
|
|
nlh->nlmsg_pid = getpid();
|
|
|
|
nlh->nlmsg_type = NLMSG_DONE;
|
2013-10-01 04:03:09 +08:00
|
|
|
nlh->nlmsg_len = size;
|
2006-08-25 15:52:06 +08:00
|
|
|
nlh->nlmsg_flags = 0;
|
|
|
|
|
|
|
|
m = NLMSG_DATA(nlh);
|
|
|
|
#if 0
|
|
|
|
ulog("%s: [%08x.%08x] len=%u, seq=%u, ack=%u.\n",
|
|
|
|
__func__, msg->id.idx, msg->id.val, msg->len, msg->seq, msg->ack);
|
|
|
|
#endif
|
|
|
|
memcpy(m, msg, sizeof(*m) + msg->len);
|
|
|
|
|
|
|
|
err = send(s, nlh, size, 0);
|
|
|
|
if (err == -1)
|
|
|
|
ulog("Failed to send: %s [%d].\n",
|
|
|
|
strerror(errno), errno);
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2009-07-18 01:14:26 +08:00
|
|
|
static void usage(void)
|
|
|
|
{
|
|
|
|
printf(
|
|
|
|
"Usage: ucon [options] [output file]\n"
|
|
|
|
"\n"
|
|
|
|
"\t-h\tthis help screen\n"
|
|
|
|
"\t-s\tsend buffers to the test module\n"
|
|
|
|
"\n"
|
|
|
|
"The default behavior of ucon is to subscribe to the test module\n"
|
|
|
|
"and wait for state messages. Any ones received are dumped to the\n"
|
|
|
|
"specified output file (or stdout). The test module is assumed to\n"
|
|
|
|
"have an id of {%u.%u}\n"
|
|
|
|
"\n"
|
|
|
|
"If you get no output, then verify the cn_test module id matches\n"
|
|
|
|
"the expected id above.\n"
|
|
|
|
, CN_TEST_IDX, CN_TEST_VAL
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2006-08-25 15:52:06 +08:00
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
|
|
|
int s;
|
|
|
|
char buf[1024];
|
|
|
|
int len;
|
|
|
|
struct nlmsghdr *reply;
|
|
|
|
struct sockaddr_nl l_local;
|
|
|
|
struct cn_msg *data;
|
|
|
|
FILE *out;
|
|
|
|
time_t tm;
|
|
|
|
struct pollfd pfd;
|
2009-07-18 01:14:26 +08:00
|
|
|
bool send_msgs = false;
|
|
|
|
|
|
|
|
while ((s = getopt(argc, argv, "hs")) != -1) {
|
|
|
|
switch (s) {
|
|
|
|
case 's':
|
|
|
|
send_msgs = true;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'h':
|
|
|
|
usage();
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
default:
|
|
|
|
/* getopt() outputs an error for us */
|
|
|
|
usage();
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
2006-08-25 15:52:06 +08:00
|
|
|
|
2009-07-18 01:14:26 +08:00
|
|
|
if (argc != optind) {
|
|
|
|
out = fopen(argv[optind], "a+");
|
2006-08-25 15:52:06 +08:00
|
|
|
if (!out) {
|
|
|
|
ulog("Unable to open %s for writing: %s\n",
|
|
|
|
argv[1], strerror(errno));
|
|
|
|
out = stdout;
|
|
|
|
}
|
2009-07-18 01:14:26 +08:00
|
|
|
} else
|
|
|
|
out = stdout;
|
2006-08-25 15:52:06 +08:00
|
|
|
|
|
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
|
|
|
|
s = socket(PF_NETLINK, SOCK_DGRAM, NETLINK_CONNECTOR);
|
|
|
|
if (s == -1) {
|
|
|
|
perror("socket");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
l_local.nl_family = AF_NETLINK;
|
2009-07-18 01:14:26 +08:00
|
|
|
l_local.nl_groups = -1; /* bitmask of requested groups */
|
2006-08-25 15:52:06 +08:00
|
|
|
l_local.nl_pid = 0;
|
|
|
|
|
2009-07-18 01:14:26 +08:00
|
|
|
ulog("subscribing to %u.%u\n", CN_TEST_IDX, CN_TEST_VAL);
|
|
|
|
|
2006-08-25 15:52:06 +08:00
|
|
|
if (bind(s, (struct sockaddr *)&l_local, sizeof(struct sockaddr_nl)) == -1) {
|
|
|
|
perror("bind");
|
|
|
|
close(s);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
{
|
|
|
|
int on = 0x57; /* Additional group number */
|
|
|
|
setsockopt(s, SOL_NETLINK, NETLINK_ADD_MEMBERSHIP, &on, sizeof(on));
|
|
|
|
}
|
|
|
|
#endif
|
2009-07-18 01:14:26 +08:00
|
|
|
if (send_msgs) {
|
2006-08-25 15:52:06 +08:00
|
|
|
int i, j;
|
|
|
|
|
|
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
|
|
|
|
data = (struct cn_msg *)buf;
|
|
|
|
|
2009-07-18 01:14:26 +08:00
|
|
|
data->id.idx = CN_TEST_IDX;
|
|
|
|
data->id.val = CN_TEST_VAL;
|
2006-08-25 15:52:06 +08:00
|
|
|
data->seq = seq++;
|
|
|
|
data->ack = 0;
|
|
|
|
data->len = 0;
|
|
|
|
|
|
|
|
for (j=0; j<10; ++j) {
|
|
|
|
for (i=0; i<1000; ++i) {
|
|
|
|
len = netlink_send(s, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
ulog("%d messages have been sent to %08x.%08x.\n", i, data->id.idx, data->id.val);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
pfd.fd = s;
|
|
|
|
|
|
|
|
while (!need_exit) {
|
|
|
|
pfd.events = POLLIN;
|
|
|
|
pfd.revents = 0;
|
|
|
|
switch (poll(&pfd, 1, -1)) {
|
|
|
|
case 0:
|
|
|
|
need_exit = 1;
|
|
|
|
break;
|
|
|
|
case -1:
|
|
|
|
if (errno != EINTR) {
|
|
|
|
need_exit = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (need_exit)
|
|
|
|
break;
|
|
|
|
|
|
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
len = recv(s, buf, sizeof(buf), 0);
|
|
|
|
if (len == -1) {
|
|
|
|
perror("recv buf");
|
|
|
|
close(s);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
reply = (struct nlmsghdr *)buf;
|
|
|
|
|
|
|
|
switch (reply->nlmsg_type) {
|
|
|
|
case NLMSG_ERROR:
|
|
|
|
fprintf(out, "Error message received.\n");
|
|
|
|
fflush(out);
|
|
|
|
break;
|
|
|
|
case NLMSG_DONE:
|
|
|
|
data = (struct cn_msg *)NLMSG_DATA(reply);
|
|
|
|
|
|
|
|
time(&tm);
|
|
|
|
fprintf(out, "%.24s : [%x.%x] [%08u.%08u].\n",
|
|
|
|
ctime(&tm), data->id.idx, data->id.val, data->seq, data->ack);
|
|
|
|
fflush(out);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
close(s);
|
|
|
|
return 0;
|
|
|
|
}
|