forked from mirrors/linux
		
	ipwireless: driver for PC Card 3G/UMTS modem
The device is manufactured by IPWireless. In some countries (for example Czech Republic, T-Mobile ISP) this card is shipped for service called UMTS 4G. It's a piece of PCMCIA "4G" UMTS PPP networking hardware that presents itself as a serial character device (i.e. looks like usual modem to userspace, accepts AT commands, etc). Rewieved-by: Jiri Slaby <jslaby@suse.cz> Signed-off-by: Ben Martel <benm@symmetric.co.nz> Signed-off-by: Stephen Blackheath <stephen@symmetric.co.nz> Signed-off-by: David Sterba <dsterba@suse.cz> Signed-off-by: Jiri Kosina <jkosina@suse.cz> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
		
							parent
							
								
									151db1fc23
								
							
						
					
					
						commit
						099dc4fb62
					
				
					 13 changed files with 3862 additions and 0 deletions
				
			
		| 
						 | 
				
			
			@ -2150,6 +2150,14 @@ M:	acme@ghostprotocols.net
 | 
			
		|||
L:	netdev@vger.kernel.org
 | 
			
		||||
S:	Maintained
 | 
			
		||||
 | 
			
		||||
IPWIRELES DRIVER
 | 
			
		||||
P:	Jiri Kosina
 | 
			
		||||
M:	jkosina@suse.cz
 | 
			
		||||
P:	David Sterba
 | 
			
		||||
M:	dsterba@suse.cz
 | 
			
		||||
S:	Maintained
 | 
			
		||||
T:	git://git.kernel.org/pub/scm/linux/kernel/git/jikos/ipwireless_cs.git
 | 
			
		||||
 | 
			
		||||
IRDA SUBSYSTEM
 | 
			
		||||
P:	Samuel Ortiz
 | 
			
		||||
