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

📄 sock.c

📁 嵌入式系统设计与实例开发实验教材二源码 多线程应用程序设计 串行端口程序设计 AD接口实验 CAN总线通信实验 GPS通信实验 Linux内核移植与编译实验 IC卡读写实验 SD驱动使
💻 C
📖 第 1 页 / 共 2 页
字号:
		if (result < 0) {			if (result == -EAGAIN) {				DDPRINTK("ncpfs: tcp: bad select ready\n");				continue;			}			return result;		}		if (result == 0) {			printk(KERN_ERR "ncpfs: tcp: EOF on socket\n");			return -EIO;		}		if (result > len) {			printk(KERN_ERR "ncpfs: tcp: bug in recvmsg\n");			return -EIO;					}		dataread += result;		buffer += result;		len -= result;	}	return 0;}	#define NCP_TCP_XMIT_MAGIC	(0x446D6454)#define NCP_TCP_XMIT_VERSION	(1)#define NCP_TCP_RCVD_MAGIC	(0x744E6350)static int do_ncp_tcp_rpc_call(struct ncp_server *server, int size,		struct ncp_reply_header* reply_buf, int max_reply_size){	struct file *file;	struct socket *sock;	int result;	struct iovec iov[2];	struct msghdr msg;	struct scm_cookie scm;	__u32 ncptcp_rcvd_hdr[2];	__u32 ncptcp_xmit_hdr[4];	int   datalen;	/* We have to check the result, so store the complete header */	struct ncp_request_header request =	*((struct ncp_request_header *) (server->packet));	file = server->ncp_filp;	sock = &file->f_dentry->d_inode->u.socket_i;		ncptcp_xmit_hdr[0] = htonl(NCP_TCP_XMIT_MAGIC);	ncptcp_xmit_hdr[1] = htonl(size + 16);	ncptcp_xmit_hdr[2] = htonl(NCP_TCP_XMIT_VERSION);	ncptcp_xmit_hdr[3] = htonl(max_reply_size + 8);	DDPRINTK("ncpfs: req.typ: %04X, con: %d, "		 "seq: %d",		 request.type,		 (request.conn_high << 8) + request.conn_low,		 request.sequence);	DDPRINTK(" func: %d\n",		 request.function);	iov[1].iov_base = (void *) server->packet;	iov[1].iov_len = size;	iov[0].iov_base = ncptcp_xmit_hdr;	iov[0].iov_len = 16;	msg.msg_name = NULL;	msg.msg_namelen = 0;	msg.msg_control = NULL;	msg.msg_iov = iov;	msg.msg_iovlen = 2;	msg.msg_flags = MSG_NOSIGNAL;	result = scm_send(sock, &msg, &scm);	if (result < 0) {		return result;	}	result = sock->ops->sendmsg(sock, &msg, size + 16, &scm);	scm_destroy(&scm);	if (result < 0) {		printk(KERN_ERR "ncpfs: tcp: Send failed: %d\n", result);		return result;	}rstrcv:	result = do_tcp_rcv(server, ncptcp_rcvd_hdr, 8);	if (result)		return result;	if (ncptcp_rcvd_hdr[0] != htonl(NCP_TCP_RCVD_MAGIC)) {		printk(KERN_ERR "ncpfs: tcp: Unexpected reply type %08X\n", ntohl(ncptcp_rcvd_hdr[0]));		return -EIO;	}	datalen = ntohl(ncptcp_rcvd_hdr[1]);	if (datalen < 8 + sizeof(*reply_buf) || datalen > max_reply_size + 8) {		printk(KERN_ERR "ncpfs: tcp: Unexpected reply len %d\n", datalen);		return -EIO;	}	datalen -= 8;	result = do_tcp_rcv(server, reply_buf, datalen);	if (result)		return result;	if (reply_buf->type != NCP_REPLY) {		DDPRINTK("ncpfs: tcp: Unexpected NCP type %02X\n", reply_buf->type);		goto rstrcv;	}	if (request.type == NCP_ALLOC_SLOT_REQUEST)		return datalen;	if (reply_buf->sequence != request.sequence) {		printk(KERN_ERR "ncpfs: tcp: Bad sequence number\n");		return -EIO;	}	if ((reply_buf->conn_low != request.conn_low) ||	    (reply_buf->conn_high != request.conn_high)) {		printk(KERN_ERR "ncpfs: tcp: Connection number mismatch\n");		return -EIO;	}	return datalen;}/* * We need the server to be locked here, so check! */static int ncp_do_request(struct ncp_server *server, int size,		void* reply, int max_reply_size){	struct file *file;	struct socket *sock;	int result;	if (server->lock == 0) {		printk(KERN_ERR "ncpfs: Server not locked!\n");		return -EIO;	}	if (!ncp_conn_valid(server)) {		return -EIO;	}#ifdef CONFIG_NCPFS_PACKET_SIGNING	if (server->sign_active)	{		sign_packet(server, &size);	}#endif /* CONFIG_NCPFS_PACKET_SIGNING */	file = server->ncp_filp;	sock = &file->f_dentry->d_inode->u.socket_i;	/* N.B. this isn't needed ... check socket type? */	if (!sock) {		printk(KERN_ERR "ncp_rpc_call: socki_lookup failed\n");		result = -EBADF;	} else {		mm_segment_t fs;		sigset_t old_set;		unsigned long mask, flags;		spin_lock_irqsave(&current->sigmask_lock, flags);		old_set = current->blocked;		if (current->flags & PF_EXITING)			mask = 0;		else			mask = sigmask(SIGKILL);		if (server->m.flags & NCP_MOUNT_INTR) {			/* FIXME: This doesn't seem right at all.  So, like,			   we can't handle SIGINT and get whatever to stop?			   What if we've blocked it ourselves?  What about			   alarms?  Why, in fact, are we mucking with the			   sigmask at all? -- r~ */			if (current->sig->action[SIGINT - 1].sa.sa_handler == SIG_DFL)				mask |= sigmask(SIGINT);			if (current->sig->action[SIGQUIT - 1].sa.sa_handler == SIG_DFL)				mask |= sigmask(SIGQUIT);		}		siginitsetinv(&current->blocked, mask);		recalc_sigpending(current);		spin_unlock_irqrestore(&current->sigmask_lock, flags);				fs = get_fs();		set_fs(get_ds());		if (sock->type == SOCK_STREAM)			result = do_ncp_tcp_rpc_call(server, size, reply, max_reply_size);		else			result = do_ncp_rpc_call(server, size, reply, max_reply_size);		set_fs(fs);		spin_lock_irqsave(&current->sigmask_lock, flags);		current->blocked = old_set;		recalc_sigpending(current);		spin_unlock_irqrestore(&current->sigmask_lock, flags);	}	DDPRINTK("do_ncp_rpc_call returned %d\n", result);	if (result < 0) {		/* There was a problem with I/O, so the connections is		 * no longer usable. */		ncp_invalidate_conn(server);	}	return result;}/* ncp_do_request assures that at least a complete reply header is * received. It assumes that server->current_size contains the ncp * request size */int ncp_request2(struct ncp_server *server, int function, 		void* rpl, int size){	struct ncp_request_header *h;	struct ncp_reply_header* reply = rpl;	int request_size = server->current_size			 - sizeof(struct ncp_request_header);	int result;	h = (struct ncp_request_header *) (server->packet);	if (server->has_subfunction != 0) {		*(__u16 *) & (h->data[0]) = htons(request_size - 2);	}	h->type = NCP_REQUEST;	server->sequence += 1;	h->sequence = server->sequence;	h->conn_low = (server->connection) & 0xff;	h->conn_high = ((server->connection) & 0xff00) >> 8;	/*	 * The server shouldn't know or care what task is making a	 * request, so we always use the same task number.	 */	h->task = 2; /* (current->pid) & 0xff; */	h->function = function;	result = ncp_do_request(server, request_size + sizeof(*h), reply, size);	if (result < 0) {		DPRINTK("ncp_request_error: %d\n", result);		goto out;	}	server->completion = reply->completion_code;	server->conn_status = reply->connection_state;	server->reply_size = result;	server->ncp_reply_size = result - sizeof(struct ncp_reply_header);	result = reply->completion_code;	if (result != 0)		PPRINTK("ncp_request: completion code=%x\n", result);out:	return result;}int ncp_connect(struct ncp_server *server){	struct ncp_request_header *h;	int result;	h = (struct ncp_request_header *) (server->packet);	h->type = NCP_ALLOC_SLOT_REQUEST;	server->sequence = 0;	h->sequence	= server->sequence;	h->conn_low	= 0xff;	h->conn_high	= 0xff;	h->task		= 2; /* see above */	h->function	= 0;	result = ncp_do_request(server, sizeof(*h), server->packet, server->packet_size);	if (result < 0)		goto out;	server->sequence = 0;	server->connection = h->conn_low + (h->conn_high * 256);	result = 0;out:	return result;}int ncp_disconnect(struct ncp_server *server){	struct ncp_request_header *h;	h = (struct ncp_request_header *) (server->packet);	h->type = NCP_DEALLOC_SLOT_REQUEST;	server->sequence += 1;	h->sequence	= server->sequence;	h->conn_low	= (server->connection) & 0xff;	h->conn_high	= ((server->connection) & 0xff00) >> 8;	h->task		= 2; /* see above */	h->function	= 0;	return ncp_do_request(server, sizeof(*h), server->packet, server->packet_size);}void ncp_lock_server(struct ncp_server *server){	down(&server->sem);	if (server->lock)		printk(KERN_WARNING "ncp_lock_server: was locked!\n");	server->lock = 1;}void ncp_unlock_server(struct ncp_server *server){	if (!server->lock) {		printk(KERN_WARNING "ncp_unlock_server: was not locked!\n");		return;	}	server->lock = 0;	up(&server->sem);}

⌨️ 快捷键说明

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