/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License, Version 1.0 only * (the "License"). You may not use this file except in compliance * with the License. * * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE * or http://www.opensolaris.org/os/licensing. * See the License for the specific language governing permissions * and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at usr/src/OPENSOLARIS.LICENSE. * If applicable, add the following below this CDDL HEADER, with the * fields enclosed by brackets "[]" replaced with your own identifying * information: Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END */ /* * Copyright 2005 Sun Microsystems, Inc. All rights reserved. * Use is subject to license terms. */ #pragma ident "%Z%%M% %I% %E% SMI" #include #include #include #include #include #include #include #include #include #include #include #include "ncaconf.h" char _depends_on[] = "fs/sockfs drv/ip"; extern caddr_t nca_g_nd; /* Head of 'named dispatch' variable list */ #define INET_NAME "nca" #define INET_STRTAB ncainfo #define INET_MODDESC "NCA STREAMS module 1.6" #define INET_DEVDESC "NCA STREAMS driver 1.6" #define INET_DEVMINOR 0 #define INET_DEVMTFLAGS D_MP #define INET_MODMTFLAGS D_MP #include "../inetddi.c" /*ARGSUSED*/ static int nca_open(queue_t *q, dev_t *devp, int flag, int sflag, cred_t *credp) { /* Reopen supported */ if (q->q_ptr != NULL) return (0); /* * NCA is not supported in non-global zones; we enforce this restriction * here. */ if (credp != NULL && crgetzoneid(credp) != GLOBAL_ZONEID) { return (ENOTSUP); } if (! (sflag & MODOPEN)) { /* Device instance */ RD(q)->q_ptr = (void *)B_TRUE; WR(q)->q_ptr = (void *)B_TRUE; } else { /* Modopen just pass through */ RD(q)->q_ptr = (void *)B_FALSE; WR(q)->q_ptr = (void *)B_FALSE; } qprocson(q); return (0); } static int nca_close(queue_t *q) { qprocsoff(q); RD(q)->q_ptr = NULL; WR(q)->q_ptr = NULL; return (0); } static void nca_rput(queue_t *q, mblk_t *mp) { /* Passthrough */ putnext(q, mp); } static void nca_wput(queue_t *q, mblk_t *mp) { struct iocblk *iocp; if (! (boolean_t)q->q_ptr) { iocp = (struct iocblk *)mp->b_rptr; if (DB_TYPE(mp) == M_IOCTL && iocp->ioc_cmd == NCA_SET_IF) { miocnak(q, mp, 0, ENOTSUP); return; } /* Module, passthrough */ putnext(q, mp); return; } switch (DB_TYPE(mp)) { case M_IOCTL: iocp = (struct iocblk *)mp->b_rptr; switch (iocp->ioc_cmd) { case ND_SET: case ND_GET: if (! nd_getset(q, nca_g_nd, mp)) { miocnak(q, mp, 0, ENOENT); return; } qreply(q, mp); break; default: miocnak(q, mp, 0, ENOTSUP); break; } break; default: freemsg(mp); break; } } static struct module_info info = { 0, "nca", 1, INFPSZ, 65536, 1024 }; static struct qinit rinit = { (pfi_t)nca_rput, NULL, nca_open, nca_close, NULL, &info }; static struct qinit winit = { (pfi_t)nca_wput, NULL, nca_open, nca_close, NULL, &info }; struct streamtab ncainfo = { &rinit, &winit }; int _init(void) { return (mod_install(&modlinkage)); } int _fini(void) { return (EBUSY); } int _info(struct modinfo *modinfop) { return (mod_info(&modlinkage, modinfop)); }