⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 common.c

📁 linux 内核源代码
💻 C
📖 第 1 页 / 共 2 页
字号:
{	struct atm_dev *dev;	struct atm_vcc *vcc = ATM_SD(sock);	int error;	pr_debug("vcc_connect (vpi %d, vci %d)\n",vpi,vci);	if (sock->state == SS_CONNECTED)		return -EISCONN;	if (sock->state != SS_UNCONNECTED)		return -EINVAL;	if (!(vpi || vci))		return -EINVAL;	if (vpi != ATM_VPI_UNSPEC && vci != ATM_VCI_UNSPEC)		clear_bit(ATM_VF_PARTIAL,&vcc->flags);	else		if (test_bit(ATM_VF_PARTIAL,&vcc->flags))			return -EINVAL;	pr_debug("vcc_connect (TX: cl %d,bw %d-%d,sdu %d; "	    "RX: cl %d,bw %d-%d,sdu %d,AAL %s%d)\n",	    vcc->qos.txtp.traffic_class,vcc->qos.txtp.min_pcr,	    vcc->qos.txtp.max_pcr,vcc->qos.txtp.max_sdu,	    vcc->qos.rxtp.traffic_class,vcc->qos.rxtp.min_pcr,	    vcc->qos.rxtp.max_pcr,vcc->qos.rxtp.max_sdu,	    vcc->qos.aal == ATM_AAL5 ? "" : vcc->qos.aal == ATM_AAL0 ? "" :	    " ??? code ",vcc->qos.aal == ATM_AAL0 ? 0 : vcc->qos.aal);	if (!test_bit(ATM_VF_HASQOS, &vcc->flags))		return -EBADFD;	if (vcc->qos.txtp.traffic_class == ATM_ANYCLASS ||	    vcc->qos.rxtp.traffic_class == ATM_ANYCLASS)		return -EINVAL;	if (likely(itf != ATM_ITF_ANY)) {		dev = try_then_request_module(atm_dev_lookup(itf), "atm-device-%d", itf);	} else {		dev = NULL;		mutex_lock(&atm_dev_mutex);		if (!list_empty(&atm_devs)) {			dev = list_entry(atm_devs.next, struct atm_dev, dev_list);			atm_dev_hold(dev);		}		mutex_unlock(&atm_dev_mutex);	}	if (!dev)		return -ENODEV;	error = __vcc_connect(vcc, dev, vpi, vci);	if (error) {		atm_dev_put(dev);		return error;	}	if (vpi == ATM_VPI_UNSPEC || vci == ATM_VCI_UNSPEC)		set_bit(ATM_VF_PARTIAL,&vcc->flags);	if (test_bit(ATM_VF_READY,&ATM_SD(sock)->flags))		sock->state = SS_CONNECTED;	return 0;}int vcc_recvmsg(struct kiocb *iocb, struct socket *sock, struct msghdr *msg,		size_t size, int flags){	struct sock *sk = sock->sk;	struct atm_vcc *vcc;	struct sk_buff *skb;	int copied, error = -EINVAL;	if (sock->state != SS_CONNECTED)		return -ENOTCONN;	if (flags & ~MSG_DONTWAIT)		/* only handle MSG_DONTWAIT */		return -EOPNOTSUPP;	vcc = ATM_SD(sock);	if (test_bit(ATM_VF_RELEASED,&vcc->flags) ||	    test_bit(ATM_VF_CLOSE,&vcc->flags) ||	    !test_bit(ATM_VF_READY, &vcc->flags))		return 0;	skb = skb_recv_datagram(sk, flags, flags & MSG_DONTWAIT, &error);	if (!skb)		return error;	copied = skb->len;	if (copied > size) {		copied = size;		msg->msg_flags |= MSG_TRUNC;	}	error = skb_copy_datagram_iovec(skb, 0, msg->msg_iov, copied);	if (error)		return error;	sock_recv_timestamp(msg, sk, skb);	pr_debug("RcvM %d -= %d\n", atomic_read(&sk->sk_rmem_alloc), skb->truesize);	atm_return(vcc, skb->truesize);	skb_free_datagram(sk, skb);	return copied;}int vcc_sendmsg(struct kiocb *iocb, struct socket *sock, struct msghdr *m,		size_t total_len){	struct sock *sk = sock->sk;	DEFINE_WAIT(wait);	struct atm_vcc *vcc;	struct sk_buff *skb;	int eff,error;	const void __user *buff;	int size;	lock_sock(sk);	if (sock->state != SS_CONNECTED) {		error = -ENOTCONN;		goto out;	}	if (m->msg_name) {		error = -EISCONN;		goto out;	}	if (m->msg_iovlen != 1) {		error = -ENOSYS; /* fix this later @@@ */		goto out;	}	buff = m->msg_iov->iov_base;	size = m->msg_iov->iov_len;	vcc = ATM_SD(sock);	if (test_bit(ATM_VF_RELEASED, &vcc->flags) ||	    test_bit(ATM_VF_CLOSE, &vcc->flags) ||	    !test_bit(ATM_VF_READY, &vcc->flags)) {		error = -EPIPE;		send_sig(SIGPIPE, current, 0);		goto out;	}	if (!size) {		error = 0;		goto out;	}	if (size < 0 || size > vcc->qos.txtp.max_sdu) {		error = -EMSGSIZE;		goto out;	}	eff = (size+3) & ~3; /* align to word boundary */	prepare_to_wait(sk->sk_sleep, &wait, TASK_INTERRUPTIBLE);	error = 0;	while (!(skb = alloc_tx(vcc,eff))) {		if (m->msg_flags & MSG_DONTWAIT) {			error = -EAGAIN;			break;		}		schedule();		if (signal_pending(current)) {			error = -ERESTARTSYS;			break;		}		if (test_bit(ATM_VF_RELEASED,&vcc->flags) ||		    test_bit(ATM_VF_CLOSE,&vcc->flags) ||		    !test_bit(ATM_VF_READY,&vcc->flags)) {			error = -EPIPE;			send_sig(SIGPIPE, current, 0);			break;		}		prepare_to_wait(sk->sk_sleep, &wait, TASK_INTERRUPTIBLE);	}	finish_wait(sk->sk_sleep, &wait);	if (error)		goto out;	skb->dev = NULL; /* for paths shared with net_device interfaces */	ATM_SKB(skb)->atm_options = vcc->atm_options;	if (copy_from_user(skb_put(skb,size),buff,size)) {		kfree_skb(skb);		error = -EFAULT;		goto out;	}	if (eff != size) memset(skb->data+size,0,eff-size);	error = vcc->dev->ops->send(vcc,skb);	error = error ? error : size;out:	release_sock(sk);	return error;}unsigned int vcc_poll(struct file *file, struct socket *sock, poll_table *wait){	struct sock *sk = sock->sk;	struct atm_vcc *vcc;	unsigned int mask;	poll_wait(file, sk->sk_sleep, wait);	mask = 0;	vcc = ATM_SD(sock);	/* exceptional events */	if (sk->sk_err)		mask = POLLERR;	if (test_bit(ATM_VF_RELEASED, &vcc->flags) ||	    test_bit(ATM_VF_CLOSE, &vcc->flags))		mask |= POLLHUP;	/* readable? */	if (!skb_queue_empty(&sk->sk_receive_queue))		mask |= POLLIN | POLLRDNORM;	/* writable? */	if (sock->state == SS_CONNECTING &&	    test_bit(ATM_VF_WAITING, &vcc->flags))		return mask;	if (vcc->qos.txtp.traffic_class != ATM_NONE &&	    vcc_writable(sk))		mask |= POLLOUT | POLLWRNORM | POLLWRBAND;	return mask;}static int atm_change_qos(struct atm_vcc *vcc,struct atm_qos *qos){	int error;	/*	 * Don't let the QoS change the already connected AAL type nor the	 * traffic class.	 */	if (qos->aal != vcc->qos.aal ||	    qos->rxtp.traffic_class != vcc->qos.rxtp.traffic_class ||	    qos->txtp.traffic_class != vcc->qos.txtp.traffic_class)		return -EINVAL;	error = adjust_tp(&qos->txtp,qos->aal);	if (!error) error = adjust_tp(&qos->rxtp,qos->aal);	if (error) return error;	if (!vcc->dev->ops->change_qos) return -EOPNOTSUPP;	if (sk_atm(vcc)->sk_family == AF_ATMPVC)		return vcc->dev->ops->change_qos(vcc,qos,ATM_MF_SET);	return svc_change_qos(vcc,qos);}static int check_tp(struct atm_trafprm *tp){	/* @@@ Should be merged with adjust_tp */	if (!tp->traffic_class || tp->traffic_class == ATM_ANYCLASS) return 0;	if (tp->traffic_class != ATM_UBR && !tp->min_pcr && !tp->pcr &&	    !tp->max_pcr) return -EINVAL;	if (tp->min_pcr == ATM_MAX_PCR) return -EINVAL;	if (tp->min_pcr && tp->max_pcr && tp->max_pcr != ATM_MAX_PCR &&	    tp->min_pcr > tp->max_pcr) return -EINVAL;	/*	 * We allow pcr to be outside [min_pcr,max_pcr], because later	 * adjustment may still push it in the valid range.	 */	return 0;}static int check_qos(struct atm_qos *qos){	int error;	if (!qos->txtp.traffic_class && !qos->rxtp.traffic_class)		return -EINVAL;	if (qos->txtp.traffic_class != qos->rxtp.traffic_class &&	    qos->txtp.traffic_class && qos->rxtp.traffic_class &&	    qos->txtp.traffic_class != ATM_ANYCLASS &&	    qos->rxtp.traffic_class != ATM_ANYCLASS) return -EINVAL;	error = check_tp(&qos->txtp);	if (error) return error;	return check_tp(&qos->rxtp);}int vcc_setsockopt(struct socket *sock, int level, int optname,		   char __user *optval, int optlen){	struct atm_vcc *vcc;	unsigned long value;	int error;	if (__SO_LEVEL_MATCH(optname, level) && optlen != __SO_SIZE(optname))		return -EINVAL;	vcc = ATM_SD(sock);	switch (optname) {		case SO_ATMQOS:			{				struct atm_qos qos;				if (copy_from_user(&qos,optval,sizeof(qos)))					return -EFAULT;				error = check_qos(&qos);				if (error) return error;				if (sock->state == SS_CONNECTED)					return atm_change_qos(vcc,&qos);				if (sock->state != SS_UNCONNECTED)					return -EBADFD;				vcc->qos = qos;				set_bit(ATM_VF_HASQOS,&vcc->flags);				return 0;			}		case SO_SETCLP:			if (get_user(value,(unsigned long __user *)optval))				return -EFAULT;			if (value) vcc->atm_options |= ATM_ATMOPT_CLP;			else vcc->atm_options &= ~ATM_ATMOPT_CLP;			return 0;		default:			if (level == SOL_SOCKET) return -EINVAL;			break;	}	if (!vcc->dev || !vcc->dev->ops->setsockopt) return -EINVAL;	return vcc->dev->ops->setsockopt(vcc,level,optname,optval,optlen);}int vcc_getsockopt(struct socket *sock, int level, int optname,		   char __user *optval, int __user *optlen){	struct atm_vcc *vcc;	int len;	if (get_user(len, optlen))		return -EFAULT;	if (__SO_LEVEL_MATCH(optname, level) && len != __SO_SIZE(optname))		return -EINVAL;	vcc = ATM_SD(sock);	switch (optname) {		case SO_ATMQOS:			if (!test_bit(ATM_VF_HASQOS,&vcc->flags))				return -EINVAL;			return copy_to_user(optval,&vcc->qos,sizeof(vcc->qos)) ?			    -EFAULT : 0;		case SO_SETCLP:			return put_user(vcc->atm_options & ATM_ATMOPT_CLP ? 1 :			  0,(unsigned long __user *)optval) ? -EFAULT : 0;		case SO_ATMPVC:			{				struct sockaddr_atmpvc pvc;				if (!vcc->dev ||				    !test_bit(ATM_VF_ADDR,&vcc->flags))					return -ENOTCONN;				pvc.sap_family = AF_ATMPVC;				pvc.sap_addr.itf = vcc->dev->number;				pvc.sap_addr.vpi = vcc->vpi;				pvc.sap_addr.vci = vcc->vci;				return copy_to_user(optval,&pvc,sizeof(pvc)) ?				    -EFAULT : 0;			}		default:			if (level == SOL_SOCKET) return -EINVAL;			break;	}	if (!vcc->dev || !vcc->dev->ops->getsockopt) return -EINVAL;	return vcc->dev->ops->getsockopt(vcc, level, optname, optval, len);}static int __init atm_init(void){	int error;	if ((error = proto_register(&vcc_proto, 0)) < 0)		goto out;	if ((error = atmpvc_init()) < 0) {		printk(KERN_ERR "atmpvc_init() failed with %d\n", error);		goto out_unregister_vcc_proto;	}	if ((error = atmsvc_init()) < 0) {		printk(KERN_ERR "atmsvc_init() failed with %d\n", error);		goto out_atmpvc_exit;	}	if ((error = atm_proc_init()) < 0) {		printk(KERN_ERR "atm_proc_init() failed with %d\n",error);		goto out_atmsvc_exit;	}	if ((error = atm_sysfs_init()) < 0) {		printk(KERN_ERR "atm_sysfs_init() failed with %d\n",error);		goto out_atmproc_exit;	}out:	return error;out_atmproc_exit:	atm_proc_exit();out_atmsvc_exit:	atmsvc_exit();out_atmpvc_exit:	atmsvc_exit();out_unregister_vcc_proto:	proto_unregister(&vcc_proto);	goto out;}static void __exit atm_exit(void){	atm_proc_exit();	atm_sysfs_exit();	atmsvc_exit();	atmpvc_exit();	proto_unregister(&vcc_proto);}subsys_initcall(atm_init);module_exit(atm_exit);MODULE_LICENSE("GPL");MODULE_ALIAS_NETPROTO(PF_ATMPVC);MODULE_ALIAS_NETPROTO(PF_ATMSVC);

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -