Support destroy in shmif.

This commit is contained in:
pooka 2010-11-17 17:51:22 +00:00
parent 48257b3e61
commit d9aab6f8fc

View File

@ -1,4 +1,4 @@
/* $NetBSD: if_shmem.c,v 1.31 2010/11/16 20:08:24 pooka Exp $ */
/* $NetBSD: if_shmem.c,v 1.32 2010/11/17 17:51:22 pooka Exp $ */
/*
* Copyright (c) 2009 Antti Kantee. All Rights Reserved.
@ -28,7 +28,7 @@
*/
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: if_shmem.c,v 1.31 2010/11/16 20:08:24 pooka Exp $");
__KERNEL_RCSID(0, "$NetBSD: if_shmem.c,v 1.32 2010/11/17 17:51:22 pooka Exp $");
#include <sys/param.h>
#include <sys/atomic.h>
@ -81,12 +81,19 @@ struct shmif_sc {
struct shmif_mem *sc_busmem;
int sc_memfd;
int sc_kq;
int sc_unit;
char *sc_backfile;
size_t sc_backfilelen;
uint64_t sc_devgen;
uint32_t sc_nextpacket;
kmutex_t sc_mtx;
kcondvar_t sc_cv;
struct lwp *sc_rcvl;
bool sc_dying;
};
static const uint32_t busversion = SHMIF_VERSION;
@ -142,18 +149,19 @@ allocif(int unit, struct shmif_sc **scp)
struct shmif_sc *sc;
struct ifnet *ifp;
uint32_t randnum;
unsigned mynum = unit;
int error;
randnum = arc4random();
memcpy(&enaddr[2], &randnum, sizeof(randnum));
sc = kmem_zalloc(sizeof(*sc), KM_SLEEP);
sc->sc_memfd = -1;
sc->sc_unit = unit;
ifp = &sc->sc_ec.ec_if;
memcpy(sc->sc_myaddr, enaddr, sizeof(enaddr));
sprintf(ifp->if_xname, "shmif%d", mynum);
sprintf(ifp->if_xname, "shmif%d", unit);
ifp->if_softc = sc;
ifp->if_flags = IFF_BROADCAST | IFF_MULTICAST;
ifp->if_init = shmif_init;
@ -162,16 +170,32 @@ allocif(int unit, struct shmif_sc **scp)
ifp->if_stop = shmif_stop;
ifp->if_mtu = ETHERMTU;
mutex_init(&sc->sc_mtx, MUTEX_DEFAULT, IPL_NONE);
cv_init(&sc->sc_cv, "shmifcv");
if_attach(ifp);
ether_ifattach(ifp, enaddr);
aprint_verbose("shmif%d: Ethernet address %s\n",
mynum, ether_sprintf(enaddr));
unit, ether_sprintf(enaddr));
if (scp)
*scp = sc;
return 0;
error = 0;
if (rump_threads) {
error = kthread_create(PRI_NONE,
KTHREAD_MPSAFE | KTHREAD_JOINABLE, NULL,
shmif_rcv, ifp, &sc->sc_rcvl, "shmif");
} else {
printf("WARNING: threads not enabled, shmif NOT working\n");
}
if (error) {
shmif_unclone(ifp);
}
return error;
}
static int
@ -221,48 +245,56 @@ initbackend(struct shmif_sc *sc, int memfd)
shmif_unlockbus(sc->sc_busmem);
sc->sc_kq = rumpuser_writewatchfile_setup(-1, memfd, 0, &error);
if (sc->sc_kq == -1)
if (sc->sc_kq == -1) {
rumpuser_unmap(sc->sc_busmem, BUSMEM_SIZE);
return error;
}
sc->sc_memfd = memfd;
return 0;
return error;
}
static void
finibackend(struct shmif_sc *sc)
{
int dummy;
kmem_free(sc->sc_backfile, sc->sc_backfilelen);
sc->sc_backfile = NULL;
sc->sc_backfilelen = 0;
if (sc->sc_backfile == NULL)
return;
if (sc->sc_backfile) {
kmem_free(sc->sc_backfile, sc->sc_backfilelen);
sc->sc_backfile = NULL;
sc->sc_backfilelen = 0;
}
rumpuser_unmap(sc->sc_busmem, BUSMEM_SIZE);
rumpuser_close(sc->sc_memfd, &dummy);
rumpuser_close(sc->sc_kq, &dummy);
rumpuser_close(sc->sc_memfd, NULL);
rumpuser_close(sc->sc_kq, NULL);
sc->sc_memfd = -1;
}
int
rump_shmif_create(const char *path, int *ifnum)
{
struct shmif_sc *sc;
int mynum, error, memfd, dummy;
int unit, error, memfd;
memfd = rumpuser_open(path, O_RDWR | O_CREAT, &error);
if (memfd == -1)
return error;
mynum = vmem_xalloc(shmif_units, 1, 0, 0, 0, 0, 0,
unit = vmem_xalloc(shmif_units, 1, 0, 0, 0, 0, 0,
VM_INSTANTFIT | VM_SLEEP) - 1;
if ((error = allocif(mynum, &sc)) != 0) {
if ((error = allocif(unit, &sc)) != 0) {
rumpuser_close(memfd, NULL);
return error;
}
error = initbackend(sc, memfd);
if (error) {
rumpuser_close(memfd, &dummy);
/* XXX: free sc */
shmif_unclone(&sc->sc_ec.ec_if);
return error;
}
@ -271,7 +303,7 @@ rump_shmif_create(const char *path, int *ifnum)
strcpy(sc->sc_backfile, path);
if (ifnum)
*ifnum = mynum;
*ifnum = unit;
return 0;
}
@ -300,8 +332,32 @@ shmif_clone(struct if_clone *ifc, int unit)
static int
shmif_unclone(struct ifnet *ifp)
{
struct shmif_sc *sc = ifp->if_softc;
return EOPNOTSUPP;
shmif_stop(ifp, 1);
if_down(ifp);
finibackend(sc);
mutex_enter(&sc->sc_mtx);
sc->sc_dying = true;
cv_broadcast(&sc->sc_cv);
mutex_exit(&sc->sc_mtx);
if (sc->sc_rcvl)
kthread_join(sc->sc_rcvl);
sc->sc_rcvl = NULL;
vmem_xfree(shmif_units, sc->sc_unit+1, 1);
ether_ifdetach(ifp);
if_detach(ifp);
cv_destroy(&sc->sc_cv);
mutex_destroy(&sc->sc_mtx);
kmem_free(sc, sizeof(*sc));
return 0;
}
static int
@ -312,15 +368,17 @@ shmif_init(struct ifnet *ifp)
if (sc->sc_memfd == -1)
return ENXIO;
if (rump_threads) {
error = kthread_create(PRI_NONE, KTHREAD_MPSAFE, NULL,
shmif_rcv, ifp, NULL, "shmif");
} else {
printf("WARNING: threads not enabled, shmif NOT working\n");
}
KASSERT(sc->sc_busmem);
ifp->if_flags |= IFF_RUNNING;
mutex_enter(&sc->sc_mtx);
sc->sc_nextpacket = sc->sc_busmem->shm_last;
sc->sc_devgen = sc->sc_busmem->shm_gen;
cv_broadcast(&sc->sc_cv);
mutex_exit(&sc->sc_mtx);
return error;
}
@ -330,7 +388,7 @@ shmif_ioctl(struct ifnet *ifp, u_long cmd, void *data)
struct shmif_sc *sc = ifp->if_softc;
struct ifdrv *ifd;
char *path;
int s, rv, memfd, dummy;
int s, rv, memfd;
s = splnet();
switch (cmd) {
@ -397,7 +455,7 @@ shmif_ioctl(struct ifnet *ifp, u_long cmd, void *data)
rv = initbackend(sc, memfd);
if (rv) {
kmem_free(path, ifd->ifd_len);
rumpuser_close(memfd, &dummy);
rumpuser_close(memfd, NULL);
break;
}
sc->sc_backfile = path;
@ -415,7 +473,7 @@ shmif_ioctl(struct ifnet *ifp, u_long cmd, void *data)
return rv;
}
/* send everything in-context */
/* send everything in-context since it's just a matter of mem-to-mem copy */
static void
shmif_start(struct ifnet *ifp)
{
@ -479,7 +537,7 @@ shmif_start(struct ifnet *ifp)
ifp->if_flags &= ~IFF_OACTIVE;
/* wakeup */
/* wakeup? */
if (wrote)
rumpuser_pwrite(sc->sc_memfd,
&busversion, sizeof(busversion), IFMEM_WAKEUP, &error);
@ -488,8 +546,18 @@ shmif_start(struct ifnet *ifp)
static void
shmif_stop(struct ifnet *ifp, int disable)
{
struct shmif_sc *sc = ifp->if_softc;
panic("%s: unimpl", __func__);
ifp->if_flags &= ~IFF_RUNNING;
membar_producer();
/*
* wakeup thread. this will of course wake up all bus
* listeners, but that's life.
*/
if (sc->sc_memfd != -1)
rumpuser_pwrite(sc->sc_memfd,
&busversion, sizeof(busversion), IFMEM_WAKEUP, NULL);
}
@ -535,14 +603,22 @@ shmif_rcv(void *arg)
{
struct ifnet *ifp = arg;
struct shmif_sc *sc = ifp->if_softc;
struct shmif_mem *busmem = sc->sc_busmem;
struct shmif_mem *busmem;
struct mbuf *m = NULL;
struct ether_header *eth;
uint32_t nextpkt;
bool wrap;
int error;
for (;;) {
reup:
mutex_enter(&sc->sc_mtx);
while ((ifp->if_flags & IFF_RUNNING) == 0 && !sc->sc_dying)
cv_wait(&sc->sc_cv, &sc->sc_mtx);
mutex_exit(&sc->sc_mtx);
busmem = sc->sc_busmem;
while (ifp->if_flags & IFF_RUNNING) {
struct shmif_pkthdr sp;
if (m == NULL) {
@ -566,6 +642,7 @@ shmif_rcv(void *arg)
rumpuser_writewatchfile_wait(sc->sc_kq, NULL, &error);
if (__predict_false(error))
printf("shmif_rcv: wait failed %d\n", error);
membar_consumer();
continue;
}
@ -620,6 +697,11 @@ shmif_rcv(void *arg)
m = NULL;
}
}
m_freem(m);
m = NULL;
panic("shmif_worker is a lazy boy %d\n", error);
if (!sc->sc_dying)
goto reup;
kthread_exit(0);
}