1*cf5a6c84SAndroid Build Coastguard Worker /* nbd-client.c - network block device client
2*cf5a6c84SAndroid Build Coastguard Worker *
3*cf5a6c84SAndroid Build Coastguard Worker * Copyright 2010 Rob Landley <[email protected]>
4*cf5a6c84SAndroid Build Coastguard Worker *
5*cf5a6c84SAndroid Build Coastguard Worker * Not in SUSv4.
6*cf5a6c84SAndroid Build Coastguard Worker
7*cf5a6c84SAndroid Build Coastguard Worker // This little dance is because a NEWTOY with - in the name tries to do
8*cf5a6c84SAndroid Build Coastguard Worker // things like prototype "nbd-client_main" which isn't a valid symbol. So
9*cf5a6c84SAndroid Build Coastguard Worker // we hide the underscore name and OLDTOY the name we want.
10*cf5a6c84SAndroid Build Coastguard Worker USE_NBD_CLIENT(NEWTOY(nbd_client, "<3>3b#<1>4294967295=4096ns", 0))
11*cf5a6c84SAndroid Build Coastguard Worker USE_NBD_CLIENT(OLDTOY(nbd-client, nbd_client, TOYFLAG_USR|TOYFLAG_BIN))
12*cf5a6c84SAndroid Build Coastguard Worker
13*cf5a6c84SAndroid Build Coastguard Worker config NBD_CLIENT
14*cf5a6c84SAndroid Build Coastguard Worker bool "nbd-client"
15*cf5a6c84SAndroid Build Coastguard Worker default y
16*cf5a6c84SAndroid Build Coastguard Worker help
17*cf5a6c84SAndroid Build Coastguard Worker usage: nbd-client [-ns] [-b BLKSZ] HOST PORT DEVICE
18*cf5a6c84SAndroid Build Coastguard Worker
19*cf5a6c84SAndroid Build Coastguard Worker -b Block size (default 4096)
20*cf5a6c84SAndroid Build Coastguard Worker -n Do not daemonize
21*cf5a6c84SAndroid Build Coastguard Worker -s nbd swap support (lock server into memory)
22*cf5a6c84SAndroid Build Coastguard Worker */
23*cf5a6c84SAndroid Build Coastguard Worker
24*cf5a6c84SAndroid Build Coastguard Worker #define FOR_nbd_client
25*cf5a6c84SAndroid Build Coastguard Worker #include "toys.h"
26*cf5a6c84SAndroid Build Coastguard Worker #include <linux/nbd.h>
27*cf5a6c84SAndroid Build Coastguard Worker
GLOBALS(long b;int nbd;)28*cf5a6c84SAndroid Build Coastguard Worker GLOBALS(
29*cf5a6c84SAndroid Build Coastguard Worker long b;
30*cf5a6c84SAndroid Build Coastguard Worker
31*cf5a6c84SAndroid Build Coastguard Worker int nbd;
32*cf5a6c84SAndroid Build Coastguard Worker )
33*cf5a6c84SAndroid Build Coastguard Worker
34*cf5a6c84SAndroid Build Coastguard Worker static void sig_cleanup(int catch)
35*cf5a6c84SAndroid Build Coastguard Worker {
36*cf5a6c84SAndroid Build Coastguard Worker // Flush on the way out
37*cf5a6c84SAndroid Build Coastguard Worker ioctl(TT.nbd, NBD_CLEAR_QUE);
38*cf5a6c84SAndroid Build Coastguard Worker ioctl(TT.nbd, NBD_CLEAR_SOCK);
39*cf5a6c84SAndroid Build Coastguard Worker _exit(catch ? 128+catch : 0);
40*cf5a6c84SAndroid Build Coastguard Worker }
41*cf5a6c84SAndroid Build Coastguard Worker
nbd_client_main(void)42*cf5a6c84SAndroid Build Coastguard Worker void nbd_client_main(void)
43*cf5a6c84SAndroid Build Coastguard Worker {
44*cf5a6c84SAndroid Build Coastguard Worker int sock = -1, flags, temp;
45*cf5a6c84SAndroid Build Coastguard Worker unsigned long timeout = 0;
46*cf5a6c84SAndroid Build Coastguard Worker char *host=toys.optargs[0], *port=toys.optargs[1], *device=toys.optargs[2];
47*cf5a6c84SAndroid Build Coastguard Worker unsigned long long devsize;
48*cf5a6c84SAndroid Build Coastguard Worker
49*cf5a6c84SAndroid Build Coastguard Worker // Daemonize in a nommu-friendly way, but retain stderr
50*cf5a6c84SAndroid Build Coastguard Worker if (toys.stacktop && !FLAG(n)) {
51*cf5a6c84SAndroid Build Coastguard Worker dup2(2, 222);
52*cf5a6c84SAndroid Build Coastguard Worker xvdaemon();
53*cf5a6c84SAndroid Build Coastguard Worker }
54*cf5a6c84SAndroid Build Coastguard Worker dup2(222, 2);
55*cf5a6c84SAndroid Build Coastguard Worker close(222);
56*cf5a6c84SAndroid Build Coastguard Worker
57*cf5a6c84SAndroid Build Coastguard Worker TT.nbd = xopen(device, O_RDWR);
58*cf5a6c84SAndroid Build Coastguard Worker xsignal(SIGINT, sig_cleanup);
59*cf5a6c84SAndroid Build Coastguard Worker xsignal(SIGTERM, sig_cleanup);
60*cf5a6c84SAndroid Build Coastguard Worker
61*cf5a6c84SAndroid Build Coastguard Worker for (;;) {
62*cf5a6c84SAndroid Build Coastguard Worker // Find and connect to server
63*cf5a6c84SAndroid Build Coastguard Worker sock = xconnectany(xgetaddrinfo(host, port, AF_UNSPEC, SOCK_STREAM, 0, 0));
64*cf5a6c84SAndroid Build Coastguard Worker temp = 1;
65*cf5a6c84SAndroid Build Coastguard Worker setsockopt(sock, IPPROTO_TCP, TCP_NODELAY, &temp, sizeof(int));
66*cf5a6c84SAndroid Build Coastguard Worker
67*cf5a6c84SAndroid Build Coastguard Worker // Read login data
68*cf5a6c84SAndroid Build Coastguard Worker xreadall(sock, toybuf, 152);
69*cf5a6c84SAndroid Build Coastguard Worker if (smemcmp(toybuf, "NBDMAGIC\x00\x00\x42\x02\x81\x86\x12\x53", 16))
70*cf5a6c84SAndroid Build Coastguard Worker error_exit("bad login %s:%s", host, port);
71*cf5a6c84SAndroid Build Coastguard Worker devsize = SWAP_BE64(*(unsigned long long *)(toybuf+16));
72*cf5a6c84SAndroid Build Coastguard Worker flags = SWAP_BE32(*(int *)(toybuf+24));
73*cf5a6c84SAndroid Build Coastguard Worker
74*cf5a6c84SAndroid Build Coastguard Worker // Use 4k block size
75*cf5a6c84SAndroid Build Coastguard Worker ioctl(TT.nbd, NBD_SET_BLKSIZE, TT.b);
76*cf5a6c84SAndroid Build Coastguard Worker ioctl(TT.nbd, NBD_SET_SIZE_BLOCKS, devsize/TT.b); // rounds down
77*cf5a6c84SAndroid Build Coastguard Worker ioctl(TT.nbd, NBD_CLEAR_SOCK);
78*cf5a6c84SAndroid Build Coastguard Worker
79*cf5a6c84SAndroid Build Coastguard Worker // Locally respect read only exports
80*cf5a6c84SAndroid Build Coastguard Worker flags = (flags>>1)&1;
81*cf5a6c84SAndroid Build Coastguard Worker xioctl(TT.nbd, BLKROSET, &flags);
82*cf5a6c84SAndroid Build Coastguard Worker
83*cf5a6c84SAndroid Build Coastguard Worker if (timeout && ioctl(TT.nbd, NBD_SET_TIMEOUT, timeout)<0) break;
84*cf5a6c84SAndroid Build Coastguard Worker if (ioctl(TT.nbd, NBD_SET_SOCK, sock) < 0) break;
85*cf5a6c84SAndroid Build Coastguard Worker
86*cf5a6c84SAndroid Build Coastguard Worker if (FLAG(s)) mlockall(MCL_CURRENT|MCL_FUTURE);
87*cf5a6c84SAndroid Build Coastguard Worker
88*cf5a6c84SAndroid Build Coastguard Worker // Open the device to force reread of the partition table.
89*cf5a6c84SAndroid Build Coastguard Worker if (!CFG_TOYBOX_FORK || !xfork()) {
90*cf5a6c84SAndroid Build Coastguard Worker char *s = strrchr(device, '/');
91*cf5a6c84SAndroid Build Coastguard Worker int i;
92*cf5a6c84SAndroid Build Coastguard Worker
93*cf5a6c84SAndroid Build Coastguard Worker // Give device up to 10 seconds to come up
94*cf5a6c84SAndroid Build Coastguard Worker sprintf(toybuf, "/sys/block/%.32s/pid", s ? s+1 : device);
95*cf5a6c84SAndroid Build Coastguard Worker for (i=0; i<100; i++) {
96*cf5a6c84SAndroid Build Coastguard Worker if (access(toybuf, F_OK)) break;
97*cf5a6c84SAndroid Build Coastguard Worker msleep(100);
98*cf5a6c84SAndroid Build Coastguard Worker }
99*cf5a6c84SAndroid Build Coastguard Worker close(open(device, O_RDONLY));
100*cf5a6c84SAndroid Build Coastguard Worker if (CFG_TOYBOX_FORK) _exit(0);
101*cf5a6c84SAndroid Build Coastguard Worker }
102*cf5a6c84SAndroid Build Coastguard Worker
103*cf5a6c84SAndroid Build Coastguard Worker // Process NBD requests until further notice.
104*cf5a6c84SAndroid Build Coastguard Worker
105*cf5a6c84SAndroid Build Coastguard Worker if (ioctl(TT.nbd, NBD_DO_IT)>=0 || errno==EBADR) break;
106*cf5a6c84SAndroid Build Coastguard Worker close(sock);
107*cf5a6c84SAndroid Build Coastguard Worker ioctl(TT.nbd, NBD_CLEAR_QUE);
108*cf5a6c84SAndroid Build Coastguard Worker }
109*cf5a6c84SAndroid Build Coastguard Worker
110*cf5a6c84SAndroid Build Coastguard Worker // Flush queue and exit.
111*cf5a6c84SAndroid Build Coastguard Worker if (CFG_TOYBOX_FREE) close(TT.nbd);
112*cf5a6c84SAndroid Build Coastguard Worker }
113