diff options
Diffstat (limited to 'drivers/firewire/ohci.c')
| -rw-r--r-- | drivers/firewire/ohci.c | 131 | 
1 files changed, 70 insertions, 61 deletions
| diff --git a/drivers/firewire/ohci.c b/drivers/firewire/ohci.c index b9ae0340b8a7..f6de0b3a9a55 100644 --- a/drivers/firewire/ohci.c +++ b/drivers/firewire/ohci.c @@ -40,6 +40,7 @@  #include "core.h"  #include "ohci.h" +#include "packet-header-definitions.h"  #define ohci_info(ohci, f, args...)	dev_info(ohci->card.device, f, ##args)  #define ohci_notice(ohci, f, args...)	dev_notice(ohci->card.device, f, ##args) @@ -393,7 +394,6 @@ MODULE_PARM_DESC(quirks, "Chip quirks (default = 0"  #define OHCI_PARAM_DEBUG_AT_AR		1  #define OHCI_PARAM_DEBUG_SELFIDS	2  #define OHCI_PARAM_DEBUG_IRQS		4 -#define OHCI_PARAM_DEBUG_BUSRESETS	8 /* only effective before chip init */  static int param_debug;  module_param_named(debug, param_debug, int, 0644); @@ -401,7 +401,6 @@ MODULE_PARM_DESC(debug, "Verbose logging (default = 0"  	", AT/AR events = "	__stringify(OHCI_PARAM_DEBUG_AT_AR)  	", self-IDs = "		__stringify(OHCI_PARAM_DEBUG_SELFIDS)  	", IRQs = "		__stringify(OHCI_PARAM_DEBUG_IRQS) -	", busReset events = "	__stringify(OHCI_PARAM_DEBUG_BUSRESETS)  	", or a combination, or all = -1)");  static bool param_remote_dma; @@ -410,12 +409,7 @@ MODULE_PARM_DESC(remote_dma, "Enable unfiltered remote DMA (default = N)");  static void log_irqs(struct fw_ohci *ohci, u32 evt)  { -	if (likely(!(param_debug & -			(OHCI_PARAM_DEBUG_IRQS | OHCI_PARAM_DEBUG_BUSRESETS)))) -		return; - -	if (!(param_debug & OHCI_PARAM_DEBUG_IRQS) && -	    !(evt & OHCI1394_busReset)) +	if (likely(!(param_debug & OHCI_PARAM_DEBUG_IRQS)))  		return;  	ohci_notice(ohci, "IRQ %08x%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s\n", evt, @@ -516,14 +510,14 @@ static const char *tcodes[] = {  static void log_ar_at_event(struct fw_ohci *ohci,  			    char dir, int speed, u32 *header, int evt)  { -	int tcode = header[0] >> 4 & 0xf; +	int tcode = async_header_get_tcode(header);  	char specific[12];  	if (likely(!(param_debug & OHCI_PARAM_DEBUG_AT_AR)))  		return;  	if (unlikely(evt >= ARRAY_SIZE(evts))) -			evt = 0x1f; +		evt = 0x1f;  	if (evt == OHCI1394_evt_bus_reset) {  		ohci_notice(ohci, "A%c evt_bus_reset, generation %d\n", @@ -532,20 +526,27 @@ static void log_ar_at_event(struct fw_ohci *ohci,  	}  	switch (tcode) { -	case 0x0: case 0x6: case 0x8: +	case TCODE_WRITE_QUADLET_REQUEST: +	case TCODE_READ_QUADLET_RESPONSE: +	case TCODE_CYCLE_START:  		snprintf(specific, sizeof(specific), " = %08x",  			 be32_to_cpu((__force __be32)header[3]));  		break; -	case 0x1: case 0x5: case 0x7: case 0x9: case 0xb: +	case TCODE_WRITE_BLOCK_REQUEST: +	case TCODE_READ_BLOCK_REQUEST: +	case TCODE_READ_BLOCK_RESPONSE: +	case TCODE_LOCK_REQUEST: +	case TCODE_LOCK_RESPONSE:  		snprintf(specific, sizeof(specific), " %x,%x", -			 header[3] >> 16, header[3] & 0xffff); +			 async_header_get_data_length(header), +			 async_header_get_extended_tcode(header));  		break;  	default:  		specific[0] = '\0';  	}  	switch (tcode) { -	case 0xa: +	case TCODE_STREAM_DATA:  		ohci_notice(ohci, "A%c %s, %s\n",  			    dir, evts[evt], tcodes[tcode]);  		break; @@ -553,19 +554,23 @@ static void log_ar_at_event(struct fw_ohci *ohci,  		ohci_notice(ohci, "A%c %s, PHY %08x %08x\n",  			    dir, evts[evt], header[1], header[2]);  		break; -	case 0x0: case 0x1: case 0x4: case 0x5: case 0x9: +	case TCODE_WRITE_QUADLET_REQUEST: +	case TCODE_WRITE_BLOCK_REQUEST: +	case TCODE_READ_QUADLET_REQUEST: +	case TCODE_READ_BLOCK_REQUEST: +	case TCODE_LOCK_REQUEST:  		ohci_notice(ohci, -			    "A%c spd %x tl %02x, %04x -> %04x, %s, %s, %04x%08x%s\n", -			    dir, speed, header[0] >> 10 & 0x3f, -			    header[1] >> 16, header[0] >> 16, evts[evt], -			    tcodes[tcode], header[1] & 0xffff, header[2], specific); +			    "A%c spd %x tl %02x, %04x -> %04x, %s, %s, %012llx%s\n", +			    dir, speed, async_header_get_tlabel(header), +			    async_header_get_source(header), async_header_get_destination(header), +			    evts[evt], tcodes[tcode], async_header_get_offset(header), specific);  		break;  	default:  		ohci_notice(ohci,  			    "A%c spd %x tl %02x, %04x -> %04x, %s, %s%s\n", -			    dir, speed, header[0] >> 10 & 0x3f, -			    header[1] >> 16, header[0] >> 16, evts[evt], -			    tcodes[tcode], specific); +			    dir, speed, async_header_get_tlabel(header), +			    async_header_get_source(header), async_header_get_destination(header), +			    evts[evt], tcodes[tcode], specific);  	}  } @@ -853,7 +858,7 @@ static __le32 *handle_ar_packet(struct ar_context *ctx, __le32 *buffer)  	p.header[1] = cond_le32_to_cpu(buffer[1]);  	p.header[2] = cond_le32_to_cpu(buffer[2]); -	tcode = (p.header[0] >> 4) & 0x0f; +	tcode = async_header_get_tcode(p.header);  	switch (tcode) {  	case TCODE_WRITE_QUADLET_REQUEST:  	case TCODE_READ_QUADLET_RESPONSE: @@ -874,7 +879,7 @@ static __le32 *handle_ar_packet(struct ar_context *ctx, __le32 *buffer)  	case TCODE_LOCK_RESPONSE:  		p.header[3] = cond_le32_to_cpu(buffer[3]);  		p.header_length = 16; -		p.payload_length = p.header[3] >> 16; +		p.payload_length = async_header_get_data_length(p.header);  		if (p.payload_length > MAX_ASYNC_PAYLOAD) {  			ar_context_abort(ctx, "invalid packet length");  			return NULL; @@ -911,8 +916,7 @@ static __le32 *handle_ar_packet(struct ar_context *ctx, __le32 *buffer)  	 * Several controllers, notably from NEC and VIA, forget to  	 * write ack_complete status at PHY packet reception.  	 */ -	if (evt == OHCI1394_evt_no_status && -	    (p.header[0] & 0xff) == (OHCI1394_phy_tcode << 4)) +	if (evt == OHCI1394_evt_no_status && tcode == OHCI1394_phy_tcode)  		p.ack = ACK_COMPLETE;  	/* @@ -1353,7 +1357,7 @@ static int at_context_queue_packet(struct context *ctx,  	 * accordingly.  	 */ -	tcode = (packet->header[0] >> 4) & 0x0f; +	tcode = async_header_get_tcode(packet->header);  	header = (__le32 *) &d[1];  	switch (tcode) {  	case TCODE_WRITE_QUADLET_REQUEST: @@ -1371,7 +1375,7 @@ static int at_context_queue_packet(struct context *ctx,  					(packet->header[0] & 0xffff0000));  		header[2] = cpu_to_le32(packet->header[2]); -		if (TCODE_IS_BLOCK_PACKET(tcode)) +		if (tcode_is_block_packet(tcode))  			header[3] = cpu_to_le32(packet->header[3]);  		else  			header[3] = (__force __le32) packet->header[3]; @@ -1550,12 +1554,6 @@ static int handle_at_packet(struct context *context,  	return 1;  } -#define HEADER_GET_DESTINATION(q)	(((q) >> 16) & 0xffff) -#define HEADER_GET_TCODE(q)		(((q) >> 4) & 0x0f) -#define HEADER_GET_OFFSET_HIGH(q)	(((q) >> 0) & 0xffff) -#define HEADER_GET_DATA_LENGTH(q)	(((q) >> 16) & 0xffff) -#define HEADER_GET_EXTENDED_TCODE(q)	(((q) >> 0) & 0xffff) -  static u32 get_cycle_time(struct fw_ohci *ohci);  static void handle_local_rom(struct fw_ohci *ohci, @@ -1564,9 +1562,9 @@ static void handle_local_rom(struct fw_ohci *ohci,  	struct fw_packet response;  	int tcode, length, i; -	tcode = HEADER_GET_TCODE(packet->header[0]); -	if (TCODE_IS_BLOCK_PACKET(tcode)) -		length = HEADER_GET_DATA_LENGTH(packet->header[3]); +	tcode = async_header_get_tcode(packet->header); +	if (tcode_is_block_packet(tcode)) +		length = async_header_get_data_length(packet->header);  	else  		length = 4; @@ -1574,7 +1572,7 @@ static void handle_local_rom(struct fw_ohci *ohci,  	if (i + length > CONFIG_ROM_SIZE) {  		fw_fill_response(&response, packet->header,  				 RCODE_ADDRESS_ERROR, NULL, 0); -	} else if (!TCODE_IS_READ_REQUEST(tcode)) { +	} else if (!tcode_is_read_request(tcode)) {  		fw_fill_response(&response, packet->header,  				 RCODE_TYPE_ERROR, NULL, 0);  	} else { @@ -1595,10 +1593,10 @@ static void handle_local_lock(struct fw_ohci *ohci,  	__be32 *payload, lock_old;  	u32 lock_arg, lock_data; -	tcode = HEADER_GET_TCODE(packet->header[0]); -	length = HEADER_GET_DATA_LENGTH(packet->header[3]); +	tcode = async_header_get_tcode(packet->header); +	length = async_header_get_data_length(packet->header);  	payload = packet->payload; -	ext_tcode = HEADER_GET_EXTENDED_TCODE(packet->header[3]); +	ext_tcode = async_header_get_extended_tcode(packet->header);  	if (tcode == TCODE_LOCK_REQUEST &&  	    ext_tcode == EXTCODE_COMPARE_SWAP && length == 8) { @@ -1646,10 +1644,7 @@ static void handle_local_request(struct context *ctx, struct fw_packet *packet)  		packet->callback(packet, &ctx->ohci->card, packet->ack);  	} -	offset = -		((unsigned long long) -		 HEADER_GET_OFFSET_HIGH(packet->header[1]) << 32) | -		packet->header[2]; +	offset = async_header_get_offset(packet->header);  	csr = offset - CSR_REGISTER_BASE;  	/* Handle config rom reads. */ @@ -1683,7 +1678,7 @@ static void at_context_transmit(struct context *ctx, struct fw_packet *packet)  	spin_lock_irqsave(&ctx->ohci->lock, flags); -	if (HEADER_GET_DESTINATION(packet->header[0]) == ctx->ohci->node_id && +	if (async_header_get_destination(packet->header) == ctx->ohci->node_id &&  	    ctx->ohci->generation == packet->generation) {  		spin_unlock_irqrestore(&ctx->ohci->lock, flags); @@ -2064,8 +2059,7 @@ static void bus_reset_work(struct work_struct *work)  	ohci->generation = generation;  	reg_write(ohci, OHCI1394_IntEventClear, OHCI1394_busReset); -	if (param_debug & OHCI_PARAM_DEBUG_BUSRESETS) -		reg_write(ohci, OHCI1394_IntMaskSet, OHCI1394_busReset); +	reg_write(ohci, OHCI1394_IntMaskSet, OHCI1394_busReset);  	if (ohci->quirks & QUIRK_RESET_PACKET)  		ohci->request_generation = generation; @@ -2137,6 +2131,7 @@ static irqreturn_t irq_handler(int irq, void *data)  	reg_write(ohci, OHCI1394_IntEventClear,  		  event & ~(OHCI1394_busReset | OHCI1394_postedWriteErr));  	log_irqs(ohci, event); +	// The flag is masked again at bus_reset_work() scheduled by selfID event.  	if (event & OHCI1394_busReset)  		reg_write(ohci, OHCI1394_IntMaskClear, OHCI1394_busReset); @@ -2476,9 +2471,8 @@ static int ohci_enable(struct fw_card *card,  		OHCI1394_cycleInconsistent |  		OHCI1394_unrecoverableError |  		OHCI1394_cycleTooLong | -		OHCI1394_masterIntEnable; -	if (param_debug & OHCI_PARAM_DEBUG_BUSRESETS) -		irqs |= OHCI1394_busReset; +		OHCI1394_masterIntEnable | +		OHCI1394_busReset;  	reg_write(ohci, OHCI1394_IntMaskSet, irqs);  	reg_write(ohci, OHCI1394_HCControlSet, @@ -3631,7 +3625,7 @@ static int pci_probe(struct pci_dev *dev,  	struct fw_ohci *ohci;  	u32 bus_options, max_receive, link_speed, version;  	u64 guid; -	int i, err; +	int i, flags, irq, err;  	size_t size;  	if (dev->vendor == PCI_VENDOR_ID_PINNACLE_SYSTEMS) { @@ -3756,18 +3750,29 @@ static int pci_probe(struct pci_dev *dev,  	guid = ((u64) reg_read(ohci, OHCI1394_GUIDHi) << 32) |  		reg_read(ohci, OHCI1394_GUIDLo); +	flags = PCI_IRQ_INTX;  	if (!(ohci->quirks & QUIRK_NO_MSI)) -		pci_enable_msi(dev); -	err = devm_request_irq(&dev->dev, dev->irq, irq_handler, -			       pci_dev_msi_enabled(dev) ? 0 : IRQF_SHARED, ohci_driver_name, ohci); +		flags |= PCI_IRQ_MSI; +	err = pci_alloc_irq_vectors(dev, 1, 1, flags); +	if (err < 0) +		return err; +	irq = pci_irq_vector(dev, 0); +	if (irq < 0) { +		err = irq; +		goto fail_msi; +	} + +	err = request_threaded_irq(irq, irq_handler, NULL, +				   pci_dev_msi_enabled(dev) ? 0 : IRQF_SHARED, ohci_driver_name, +				   ohci);  	if (err < 0) { -		ohci_err(ohci, "failed to allocate interrupt %d\n", dev->irq); +		ohci_err(ohci, "failed to allocate interrupt %d\n", irq);  		goto fail_msi;  	}  	err = fw_card_add(&ohci->card, max_receive, link_speed, guid);  	if (err) -		goto fail_msi; +		goto fail_irq;  	version = reg_read(ohci, OHCI1394_Version) & 0x00ff00ff;  	ohci_notice(ohci, @@ -3780,9 +3785,10 @@ static int pci_probe(struct pci_dev *dev,  	return 0; + fail_irq: +	free_irq(irq, ohci);   fail_msi: -	devm_free_irq(&dev->dev, dev->irq, ohci); -	pci_disable_msi(dev); +	pci_free_irq_vectors(dev);  	return err;  } @@ -3790,6 +3796,7 @@ static int pci_probe(struct pci_dev *dev,  static void pci_remove(struct pci_dev *dev)  {  	struct fw_ohci *ohci = pci_get_drvdata(dev); +	int irq;  	/*  	 * If the removal is happening from the suspend state, LPS won't be @@ -3809,8 +3816,10 @@ static void pci_remove(struct pci_dev *dev)  	software_reset(ohci); -	devm_free_irq(&dev->dev, dev->irq, ohci); -	pci_disable_msi(dev); +	irq = pci_irq_vector(dev, 0); +	if (irq >= 0) +		free_irq(irq, ohci); +	pci_free_irq_vectors(dev);  	dev_notice(&dev->dev, "removing fw-ohci device\n");  } | 