M:	samuel@sortiz.org
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -43,5 +43,14 @@ config CARDMAN_4040
 | 
			
		|||
	  (http://www.omnikey.com/), or a current development version of OpenCT
 | 
			
		||||
	  (http://www.opensc.org/).
 | 
			
		||||
 | 
			
		||||
config IPWIRELESS
 | 
			
		||||
	tristate "IPWireless 3G UMTS PCMCIA card support"
 | 
			
		||||
	depends on PCMCIA
 | 
			
		||||
	select PPP
 | 
			
		||||
	help
 | 
			
		||||
	  This is a driver for 3G UMTS PCMCIA card from IPWireless company. In
 | 
			
		||||
	  some countries (for example Czech Republic, T-Mobile ISP) this card
 | 
			
		||||
	  is shipped for service called UMTS 4G.
 | 
			
		||||
 | 
			
		||||
endmenu
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -4,6 +4,8 @@
 | 
			
		|||
# Makefile for the Linux PCMCIA char device drivers.
 | 
			
		||||
#
 | 
			
		||||
 | 
			
		||||
obj-y += ipwireless/
 | 
			
		||||
 | 
			
		||||
obj-$(CONFIG_SYNCLINK_CS) += synclink_cs.o
 | 
			
		||||
obj-$(CONFIG_CARDMAN_4000) += cm4000_cs.o
 | 
			
		||||
obj-$(CONFIG_CARDMAN_4040) += cm4040_cs.o
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										10
									
								
								drivers/char/pcmcia/ipwireless/Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								drivers/char/pcmcia/ipwireless/Makefile
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,10 @@
 | 
			
		|||
#
 | 
			
		||||
# drivers/char/pcmcia/ipwireless/Makefile
 | 
			
		||||
#
 | 
			
		||||
# Makefile for the IPWireless driver
 | 
			
		||||
#
 | 
			
		||||
 | 
			
		||||
obj-$(CONFIG_IPWIRELESS) += ipwireless.o
 | 
			
		||||
 | 
			
		||||
ipwireless-objs := hardware.o main.o network.o tty.o
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										1787
									
								
								drivers/char/pcmcia/ipwireless/hardware.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1787
									
								
								drivers/char/pcmcia/ipwireless/hardware.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							
							
								
								
									
										64
									
								
								drivers/char/pcmcia/ipwireless/hardware.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								drivers/char/pcmcia/ipwireless/hardware.h
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,64 @@
 | 
			
		|||
/*
 | 
			
		||||
 * IPWireless 3G PCMCIA Network Driver
 | 
			
		||||
 *
 | 
			
		||||
 * Original code
 | 
			
		||||
 *   by Stephen Blackheath <stephen@blacksapphire.com>,
 | 
			
		||||
 *      Ben Martel <benm@symmetric.co.nz>
 | 
			
		||||
 *
 | 
			
		||||
 * Copyrighted as follows:
 | 
			
		||||
 *   Copyright (C) 2004 by Symmetric Systems Ltd (NZ)
 | 
			
		||||
 *
 | 
			
		||||
 * Various driver changes and rewrites, port to new kernels
 | 
			
		||||
 *   Copyright (C) 2006-2007 Jiri Kosina
 | 
			
		||||
 *
 | 
			
		||||
 * Misc code cleanups and updates
 | 
			
		||||
 *   Copyright (C) 2007 David Sterba
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
#ifndef _IPWIRELESS_CS_HARDWARE_H_
 | 
			
		||||
#define _IPWIRELESS_CS_HARDWARE_H_
 | 
			
		||||
 | 
			
		||||
#include <linux/types.h>
 | 
			
		||||
#include <linux/sched.h>
 | 
			
		||||
#include <linux/interrupt.h>
 | 
			
		||||
 | 
			
		||||
#define IPW_CONTROL_LINE_CTS 0x0001
 | 
			
		||||
#define IPW_CONTROL_LINE_DCD 0x0002
 | 
			
		||||
#define IPW_CONTROL_LINE_DSR 0x0004
 | 
			
		||||
#define IPW_CONTROL_LINE_RI  0x0008
 | 
			
		||||
#define IPW_CONTROL_LINE_DTR 0x0010
 | 
			
		||||
#define IPW_CONTROL_LINE_RTS 0x0020
 | 
			
		||||
 | 
			
		||||
struct ipw_hardware;
 | 
			
		||||
struct ipw_network;
 | 
			
		||||
 | 
			
		||||
struct ipw_hardware *ipwireless_hardware_create(void);
 | 
			
		||||
void ipwireless_hardware_free(struct ipw_hardware *hw);
 | 
			
		||||
irqreturn_t ipwireless_interrupt(int irq, void *dev_id, struct pt_regs *regs);
 | 
			
		||||
int ipwireless_set_DTR(struct ipw_hardware *hw, unsigned int channel_idx,
 | 
			
		||||
		int state);
 | 
			
		||||
int ipwireless_set_RTS(struct ipw_hardware *hw, unsigned int channel_idx,
 | 
			
		||||
		int state);
 | 
			
		||||
int ipwireless_send_packet(struct ipw_hardware *hw,
 | 
			
		||||
			    unsigned int channel_idx,
 | 
			
		||||
			    unsigned char *data,
 | 
			
		||||
			    unsigned int length,
 | 
			
		||||
			    void (*packet_sent_callback) (void *cb,
 | 
			
		||||
							  unsigned int length),
 | 
			
		||||
			    void *sent_cb_data);
 | 
			
		||||
void ipwireless_associate_network(struct ipw_hardware *hw,
 | 
			
		||||
		struct ipw_network *net);
 | 
			
		||||
void ipwireless_stop_interrupts(struct ipw_hardware *hw);
 | 
			
		||||
void ipwireless_init_hardware_v1(struct ipw_hardware *hw,
 | 
			
		||||
				 unsigned int base_port,
 | 
			
		||||
				 void __iomem *attr_memory,
 | 
			
		||||
				 void __iomem *common_memory,
 | 
			
		||||
				 int is_v2_card,
 | 
			
		||||
				 void (*reboot_cb) (void *data),
 | 
			
		||||
				 void *reboot_cb_data);
 | 
			
		||||
void ipwireless_init_hardware_v2_v3(struct ipw_hardware *hw);
 | 
			
		||||
void ipwireless_sleep(unsigned int tenths);
 | 
			
		||||
int ipwireless_dump_hardware_state(char *p, size_t limit,
 | 
			
		||||
				   struct ipw_hardware *hw);
 | 
			
		||||
 | 
			
		||||
#endif
 | 
			
		||||
							
								
								
									
										501
									
								
								drivers/char/pcmcia/ipwireless/main.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										501
									
								
								drivers/char/pcmcia/ipwireless/main.c
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,501 @@
 | 
			
		|||
/*
 | 
			
		||||
 * IPWireless 3G PCMCIA Network Driver
 | 
			
		||||
 *
 | 
			
		||||
 * Original code
 | 
			
		||||
 *   by Stephen Blackheath <stephen@blacksapphire.com>,
 | 
			
		||||
 *      Ben Martel <benm@symmetric.co.nz>
 | 
			
		||||
 *
 | 
			
		||||
 * Copyrighted as follows:
 | 
			
		||||
 *   Copyright (C) 2004 by Symmetric Systems Ltd (NZ)
 | 
			
		||||
 *
 | 
			
		||||
 * Various driver changes and rewrites, port to new kernels
 | 
			
		||||
 *   Copyright (C) 2006-2007 Jiri Kosina
 | 
			
		||||
 *
 | 
			
		||||
 * Misc code cleanups and updates
 | 
			
		||||
 *   Copyright (C) 2007 David Sterba
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
#include "hardware.h"
 | 
			
		||||
#include "network.h"
 | 
			
		||||
#include "main.h"
 | 
			
		||||
#include "tty.h"
 | 
			
		||||
 | 
			
		||||
#include <linux/delay.h>
 | 
			
		||||
#include <linux/init.h>
 | 
			
		||||
#include <linux/io.h>
 | 
			
		||||
#include <linux/kernel.h>
 | 
			
		||||
#include <linux/module.h>
 | 
			
		||||
#include <linux/sched.h>
 | 
			
		||||
#include <linux/slab.h>
 | 
			
		||||
 | 
			
		||||
#include <pcmcia/version.h>
 | 
			
		||||
#include <pcmcia/cisreg.h>
 | 
			
		||||
#include <pcmcia/device_id.h>
 | 
			
		||||
#include <pcmcia/ss.h>
 | 
			
		||||
#include <pcmcia/ds.h>
 | 
			
		||||
#include <pcmcia/cs.h>
 | 
			
		||||
 | 
			
		||||
static struct pcmcia_device_id ipw_ids[] = {
 | 
			
		||||
	PCMCIA_DEVICE_MANF_CARD(0x02f2, 0x0100),
 | 
			
		||||
	PCMCIA_DEVICE_MANF_CARD(0x02f2, 0x0200),
 | 
			
		||||
	PCMCIA_DEVICE_NULL
 | 
			
		||||
};
 | 
			
		||||
MODULE_DEVICE_TABLE(pcmcia, ipw_ids);
 | 
			
		||||
 | 
			
		||||
static void ipwireless_detach(struct pcmcia_device *link);
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * Module params
 | 
			
		||||
 */
 | 
			
		||||
/* Debug mode: more verbose, print sent/recv bytes */
 | 
			
		||||
int ipwireless_debug;
 | 
			
		||||
int ipwireless_loopback;
 | 
			
		||||
int ipwireless_out_queue = 1;
 | 
			
		||||
 | 
			
		||||
module_param_named(debug, ipwireless_debug, int, 0);
 | 
			
		||||
module_param_named(loopback, ipwireless_loopback, int, 0);
 | 
			
		||||
module_param_named(out_queue, ipwireless_out_queue, int, 0);
 | 
			
		||||
MODULE_PARM_DESC(debug, "switch on debug messages [0]");
 | 
			
		||||
MODULE_PARM_DESC(loopback,
 | 
			
		||||
		"debug: enable ras_raw channel [0]");
 | 
			
		||||
MODULE_PARM_DESC(out_queue, "debug: set size of outgoing queue [1]");
 | 
			
		||||
 | 
			
		||||
/* Executes in process context. */
 | 
			
		||||
static void signalled_reboot_work(struct work_struct *work_reboot)
 | 
			
		||||
{
 | 
			
		||||
	struct ipw_dev *ipw = container_of(work_reboot, struct ipw_dev,
 | 
			
		||||
			work_reboot);
 | 
			
		||||
	struct pcmcia_device *link = ipw->link;
 | 
			
		||||
	int ret = pccard_reset_card(link->socket);
 | 
			
		||||
 | 
			
		||||
	if (ret != CS_SUCCESS)
 | 
			
		||||
		cs_error(link, ResetCard, ret);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void signalled_reboot_callback(void *callback_data)
 | 
			
		||||
{
 | 
			
		||||
	struct ipw_dev *ipw = (struct ipw_dev *) callback_data;
 | 
			
		||||
 | 
			
		||||
	/* Delegate to process context. */
 | 
			
		||||
	schedule_work(&ipw->work_reboot);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int config_ipwireless(struct ipw_dev *ipw)
 | 
			
		||||
{
 | 
			
		||||
	struct pcmcia_device *link = ipw->link;
 | 
			
		||||
	int ret;
 | 
			
		||||
	config_info_t conf;
 | 
			
		||||
	tuple_t tuple;
 | 
			
		||||
	unsigned short buf[64];
 | 
			
		||||
	cisparse_t parse;
 | 
			
		||||
	unsigned short cor_value;
 | 
			
		||||
	win_req_t request_attr_memory;
 | 
			
		||||
	win_req_t request_common_memory;
 | 
			
		||||
	memreq_t memreq_attr_memory;
 | 
			
		||||
	memreq_t memreq_common_memory;
 | 
			
		||||
 | 
			
		||||
	ipw->is_v2_card = 0;
 | 
			
		||||
 | 
			
		||||
	tuple.Attributes = 0;
 | 
			
		||||
	tuple.TupleData = (cisdata_t *) buf;
 | 
			
		||||
	tuple.TupleDataMax = sizeof(buf);
 | 
			
		||||
	tuple.TupleOffset = 0;
 | 
			
		||||
 | 
			
		||||
	tuple.DesiredTuple = RETURN_FIRST_TUPLE;
 | 
			
		||||
 | 
			
		||||
	ret = pcmcia_get_first_tuple(link, &tuple);
 | 
			
		||||
 | 
			
		||||
	while (ret == 0) {
 | 
			
		||||
		ret = pcmcia_get_tuple_data(link, &tuple);
 | 
			
		||||
 | 
			
		||||
		if (ret != CS_SUCCESS) {
 | 
			
		||||
			cs_error(link, GetTupleData, ret);
 | 
			
		||||
			goto exit0;
 | 
			
		||||
		}
 | 
			
		||||
		ret = pcmcia_get_next_tuple(link, &tuple);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	tuple.DesiredTuple = CISTPL_CFTABLE_ENTRY;
 | 
			
		||||
 | 
			
		||||
	ret = pcmcia_get_first_tuple(link, &tuple);
 | 
			
		||||
 | 
			
		||||
	if (ret != CS_SUCCESS) {
 | 
			
		||||
		cs_error(link, GetFirstTuple, ret);
 | 
			
		||||
		goto exit0;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	ret = pcmcia_get_tuple_data(link, &tuple);
 | 
			
		||||
 | 
			
		||||
	if (ret != CS_SUCCESS) {
 | 
			
		||||
		cs_error(link, GetTupleData, ret);
 | 
			
		||||
		goto exit0;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	ret = pcmcia_parse_tuple(link, &tuple, &parse);
 | 
			
		||||
 | 
			
		||||
	if (ret != CS_SUCCESS) {
 | 
			
		||||
		cs_error(link, ParseTuple, ret);
 | 
			
		||||
		goto exit0;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	link->io.Attributes1 = IO_DATA_PATH_WIDTH_AUTO;
 | 
			
		||||
	link->io.BasePort1 = parse.cftable_entry.io.win[0].base;
 | 
			
		||||
	link->io.NumPorts1 = parse.cftable_entry.io.win[0].len;
 | 
			
		||||
	link->io.IOAddrLines = 16;
 | 
			
		||||
 | 
			
		||||
	link->irq.IRQInfo1 = parse.cftable_entry.irq.IRQInfo1;
 | 
			
		||||
 | 
			
		||||
	/* 0x40 causes it to generate level mode interrupts. */
 | 
			
		||||
	/* 0x04 enables IREQ pin. */
 | 
			
		||||
	cor_value = parse.cftable_entry.index | 0x44;
 | 
			
		||||
	link->conf.ConfigIndex = cor_value;
 | 
			
		||||
 | 
			
		||||
	/* IRQ and I/O settings */
 | 
			
		||||
	tuple.DesiredTuple = CISTPL_CONFIG;
 | 
			
		||||
 | 
			
		||||
	ret = pcmcia_get_first_tuple(link, &tuple);
 | 
			
		||||
 | 
			
		||||
	if (ret != CS_SUCCESS) {
 | 
			
		||||
		cs_error(link, GetFirstTuple, ret);
 | 
			
		||||
		goto exit0;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	ret = pcmcia_get_tuple_data(link, &tuple);
 | 
			
		||||
 | 
			
		||||
	if (ret != CS_SUCCESS) {
 | 
			
		||||
		cs_error(link, GetTupleData, ret);
 | 
			
		||||
		goto exit0;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	ret = pcmcia_parse_tuple(link, &tuple, &parse);
 | 
			
		||||
 | 
			
		||||
	if (ret != CS_SUCCESS) {
 | 
			
		||||
		cs_error(link, GetTupleData, ret);
 | 
			
		||||
		goto exit0;
 | 
			
		||||
	}
 | 
			
		||||
	link->conf.Attributes = CONF_ENABLE_IRQ;
 | 
			
		||||
	link->conf.ConfigBase = parse.config.base;
 | 
			
		||||
	link->conf.Present = parse.config.rmask[0];
 | 
			
		||||
	link->conf.IntType = INT_MEMORY_AND_IO;
 | 
			
		||||
 | 
			
		||||
	link->irq.Attributes = IRQ_TYPE_DYNAMIC_SHARING | IRQ_HANDLE_PRESENT;
 | 
			
		||||
	link->irq.Handler = ipwireless_interrupt;
 | 
			
		||||
	link->irq.Instance = ipw->hardware;
 | 
			
		||||
 | 
			
		||||
	ret = pcmcia_request_io(link, &link->io);
 | 
			
		||||
 | 
			
		||||
	if (ret != CS_SUCCESS) {
 | 
			
		||||
		cs_error(link, RequestIO, ret);
 | 
			
		||||
		goto exit0;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* memory settings */
 | 
			
		||||
 | 
			
		||||
	tuple.DesiredTuple = CISTPL_CFTABLE_ENTRY;
 | 
			
		||||
 | 
			
		||||
	ret = pcmcia_get_first_tuple(link, &tuple);
 | 
			
		||||
 | 
			
		||||
	if (ret != CS_SUCCESS) {
 | 
			
		||||
		cs_error(link, GetFirstTuple, ret);
 | 
			
		||||
		goto exit1;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	ret = pcmcia_get_tuple_data(link, &tuple);
 | 
			
		||||
 | 
			
		||||
	if (ret != CS_SUCCESS) {
 | 
			
		||||
		cs_error(link, GetTupleData, ret);
 | 
			
		||||
		goto exit1;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	ret = pcmcia_parse_tuple(link, &tuple, &parse);
 | 
			
		||||
 | 
			
		||||
	if (ret != CS_SUCCESS) {
 | 
			
		||||
		cs_error(link, ParseTuple, ret);
 | 
			
		||||
		goto exit1;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (parse.cftable_entry.mem.nwin > 0) {
 | 
			
		||||
		request_common_memory.Attributes =
 | 
			
		||||
			WIN_DATA_WIDTH_16 | WIN_MEMORY_TYPE_CM | WIN_ENABLE;
 | 
			
		||||
		request_common_memory.Base =
 | 
			
		||||
			parse.cftable_entry.mem.win[0].host_addr;
 | 
			
		||||
		request_common_memory.Size = parse.cftable_entry.mem.win[0].len;
 | 
			
		||||
		if (request_common_memory.Size < 0x1000)
 | 
			
		||||
			request_common_memory.Size = 0x1000;
 | 
			
		||||
		request_common_memory.AccessSpeed = 0;
 | 
			
		||||
 | 
			
		||||
		ret = pcmcia_request_window(&link, &request_common_memory,
 | 
			
		||||
				&ipw->handle_common_memory);
 | 
			
		||||
 | 
			
		||||
		if (ret != CS_SUCCESS) {
 | 
			
		||||
			cs_error(link, RequestWindow, ret);
 | 
			
		||||
			goto exit1;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		memreq_common_memory.CardOffset =
 | 
			
		||||
			parse.cftable_entry.mem.win[0].card_addr;
 | 
			
		||||
		memreq_common_memory.Page = 0;
 | 
			
		||||
 | 
			
		||||
		ret = pcmcia_map_mem_page(ipw->handle_common_memory,
 | 
			
		||||
				&memreq_common_memory);
 | 
			
		||||
 | 
			
		||||
		if (ret != CS_SUCCESS) {
 | 
			
		||||
			cs_error(link, MapMemPage, ret);
 | 
			
		||||
			goto exit1;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		ipw->is_v2_card =
 | 
			
		||||
			parse.cftable_entry.mem.win[0].len == 0x100;
 | 
			
		||||
 | 
			
		||||
		ipw->common_memory = ioremap(request_common_memory.Base,
 | 
			
		||||
				request_common_memory.Size);
 | 
			
		||||
 | 
			
		||||
		request_attr_memory.Attributes =
 | 
			
		||||
			WIN_DATA_WIDTH_16 | WIN_MEMORY_TYPE_AM | WIN_ENABLE;
 | 
			
		||||
		request_attr_memory.Base = 0;
 | 
			
		||||
		request_attr_memory.Size = 0;	/* this used to be 0x1000 */
 | 
			
		||||
		request_attr_memory.AccessSpeed = 0;
 | 
			
		||||
 | 
			
		||||
		ret = pcmcia_request_window(&link, &request_attr_memory,
 | 
			
		||||
				&ipw->handle_attr_memory);
 | 
			
		||||
 | 
			
		||||
		if (ret != CS_SUCCESS) {
 | 
			
		||||
			cs_error(link, RequestWindow, ret);
 | 
			
		||||
			goto exit2;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		memreq_attr_memory.CardOffset = 0;
 | 
			
		||||
		memreq_attr_memory.Page = 0;
 | 
			
		||||
 | 
			
		||||
		ret = pcmcia_map_mem_page(ipw->handle_attr_memory,
 | 
			
		||||
				&memreq_attr_memory);
 | 
			
		||||
 | 
			
		||||
		if (ret != CS_SUCCESS) {
 | 
			
		||||
			cs_error(link, MapMemPage, ret);
 | 
			
		||||
			goto exit2;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		ipw->attr_memory = ioremap(request_attr_memory.Base,
 | 
			
		||||
				request_attr_memory.Size);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	INIT_WORK(&ipw->work_reboot, signalled_reboot_work);
 | 
			
		||||
 | 
			
		||||
	ipwireless_init_hardware_v1(ipw->hardware, link->io.BasePort1,
 | 
			
		||||
				    ipw->attr_memory, ipw->common_memory,
 | 
			
		||||
				    ipw->is_v2_card, signalled_reboot_callback,
 | 
			
		||||
				    ipw);
 | 
			
		||||
 | 
			
		||||
	ret = pcmcia_request_irq(link, &link->irq);
 | 
			
		||||
 | 
			
		||||
	if (ret != CS_SUCCESS) {
 | 
			
		||||
		cs_error(link, RequestIRQ, ret);
 | 
			
		||||
		goto exit3;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* Look up current Vcc */
 | 
			
		||||
 | 
			
		||||
	ret = pcmcia_get_configuration_info(link, &conf);
 | 
			
		||||
 | 
			
		||||
	if (ret != CS_SUCCESS) {
 | 
			
		||||
		cs_error(link, GetConfigurationInfo, ret);
 | 
			
		||||
		goto exit4;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	printk(KERN_INFO IPWIRELESS_PCCARD_NAME ": Card type %s\n",
 | 
			
		||||
			ipw->is_v2_card ? "V2/V3" : "V1");
 | 
			
		||||
	printk(KERN_INFO IPWIRELESS_PCCARD_NAME
 | 
			
		||||
			": I/O ports 0x%04x-0x%04x, irq %d\n",
 | 
			
		||||
			(unsigned int) link->io.BasePort1,
 | 
			
		||||
			(unsigned int) (link->io.BasePort1 +
 | 
			
		||||
				link->io.NumPorts1 - 1),
 | 
			
		||||
			(unsigned int) link->irq.AssignedIRQ);
 | 
			
		||||
	if (ipw->attr_memory && ipw->common_memory)
 | 
			
		||||
		printk(KERN_INFO IPWIRELESS_PCCARD_NAME
 | 
			
		||||
				": attr memory 0x%08lx-0x%08lx, "
 | 
			
		||||
				"common memory 0x%08lx-0x%08lx\n",
 | 
			
		||||
				request_attr_memory.Base,
 | 
			
		||||
				request_attr_memory.Base
 | 
			
		||||
				+ request_attr_memory.Size - 1,
 | 
			
		||||
				request_common_memory.Base,
 | 
			
		||||
				request_common_memory.Base
 | 
			
		||||
				+ request_common_memory.Size - 1);
 | 
			
		||||
 | 
			
		||||
	ipw->network = ipwireless_network_create(ipw->hardware);
 | 
			
		||||
	if (!ipw->network)
 | 
			
		||||
		goto exit3;
 | 
			
		||||
 | 
			
		||||
	ipw->tty = ipwireless_tty_create(ipw->hardware, ipw->network,
 | 
			
		||||
			ipw->nodes);
 | 
			
		||||
	if (!ipw->tty)
 | 
			
		||||
		goto exit3;
 | 
			
		||||
 | 
			
		||||
	ipwireless_init_hardware_v2_v3(ipw->hardware);
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * Do the RequestConfiguration last, because it enables interrupts.
 | 
			
		||||
	 * Then we don't get any interrupts before we're ready for them.
 | 
			
		||||
	 */
 | 
			
		||||
	ret = pcmcia_request_configuration(link, &link->conf);
 | 
			
		||||
 | 
			
		||||
	if (ret != CS_SUCCESS) {
 | 
			
		||||
		cs_error(link, RequestConfiguration, ret);
 | 
			
		||||
		goto exit4;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	link->dev_node = &ipw->nodes[0];
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
 | 
			
		||||
exit4:
 | 
			
		||||
	pcmcia_disable_device(link);
 | 
			
		||||
exit3:
 | 
			
		||||
	if (ipw->attr_memory) {
 | 
			
		||||
		iounmap(ipw->attr_memory);
 | 
			
		||||
		pcmcia_release_window(ipw->handle_attr_memory);
 | 
			
		||||
		pcmcia_disable_device(link);
 | 
			
		||||
	}
 | 
			
		||||
exit2:
 | 
			
		||||
	if (ipw->common_memory) {
 | 
			
		||||
		iounmap(ipw->common_memory);
 | 
			
		||||
		pcmcia_release_window(ipw->handle_common_memory);
 | 
			
		||||
	}
 | 
			
		||||
exit1:
 | 
			
		||||
	pcmcia_disable_device(link);
 | 
			
		||||
exit0:
 | 
			
		||||
	return -1;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void release_ipwireless(struct ipw_dev *ipw)
 | 
			
		||||
{
 | 
			
		||||
	struct pcmcia_device *link = ipw->link;
 | 
			
		||||
 | 
			
		||||
	pcmcia_disable_device(link);
 | 
			
		||||
 | 
			
		||||
	if (ipw->common_memory)
 | 
			
		||||
		iounmap(ipw->common_memory);
 | 
			
		||||
	if (ipw->attr_memory)
 | 
			
		||||
		iounmap(ipw->attr_memory);
 | 
			
		||||
	if (ipw->common_memory)
 | 
			
		||||
		pcmcia_release_window(ipw->handle_common_memory);
 | 
			
		||||
	if (ipw->attr_memory)
 | 
			
		||||
		pcmcia_release_window(ipw->handle_attr_memory);
 | 
			
		||||
	pcmcia_disable_device(link);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * ipwireless_attach() creates an "instance" of the driver, allocating
 | 
			
		||||
 * local data structures for one device (one interface).  The device
 | 
			
		||||
 * is registered with Card Services.
 | 
			
		||||
 *
 | 
			
		||||
 * The pcmcia_device structure is initialized, but we don't actually
 | 
			
		||||
 * configure the card at this point -- we wait until we receive a
 | 
			
		||||
 * card insertion event.
 | 
			
		||||
 */
 | 
			
		||||
static int ipwireless_attach(struct pcmcia_device *link)
 | 
			
		||||
{
 | 
			
		||||
	struct ipw_dev *ipw;
 | 
			
		||||
	int ret;
 | 
			
		||||
 | 
			
		||||
	ipw = kzalloc(sizeof(struct ipw_dev), GFP_KERNEL);
 | 
			
		||||
	if (!ipw)
 | 
			
		||||
		return -ENOMEM;
 | 
			
		||||
 | 
			
		||||
	ipw->link = link;
 | 
			
		||||
	link->priv = ipw;
 | 
			
		||||
	link->irq.Instance = ipw;
 | 
			
		||||
 | 
			
		||||
	/* Link this device into our device list. */
 | 
			
		||||
	link->dev_node = &ipw->nodes[0];
 | 
			
		||||
 | 
			
		||||
	ipw->hardware = ipwireless_hardware_create();
 | 
			
		||||
	if (!ipw->hardware) {
 | 
			
		||||
		kfree(ipw);
 | 
			
		||||
		return -ENOMEM;
 | 
			
		||||
	}
 | 
			
		||||
	/* RegisterClient will call config_ipwireless */
 | 
			
		||||
 | 
			
		||||
	ret = config_ipwireless(ipw);
 | 
			
		||||
 | 
			
		||||
	if (ret != 0) {
 | 
			
		||||
		cs_error(link, RegisterClient, ret);
 | 
			
		||||
		ipwireless_detach(link);
 | 
			
		||||
		return ret;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * This deletes a driver "instance".  The device is de-registered with
 | 
			
		||||
 * Card Services.  If it has been released, all local data structures
 | 
			
		||||
 * are freed.  Otherwise, the structures will be freed when the device
 | 
			
		||||
 * is released.
 | 
			
		||||
 */
 | 
			
		||||
static void ipwireless_detach(struct pcmcia_device *link)
 | 
			
		||||
{
 | 
			
		||||
	struct ipw_dev *ipw = link->priv;
 | 
			
		||||
 | 
			
		||||
	release_ipwireless(ipw);
 | 
			
		||||
 | 
			
		||||
	/* Break the link with Card Services */
 | 
			
		||||
	if (link)
 | 
			
		||||
		pcmcia_disable_device(link);
 | 
			
		||||
 | 
			
		||||
	if (ipw->tty != NULL)
 | 
			
		||||
		ipwireless_tty_free(ipw->tty);
 | 
			
		||||
	if (ipw->network != NULL)
 | 
			
		||||
		ipwireless_network_free(ipw->network);
 | 
			
		||||
	if (ipw->hardware != NULL)
 | 
			
		||||
		ipwireless_hardware_free(ipw->hardware);
 | 
			
		||||
	kfree(ipw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static struct pcmcia_driver me = {
 | 
			
		||||
	.owner		= THIS_MODULE,
 | 
			
		||||
	.probe          = ipwireless_attach,
 | 
			
		||||
	.remove         = ipwireless_detach,
 | 
			
		||||
	.drv = { .name  = IPWIRELESS_PCCARD_NAME },
 | 
			
		||||
	.id_table       = ipw_ids
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * Module insertion : initialisation of the module.
 | 
			
		||||
 * Register the card with cardmgr...
 | 
			
		||||
 */
 | 
			
		||||
static int __init init_ipwireless(void)
 | 
			
		||||
{
 | 
			
		||||
	int ret;
 | 
			
		||||
 | 
			
		||||
	printk(KERN_INFO IPWIRELESS_PCCARD_NAME " "
 | 
			
		||||
	       IPWIRELESS_PCMCIA_VERSION " by " IPWIRELESS_PCMCIA_AUTHOR "\n");
 | 
			
		||||
 | 
			
		||||
	ret = ipwireless_tty_init();
 | 
			
		||||
	if (ret != 0)
 | 
			
		||||
		return ret;
 | 
			
		||||
 | 
			
		||||
	ret = pcmcia_register_driver(&me);
 | 
			
		||||
	if (ret != 0)
 | 
			
		||||
		ipwireless_tty_release();
 | 
			
		||||
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * Module removal
 | 
			
		||||
 */
 | 
			
		||||
static void __exit exit_ipwireless(void)
 | 
			
		||||
{
 | 
			
		||||
	printk(KERN_INFO IPWIRELESS_PCCARD_NAME " "
 | 
			
		||||
			IPWIRELESS_PCMCIA_VERSION " removed\n");
 | 
			
		||||
 | 
			
		||||
	pcmcia_unregister_driver(&me);
 | 
			
		||||
	ipwireless_tty_release();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
module_init(init_ipwireless);
 | 
			
		||||
module_exit(exit_ipwireless);
 | 
			
		||||
 | 
			
		||||
MODULE_AUTHOR(IPWIRELESS_PCMCIA_AUTHOR);
 | 
			
		||||
MODULE_DESCRIPTION(IPWIRELESS_PCCARD_NAME " " IPWIRELESS_PCMCIA_VERSION);
 | 
			
		||||
MODULE_LICENSE("GPL");
 | 
			
		||||
							
								
								
									
										70
									
								
								drivers/char/pcmcia/ipwireless/main.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								drivers/char/pcmcia/ipwireless/main.h
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,70 @@
 | 
			
		|||
/*
 | 
			
		||||
 * IPWireless 3G PCMCIA Network Driver
 | 
			
		||||
 *
 | 
			
		||||
 * Original code
 | 
			
		||||
 *   by Stephen Blackheath <stephen@blacksapphire.com>,
 | 
			
		||||
 *      Ben Martel <benm@symmetric.co.nz>
 | 
			
		||||
 *
 | 
			
		||||
 * Copyrighted as follows:
 | 
			
		||||
 *   Copyright (C) 2004 by Symmetric Systems Ltd (NZ)
 | 
			
		||||
 *
 | 
			
		||||
 * Various driver changes and rewrites, port to new kernels
 | 
			
		||||
 *   Copyright (C) 2006-2007 Jiri Kosina
 | 
			
		||||
 *
 | 
			
		||||
 * Misc code cleanups and updates
 | 
			
		||||
 *   Copyright (C) 2007 David Sterba
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
#ifndef _IPWIRELESS_CS_H_
 | 
			
		||||
#define _IPWIRELESS_CS_H_
 | 
			
		||||
 | 
			
		||||
#include <linux/sched.h>
 | 
			
		||||
#include <linux/types.h>
 | 
			
		||||
 | 
			
		||||
#include <pcmcia/cs_types.h>
 | 
			
		||||
#include <pcmcia/cs.h>
 | 
			
		||||
#include <pcmcia/cistpl.h>
 | 
			
		||||
#include <pcmcia/ds.h>
 | 
			
		||||
 | 
			
		||||
#include "hardware.h"
 | 
			
		||||
 | 
			
		||||
#define IPWIRELESS_PCCARD_NAME		"ipwireless"
 | 
			
		||||
#define IPWIRELESS_PCMCIA_VERSION	"1.1"
 | 
			
		||||
#define IPWIRELESS_PCMCIA_AUTHOR        \
 | 
			
		||||
	"Stephen Blackheath, Ben Martel, Jiri Kosina and David Sterba"
 | 
			
		||||
 | 
			
		||||
#define IPWIRELESS_TX_QUEUE_SIZE  262144
 | 
			
		||||
#define IPWIRELESS_RX_QUEUE_SIZE  262144
 | 
			
		||||
 | 
			
		||||
#define IPWIRELESS_STATE_DEBUG
 | 
			
		||||
 | 
			
		||||
struct ipw_hardware;
 | 
			
		||||
struct ipw_network;
 | 
			
		||||
struct ipw_tty;
 | 
			
		||||
 | 
			
		||||
struct ipw_dev {
 | 
			
		||||
	struct pcmcia_device *link;
 | 
			
		||||
	int is_v2_card;
 | 
			
		||||
	window_handle_t handle_attr_memory;
 | 
			
		||||
	void __iomem *attr_memory;
 | 
			
		||||
	window_handle_t handle_common_memory;
 | 
			
		||||
	void __iomem *common_memory;
 | 
			
		||||
	dev_node_t nodes[2];
 | 
			
		||||
	/* Reference to attribute memory, containing CIS data */
 | 
			
		||||
	void *attribute_memory;
 | 
			
		||||
 | 
			
		||||
	/* Hardware context */
 | 
			
		||||
	struct ipw_hardware *hardware;
 | 
			
		||||
	/* Network layer context */
 | 
			
		||||
	struct ipw_network *network;
 | 
			
		||||
	/* TTY device context */
 | 
			
		||||
	struct ipw_tty *tty;
 | 
			
		||||
	struct work_struct work_reboot;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/* Module parametres */
 | 
			
		||||
extern int ipwireless_debug;
 | 
			
		||||
extern int ipwireless_loopback;
 | 
			
		||||
extern int ipwireless_out_queue;
 | 
			
		||||
 | 
			
		||||
#endif
 | 
			
		||||
							
								
								
									
										512
									
								
								drivers/char/pcmcia/ipwireless/network.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										512
									
								
								drivers/char/pcmcia/ipwireless/network.c
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,512 @@
 | 
			
		|||
/*
 | 
			
		||||
 * IPWireless 3G PCMCIA Network Driver
 | 
			
		||||
 *
 | 
			
		||||
 * Original code
 | 
			
		||||
 *   by Stephen Blackheath <stephen@blacksapphire.com>,
 | 
			
		||||
 *      Ben Martel <benm@symmetric.co.nz>
 | 
			
		||||
 *
 | 
			
		||||
 * Copyrighted as follows:
 | 
			
		||||
 *   Copyright (C) 2004 by Symmetric Systems Ltd (NZ)
 | 
			
		||||
 *
 | 
			
		||||
 * Various driver changes and rewrites, port to new kernels
 | 
			
		||||
 *   Copyright (C) 2006-2007 Jiri Kosina
 | 
			
		||||
 *
 | 
			
		||||
 * Misc code cleanups and updates
 | 
			
		||||
 *   Copyright (C) 2007 David Sterba
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
#include <linux/interrupt.h>
 | 
			
		||||
#include <linux/kernel.h>
 | 
			
		||||
#include <linux/mutex.h>
 | 
			
		||||
#include <linux/netdevice.h>
 | 
			
		||||
#include <linux/ppp_channel.h>
 | 
			
		||||
#include <linux/ppp_defs.h>
 | 
			
		||||
#include <linux/if_ppp.h>
 | 
			
		||||
#include <linux/skbuff.h>
 | 
			
		||||
 | 
			
		||||
#include "network.h"
 | 
			
		||||
#include "hardware.h"
 | 
			
		||||
#include "main.h"
 | 
			
		||||
#include "tty.h"
 | 
			
		||||
 | 
			
		||||
#define MAX_OUTGOING_PACKETS_QUEUED   ipwireless_out_queue
 | 
			
		||||
#define MAX_ASSOCIATED_TTYS 2
 | 
			
		||||
 | 
			
		||||
#define SC_RCV_BITS     (SC_RCV_B7_1|SC_RCV_B7_0|SC_RCV_ODDP|SC_RCV_EVNP)
 | 
			
		||||
 | 
			
		||||
struct ipw_network {
 | 
			
		||||
	/* Hardware context, used for calls to hardware layer. */
 | 
			
		||||
	struct ipw_hardware *hardware;
 | 
			
		||||
	/* Context for kernel 'generic_ppp' functionality */
 | 
			
		||||
	struct ppp_channel *ppp_channel;
 | 
			
		||||
	/* tty context connected with IPW console */
 | 
			
		||||
	struct ipw_tty *associated_ttys[NO_OF_IPW_CHANNELS][MAX_ASSOCIATED_TTYS];
 | 
			
		||||
	/* True if ppp needs waking up once we're ready to xmit */
 | 
			
		||||
	int ppp_blocked;
 | 
			
		||||
	/* Number of packets queued up in hardware module. */
 | 
			
		||||
	int outgoing_packets_queued;
 | 
			
		||||
	/* Spinlock to avoid interrupts during shutdown */
 | 
			
		||||
	spinlock_t spinlock;
 | 
			
		||||
	struct mutex close_lock;
 | 
			
		||||
 | 
			
		||||
	/* PPP ioctl data, not actually used anywere */
 | 
			
		||||
	unsigned int flags;
 | 
			
		||||
	unsigned int rbits;
 | 
			
		||||
	u32 xaccm[8];
 | 
			
		||||
	u32 raccm;
 | 
			
		||||
	int mru;
 | 
			
		||||
 | 
			
		||||
	int shutting_down;
 | 
			
		||||
	unsigned int ras_control_lines;
 | 
			
		||||
 | 
			
		||||
	struct work_struct work_go_online;
 | 
			
		||||
	struct work_struct work_go_offline;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#ifdef IPWIRELESS_STATE_DEBUG
 | 
			
		||||
int ipwireless_dump_network_state(char *p, size_t limit,
 | 
			
		||||
				  struct ipw_network *network)
 | 
			
		||||
{
 | 
			
		||||
	return snprintf(p, limit,
 | 
			
		||||
			"debug: ppp_blocked=%d\n"
 | 
			
		||||
			"debug: outgoing_packets_queued=%d\n"
 | 
			
		||||
			"debug: network.shutting_down=%d\n",
 | 
			
		||||
			network->ppp_blocked,
 | 
			
		||||
			network->outgoing_packets_queued,
 | 
			
		||||
			network->shutting_down);
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
static void notify_packet_sent(void *callback_data, unsigned int packet_length)
 | 
			
		||||
{
 | 
			
		||||
	struct ipw_network *network = callback_data;
 | 
			
		||||
	unsigned long flags;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irqsave(&network->spinlock, flags);
 | 
			
		||||
	network->outgoing_packets_queued--;
 | 
			
		||||
	if (network->ppp_channel != NULL) {
 | 
			
		||||
		if (network->ppp_blocked) {
 | 
			
		||||
			network->ppp_blocked = 0;
 | 
			
		||||
			spin_unlock_irqrestore(&network->spinlock, flags);
 | 
			
		||||
			ppp_output_wakeup(network->ppp_channel);
 | 
			
		||||
			if (ipwireless_debug)
 | 
			
		||||
				printk(KERN_INFO IPWIRELESS_PCCARD_NAME
 | 
			
		||||
				       ": ppp unblocked\n");
 | 
			
		||||
		} else
 | 
			
		||||
			spin_unlock_irqrestore(&network->spinlock, flags);
 | 
			
		||||
	} else
 | 
			
		||||
		spin_unlock_irqrestore(&network->spinlock, flags);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * Called by the ppp system when it has a packet to send to the hardware.
 | 
			
		||||
 */
 | 
			
		||||
static int ipwireless_ppp_start_xmit(struct ppp_channel *ppp_channel,
 | 
			
		||||
				     struct sk_buff *skb)
 | 
			
		||||
{
 | 
			
		||||
	struct ipw_network *network = ppp_channel->private;
 | 
			
		||||
	unsigned long flags;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irqsave(&network->spinlock, flags);
 | 
			
		||||
	if (network->outgoing_packets_queued < MAX_OUTGOING_PACKETS_QUEUED) {
 | 
			
		||||
		unsigned char *buf;
 | 
			
		||||
		static unsigned char header[] = {
 | 
			
		||||
			PPP_ALLSTATIONS, /* 0xff */
 | 
			
		||||
			PPP_UI,		 /* 0x03 */
 | 
			
		||||
		};
 | 
			
		||||
		int ret;
 | 
			
		||||
 | 
			
		||||
		network->outgoing_packets_queued++;
 | 
			
		||||
		spin_unlock_irqrestore(&network->spinlock, flags);
 | 
			
		||||
 | 
			
		||||
		/*
 | 
			
		||||
		 * If we have the requested amount of headroom in the skb we
 | 
			
		||||
		 * were handed, then we can add the header efficiently.
 | 
			
		||||
		 */
 | 
			
		||||
		if (skb_headroom(skb) >= 2) {
 | 
			
		||||
			memcpy(skb_push(skb, 2), header, 2);
 | 
			
		||||
			ret = ipwireless_send_packet(network->hardware,
 | 
			
		||||
					       IPW_CHANNEL_RAS, skb->data,
 | 
			
		||||
					       skb->len,
 | 
			
		||||
					       notify_packet_sent,
 | 
			
		||||
					       network);
 | 
			
		||||
			if (ret == -1) {
 | 
			
		||||
				skb_pull(skb, 2);
 | 
			
		||||
				return 0;
 | 
			
		||||
			}
 | 
			
		||||
		} else {
 | 
			
		||||
			/* Otherwise (rarely) we do it inefficiently. */
 | 
			
		||||
			buf = kmalloc(skb->len + 2, GFP_ATOMIC);
 | 
			
		||||
			if (!buf)
 | 
			
		||||
				return 0;
 | 
			
		||||
			memcpy(buf + 2, skb->data, skb->len);
 | 
			
		||||
			memcpy(buf, header, 2);
 | 
			
		||||
			ret = ipwireless_send_packet(network->hardware,
 | 
			
		||||
					       IPW_CHANNEL_RAS, buf,
 | 
			
		||||
					       skb->len + 2,
 | 
			
		||||
					       notify_packet_sent,
 | 
			
		||||
					       network);
 | 
			
		||||
			kfree(buf);
 | 
			
		||||
			if (ret == -1)
 | 
			
		||||
				return 0;
 | 
			
		||||
		}
 | 
			
		||||
		kfree_skb(skb);
 | 
			
		||||
		return 1;
 | 
			
		||||
	} else {
 | 
			
		||||
		/*
 | 
			
		||||
		 * Otherwise reject the packet, and flag that the ppp system
 | 
			
		||||
		 * needs to be unblocked once we are ready to send.
 | 
			
		||||
		 */
 | 
			
		||||
		network->ppp_blocked = 1;
 | 
			
		||||
		spin_unlock_irqrestore(&network->spinlock, flags);
 | 
			
		||||
		return 0;
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* Handle an ioctl call that has come in via ppp. (copy of ppp_async_ioctl() */
 | 
			
		||||
static int ipwireless_ppp_ioctl(struct ppp_channel *ppp_channel,
 | 
			
		||||
				unsigned int cmd, unsigned long arg)
 | 
			
		||||
{
 | 
			
		||||
	struct ipw_network *network = ppp_channel->private;
 | 
			
		||||
	int err, val;
 | 
			
		||||
	u32 accm[8];
 | 
			
		||||
	int __user *user_arg = (int __user *) arg;
 | 
			
		||||
 | 
			
		||||
	err = -EFAULT;
 | 
			
		||||
	switch (cmd) {
 | 
			
		||||
	case PPPIOCGFLAGS:
 | 
			
		||||
		val = network->flags | network->rbits;
 | 
			
		||||
		if (put_user(val, user_arg))
 | 
			
		||||
			break;
 | 
			
		||||
		err = 0;
 | 
			
		||||
		break;
 | 
			
		||||
 | 
			
		||||
	case PPPIOCSFLAGS:
 | 
			
		||||
		if (get_user(val, user_arg))
 | 
			
		||||
			break;
 | 
			
		||||
		network->flags = val & ~SC_RCV_BITS;
 | 
			
		||||
		network->rbits = val & SC_RCV_BITS;
 | 
			
		||||
		err = 0;
 | 
			
		||||
		break;
 | 
			
		||||
 | 
			
		||||
	case PPPIOCGASYNCMAP:
 | 
			
		||||
		if (put_user(network->xaccm[0], user_arg))
 | 
			
		||||
			break;
 | 
			
		||||
		err = 0;
 | 
			
		||||
		break;
 | 
			
		||||
 | 
			
		||||
	case PPPIOCSASYNCMAP:
 | 
			
		||||
		if (get_user(network->xaccm[0], user_arg))
 | 
			
		||||
			break;
 | 
			
		||||
		err = 0;
 | 
			
		||||
		break;
 | 
			
		||||
 | 
			
		||||
	case PPPIOCGRASYNCMAP:
 | 
			
		||||
		if (put_user(network->raccm, user_arg))
 | 
			
		||||
			break;
 | 
			
		||||
		err = 0;
 | 
			
		||||
		break;
 | 
			
		||||
 | 
			
		||||
	case PPPIOCSRASYNCMAP:
 | 
			
		||||
		if (get_user(network->raccm, user_arg))
 | 
			
		||||
			break;
 | 
			
		||||
		err = 0;
 | 
			
		||||
		break;
 | 
			
		||||
 | 
			
		||||
	case PPPIOCGXASYNCMAP:
 | 
			
		||||
		if (copy_to_user((void __user *) arg, network->xaccm,
 | 
			
		||||
					sizeof(network->xaccm)))
 | 
			
		||||
			break;
 | 
			
		||||
		err = 0;
 | 
			
		||||
		break;
 | 
			
		||||
 | 
			
		||||
	case PPPIOCSXASYNCMAP:
 | 
			
		||||
		if (copy_from_user(accm, (void __user *) arg, sizeof(accm)))
 | 
			
		||||
			break;
 | 
			
		||||
		accm[2] &= ~0x40000000U;	/* can't escape 0x5e */
 | 
			
		||||
		accm[3] |= 0x60000000U;	/* must escape 0x7d, 0x7e */
 | 
			
		||||
		memcpy(network->xaccm, accm, sizeof(network->xaccm));
 | 
			
		||||
		err = 0;
 | 
			
		||||
		break;
 | 
			
		||||
 | 
			
		||||
	case PPPIOCGMRU:
 | 
			
		||||
		if (put_user(network->mru, user_arg))
 | 
			
		||||
			break;
 | 
			
		||||
		err = 0;
 | 
			
		||||
		break;
 | 
			
		||||
 | 
			
		||||
	case PPPIOCSMRU:
 | 
			
		||||
		if (get_user(val, user_arg))
 | 
			
		||||
			break;
 | 
			
		||||
		if (val < PPP_MRU)
 | 
			
		||||
			val = PPP_MRU;
 | 
			
		||||
		network->mru = val;
 | 
			
		||||
		err = 0;
 | 
			
		||||
		break;
 | 
			
		||||
 | 
			
		||||
	default:
 | 
			
		||||
		err = -ENOTTY;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return err;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static struct ppp_channel_ops ipwireless_ppp_channel_ops = {
 | 
			
		||||
	.start_xmit = ipwireless_ppp_start_xmit,
 | 
			
		||||
	.ioctl      = ipwireless_ppp_ioctl
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static void do_go_online(struct work_struct *work_go_online)
 | 
			
		||||
{
 | 
			
		||||
	struct ipw_network *network =
 | 
			
		||||
		container_of(work_go_online, struct ipw_network,
 | 
			
		||||
				work_go_online);
 | 
			
		||||
	unsigned long flags;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irqsave(&network->spinlock, flags);
 | 
			
		||||
	if (!network->ppp_channel) {
 | 
			
		||||
		struct ppp_channel *channel;
 | 
			
		||||
 | 
			
		||||
		spin_unlock_irqrestore(&network->spinlock, flags);
 | 
			
		||||
		channel = kzalloc(sizeof(struct ppp_channel), GFP_KERNEL);
 | 
			
		||||
		if (!channel) {
 | 
			
		||||
			printk(KERN_ERR IPWIRELESS_PCCARD_NAME
 | 
			
		||||
					": unable to allocate PPP channel\n");
 | 
			
		||||
			return;
 | 
			
		||||
		}
 | 
			
		||||
		channel->private = network;
 | 
			
		||||
		channel->mtu = 16384;	/* Wild guess */
 | 
			
		||||
		channel->hdrlen = 2;
 | 
			
		||||
		channel->ops = &ipwireless_ppp_channel_ops;
 | 
			
		||||
 | 
			
		||||
		network->flags = 0;
 | 
			
		||||
		network->rbits = 0;
 | 
			
		||||
		network->mru = PPP_MRU;
 | 
			
		||||
		memset(network->xaccm, 0, sizeof(network->xaccm));
 | 
			
		||||
		network->xaccm[0] = ~0U;
 | 
			
		||||
		network->xaccm[3] = 0x60000000U;
 | 
			
		||||
		network->raccm = ~0U;
 | 
			
		||||
		ppp_register_channel(channel);
 | 
			
		||||
		spin_lock_irqsave(&network->spinlock, flags);
 | 
			
		||||
		network->ppp_channel = channel;
 | 
			
		||||
	}
 | 
			
		||||
	spin_unlock_irqrestore(&network->spinlock, flags);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void do_go_offline(struct work_struct *work_go_offline)
 | 
			
		||||
{
 | 
			
		||||
	struct ipw_network *network =
 | 
			
		||||
		container_of(work_go_offline, struct ipw_network,
 | 
			
		||||
				work_go_offline);
 | 
			
		||||
	unsigned long flags;
 | 
			
		||||
 | 
			
		||||
	mutex_lock(&network->close_lock);
 | 
			
		||||
	spin_lock_irqsave(&network->spinlock, flags);
 | 
			
		||||
	if (network->ppp_channel != NULL) {
 | 
			
		||||
		struct ppp_channel *channel = network->ppp_channel;
 | 
			
		||||
 | 
			
		||||
		network->ppp_channel = NULL;
 | 
			
		||||
		spin_unlock_irqrestore(&network->spinlock, flags);
 | 
			
		||||
		mutex_unlock(&network->close_lock);
 | 
			
		||||
		ppp_unregister_channel(channel);
 | 
			
		||||
	} else {
 | 
			
		||||
		spin_unlock_irqrestore(&network->spinlock, flags);
 | 
			
		||||
		mutex_unlock(&network->close_lock);
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ipwireless_network_notify_control_line_change(struct ipw_network *network,
 | 
			
		||||
						   unsigned int channel_idx,
 | 
			
		||||
						   unsigned int control_lines,
 | 
			
		||||
						   unsigned int changed_mask)
 | 
			
		||||
{
 | 
			
		||||
	int i;
 | 
			
		||||
 | 
			
		||||
	if (channel_idx == IPW_CHANNEL_RAS)
 | 
			
		||||
		network->ras_control_lines = control_lines;
 | 
			
		||||
 | 
			
		||||
	for (i = 0; i < MAX_ASSOCIATED_TTYS; i++) {
 | 
			
		||||
		struct ipw_tty *tty =
 | 
			
		||||
			network->associated_ttys[channel_idx][i];
 | 
			
		||||
 | 
			
		||||
		/*
 | 
			
		||||
		 * If it's associated with a tty (other than the RAS channel
 | 
			
		||||
		 * when we're online), then send the data to that tty.  The RAS
 | 
			
		||||
		 * channel's data is handled above - it always goes through
 | 
			
		||||
		 * ppp_generic.
 | 
			
		||||
		 */
 | 
			
		||||
		if (tty)
 | 
			
		||||
			ipwireless_tty_notify_control_line_change(tty,
 | 
			
		||||
								  channel_idx,
 | 
			
		||||
								  control_lines,
 | 
			
		||||
								  changed_mask);
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * Some versions of firmware stuff packets with 0xff 0x03 (PPP: ALLSTATIONS, UI)
 | 
			
		||||
 * bytes, which are required on sent packet, but not always present on received
 | 
			
		||||
 * packets
 | 
			
		||||
 */
 | 
			
		||||
static struct sk_buff *ipw_packet_received_skb(unsigned char *data,
 | 
			
		||||
					       unsigned int length)
 | 
			
		||||
{
 | 
			
		||||
	struct sk_buff *skb;
 | 
			
		||||
 | 
			
		||||
	if (length > 2 && data[0] == PPP_ALLSTATIONS && data[1] == PPP_UI) {
 | 
			
		||||
		length -= 2;
 | 
			
		||||
		data += 2;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	skb = dev_alloc_skb(length + 4);
 | 
			
		||||
	skb_reserve(skb, 2);
 | 
			
		||||
	memcpy(skb_put(skb, length), data, length);
 | 
			
		||||
 | 
			
		||||
	return skb;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ipwireless_network_packet_received(struct ipw_network *network,
 | 
			
		||||
					unsigned int channel_idx,
 | 
			
		||||
					unsigned char *data,
 | 
			
		||||
					unsigned int length)
 | 
			
		||||
{
 | 
			
		||||
	int i;
 | 
			
		||||
	unsigned long flags;
 | 
			
		||||
 | 
			
		||||
	for (i = 0; i < MAX_ASSOCIATED_TTYS; i++) {
 | 
			
		||||
		struct ipw_tty *tty = network->associated_ttys[channel_idx][i];
 | 
			
		||||
 | 
			
		||||
		/*
 | 
			
		||||
		 * If it's associated with a tty (other than the RAS channel
 | 
			
		||||
		 * when we're online), then send the data to that tty.  The RAS
 | 
			
		||||
		 * channel's data is handled above - it always goes through
 | 
			
		||||
		 * ppp_generic.
 | 
			
		||||
		 */
 | 
			
		||||
		if (tty && channel_idx == IPW_CHANNEL_RAS
 | 
			
		||||
				&& (network->ras_control_lines &
 | 
			
		||||
					IPW_CONTROL_LINE_DCD) != 0
 | 
			
		||||
				&& ipwireless_tty_is_modem(tty)) {
 | 
			
		||||
			/*
 | 
			
		||||
			 * If data came in on the RAS channel and this tty is
 | 
			
		||||
			 * the modem tty, and we are online, then we send it to
 | 
			
		||||
			 * the PPP layer.
 | 
			
		||||
			 */
 | 
			
		||||
			mutex_lock(&network->close_lock);
 | 
			
		||||
			spin_lock_irqsave(&network->spinlock, flags);
 | 
			
		||||
			if (network->ppp_channel != NULL) {
 | 
			
		||||
				struct sk_buff *skb;
 | 
			
		||||
 | 
			
		||||
				spin_unlock_irqrestore(&network->spinlock,
 | 
			
		||||
						flags);
 | 
			
		||||
 | 
			
		||||
				/* Send the data to the ppp_generic module. */
 | 
			
		||||
				skb = ipw_packet_received_skb(data, length);
 | 
			
		||||
				ppp_input(network->ppp_channel, skb);
 | 
			
		||||
			} else
 | 
			
		||||
				spin_unlock_irqrestore(&network->spinlock,
 | 
			
		||||
						flags);
 | 
			
		||||
			mutex_unlock(&network->close_lock);
 | 
			
		||||
		}
 | 
			
		||||
		/* Otherwise we send it out the tty. */
 | 
			
		||||
		else
 | 
			
		||||
			ipwireless_tty_received(tty, data, length);
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
struct ipw_network *ipwireless_network_create(struct ipw_hardware *hw)
 | 
			
		||||
{
 | 
			
		||||
	struct ipw_network *network =
 | 
			
		||||
		kzalloc(sizeof(struct ipw_network), GFP_ATOMIC);
 | 
			
		||||
 | 
			
		||||
	if (!network)
 | 
			
		||||
		return NULL;
 | 
			
		||||
 | 
			
		||||
	spin_lock_init(&network->spinlock);
 | 
			
		||||
	mutex_init(&network->close_lock);
 | 
			
		||||
 | 
			
		||||
	network->hardware = hw;
 | 
			
		||||
 | 
			
		||||
	INIT_WORK(&network->work_go_online, do_go_online);
 | 
			
		||||
	INIT_WORK(&network->work_go_offline, do_go_offline);
 | 
			
		||||
 | 
			
		||||
	ipwireless_associate_network(hw, network);
 | 
			
		||||
 | 
			
		||||
	return network;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ipwireless_network_free(struct ipw_network *network)
 | 
			
		||||
{
 | 
			
		||||
	network->shutting_down = 1;
 | 
			
		||||
 | 
			
		||||
	ipwireless_ppp_close(network);
 | 
			
		||||
	flush_scheduled_work();
 | 
			
		||||
 | 
			
		||||
	ipwireless_stop_interrupts(network->hardware);
 | 
			
		||||
	ipwireless_associate_network(network->hardware, NULL);
 | 
			
		||||
 | 
			
		||||
	kfree(network);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ipwireless_associate_network_tty(struct ipw_network *network,
 | 
			
		||||
				      unsigned int channel_idx,
 | 
			
		||||
				      struct ipw_tty *tty)
 | 
			
		||||
{
 | 
			
		||||
	int i;
 | 
			
		||||
 | 
			
		||||
	for (i = 0; i < MAX_ASSOCIATED_TTYS; i++)
 | 
			
		||||
		if (network->associated_ttys[channel_idx][i] == NULL) {
 | 
			
		||||
			network->associated_ttys[channel_idx][i] = tty;
 | 
			
		||||
			break;
 | 
			
		||||
		}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ipwireless_disassociate_network_ttys(struct ipw_network *network,
 | 
			
		||||
					  unsigned int channel_idx)
 | 
			
		||||
{
 | 
			
		||||
	int i;
 | 
			
		||||
 | 
			
		||||
	for (i = 0; i < MAX_ASSOCIATED_TTYS; i++)
 | 
			
		||||
		network->associated_ttys[channel_idx][i] = NULL;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ipwireless_ppp_open(struct ipw_network *network)
 | 
			
		||||
{
 | 
			
		||||
	if (ipwireless_debug)
 | 
			
		||||
		printk(KERN_DEBUG IPWIRELESS_PCCARD_NAME ": online\n");
 | 
			
		||||
	schedule_work(&network->work_go_online);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ipwireless_ppp_close(struct ipw_network *network)
 | 
			
		||||
{
 | 
			
		||||
	/* Disconnect from the wireless network. */
 | 
			
		||||
	if (ipwireless_debug)
 | 
			
		||||
		printk(KERN_DEBUG IPWIRELESS_PCCARD_NAME ": offline\n");
 | 
			
		||||
	schedule_work(&network->work_go_offline);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int ipwireless_ppp_channel_index(struct ipw_network *network)
 | 
			
		||||
{
 | 
			
		||||
	int ret = -1;
 | 
			
		||||
	unsigned long flags;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irqsave(&network->spinlock, flags);
 | 
			
		||||
	if (network->ppp_channel != NULL)
 | 
			
		||||
		ret = ppp_channel_index(network->ppp_channel);
 | 
			
		||||
	spin_unlock_irqrestore(&network->spinlock, flags);
 | 
			
		||||
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int ipwireless_ppp_unit_number(struct ipw_network *network)
 | 
			
		||||
{
 | 
			
		||||
	int ret = -1;
 | 
			
		||||
	unsigned long flags;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irqsave(&network->spinlock, flags);
 | 
			
		||||
	if (network->ppp_channel != NULL)
 | 
			
		||||
		ret = ppp_unit_number(network->ppp_channel);
 | 
			
		||||
	spin_unlock_irqrestore(&network->spinlock, flags);
 | 
			
		||||
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										55
									
								
								drivers/char/pcmcia/ipwireless/network.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								drivers/char/pcmcia/ipwireless/network.h
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,55 @@
 | 
			
		|||
/*
 | 
			
		||||
 * IPWireless 3G PCMCIA Network Driver
 | 
			
		||||
 *
 | 
			
		||||
 * Original code
 | 
			
		||||
 *   by Stephen Blackheath <stephen@blacksapphire.com>,
 | 
			
		||||
 *      Ben Martel <benm@symmetric.co.nz>
 | 
			
		||||
 *
 | 
			
		||||
 * Copyrighted as follows:
 | 
			
		||||
 *   Copyright (C) 2004 by Symmetric Systems Ltd (NZ)
 | 
			
		||||
 *
 | 
			
		||||
 * Various driver changes and rewrites, port to new kernels
 | 
			
		||||
 *   Copyright (C) 2006-2007 Jiri Kosina
 | 
			
		||||
 *
 | 
			
		||||
 * Misc code cleanups and updates
 | 
			
		||||
 *   Copyright (C) 2007 David Sterba
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
#ifndef _IPWIRELESS_CS_NETWORK_H_
 | 
			
		||||
#define _IPWIRELESS_CS_NETWORK_H_
 | 
			
		||||
 | 
			
		||||
#include <linux/types.h>
 | 
			
		||||
 | 
			
		||||
struct ipw_network;
 | 
			
		||||
struct ipw_tty;
 | 
			
		||||
struct ipw_hardware;
 | 
			
		||||
 | 
			
		||||
/* Definitions of the different channels on the PCMCIA UE */
 | 
			
		||||
#define IPW_CHANNEL_RAS      0
 | 
			
		||||
#define IPW_CHANNEL_DIALLER  1
 | 
			
		||||
#define IPW_CHANNEL_CONSOLE  2
 | 
			
		||||
#define NO_OF_IPW_CHANNELS   5
 | 
			
		||||
 | 
			
		||||
void ipwireless_network_notify_control_line_change(struct ipw_network *net,
 | 
			
		||||
		unsigned int channel_idx, unsigned int control_lines,
 | 
			
		||||
		unsigned int control_mask);
 | 
			
		||||
void ipwireless_network_packet_received(struct ipw_network *net,
 | 
			
		||||
		unsigned int channel_idx, unsigned char *data,
 | 
			
		||||
		unsigned int length);
 | 
			
		||||
struct ipw_network *ipwireless_network_create(struct ipw_hardware *hw);
 | 
			
		||||
void ipwireless_network_free(struct ipw_network *net);
 | 
			
		||||
void ipwireless_associate_network_tty(struct ipw_network *net,
 | 
			
		||||
		unsigned int channel_idx, struct ipw_tty *tty);
 | 
			
		||||
void ipwireless_disassociate_network_ttys(struct ipw_network *net,
 | 
			
		||||
		unsigned int channel_idx);
 | 
			
		||||
 | 
			
		||||
void ipwireless_ppp_open(struct ipw_network *net);
 | 
			
		||||
 | 
			
		||||
void ipwireless_ppp_close(struct ipw_network *net);
 | 
			
		||||
int ipwireless_ppp_channel_index(struct ipw_network *net);
 | 
			
		||||
int ipwireless_ppp_unit_number(struct ipw_network *net);
 | 
			
		||||
 | 
			
		||||
int ipwireless_dump_network_state(char *p, size_t limit,
 | 
			
		||||
				  struct ipw_network *net);
 | 
			
		||||
 | 
			
		||||
#endif
 | 
			
		||||
							
								
								
									
										108
									
								
								drivers/char/pcmcia/ipwireless/setup_protocol.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										108
									
								
								drivers/char/pcmcia/ipwireless/setup_protocol.h
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,108 @@
 | 
			
		|||
/*
 | 
			
		||||
 * IPWireless 3G PCMCIA Network Driver
 | 
			
		||||
 *
 | 
			
		||||
 * Original code
 | 
			
		||||
 *   by Stephen Blackheath <stephen@blacksapphire.com>,
 | 
			
		||||
 *      Ben Martel <benm@symmetric.co.nz>
 | 
			
		||||
 *
 | 
			
		||||
 * Copyrighted as follows:
 | 
			
		||||
 *   Copyright (C) 2004 by Symmetric Systems Ltd (NZ)
 | 
			
		||||
 *
 | 
			
		||||
 * Various driver changes and rewrites, port to new kernels
 | 
			
		||||
 *   Copyright (C) 2006-2007 Jiri Kosina
 | 
			
		||||
 *
 | 
			
		||||
 * Misc code cleanups and updates
 | 
			
		||||
 *   Copyright (C) 2007 David Sterba
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
#ifndef _IPWIRELESS_CS_SETUP_PROTOCOL_H_
 | 
			
		||||
#define _IPWIRELESS_CS_SETUP_PROTOCOL_H_
 | 
			
		||||
 | 
			
		||||
/* Version of the setup protocol and transport protocols */
 | 
			
		||||
#define TL_SETUP_VERSION		1
 | 
			
		||||
 | 
			
		||||
#define TL_SETUP_VERSION_QRY_TMO	1000
 | 
			
		||||
#define TL_SETUP_MAX_VERSION_QRY	30
 | 
			
		||||
 | 
			
		||||
/* Message numbers 0-9 are obsoleted and must not be reused! */
 | 
			
		||||
#define TL_SETUP_SIGNO_GET_VERSION_QRY	10
 | 
			
		||||
#define TL_SETUP_SIGNO_GET_VERSION_RSP	11
 | 
			
		||||
#define TL_SETUP_SIGNO_CONFIG_MSG	12
 | 
			
		||||
#define TL_SETUP_SIGNO_CONFIG_DONE_MSG	13
 | 
			
		||||
#define TL_SETUP_SIGNO_OPEN_MSG		14
 | 
			
		||||
#define TL_SETUP_SIGNO_CLOSE_MSG	15
 | 
			
		||||
 | 
			
		||||
#define TL_SETUP_SIGNO_INFO_MSG     20
 | 
			
		||||
#define TL_SETUP_SIGNO_INFO_MSG_ACK 21
 | 
			
		||||
 | 
			
		||||
#define TL_SETUP_SIGNO_REBOOT_MSG      22
 | 
			
		||||
#define TL_SETUP_SIGNO_REBOOT_MSG_ACK  23
 | 
			
		||||
 | 
			
		||||
/* Synchronous start-messages */
 | 
			
		||||
struct tl_setup_get_version_qry {
 | 
			
		||||
	unsigned char sig_no;		/* TL_SETUP_SIGNO_GET_VERSION_QRY */
 | 
			
		||||
} __attribute__ ((__packed__));
 | 
			
		||||
 | 
			
		||||
struct tl_setup_get_version_rsp {
 | 
			
		||||
	unsigned char sig_no;		/* TL_SETUP_SIGNO_GET_VERSION_RSP */
 | 
			
		||||
	unsigned char version;		/* TL_SETUP_VERSION */
 | 
			
		||||
} __attribute__ ((__packed__));
 | 
			
		||||
 | 
			
		||||
struct tl_setup_config_msg {
 | 
			
		||||
	unsigned char sig_no;		/* TL_SETUP_SIGNO_CONFIG_MSG */
 | 
			
		||||
	unsigned char port_no;
 | 
			
		||||
	unsigned char prio_data;
 | 
			
		||||
	unsigned char prio_ctrl;
 | 
			
		||||
} __attribute__ ((__packed__));
 | 
			
		||||
 | 
			
		||||
struct tl_setup_config_done_msg {
 | 
			
		||||
	unsigned char sig_no;		/* TL_SETUP_SIGNO_CONFIG_DONE_MSG */
 | 
			
		||||
} __attribute__ ((__packed__));
 | 
			
		||||
 | 
			
		||||
/* Asyncronous messages */
 | 
			
		||||
struct tl_setup_open_msg {
 | 
			
		||||
	unsigned char sig_no;		/* TL_SETUP_SIGNO_OPEN_MSG */
 | 
			
		||||
	unsigned char port_no;
 | 
			
		||||
} __attribute__ ((__packed__));
 | 
			
		||||
 | 
			
		||||
struct tl_setup_close_msg {
 | 
			
		||||
	unsigned char sig_no;		/* TL_SETUP_SIGNO_CLOSE_MSG */
 | 
			
		||||
	unsigned char port_no;
 | 
			
		||||
} __attribute__ ((__packed__));
 | 
			
		||||
 | 
			
		||||
/* Driver type  - for use in tl_setup_info_msg.driver_type */
 | 
			
		||||
#define COMM_DRIVER     0
 | 
			
		||||
#define NDISWAN_DRIVER  1
 | 
			
		||||
#define NDISWAN_DRIVER_MAJOR_VERSION  2
 | 
			
		||||
#define NDISWAN_DRIVER_MINOR_VERSION  0
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * It should not matter when this message comes over as we just store the
 | 
			
		||||
 * results and send the ACK.
 | 
			
		||||
 */
 | 
			
		||||
struct tl_setup_info_msg {
 | 
			
		||||
	unsigned char sig_no;		/* TL_SETUP_SIGNO_INFO_MSG */
 | 
			
		||||
	unsigned char driver_type;
 | 
			
		||||
	unsigned char major_version;
 | 
			
		||||
	unsigned char minor_version;
 | 
			
		||||
} __attribute__ ((__packed__));
 | 
			
		||||
 | 
			
		||||
struct tl_setup_info_msgAck {
 | 
			
		||||
	unsigned char sig_no;		/* TL_SETUP_SIGNO_INFO_MSG_ACK */
 | 
			
		||||
} __attribute__ ((__packed__));
 | 
			
		||||
 | 
			
		||||
struct TlSetupRebootMsgAck {
 | 
			
		||||
	unsigned char sig_no;		/* TL_SETUP_SIGNO_REBOOT_MSG_ACK */
 | 
			
		||||
} __attribute__ ((__packed__));
 | 
			
		||||
 | 
			
		||||
/* Define a union of all the msgs that the driver can receive from the card.*/
 | 
			
		||||
union ipw_setup_rx_msg {
 | 
			
		||||
	unsigned char sig_no;
 | 
			
		||||
	struct tl_setup_get_version_rsp version_rsp_msg;
 | 
			
		||||
	struct tl_setup_open_msg open_msg;
 | 
			
		||||
	struct tl_setup_close_msg close_msg;
 | 
			
		||||
	struct tl_setup_info_msg InfoMsg;
 | 
			
		||||
	struct tl_setup_info_msgAck info_msg_ack;
 | 
			
		||||
} __attribute__ ((__packed__));
 | 
			
		||||
 | 
			
		||||
#endif				/* _IPWIRELESS_CS_SETUP_PROTOCOL_H_ */
 | 
			
		||||
							
								
								
									
										688
									
								
								drivers/char/pcmcia/ipwireless/tty.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										688
									
								
								drivers/char/pcmcia/ipwireless/tty.c
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,688 @@
 | 
			
		|||
/*
 | 
			
		||||
 * IPWireless 3G PCMCIA Network Driver
 | 
			
		||||
 *
 | 
			
		||||
 * Original code
 | 
			
		||||
 *   by Stephen Blackheath <stephen@blacksapphire.com>,
 | 
			
		||||
 *      Ben Martel <benm@symmetric.co.nz>
 | 
			
		||||
 *
 | 
			
		||||
 * Copyrighted as follows:
 | 
			
		||||
 *   Copyright (C) 2004 by Symmetric Systems Ltd (NZ)
 | 
			
		||||
 *
 | 
			
		||||
 * Various driver changes and rewrites, port to new kernels
 | 
			
		||||
 *   Copyright (C) 2006-2007 Jiri Kosina
 | 
			
		||||
 *
 | 
			
		||||
 * Misc code cleanups and updates
 | 
			
		||||
 *   Copyright (C) 2007 David Sterba
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
#include <linux/init.h>
 | 
			
		||||
#include <linux/kernel.h>
 | 
			
		||||
#include <linux/module.h>
 | 
			
		||||
#include <linux/mutex.h>
 | 
			
		||||
#include <linux/ppp_defs.h>
 | 
			
		||||
#include <linux/if.h>
 | 
			
		||||
#include <linux/if_ppp.h>
 | 
			
		||||
#include <linux/sched.h>
 | 
			
		||||
#include <linux/serial.h>
 | 
			
		||||
#include <linux/slab.h>
 | 
			
		||||
#include <linux/tty.h>
 | 
			
		||||
#include <linux/tty_driver.h>
 | 
			
		||||
#include <linux/tty_flip.h>
 | 
			
		||||
#include <linux/uaccess.h>
 | 
			
		||||
#include <linux/version.h>
 | 
			
		||||
 | 
			
		||||
#include "tty.h"
 | 
			
		||||
#include "network.h"
 | 
			
		||||
#include "hardware.h"
 | 
			
		||||
#include "main.h"
 | 
			
		||||
 | 
			
		||||
#define IPWIRELESS_PCMCIA_START 	(0)
 | 
			
		||||
#define IPWIRELESS_PCMCIA_MINORS	(24)
 | 
			
		||||
#define IPWIRELESS_PCMCIA_MINOR_RANGE	(8)
 | 
			
		||||
 | 
			
		||||
#define TTYTYPE_MODEM    (0)
 | 
			
		||||
#define TTYTYPE_MONITOR  (1)
 | 
			
		||||
#define TTYTYPE_RAS_RAW  (2)
 | 
			
		||||
 | 
			
		||||
struct ipw_tty {
 | 
			
		||||
	int index;
 | 
			
		||||
	struct ipw_hardware *hardware;
 | 
			
		||||
	unsigned int channel_idx;
 | 
			
		||||
	unsigned int secondary_channel_idx;
 | 
			
		||||
	int tty_type;
 | 
			
		||||
	struct ipw_network *network;
 | 
			
		||||
	struct tty_struct *linux_tty;
 | 
			
		||||
	int open_count;
 | 
			
		||||
	unsigned int control_lines;
 | 
			
		||||
	struct mutex ipw_tty_mutex;
 | 
			
		||||
	int tx_bytes_queued;
 | 
			
		||||
	int closing;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static struct ipw_tty *ttys[IPWIRELESS_PCMCIA_MINORS];
 | 
			
		||||
 | 
			
		||||
static struct tty_driver *ipw_tty_driver;
 | 
			
		||||
 | 
			
		||||
static char *tty_type_name(int tty_type)
 | 
			
		||||
{
 | 
			
		||||
	static char *channel_names[] = {
 | 
			
		||||
		"modem",
 | 
			
		||||
		"monitor",
 | 
			
		||||
		"RAS-raw"
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	return channel_names[tty_type];
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void report_registering(struct ipw_tty *tty)
 | 
			
		||||
{
 | 
			
		||||
	char *iftype = tty_type_name(tty->tty_type);
 | 
			
		||||
 | 
			
		||||
	printk(KERN_INFO IPWIRELESS_PCCARD_NAME
 | 
			
		||||
	       ": registering %s device ttyIPWp%d\n", iftype, tty->index);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void report_deregistering(struct ipw_tty *tty)
 | 
			
		||||
{
 | 
			
		||||
	char *iftype = tty_type_name(tty->tty_type);
 | 
			
		||||
 | 
			
		||||
	printk(KERN_INFO IPWIRELESS_PCCARD_NAME
 | 
			
		||||
	       ": deregistering %s device ttyIPWp%d\n", iftype,
 | 
			
		||||
	       tty->index);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static struct ipw_tty *get_tty(int minor)
 | 
			
		||||
{
 | 
			
		||||
	if (minor < ipw_tty_driver->minor_start
 | 
			
		||||
			|| minor >= ipw_tty_driver->minor_start +
 | 
			
		||||
			IPWIRELESS_PCMCIA_MINORS)
 | 
			
		||||
		return NULL;
 | 
			
		||||
	else {
 | 
			
		||||
		int minor_offset = minor - ipw_tty_driver->minor_start;
 | 
			
		||||
 | 
			
		||||
		/*
 | 
			
		||||
		 * The 'ras_raw' channel is only available when 'loopback' mode
 | 
			
		||||
		 * is enabled.
 | 
			
		||||
		 * Number of minor starts with 16 (_RANGE * _RAS_RAW).
 | 
			
		||||
		 */
 | 
			
		||||
		if (!ipwireless_loopback &&
 | 
			
		||||
				minor_offset >=
 | 
			
		||||
				 IPWIRELESS_PCMCIA_MINOR_RANGE * TTYTYPE_RAS_RAW)
 | 
			
		||||
			return NULL;
 | 
			
		||||
 | 
			
		||||
		return ttys[minor_offset];
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int ipw_open(struct tty_struct *linux_tty, struct file *filp)
 | 
			
		||||
{
 | 
			
		||||
	int minor = linux_tty->index;
 | 
			
		||||
	struct ipw_tty *tty = get_tty(minor);
 | 
			
		||||
 | 
			
		||||
	if (!tty)
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
 | 
			
		||||
	mutex_lock(&tty->ipw_tty_mutex);
 | 
			
		||||
 | 
			
		||||
	if (tty->closing) {
 | 
			
		||||
		mutex_unlock(&tty->ipw_tty_mutex);
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
	if (tty->open_count == 0)
 | 
			
		||||
		tty->tx_bytes_queued = 0;
 | 
			
		||||
 | 
			
		||||
	tty->open_count++;
 | 
			
		||||
 | 
			
		||||
	tty->linux_tty = linux_tty;
 | 
			
		||||
	linux_tty->driver_data = tty;
 | 
			
		||||
	linux_tty->low_latency = 1;
 | 
			
		||||
 | 
			
		||||
	if (tty->tty_type == TTYTYPE_MODEM)
 | 
			
		||||
		ipwireless_ppp_open(tty->network);
 | 
			
		||||
 | 
			
		||||
	mutex_unlock(&tty->ipw_tty_mutex);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void do_ipw_close(struct ipw_tty *tty)
 | 
			
		||||
{
 | 
			
		||||
	tty->open_count--;
 | 
			
		||||
 | 
			
		||||
	if (tty->open_count == 0) {
 | 
			
		||||
		struct tty_struct *linux_tty = tty->linux_tty;
 | 
			
		||||
 | 
			
		||||
		if (linux_tty != NULL) {
 | 
			
		||||
			tty->linux_tty = NULL;
 | 
			
		||||
			linux_tty->driver_data = NULL;
 | 
			
		||||
 | 
			
		||||
			if (tty->tty_type == TTYTYPE_MODEM)
 | 
			
		||||
				ipwireless_ppp_close(tty->network);
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void ipw_hangup(struct tty_struct *linux_tty)
 | 
			
		||||
{
 | 
			
		||||
	struct ipw_tty *tty = linux_tty->driver_data;
 | 
			
		||||
 | 
			
		||||
	if (!tty)
 | 
			
		||||
		return;
 | 
			
		||||
 | 
			
		||||
	mutex_lock(&tty->ipw_tty_mutex);
 | 
			
		||||
	if (tty->open_count == 0) {
 | 
			
		||||
		mutex_unlock(&tty->ipw_tty_mutex);
 | 
			
		||||
		return;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	do_ipw_close(tty);
 | 
			
		||||
 | 
			
		||||
	mutex_unlock(&tty->ipw_tty_mutex);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void ipw_close(struct tty_struct *linux_tty, struct file *filp)
 | 
			
		||||
{
 | 
			
		||||
	ipw_hangup(linux_tty);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* Take data received from hardware, and send it out the tty */
 | 
			
		||||
void ipwireless_tty_received(struct ipw_tty *tty, unsigned char *data,
 | 
			
		||||
			unsigned int length)
 | 
			
		||||
{
 | 
			
		||||
	struct tty_struct *linux_tty;
 | 
			
		||||
	int work = 0;
 | 
			
		||||
 | 
			
		||||
	mutex_lock(&tty->ipw_tty_mutex);
 | 
			
		||||
	linux_tty = tty->linux_tty;
 | 
			
		||||
	if (linux_tty == NULL) {
 | 
			
		||||
		mutex_unlock(&tty->ipw_tty_mutex);
 | 
			
		||||
		return;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (!tty->open_count) {
 | 
			
		||||
		mutex_unlock(&tty->ipw_tty_mutex);
 | 
			
		||||
		return;
 | 
			
		||||
	}
 | 
			
		||||
	mutex_unlock(&tty->ipw_tty_mutex);
 | 
			
		||||
 | 
			
		||||
	work = tty_insert_flip_string(linux_tty, data, length);
 | 
			
		||||
 | 
			
		||||
	if (work != length)
 | 
			
		||||
		printk(KERN_DEBUG IPWIRELESS_PCCARD_NAME
 | 
			
		||||
				": %d chars not inserted to flip buffer!\n",
 | 
			
		||||
				length - work);
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * This may sleep if ->low_latency is set
 | 
			
		||||
	 */
 | 
			
		||||
	if (work)
 | 
			
		||||
		tty_flip_buffer_push(linux_tty);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void ipw_write_packet_sent_callback(void *callback_data,
 | 
			
		||||
					   unsigned int packet_length)
 | 
			
		||||
{
 | 
			
		||||
	struct ipw_tty *tty = callback_data;
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * Packet has been sent, so we subtract the number of bytes from our
 | 
			
		||||
	 * tally of outstanding TX bytes.
 | 
			
		||||
	 */
 | 
			
		||||
	tty->tx_bytes_queued -= packet_length;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int ipw_write(struct tty_struct *linux_tty,
 | 
			
		||||
		     const unsigned char *buf, int count)
 | 
			
		||||
{
 | 
			
		||||
	struct ipw_tty *tty = linux_tty->driver_data;
 | 
			
		||||
	int room, ret;
 | 
			
		||||
 | 
			
		||||
	if (!tty)
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
 | 
			
		||||
	mutex_lock(&tty->ipw_tty_mutex);
 | 
			
		||||
	if (!tty->open_count) {
 | 
			
		||||
		mutex_unlock(&tty->ipw_tty_mutex);
 | 
			
		||||
		return -EINVAL;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	room = IPWIRELESS_TX_QUEUE_SIZE - tty->tx_bytes_queued;
 | 
			
		||||
	if (room < 0)
 | 
			
		||||
		room = 0;
 | 
			
		||||
	/* Don't allow caller to write any more than we have room for */
 | 
			
		||||
	if (count > room)
 | 
			
		||||
		count = room;
 | 
			
		||||
 | 
			
		||||
	if (count == 0) {
 | 
			
		||||
		mutex_unlock(&tty->ipw_tty_mutex);
 | 
			
		||||
		return 0;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	ret = ipwireless_send_packet(tty->hardware, IPW_CHANNEL_RAS,
 | 
			
		||||
			       (unsigned char *) buf, count,
 | 
			
		||||
			       ipw_write_packet_sent_callback, tty);
 | 
			
		||||
	if (ret == -1) {
 | 
			
		||||
		mutex_unlock(&tty->ipw_tty_mutex);
 | 
			
		||||
		return 0;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	tty->tx_bytes_queued += count;
 | 
			
		||||
	mutex_unlock(&tty->ipw_tty_mutex);
 | 
			
		||||
 | 
			
		||||
	return count;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int ipw_write_room(struct tty_struct *linux_tty)
 | 
			
		||||
{
 | 
			
		||||
	struct ipw_tty *tty = linux_tty->driver_data;
 | 
			
		||||
	int room;
 | 
			
		||||
 | 
			
		||||
	if (!tty)
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
 | 
			
		||||
	if (!tty->open_count)
 | 
			
		||||
		return -EINVAL;
 | 
			
		||||
 | 
			
		||||
	room = IPWIRELESS_TX_QUEUE_SIZE - tty->tx_bytes_queued;
 | 
			
		||||
	if (room < 0)
 | 
			
		||||
		room = 0;
 | 
			
		||||
 | 
			
		||||
	return room;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int ipwireless_get_serial_info(struct ipw_tty *tty,
 | 
			
		||||
				      struct serial_struct __user *retinfo)
 | 
			
		||||
{
 | 
			
		||||
	struct serial_struct tmp;
 | 
			
		||||
 | 
			
		||||
	if (!retinfo)
 | 
			
		||||
		return (-EFAULT);
 | 
			
		||||
 | 
			
		||||
	memset(&tmp, 0, sizeof(tmp));
 | 
			
		||||
	tmp.type = PORT_UNKNOWN;
 | 
			
		||||
	tmp.line = tty->index;
 | 
			
		||||
	tmp.port = 0;
 | 
			
		||||
	tmp.irq = 0;
 | 
			
		||||
	tmp.flags = 0;
 | 
			
		||||
	tmp.baud_base = 115200;
 | 
			
		||||
	tmp.close_delay = 0;
 | 
			
		||||
	tmp.closing_wait = 0;
 | 
			
		||||
	tmp.custom_divisor = 0;
 | 
			
		||||
	tmp.hub6 = 0;
 | 
			
		||||
	if (copy_to_user(retinfo, &tmp, sizeof(*retinfo)))
 | 
			
		||||
		return -EFAULT;
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int ipw_chars_in_buffer(struct tty_struct *linux_tty)
 | 
			
		||||
{
 | 
			
		||||
	struct ipw_tty *tty = linux_tty->driver_data;
 | 
			
		||||
 | 
			
		||||
	if (!tty)
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
 | 
			
		||||
	if (!tty->open_count)
 | 
			
		||||
		return -EINVAL;
 | 
			
		||||
 | 
			
		||||
	return tty->tx_bytes_queued;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int get_control_lines(struct ipw_tty *tty)
 | 
			
		||||
{
 | 
			
		||||
	unsigned int my = tty->control_lines;
 | 
			
		||||
	unsigned int out = 0;
 | 
			
		||||
 | 
			
		||||
	if (my & IPW_CONTROL_LINE_RTS)
 | 
			
		||||
		out |= TIOCM_RTS;
 | 
			
		||||
	if (my & IPW_CONTROL_LINE_DTR)
 | 
			
		||||
		out |= TIOCM_DTR;
 | 
			
		||||
	if (my & IPW_CONTROL_LINE_CTS)
 | 
			
		||||
		out |= TIOCM_CTS;
 | 
			
		||||
	if (my & IPW_CONTROL_LINE_DSR)
 | 
			
		||||
		out |= TIOCM_DSR;
 | 
			
		||||
	if (my & IPW_CONTROL_LINE_DCD)
 | 
			
		||||
		out |= TIOCM_CD;
 | 
			
		||||
 | 
			
		||||
	return out;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int set_control_lines(struct ipw_tty *tty, unsigned int set,
 | 
			
		||||
			     unsigned int clear)
 | 
			
		||||
{
 | 
			
		||||
	int ret;
 | 
			
		||||
 | 
			
		||||
	if (set & TIOCM_RTS) {
 | 
			
		||||
		ret = ipwireless_set_RTS(tty->hardware, tty->channel_idx, 1);
 | 
			
		||||
		if (ret)
 | 
			
		||||
			return ret;
 | 
			
		||||
		if (tty->secondary_channel_idx != -1) {
 | 
			
		||||
			ret = ipwireless_set_RTS(tty->hardware,
 | 
			
		||||
					  tty->secondary_channel_idx, 1);
 | 
			
		||||
			if (ret)
 | 
			
		||||
				return ret;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	if (set & TIOCM_DTR) {
 | 
			
		||||
		ret = ipwireless_set_DTR(tty->hardware, tty->channel_idx, 1);
 | 
			
		||||
		if (ret)
 | 
			
		||||
			return ret;
 | 
			
		||||
		if (tty->secondary_channel_idx != -1) {
 | 
			
		||||
			ret = ipwireless_set_DTR(tty->hardware,
 | 
			
		||||
					  tty->secondary_channel_idx, 1);
 | 
			
		||||
			if (ret)
 | 
			
		||||
				return ret;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	if (clear & TIOCM_RTS) {
 | 
			
		||||
		ret = ipwireless_set_RTS(tty->hardware, tty->channel_idx, 0);
 | 
			
		||||
		if (tty->secondary_channel_idx != -1) {
 | 
			
		||||
			ret = ipwireless_set_RTS(tty->hardware,
 | 
			
		||||
					  tty->secondary_channel_idx, 0);
 | 
			
		||||
			if (ret)
 | 
			
		||||
				return ret;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	if (clear & TIOCM_DTR) {
 | 
			
		||||
		ret = ipwireless_set_DTR(tty->hardware, tty->channel_idx, 0);
 | 
			
		||||
		if (tty->secondary_channel_idx != -1) {
 | 
			
		||||
			ret = ipwireless_set_DTR(tty->hardware,
 | 
			
		||||
					  tty->secondary_channel_idx, 0);
 | 
			
		||||
			if (ret)
 | 
			
		||||
				return ret;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int ipw_tiocmget(struct tty_struct *linux_tty, struct file *file)
 | 
			
		||||
{
 | 
			
		||||
	struct ipw_tty *tty = linux_tty->driver_data;
 | 
			
		||||
 | 
			
		||||
	if (!tty)
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
 | 
			
		||||
	if (!tty->open_count)
 | 
			
		||||
		return -EINVAL;
 | 
			
		||||
 | 
			
		||||
	return get_control_lines(tty);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int
 | 
			
		||||
ipw_tiocmset(struct tty_struct *linux_tty, struct file *file,
 | 
			
		||||
	     unsigned int set, unsigned int clear)
 | 
			
		||||
{
 | 
			
		||||
	struct ipw_tty *tty = linux_tty->driver_data;
 | 
			
		||||
 | 
			
		||||
	if (!tty)
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
 | 
			
		||||
	if (!tty->open_count)
 | 
			
		||||
		return -EINVAL;
 | 
			
		||||
 | 
			
		||||
	return set_control_lines(tty, set, clear);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int ipw_ioctl(struct tty_struct *linux_tty, struct file *file,
 | 
			
		||||
		     unsigned int cmd, unsigned long arg)
 | 
			
		||||
{
 | 
			
		||||
	struct ipw_tty *tty = linux_tty->driver_data;
 | 
			
		||||
 | 
			
		||||
	if (!tty)
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
 | 
			
		||||
	if (!tty->open_count)
 | 
			
		||||
		return -EINVAL;
 | 
			
		||||
 | 
			
		||||
	switch (cmd) {
 | 
			
		||||
	case TIOCGSERIAL:
 | 
			
		||||
		return ipwireless_get_serial_info(tty, (void __user *) arg);
 | 
			
		||||
 | 
			
		||||
	case TIOCSSERIAL:
 | 
			
		||||
		return 0;	/* Keeps the PCMCIA scripts happy. */
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (tty->tty_type == TTYTYPE_MODEM) {
 | 
			
		||||
		switch (cmd) {
 | 
			
		||||
		case PPPIOCGCHAN:
 | 
			
		||||
			{
 | 
			
		||||
				int chan = ipwireless_ppp_channel_index(
 | 
			
		||||
							tty->network);
 | 
			
		||||
 | 
			
		||||
				if (chan < 0)
 | 
			
		||||
					return -ENODEV;
 | 
			
		||||
				if (put_user(chan, (int __user *) arg))
 | 
			
		||||
					return -EFAULT;
 | 
			
		||||
			}
 | 
			
		||||
			return 0;
 | 
			
		||||
 | 
			
		||||
		case PPPIOCGUNIT:
 | 
			
		||||
			{
 | 
			
		||||
				int unit = ipwireless_ppp_unit_number(
 | 
			
		||||
						tty->network);
 | 
			
		||||
 | 
			
		||||
				if (unit < 0)
 | 
			
		||||
					return -ENODEV;
 | 
			
		||||
				if (put_user(unit, (int __user *) arg))
 | 
			
		||||
					return -EFAULT;
 | 
			
		||||
			}
 | 
			
		||||
			return 0;
 | 
			
		||||
 | 
			
		||||
		case TCGETS:
 | 
			
		||||
		case TCGETA:
 | 
			
		||||
			return n_tty_ioctl(linux_tty, file, cmd, arg);
 | 
			
		||||
 | 
			
		||||
		case TCFLSH:
 | 
			
		||||
			return n_tty_ioctl(linux_tty, file, cmd, arg);
 | 
			
		||||
 | 
			
		||||
		case FIONREAD:
 | 
			
		||||
			{
 | 
			
		||||
				int val = 0;
 | 
			
		||||
 | 
			
		||||
				if (put_user(val, (int __user *) arg))
 | 
			
		||||
					return -EFAULT;
 | 
			
		||||
			}
 | 
			
		||||
			return 0;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return -ENOIOCTLCMD;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int add_tty(dev_node_t *nodesp, int j,
 | 
			
		||||
		    struct ipw_hardware *hardware,
 | 
			
		||||
		    struct ipw_network *network, int channel_idx,
 | 
			
		||||
		    int secondary_channel_idx, int tty_type)
 | 
			
		||||
{
 | 
			
		||||
	ttys[j] = kzalloc(sizeof(struct ipw_tty), GFP_KERNEL);
 | 
			
		||||
	if (!ttys[j])
 | 
			
		||||
		return -ENOMEM;
 | 
			
		||||
	ttys[j]->index = j;
 | 
			
		||||
	ttys[j]->hardware = hardware;
 | 
			
		||||
	ttys[j]->channel_idx = channel_idx;
 | 
			
		||||
	ttys[j]->secondary_channel_idx = secondary_channel_idx;
 | 
			
		||||
	ttys[j]->network = network;
 | 
			
		||||
	ttys[j]->tty_type = tty_type;
 | 
			
		||||
	mutex_init(&ttys[j]->ipw_tty_mutex);
 | 
			
		||||
 | 
			
		||||
	tty_register_device(ipw_tty_driver, j, NULL);
 | 
			
		||||
	ipwireless_associate_network_tty(network, channel_idx, ttys[j]);
 | 
			
		||||
 | 
			
		||||
	if (secondary_channel_idx != -1)
 | 
			
		||||
		ipwireless_associate_network_tty(network,
 | 
			
		||||
						 secondary_channel_idx,
 | 
			
		||||
						 ttys[j]);
 | 
			
		||||
	if (nodesp != NULL) {
 | 
			
		||||
		sprintf(nodesp->dev_name, "ttyIPWp%d", j);
 | 
			
		||||
		nodesp->major = ipw_tty_driver->major;
 | 
			
		||||
		nodesp->minor = j + ipw_tty_driver->minor_start;
 | 
			
		||||
	}
 | 
			
		||||
	if (get_tty(j + ipw_tty_driver->minor_start) == ttys[j])
 | 
			
		||||
		report_registering(ttys[j]);
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
struct ipw_tty *ipwireless_tty_create(struct ipw_hardware *hardware,
 | 
			
		||||
				      struct ipw_network *network,
 | 
			
		||||
				      dev_node_t *nodes)
 | 
			
		||||
{
 | 
			
		||||
	int i, j;
 | 
			
		||||
 | 
			
		||||
	for (i = 0; i < IPWIRELESS_PCMCIA_MINOR_RANGE; i++) {
 | 
			
		||||
		int allfree = 1;
 | 
			
		||||
 | 
			
		||||
		for (j = i; j < IPWIRELESS_PCMCIA_MINORS;
 | 
			
		||||
				j += IPWIRELESS_PCMCIA_MINOR_RANGE)
 | 
			
		||||
			if (ttys[j] != NULL) {
 | 
			
		||||
				allfree = 0;
 | 
			
		||||
				break;
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
		if (allfree) {
 | 
			
		||||
			j = i;
 | 
			
		||||
 | 
			
		||||
			if (add_tty(&nodes[0], j, hardware, network,
 | 
			
		||||
					IPW_CHANNEL_DIALLER, IPW_CHANNEL_RAS,
 | 
			
		||||
					TTYTYPE_MODEM))
 | 
			
		||||
				return NULL;
 | 
			
		||||
 | 
			
		||||
			j += IPWIRELESS_PCMCIA_MINOR_RANGE;
 | 
			
		||||
			if (add_tty(&nodes[1], j, hardware, network,
 | 
			
		||||
					IPW_CHANNEL_DIALLER, -1,
 | 
			
		||||
					TTYTYPE_MONITOR))
 | 
			
		||||
				return NULL;
 | 
			
		||||
 | 
			
		||||
			j += IPWIRELESS_PCMCIA_MINOR_RANGE;
 | 
			
		||||
			if (add_tty(NULL, j, hardware, network,
 | 
			
		||||
					IPW_CHANNEL_RAS, -1,
 | 
			
		||||
					TTYTYPE_RAS_RAW))
 | 
			
		||||
				return NULL;
 | 
			
		||||
 | 
			
		||||
			nodes[0].next = &nodes[1];
 | 
			
		||||
			nodes[1].next = NULL;
 | 
			
		||||
 | 
			
		||||
			return ttys[i];
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	return NULL;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * Must be called before ipwireless_network_free().
 | 
			
		||||
 */
 | 
			
		||||
void ipwireless_tty_free(struct ipw_tty *tty)
 | 
			
		||||
{
 | 
			
		||||
	int j;
 | 
			
		||||
	struct ipw_network *network = ttys[tty->index]->network;
 | 
			
		||||
 | 
			
		||||
	for (j = tty->index; j < IPWIRELESS_PCMCIA_MINORS;
 | 
			
		||||
			j += IPWIRELESS_PCMCIA_MINOR_RANGE) {
 | 
			
		||||
		struct ipw_tty *ttyj = ttys[j];
 | 
			
		||||
 | 
			
		||||
		if (ttyj) {
 | 
			
		||||
			mutex_lock(&ttyj->ipw_tty_mutex);
 | 
			
		||||
			if (get_tty(j + ipw_tty_driver->minor_start) == ttyj)
 | 
			
		||||
				report_deregistering(ttyj);
 | 
			
		||||
			ttyj->closing = 1;
 | 
			
		||||
			if (ttyj->linux_tty != NULL) {
 | 
			
		||||
				mutex_unlock(&ttyj->ipw_tty_mutex);
 | 
			
		||||
				tty_hangup(ttyj->linux_tty);
 | 
			
		||||
				/* Wait till the tty_hangup has completed */
 | 
			
		||||
				flush_scheduled_work();
 | 
			
		||||
				mutex_lock(&ttyj->ipw_tty_mutex);
 | 
			
		||||
			}
 | 
			
		||||
			while (ttyj->open_count)
 | 
			
		||||
				do_ipw_close(ttyj);
 | 
			
		||||
			ipwireless_disassociate_network_ttys(network,
 | 
			
		||||
							     ttyj->channel_idx);
 | 
			
		||||
			tty_unregister_device(ipw_tty_driver, j);
 | 
			
		||||
			ttys[j] = NULL;
 | 
			
		||||
			mutex_unlock(&ttyj->ipw_tty_mutex);
 | 
			
		||||
			kfree(ttyj);
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static struct tty_operations tty_ops = {
 | 
			
		||||
	.open = ipw_open,
 | 
			
		||||
	.close = ipw_close,
 | 
			
		||||
	.hangup = ipw_hangup,
 | 
			
		||||
	.write = ipw_write,
 | 
			
		||||
	.write_room = ipw_write_room,
 | 
			
		||||
	.ioctl = ipw_ioctl,
 | 
			
		||||
	.chars_in_buffer = ipw_chars_in_buffer,
 | 
			
		||||
	.tiocmget = ipw_tiocmget,
 | 
			
		||||
	.tiocmset = ipw_tiocmset,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
int ipwireless_tty_init(void)
 | 
			
		||||
{
 | 
			
		||||
	int result;
 | 
			
		||||
 | 
			
		||||
	ipw_tty_driver = alloc_tty_driver(IPWIRELESS_PCMCIA_MINORS);
 | 
			
		||||
	if (!ipw_tty_driver)
 | 
			
		||||
		return -ENOMEM;
 | 
			
		||||
 | 
			
		||||
	ipw_tty_driver->owner = THIS_MODULE;
 | 
			
		||||
	ipw_tty_driver->driver_name = IPWIRELESS_PCCARD_NAME;
 | 
			
		||||
	ipw_tty_driver->name = "ttyIPWp";
 | 
			
		||||
	ipw_tty_driver->major = 0;
 | 
			
		||||
	ipw_tty_driver->minor_start = IPWIRELESS_PCMCIA_START;
 | 
			
		||||
	ipw_tty_driver->type = TTY_DRIVER_TYPE_SERIAL;
 | 
			
		||||
	ipw_tty_driver->subtype = SERIAL_TYPE_NORMAL;
 | 
			
		||||
	ipw_tty_driver->flags = TTY_DRIVER_REAL_RAW | TTY_DRIVER_DYNAMIC_DEV;
 | 
			
		||||
	ipw_tty_driver->init_termios = tty_std_termios;
 | 
			
		||||
	ipw_tty_driver->init_termios.c_cflag =
 | 
			
		||||
	    B9600 | CS8 | CREAD | HUPCL | CLOCAL;
 | 
			
		||||
	ipw_tty_driver->init_termios.c_ispeed = 9600;
 | 
			
		||||
	ipw_tty_driver->init_termios.c_ospeed = 9600;
 | 
			
		||||
	tty_set_operations(ipw_tty_driver, &tty_ops);
 | 
			
		||||
	result = tty_register_driver(ipw_tty_driver);
 | 
			
		||||
	if (result) {
 | 
			
		||||
		printk(KERN_ERR IPWIRELESS_PCCARD_NAME
 | 
			
		||||
		       ": failed to register tty driver\n");
 | 
			
		||||
		put_tty_driver(ipw_tty_driver);
 | 
			
		||||
		return result;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ipwireless_tty_release(void)
 | 
			
		||||
{
 | 
			
		||||
	int ret;
 | 
			
		||||
 | 
			
		||||
	ret = tty_unregister_driver(ipw_tty_driver);
 | 
			
		||||
	put_tty_driver(ipw_tty_driver);
 | 
			
		||||
	if (ret != 0)
 | 
			
		||||
		printk(KERN_ERR IPWIRELESS_PCCARD_NAME
 | 
			
		||||
			": tty_unregister_driver failed with code %d\n", ret);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int ipwireless_tty_is_modem(struct ipw_tty *tty)
 | 
			
		||||
{
 | 
			
		||||
	return tty->tty_type == TTYTYPE_MODEM;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void
 | 
			
		||||
ipwireless_tty_notify_control_line_change(struct ipw_tty *tty,
 | 
			
		||||
					  unsigned int channel_idx,
 | 
			
		||||
					  unsigned int control_lines,
 | 
			
		||||
					  unsigned int changed_mask)
 | 
			
		||||
{
 | 
			
		||||
	unsigned int old_control_lines = tty->control_lines;
 | 
			
		||||
 | 
			
		||||
	tty->control_lines = (tty->control_lines & ~changed_mask)
 | 
			
		||||
		| (control_lines & changed_mask);
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * If DCD is de-asserted, we close the tty so pppd can tell that we
 | 
			
		||||
	 * have gone offline.
 | 
			
		||||
	 */
 | 
			
		||||
	if ((old_control_lines & IPW_CONTROL_LINE_DCD)
 | 
			
		||||
			&& !(tty->control_lines & IPW_CONTROL_LINE_DCD)
 | 
			
		||||
			&& tty->linux_tty) {
 | 
			
		||||
		tty_hangup(tty->linux_tty);
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										48
									
								
								drivers/char/pcmcia/ipwireless/tty.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								drivers/char/pcmcia/ipwireless/tty.h
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,48 @@
 | 
			
		|||
/*
 | 
			
		||||
 * IPWireless 3G PCMCIA Network Driver
 | 
			
		||||
 *
 | 
			
		||||
 * Original code
 | 
			
		||||
 *   by Stephen Blackheath <stephen@blacksapphire.com>,
 | 
			
		||||
 *      Ben Martel <benm@symmetric.co.nz>
 | 
			
		||||
 *
 | 
			
		||||
 * Copyrighted as follows:
 | 
			
		||||
 *   Copyright (C) 2004 by Symmetric Systems Ltd (NZ)
 | 
			
		||||
 *
 | 
			
		||||
 * Various driver changes and rewrites, port to new kernels
 | 
			
		||||
 *   Copyright (C) 2006-2007 Jiri Kosina
 | 
			
		||||
 *
 | 
			
		||||
 * Misc code cleanups and updates
 | 
			
		||||
 *   Copyright (C) 2007 David Sterba
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
#ifndef _IPWIRELESS_CS_TTY_H_
 | 
			
		||||
#define _IPWIRELESS_CS_TTY_H_
 | 
			
		||||
 | 
			
		||||
#include <linux/types.h>
 | 
			
		||||
#include <linux/sched.h>
 | 
			
		||||
 | 
			
		||||
#include <pcmcia/cs_types.h>
 | 
			
		||||
#include <pcmcia/cs.h>
 | 
			
		||||
#include <pcmcia/cistpl.h>
 | 
			
		||||
#include <pcmcia/ds.h>
 | 
			
		||||
 | 
			
		||||
struct ipw_tty;
 | 
			
		||||
struct ipw_network;
 | 
			
		||||
struct ipw_hardware;
 | 
			
		||||
 | 
			
		||||
int ipwireless_tty_init(void);
 | 
			
		||||
void ipwireless_tty_release(void);
 | 
			
		||||
 | 
			
		||||
struct ipw_tty *ipwireless_tty_create(struct ipw_hardware *hw,
 | 
			
		||||
				      struct ipw_network *net,
 | 
			
		||||
				      dev_node_t *nodes);
 | 
			
		||||
void ipwireless_tty_free(struct ipw_tty *tty);
 | 
			
		||||
void ipwireless_tty_received(struct ipw_tty *tty, unsigned char *data,
 | 
			
		||||
			     unsigned int length);
 | 
			
		||||
int ipwireless_tty_is_modem(struct ipw_tty *tty);
 | 
			
		||||
void ipwireless_tty_notify_control_line_change(struct ipw_tty *tty,
 | 
			
		||||
					       unsigned int channel_idx,
 | 
			
		||||
					       unsigned int control_lines,
 | 
			
		||||
					       unsigned int changed_mask);
 | 
			
		||||
 | 
			
		||||
#endif
 | 
			
		||||
		Loading…
	
		Reference in a new issue