ref: babf901b4a508c3ec5d1f89655f10377bbdf9637
dir: /appl/cmd/ip/dhcp.b/
implement Dhcp;
#
# configure an interface using DHCP
#
include "sys.m";
sys: Sys;
include "draw.m";
include "ip.m";
ip: IP;
IPv4off, IPaddrlen: import IP;
IPaddr: import ip;
get2, get4, put2, put4: import ip;
include "dhcp.m";
dhcpclient: Dhcpclient;
Bootconf, Lease: import dhcpclient;
include "arg.m";
Dhcp: module
{
init: fn(nil: ref Draw->Context, nil: list of string);
};
RetryTime: con 10*1000; # msec
init(nil: ref Draw->Context, args: list of string)
{
sys = load Sys Sys->PATH;
ip = load IP IP->PATH;
dhcpclient = load Dhcpclient Dhcpclient->PATH;
sys->pctl(Sys->NEWFD|Sys->NEWPGRP, 0 :: 1 :: 2 :: nil);
arg := load Arg Arg->PATH;
arg->init(args);
arg->setusage("dhcp [-bdmnpr] [-g ipgw] [-h hostname] [-x /net] ifcdir [ip [ipmask]]");
trace := 0;
pcfg := 0;
bootp := 0;
monitor := 0;
retry := 0;
noctl := 0;
netdir := "/net";
cfg := Bootconf.new();
while((o := arg->opt()) != 0)
case o {
'b' => bootp = 1;
'd' => trace++;
'g' => cfg.ipgw = arg->earg();
'h' => cfg.puts(Dhcpclient->Ohostname, arg->earg());
'm' => monitor = 1;
'n' => noctl = 1;
'p' => pcfg = 1;
'r' => retry = 1;
'x' => netdir = arg->earg();
* => arg->usage();
}
args = arg->argv();
if(len args == 0)
arg->usage();
ifcdir := hd args;
args = tl args;
if(args != nil){
cfg.ip = hd args;
args = tl args;
if(args != nil){
cfg.ipmask = hd args;
args = tl args;
if(args != nil)
arg->usage();
}
}
arg = nil;
ifcctl: ref Sys->FD;
if(noctl == 0){
ifcctl = sys->open(ifcdir+"/ctl", Sys->OWRITE);
if(ifcctl == nil)
err(sys->sprint("cannot open %s/ctl: %r", ifcdir));
}
etherdir := finddev(ifcdir);
if(etherdir == nil)
err(sys->sprint("cannot find network device in %s/status: %r", ifcdir));
if(etherdir[0] != '/' && etherdir[0] != '#')
etherdir = netdir+"/"+etherdir;
ip->init();
dhcpclient->init();
dhcpclient->tracing(trace);
e: string;
lease: ref Lease;
for(;;){
if(bootp){
(cfg, e) = dhcpclient->bootp(netdir, ifcctl, etherdir+"/addr", cfg);
if(e == nil){
if(cfg != nil)
dhcpclient->applycfg(netdir, ifcctl, cfg);
if(pcfg)
printcfg(cfg);
break;
}
}else{
(cfg, lease, e) = dhcpclient->dhcp(netdir, ifcctl, etherdir+"/addr", cfg, nil); # last is array of int options
if(e == nil){
if(pcfg)
printcfg(cfg);
if(cfg.lease > 0 && monitor)
leasemon(lease.configs, pcfg);
break;
}
}
if(!retry)
err("failed to configure network: "+e);
sys->fprint(sys->fildes(2), "dhcp: failed to configure network: %s; retrying", e);
sys->sleep(RetryTime);
}
}
leasemon(configs: chan of (ref Bootconf, string), pcfg: int)
{
for(;;){
(cfg, e) := <-configs;
if(e != nil)
sys->fprint(sys->fildes(2), "dhcp: %s", e);
if(pcfg)
printcfg(cfg);
}
}
printcfg(cfg: ref Bootconf)
{
sys->print("ip=%s ipmask=%s ipgw=%s iplease=%d\n", cfg.ip, cfg.ipmask, cfg.ipgw, cfg.lease);
}
finddev(ifcdir: string): string
{
fd := sys->open(ifcdir+"/status", Sys->OREAD);
if(fd == nil)
return nil;
buf := array[1024] of byte;
n := sys->read(fd, buf, len buf);
if(n < 0)
return nil;
(nf, l) := sys->tokenize(string buf[0:n], " \n");
if(nf < 2){
sys->werrstr("unexpected format for status file");
return nil;
}
return hd tl l;
}
err(s: string)
{
sys->fprint(sys->fildes(2), "dhcp: %s\n", s);
raise "fail:error";
}