#include <sys/param.h>
#include <sys/socket.h>
#include <sys/stdarg.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"
#ifndef NETPROTO_DEFAULT
# define NETPROTO_DEFAULT NET_NFS
#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 = {
.dv_name = "net",
.dv_type = DEVT_NET,
.dv_init = net_init,
.dv_strategy = net_strategy,
.dv_open = net_open,
.dv_close = net_close,
.dv_ioctl = noioctl,
.dv_print = net_print,
.dv_cleanup = 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);
DEBUG_PRINTF(1,("%s: netif_open() succeeded %#x\n",
__func__, rootip.s_addr));
}
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 if (netproto == NET_NFS) {
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);
}
DEBUG_PRINTF(1,("%s: netproto=%d\n", __func__, netproto));
}
netdev_opens++;
dev->d_opendata = &netdev_sock;
return (error);
}
static int
net_close(struct open_file *f)
{
struct devdesc *dev;
DEBUG_PRINTF(2,("%s: opens=%d\n", __func__, netdev_opens));
dev = f->f_devdata;
dev->d_opendata = NULL;
return (0);
}
static void
net_cleanup(void)
{
if (netdev_sock >= 0) {
DEBUG_PRINTF(1,("%s: calling netif_close()\n", __func__));
rootip.s_addr = 0;
free(netdev_name);
netif_close(netdev_sock);
netdev_sock = -1;
}
}
static int
net_strategy(void *devdata, int rw, daddr_t blk, size_t size, char *buf,
size_t *rsize)
{
return (EIO);
}
#define SUPPORT_BOOTP
#ifdef SUPPORT_BOOTP
int try_bootp = 1;
#endif
extern n_long ip_convertaddr(char *p);
static int
net_getparams(int sock)
{
char buf[MAXHOSTNAMELEN];
n_long rootaddr, smask;
#ifdef SUPPORT_BOOTP
if (try_bootp)
bootp(sock);
if (myip.s_addr != 0)
goto exit;
DEBUG_PRINTF(1,("%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);
}
DEBUG_PRINTF(1,("%s: client name: %s\n", __func__, hostname));
smask = 0;
gateip.s_addr = 0;
if (bp_getfile(sock, "gateway", &gateip, buf) == 0) {
smask = ip_convertaddr(buf);
}
if (smask) {
netmask = smask;
DEBUG_PRINTF(1,("%s: subnet mask: %s\n", __func__,
intoa(netmask)));
}
if (gateip.s_addr)
DEBUG_PRINTF(1,("%s: net gateway: %s\n", __func__,
inet_ntoa(gateip)));
if (bp_getfile(sock, "root", &rootip, rootpath)) {
printf("%s: bootparam/getfile RPC failed\n", __func__);
return (EIO);
}
exit:
if ((rootaddr = net_parse_rootpath()) != htonl(INADDR_NONE))
rootip.s_addr = rootaddr;
DEBUG_PRINTF(1,("%s: proto: %d\n", __func__, netproto));
DEBUG_PRINTF(1,("%s: server addr: %s\n", __func__, inet_ntoa(rootip)));
DEBUG_PRINTF(1,("%s: server port: %d\n", __func__, rootport));
DEBUG_PRINTF(1,("%s: server path: %s\n", __func__, rootpath));
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);
}
bool
is_tftp(void)
{
return (netproto == NET_TFTP);
}
uint32_t
net_parse_rootpath(void)
{
n_long addr = 0;
size_t i;
char ip[FNAME_SIZE];
char *ptr, *portp, *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) {
netproto = NETPROTO_DEFAULT;
(void)strsep(&ptr, ":");
if (ptr != NULL) {
addr = inet_addr(rootpath);
DEBUG_PRINTF(1,("rootpath=%s addr=%#x\n",
rootpath, addr));
bcopy(ptr, rootpath, strlen(ptr) + 1);
}
} else {
ptr += strlen(uri_schemes[i].scheme);
if (*ptr == '/') {
ptr++;
portp = val = strchr(ptr, ':');
if (val != NULL) {
val++;
rootport = strtol(val, NULL, 10);
}
val = strchr(ptr, '/');
if (val != NULL) {
if (portp == NULL)
portp = val;
snprintf(ip, sizeof(ip), "%.*s",
(int)(portp - ptr),
ptr);
addr = inet_addr(ip);
DEBUG_PRINTF(1,("ip=%s addr=%#x\n",
ip, addr));
bcopy(val, rootpath, strlen(val) + 1);
}
} else {
ptr--;
bcopy(ptr, rootpath, strlen(ptr) + 1);
}
}
if (addr == 0)
addr = htonl(INADDR_NONE);
return (addr);
}