#include <sys/cdefs.h>
#include <machine/stdarg.h>
#include <sys/param.h>
#include <sys/socket.h>
#include <net/if.h>
#include <netinet/in.h>
#include <netinet/in_systm.h>
#include <stand.h>
#include <stddef.h>
#include <string.h>
#include <net.h>
#include <netif.h>
#include <bootp.h>
#include <bootparam.h>
#include "dev_net.h"
#include "bootstrap.h"
#ifdef NETIF_DEBUG
int debug = 0;
#endif
static char *netdev_name;
static int netdev_sock = -1;
static int netdev_opens;
static int net_init(void);
static int net_open(struct open_file *, ...);
static int net_close(struct open_file *);
static void net_cleanup(void);
static int net_strategy(void *, int, daddr_t, size_t, char *, size_t *);
static int net_print(int);
static int net_getparams(int sock);
struct devsw netdev = {
"net",
DEVT_NET,
net_init,
net_strategy,
net_open,
net_close,
noioctl,
net_print,
net_cleanup
};
static struct uri_scheme {
const char *scheme;
int proto;
} uri_schemes[] = {
{ "tftp:/", NET_TFTP },
{ "nfs:/", NET_NFS },
};
static int
net_init(void)
{
return (0);
}
static int
net_open(struct open_file *f, ...)
{
struct iodesc *d;
va_list args;
struct devdesc *dev;
const char *devname;
int error = 0;
va_start(args, f);
dev = va_arg(args, struct devdesc *);
va_end(args);
devname = dev->d_dev->dv_name;
if (netdev_sock >= 0 && strcmp(devname, netdev_name) != 0)
net_cleanup();
if (netdev_opens == 0) {
if (netdev_sock < 0) {
netdev_sock = netif_open(dev);
if (netdev_sock < 0) {
printf("%s: netif_open() failed\n", __func__);
return (ENXIO);
}
netdev_name = strdup(devname);
#ifdef NETIF_DEBUG
if (debug)
printf("%s: netif_open() succeeded\n",
__func__);
#endif
}
if (rootip.s_addr == 0) {
error = net_getparams(netdev_sock);
if (error) {
free(netdev_name);
netif_close(netdev_sock);
netdev_sock = -1;
return (error);
}
}
d = socktodesc(netdev_sock);
setenv("boot.netif.hwaddr", ether_sprintf(d->myea), 1);
setenv("boot.netif.ip", inet_ntoa(myip), 1);
setenv("boot.netif.netmask", intoa(netmask), 1);
setenv("boot.netif.gateway", inet_ntoa(gateip), 1);
setenv("boot.netif.server", inet_ntoa(rootip), 1);
if (netproto == NET_TFTP) {
setenv("boot.tftproot.server", inet_ntoa(rootip), 1);
setenv("boot.tftproot.path", rootpath, 1);
} else {
setenv("boot.nfsroot.server", inet_ntoa(rootip), 1);
setenv("boot.nfsroot.path", rootpath, 1);
}
if (intf_mtu != 0) {
char mtu[16];
snprintf(mtu, sizeof (mtu), "%u", intf_mtu);
setenv("boot.netif.mtu", mtu, 1);
}
}
netdev_opens++;
dev->d_opendata = &netdev_sock;
return (error);
}
static int
net_close(struct open_file *f)
{
struct devdesc *dev;
#ifdef NETIF_DEBUG
if (debug)
printf("%s: opens=%d\n", __func__, netdev_opens);
#endif
dev = f->f_devdata;
dev->d_opendata = NULL;
return (0);
}
static void
net_cleanup(void)
{
if (netdev_sock >= 0) {
#ifdef NETIF_DEBUG
if (debug)
printf("%s: calling netif_close()\n", __func__);
#endif
rootip.s_addr = 0;
free(netdev_name);
netif_close(netdev_sock);
netdev_sock = -1;
}
}
static int
net_strategy(void *devdata __unused, int rw __unused, daddr_t blk __unused,
size_t size __unused, char *buf __unused, size_t *rsize __unused)
{
return (EIO);
}
extern n_long ip_convertaddr(char *p);
static int
net_getparams(int sock)
{
char buf[MAXHOSTNAMELEN];
n_long rootaddr, smask;
bootp(sock);
if (myip.s_addr != 0)
goto exit;
#ifdef NETIF_DEBUG
if (debug)
printf("%s: BOOTP failed, trying RARP/RPC...\n", __func__);
#endif
if (rarp_getipaddress(sock)) {
printf("%s: RARP failed\n", __func__);
return (EIO);
}
printf("%s: client addr: %s\n", __func__, inet_ntoa(myip));
if (bp_whoami(sock)) {
printf("%s: bootparam/whoami RPC failed\n", __func__);
return (EIO);
}
#ifdef NETIF_DEBUG
if (debug)
printf("%s: client name: %s\n", __func__, hostname);
#endif
smask = 0;
gateip.s_addr = 0;
if (bp_getfile(sock, "gateway", &gateip, buf) == 0) {
smask = ip_convertaddr(buf);
}
if (smask) {
netmask = smask;
#ifdef NETIF_DEBUG
if (debug)
printf("%s: subnet mask: %s\n", __func__,
intoa(netmask));
#endif
}
#ifdef NETIF_DEBUG
if (gateip.s_addr && debug)
printf("%s: net gateway: %s\n", __func__, inet_ntoa(gateip));
#endif
if (bp_getfile(sock, "root", &rootip, rootpath)) {
printf("%s: bootparam/getfile RPC failed\n", __func__);
return (EIO);
}
exit:
if ((rootaddr = net_parse_rootpath()) != INADDR_NONE)
rootip.s_addr = rootaddr;
#ifdef NETIF_DEBUG
if (debug) {
printf("%s: server addr: %s\n", __func__,
inet_ntoa(rootip));
printf("%s: server path: %s\n", __func__, rootpath);
}
#endif
return (0);
}
static int
net_print(int verbose)
{
struct netif_driver *drv;
int i, d, cnt;
int ret = 0;
if (netif_drivers[0] == NULL)
return (ret);
printf("%s devices:", netdev.dv_name);
if ((ret = pager_output("\n")) != 0)
return (ret);
cnt = 0;
for (d = 0; netif_drivers[d]; d++) {
drv = netif_drivers[d];
for (i = 0; i < drv->netif_nifs; i++) {
printf("\t%s%d:", netdev.dv_name, cnt++);
if (verbose) {
printf(" (%s%d)", drv->netif_bname,
drv->netif_ifs[i].dif_unit);
}
if ((ret = pager_output("\n")) != 0)
return (ret);
}
}
return (ret);
}
uint32_t
net_parse_rootpath(void)
{
n_long addr = htonl(INADDR_NONE);
size_t i;
char ip[FNAME_SIZE];
char *ptr, *val;
netproto = NET_NONE;
for (i = 0; i < nitems(uri_schemes); i++) {
if (strncmp(rootpath, uri_schemes[i].scheme,
strlen(uri_schemes[i].scheme)) != 0)
continue;
netproto = uri_schemes[i].proto;
break;
}
ptr = rootpath;
if (netproto == NET_NONE) {
if (strcmp(rootpath, "/") == 0) {
netproto = NET_TFTP;
} else {
netproto = NET_NFS;
(void) strsep(&ptr, ":");
if (ptr != NULL) {
addr = inet_addr(rootpath);
bcopy(ptr, rootpath, strlen(ptr) + 1);
}
}
} else {
ptr += strlen(uri_schemes[i].scheme);
if (*ptr == '/') {
ptr++;
val = strchr(ptr, '/');
if (val == NULL) {
strlcat(rootpath, "/", sizeof (rootpath));
val = strchr(ptr, '/');
}
if (val != NULL) {
snprintf(ip, sizeof (ip), "%.*s",
(int)((uintptr_t)val - (uintptr_t)ptr),
ptr);
addr = inet_addr(ip);
if (addr == htonl(INADDR_NONE)) {
printf("Bad IP address: %s\n", ip);
}
bcopy(val, rootpath, strlen(val) + 1);
}
} else {
ptr--;
bcopy(ptr, rootpath, strlen(ptr) + 1);
}
}
return (addr);
}