#include <sys/types.h>
#include <sys/param.h>
#include <sys/errno.h>
#include <sys/stream.h>
#include <sys/stropts.h>
#include <sys/debug.h>
#include <sys/strredir.h>
#include <sys/strsubr.h>
#include <sys/strsun.h>
#include <sys/conf.h>
#include <sys/ddi.h>
#include <sys/sunddi.h>
#include <sys/modctl.h>
static int wcmopen(queue_t *, dev_t *, int, int, cred_t *);
static int wcmclose(queue_t *, int, cred_t *);
static int wcmrput(queue_t *, mblk_t *);
static int wcmwput(queue_t *, mblk_t *);
static struct module_info wcminfo = {
STRREDIR_MODID,
STRREDIR_MOD,
0,
INFPSZ,
5120,
1024
};
static struct qinit wcmrinit = {
wcmrput,
NULL,
wcmopen,
wcmclose,
NULL,
&wcminfo,
NULL
};
static struct qinit wcmwinit = {
wcmwput,
NULL,
wcmopen,
wcmclose,
NULL,
&wcminfo,
NULL
};
static struct streamtab redirminfo = {
&wcmrinit,
&wcmwinit,
NULL,
NULL
};
static struct fmodsw fsw = {
"redirmod",
&redirminfo,
D_MP
};
static struct modlstrmod modlstrmod = {
&mod_strmodops,
"redirection module",
&fsw
};
static struct modlinkage modlinkage = {
MODREV_1, &modlstrmod, NULL
};
int
_init()
{
return (mod_install(&modlinkage));
}
int
_fini()
{
return (EBUSY);
}
int
_info(struct modinfo *modinfop)
{
return (mod_info(&modlinkage, modinfop));
}
static int
wcmopen(queue_t *q, dev_t *dev, int flag, int sflag, cred_t *cred)
{
if (sflag != MODOPEN)
return (EINVAL);
qprocson(q);
return (0);
}
static int
wcmclose(queue_t *q, int flag, cred_t *cred)
{
qprocsoff(q);
srpop(q->q_stream->sd_vnode, B_TRUE);
return (0);
}
static int
wcmrput(queue_t *q, mblk_t *mp)
{
if (DB_TYPE(mp) == M_HANGUP)
srpop(q->q_stream->sd_vnode, B_FALSE);
putnext(q, mp);
return (0);
}
static int
wcmwput(queue_t *q, mblk_t *mp)
{
putnext(q, mp);
return (0);
}