1133 lines
		
	
	
		
			28 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1133 lines
		
	
	
		
			28 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| /* $Id: os_4bri.c,v 1.28.4.4 2005/02/11 19:40:25 armin Exp $ */
 | |
| 
 | |
| #include "platform.h"
 | |
| #include "debuglib.h"
 | |
| #include "cardtype.h"
 | |
| #include "pc.h"
 | |
| #include "pr_pc.h"
 | |
| #include "di_defs.h"
 | |
| #include "dsp_defs.h"
 | |
| #include "di.h"
 | |
| #include "io.h"
 | |
| 
 | |
| #include "xdi_msg.h"
 | |
| #include "xdi_adapter.h"
 | |
| #include "os_4bri.h"
 | |
| #include "diva_pci.h"
 | |
| #include "mi_pc.h"
 | |
| #include "dsrv4bri.h"
 | |
| #include "helpers.h"
 | |
| 
 | |
| static void *diva_xdiLoadFileFile = NULL;
 | |
| static dword diva_xdiLoadFileLength = 0;
 | |
| 
 | |
| /*
 | |
| **  IMPORTS
 | |
| */
 | |
| extern void prepare_qBri_functions(PISDN_ADAPTER IoAdapter);
 | |
| extern void prepare_qBri2_functions(PISDN_ADAPTER IoAdapter);
 | |
| extern void diva_xdi_display_adapter_features(int card);
 | |
| extern void diva_add_slave_adapter(diva_os_xdi_adapter_t *a);
 | |
| 
 | |
| extern int qBri_FPGA_download(PISDN_ADAPTER IoAdapter);
 | |
| extern void start_qBri_hardware(PISDN_ADAPTER IoAdapter);
 | |
| 
 | |
| extern int diva_card_read_xlog(diva_os_xdi_adapter_t *a);
 | |
| 
 | |
| /*
 | |
| **  LOCALS
 | |
| */
 | |
| static unsigned long _4bri_bar_length[4] = {
 | |
| 	0x100,
 | |
| 	0x100,			/* I/O */
 | |
| 	MQ_MEMORY_SIZE,
 | |
| 	0x2000
 | |
| };
 | |
| static unsigned long _4bri_v2_bar_length[4] = {
 | |
| 	0x100,
 | |
| 	0x100,			/* I/O */
 | |
| 	MQ2_MEMORY_SIZE,
 | |
| 	0x10000
 | |
| };
 | |
| static unsigned long _4bri_v2_bri_bar_length[4] = {
 | |
| 	0x100,
 | |
| 	0x100,			/* I/O */
 | |
| 	BRI2_MEMORY_SIZE,
 | |
| 	0x10000
 | |
| };
 | |
| 
 | |
| 
 | |
| static int diva_4bri_cleanup_adapter(diva_os_xdi_adapter_t *a);
 | |
| static int _4bri_get_serial_number(diva_os_xdi_adapter_t *a);
 | |
| static int diva_4bri_cmd_card_proc(struct _diva_os_xdi_adapter *a,
 | |
| 				   diva_xdi_um_cfg_cmd_t *cmd,
 | |
| 				   int length);
 | |
| static int diva_4bri_cleanup_slave_adapters(diva_os_xdi_adapter_t *a);
 | |
| static int diva_4bri_write_fpga_image(diva_os_xdi_adapter_t *a,
 | |
| 				      byte *data, dword length);
 | |
| static int diva_4bri_reset_adapter(PISDN_ADAPTER IoAdapter);
 | |
| static int diva_4bri_write_sdram_block(PISDN_ADAPTER IoAdapter,
 | |
| 				       dword address,
 | |
| 				       const byte *data,
 | |
| 				       dword length, dword limit);
 | |
| static int diva_4bri_start_adapter(PISDN_ADAPTER IoAdapter,
 | |
| 				   dword start_address, dword features);
 | |
| static int check_qBri_interrupt(PISDN_ADAPTER IoAdapter);
 | |
| static int diva_4bri_stop_adapter(diva_os_xdi_adapter_t *a);
 | |
| 
 | |
| static int _4bri_is_rev_2_card(int card_ordinal)
 | |
| {
 | |
| 	switch (card_ordinal) {
 | |
| 	case CARDTYPE_DIVASRV_Q_8M_V2_PCI:
 | |
| 	case CARDTYPE_DIVASRV_VOICE_Q_8M_V2_PCI:
 | |
| 	case CARDTYPE_DIVASRV_B_2M_V2_PCI:
 | |
| 	case CARDTYPE_DIVASRV_B_2F_PCI:
 | |
| 	case CARDTYPE_DIVASRV_VOICE_B_2M_V2_PCI:
 | |
| 		return (1);
 | |
| 	}
 | |
| 	return (0);
 | |
| }
 | |
| 
 | |
| static int _4bri_is_rev_2_bri_card(int card_ordinal)
 | |
| {
 | |
| 	switch (card_ordinal) {
 | |
| 	case CARDTYPE_DIVASRV_B_2M_V2_PCI:
 | |
| 	case CARDTYPE_DIVASRV_B_2F_PCI:
 | |
| 	case CARDTYPE_DIVASRV_VOICE_B_2M_V2_PCI:
 | |
| 		return (1);
 | |
| 	}
 | |
| 	return (0);
 | |
| }
 | |
| 
 | |
| static void diva_4bri_set_addresses(diva_os_xdi_adapter_t *a)
 | |
| {
 | |
| 	dword offset = a->resources.pci.qoffset;
 | |
| 	dword c_offset = offset * a->xdi_adapter.ControllerNumber;
 | |
| 
 | |
| 	a->resources.pci.mem_type_id[MEM_TYPE_RAM] = 2;
 | |
| 	a->resources.pci.mem_type_id[MEM_TYPE_ADDRESS] = 2;
 | |
| 	a->resources.pci.mem_type_id[MEM_TYPE_CONTROL] = 2;
 | |
| 	a->resources.pci.mem_type_id[MEM_TYPE_RESET] = 0;
 | |
| 	a->resources.pci.mem_type_id[MEM_TYPE_CTLREG] = 3;
 | |
| 	a->resources.pci.mem_type_id[MEM_TYPE_PROM] = 0;
 | |
| 
 | |
| 	/*
 | |
| 	  Set up hardware related pointers
 | |
| 	*/
 | |
| 	a->xdi_adapter.Address = a->resources.pci.addr[2];	/* BAR2 SDRAM  */
 | |
| 	a->xdi_adapter.Address += c_offset;
 | |
| 
 | |
| 	a->xdi_adapter.Control = a->resources.pci.addr[2];	/* BAR2 SDRAM  */
 | |
| 
 | |
| 	a->xdi_adapter.ram = a->resources.pci.addr[2];	/* BAR2 SDRAM  */
 | |
| 	a->xdi_adapter.ram += c_offset + (offset - MQ_SHARED_RAM_SIZE);
 | |
| 
 | |
| 	a->xdi_adapter.reset = a->resources.pci.addr[0];	/* BAR0 CONFIG */
 | |
| 	/*
 | |
| 	  ctlReg contains the register address for the MIPS CPU reset control
 | |
| 	*/
 | |
| 	a->xdi_adapter.ctlReg = a->resources.pci.addr[3];	/* BAR3 CNTRL  */
 | |
| 	/*
 | |
| 	  prom contains the register address for FPGA and EEPROM programming
 | |
| 	*/
 | |
| 	a->xdi_adapter.prom = &a->xdi_adapter.reset[0x6E];
 | |
| }
 | |
| 
 | |
| /*
 | |
| **  BAR0 - MEM - 0x100    - CONFIG MEM
 | |
| **  BAR1 - I/O - 0x100    - UNUSED
 | |
| **  BAR2 - MEM - MQ_MEMORY_SIZE (MQ2_MEMORY_SIZE on Rev.2) - SDRAM
 | |
| **  BAR3 - MEM - 0x2000 (0x10000 on Rev.2)   - CNTRL
 | |
| **
 | |
| **  Called by master adapter, that will initialize and add slave adapters
 | |
| */
 | |
| int diva_4bri_init_card(diva_os_xdi_adapter_t *a)
 | |
| {
 | |
| 	int bar, i;
 | |
| 	byte __iomem *p;
 | |
| 	PADAPTER_LIST_ENTRY quadro_list;
 | |
| 	diva_os_xdi_adapter_t *diva_current;
 | |
| 	diva_os_xdi_adapter_t *adapter_list[4];
 | |
| 	PISDN_ADAPTER Slave;
 | |
| 	unsigned long bar_length[ARRAY_SIZE(_4bri_bar_length)];
 | |
| 	int v2 = _4bri_is_rev_2_card(a->CardOrdinal);
 | |
| 	int tasks = _4bri_is_rev_2_bri_card(a->CardOrdinal) ? 1 : MQ_INSTANCE_COUNT;
 | |
| 	int factor = (tasks == 1) ? 1 : 2;
 | |
| 
 | |
| 	if (v2) {
 | |
| 		if (_4bri_is_rev_2_bri_card(a->CardOrdinal)) {
 | |
| 			memcpy(bar_length, _4bri_v2_bri_bar_length,
 | |
| 			       sizeof(bar_length));
 | |
| 		} else {
 | |
| 			memcpy(bar_length, _4bri_v2_bar_length,
 | |
| 			       sizeof(bar_length));
 | |
| 		}
 | |
| 	} else {
 | |
| 		memcpy(bar_length, _4bri_bar_length, sizeof(bar_length));
 | |
| 	}
 | |
| 	DBG_TRC(("SDRAM_LENGTH=%08x, tasks=%d, factor=%d",
 | |
| 		 bar_length[2], tasks, factor))
 | |
| 
 | |
| 		/*
 | |
| 		  Get Serial Number
 | |
| 		  The serial number of 4BRI is accessible in accordance with PCI spec
 | |
| 		  via command register located in configuration space, also we do not
 | |
| 		  have to map any BAR before we can access it
 | |
| 		*/
 | |
| 		if (!_4bri_get_serial_number(a)) {
 | |
| 			DBG_ERR(("A: 4BRI can't get Serial Number"))
 | |
| 				diva_4bri_cleanup_adapter(a);
 | |
| 			return (-1);
 | |
| 		}
 | |
| 
 | |
| 	/*
 | |
| 	  Set properties
 | |
| 	*/
 | |
| 	a->xdi_adapter.Properties = CardProperties[a->CardOrdinal];
 | |
| 	DBG_LOG(("Load %s, SN:%ld, bus:%02x, func:%02x",
 | |
| 		 a->xdi_adapter.Properties.Name,
 | |
| 		 a->xdi_adapter.serialNo,
 | |
| 		 a->resources.pci.bus, a->resources.pci.func))
 | |
| 
 | |
| 		/*
 | |
| 		  First initialization step: get and check hardware resoures.
 | |
| 		  Do not map resources and do not access card at this step
 | |
| 		*/
 | |
| 		for (bar = 0; bar < 4; bar++) {
 | |
| 			a->resources.pci.bar[bar] =
 | |
| 				divasa_get_pci_bar(a->resources.pci.bus,
 | |
| 						   a->resources.pci.func, bar,
 | |
| 						   a->resources.pci.hdev);
 | |
| 			if (!a->resources.pci.bar[bar]
 | |
| 			    || (a->resources.pci.bar[bar] == 0xFFFFFFF0)) {
 | |
| 				DBG_ERR(
 | |
| 					("A: invalid bar[%d]=%08x", bar,
 | |
| 					 a->resources.pci.bar[bar]))
 | |
| 					return (-1);
 | |
| 			}
 | |
| 		}
 | |
| 	a->resources.pci.irq =
 | |
| 		(byte) divasa_get_pci_irq(a->resources.pci.bus,
 | |
| 					  a->resources.pci.func,
 | |
| 					  a->resources.pci.hdev);
 | |
| 	if (!a->resources.pci.irq) {
 | |
| 		DBG_ERR(("A: invalid irq"));
 | |
| 		return (-1);
 | |
| 	}
 | |
| 
 | |
| 	a->xdi_adapter.sdram_bar = a->resources.pci.bar[2];
 | |
| 
 | |
| 	/*
 | |
| 	  Map all MEMORY BAR's
 | |
| 	*/
 | |
| 	for (bar = 0; bar < 4; bar++) {
 | |
| 		if (bar != 1) {	/* ignore I/O */
 | |
| 			a->resources.pci.addr[bar] =
 | |
| 				divasa_remap_pci_bar(a, bar, a->resources.pci.bar[bar],
 | |
| 						     bar_length[bar]);
 | |
| 			if (!a->resources.pci.addr[bar]) {
 | |
| 				DBG_ERR(("A: 4BRI: can't map bar[%d]", bar))
 | |
| 					diva_4bri_cleanup_adapter(a);
 | |
| 				return (-1);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	  Register I/O port
 | |
| 	*/
 | |
| 	sprintf(&a->port_name[0], "DIVA 4BRI %ld", (long) a->xdi_adapter.serialNo);
 | |
| 
 | |
| 	if (diva_os_register_io_port(a, 1, a->resources.pci.bar[1],
 | |
| 				     bar_length[1], &a->port_name[0], 1)) {
 | |
| 		DBG_ERR(("A: 4BRI: can't register bar[1]"))
 | |
| 			diva_4bri_cleanup_adapter(a);
 | |
| 		return (-1);
 | |
| 	}
 | |
| 
 | |
| 	a->resources.pci.addr[1] =
 | |
| 		(void *) (unsigned long) a->resources.pci.bar[1];
 | |
| 
 | |
| 	/*
 | |
| 	  Set cleanup pointer for base adapter only, so slave adapter
 | |
| 	  will be unable to get cleanup
 | |
| 	*/
 | |
| 	a->interface.cleanup_adapter_proc = diva_4bri_cleanup_adapter;
 | |
| 
 | |
| 	/*
 | |
| 	  Create slave adapters
 | |
| 	*/
 | |
| 	if (tasks > 1) {
 | |
| 		if (!(a->slave_adapters[0] =
 | |
| 		      (diva_os_xdi_adapter_t *) diva_os_malloc(0, sizeof(*a))))
 | |
| 		{
 | |
| 			diva_4bri_cleanup_adapter(a);
 | |
| 			return (-1);
 | |
| 		}
 | |
| 		if (!(a->slave_adapters[1] =
 | |
| 		      (diva_os_xdi_adapter_t *) diva_os_malloc(0, sizeof(*a))))
 | |
| 		{
 | |
| 			diva_os_free(0, a->slave_adapters[0]);
 | |
| 			a->slave_adapters[0] = NULL;
 | |
| 			diva_4bri_cleanup_adapter(a);
 | |
| 			return (-1);
 | |
| 		}
 | |
| 		if (!(a->slave_adapters[2] =
 | |
| 		      (diva_os_xdi_adapter_t *) diva_os_malloc(0, sizeof(*a))))
 | |
| 		{
 | |
| 			diva_os_free(0, a->slave_adapters[0]);
 | |
| 			diva_os_free(0, a->slave_adapters[1]);
 | |
| 			a->slave_adapters[0] = NULL;
 | |
| 			a->slave_adapters[1] = NULL;
 | |
| 			diva_4bri_cleanup_adapter(a);
 | |
| 			return (-1);
 | |
| 		}
 | |
| 		memset(a->slave_adapters[0], 0x00, sizeof(*a));
 | |
| 		memset(a->slave_adapters[1], 0x00, sizeof(*a));
 | |
| 		memset(a->slave_adapters[2], 0x00, sizeof(*a));
 | |
| 	}
 | |
| 
 | |
| 	adapter_list[0] = a;
 | |
| 	adapter_list[1] = a->slave_adapters[0];
 | |
| 	adapter_list[2] = a->slave_adapters[1];
 | |
| 	adapter_list[3] = a->slave_adapters[2];
 | |
| 
 | |
| 	/*
 | |
| 	  Allocate slave list
 | |
| 	*/
 | |
| 	quadro_list =
 | |
| 		(PADAPTER_LIST_ENTRY) diva_os_malloc(0, sizeof(*quadro_list));
 | |
| 	if (!(a->slave_list = quadro_list)) {
 | |
| 		for (i = 0; i < (tasks - 1); i++) {
 | |
| 			diva_os_free(0, a->slave_adapters[i]);
 | |
| 			a->slave_adapters[i] = NULL;
 | |
| 		}
 | |
| 		diva_4bri_cleanup_adapter(a);
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	memset(quadro_list, 0x00, sizeof(*quadro_list));
 | |
| 
 | |
| 	/*
 | |
| 	  Set interfaces
 | |
| 	*/
 | |
| 	a->xdi_adapter.QuadroList = quadro_list;
 | |
| 	for (i = 0; i < tasks; i++) {
 | |
| 		adapter_list[i]->xdi_adapter.ControllerNumber = i;
 | |
| 		adapter_list[i]->xdi_adapter.tasks = tasks;
 | |
| 		quadro_list->QuadroAdapter[i] =
 | |
| 			&adapter_list[i]->xdi_adapter;
 | |
| 	}
 | |
| 
 | |
| 	for (i = 0; i < tasks; i++) {
 | |
| 		diva_current = adapter_list[i];
 | |
| 
 | |
| 		diva_current->dsp_mask = 0x00000003;
 | |
| 
 | |
| 		diva_current->xdi_adapter.a.io =
 | |
| 			&diva_current->xdi_adapter;
 | |
| 		diva_current->xdi_adapter.DIRequest = request;
 | |
| 		diva_current->interface.cmd_proc = diva_4bri_cmd_card_proc;
 | |
| 		diva_current->xdi_adapter.Properties =
 | |
| 			CardProperties[a->CardOrdinal];
 | |
| 		diva_current->CardOrdinal = a->CardOrdinal;
 | |
| 
 | |
| 		diva_current->xdi_adapter.Channels =
 | |
| 			CardProperties[a->CardOrdinal].Channels;
 | |
| 		diva_current->xdi_adapter.e_max =
 | |
| 			CardProperties[a->CardOrdinal].E_info;
 | |
| 		diva_current->xdi_adapter.e_tbl =
 | |
| 			diva_os_malloc(0,
 | |
| 				       diva_current->xdi_adapter.e_max *
 | |
| 				       sizeof(E_INFO));
 | |
| 
 | |
| 		if (!diva_current->xdi_adapter.e_tbl) {
 | |
| 			diva_4bri_cleanup_slave_adapters(a);
 | |
| 			diva_4bri_cleanup_adapter(a);
 | |
| 			for (i = 1; i < (tasks - 1); i++) {
 | |
| 				diva_os_free(0, adapter_list[i]);
 | |
| 			}
 | |
| 			return (-1);
 | |
| 		}
 | |
| 		memset(diva_current->xdi_adapter.e_tbl, 0x00,
 | |
| 		       diva_current->xdi_adapter.e_max * sizeof(E_INFO));
 | |
| 
 | |
| 		if (diva_os_initialize_spin_lock(&diva_current->xdi_adapter.isr_spin_lock, "isr")) {
 | |
| 			diva_4bri_cleanup_slave_adapters(a);
 | |
| 			diva_4bri_cleanup_adapter(a);
 | |
| 			for (i = 1; i < (tasks - 1); i++) {
 | |
| 				diva_os_free(0, adapter_list[i]);
 | |
| 			}
 | |
| 			return (-1);
 | |
| 		}
 | |
| 		if (diva_os_initialize_spin_lock(&diva_current->xdi_adapter.data_spin_lock, "data")) {
 | |
| 			diva_4bri_cleanup_slave_adapters(a);
 | |
| 			diva_4bri_cleanup_adapter(a);
 | |
| 			for (i = 1; i < (tasks - 1); i++) {
 | |
| 				diva_os_free(0, adapter_list[i]);
 | |
| 			}
 | |
| 			return (-1);
 | |
| 		}
 | |
| 
 | |
| 		strcpy(diva_current->xdi_adapter.req_soft_isr. dpc_thread_name, "kdivas4brid");
 | |
| 
 | |
| 		if (diva_os_initialize_soft_isr(&diva_current->xdi_adapter.req_soft_isr, DIDpcRoutine,
 | |
| 						&diva_current->xdi_adapter)) {
 | |
| 			diva_4bri_cleanup_slave_adapters(a);
 | |
| 			diva_4bri_cleanup_adapter(a);
 | |
| 			for (i = 1; i < (tasks - 1); i++) {
 | |
| 				diva_os_free(0, adapter_list[i]);
 | |
| 			}
 | |
| 			return (-1);
 | |
| 		}
 | |
| 
 | |
| 		/*
 | |
| 		  Do not initialize second DPC - only one thread will be created
 | |
| 		*/
 | |
| 		diva_current->xdi_adapter.isr_soft_isr.object =
 | |
| 			diva_current->xdi_adapter.req_soft_isr.object;
 | |
| 	}
 | |
| 
 | |
| 	if (v2) {
 | |
| 		prepare_qBri2_functions(&a->xdi_adapter);
 | |
| 	} else {
 | |
| 		prepare_qBri_functions(&a->xdi_adapter);
 | |
| 	}
 | |
| 
 | |
| 	for (i = 0; i < tasks; i++) {
 | |
| 		diva_current = adapter_list[i];
 | |
| 		if (i)
 | |
| 			memcpy(&diva_current->resources, &a->resources, sizeof(divas_card_resources_t));
 | |
| 		diva_current->resources.pci.qoffset = (a->xdi_adapter.MemorySize >> factor);
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	  Set up hardware related pointers
 | |
| 	*/
 | |
| 	a->xdi_adapter.cfg = (void *) (unsigned long) a->resources.pci.bar[0];	/* BAR0 CONFIG */
 | |
| 	a->xdi_adapter.port = (void *) (unsigned long) a->resources.pci.bar[1];	/* BAR1        */
 | |
| 	a->xdi_adapter.ctlReg = (void *) (unsigned long) a->resources.pci.bar[3];	/* BAR3 CNTRL  */
 | |
| 
 | |
| 	for (i = 0; i < tasks; i++) {
 | |
| 		diva_current = adapter_list[i];
 | |
| 		diva_4bri_set_addresses(diva_current);
 | |
| 		Slave = a->xdi_adapter.QuadroList->QuadroAdapter[i];
 | |
| 		Slave->MultiMaster = &a->xdi_adapter;
 | |
| 		Slave->sdram_bar = a->xdi_adapter.sdram_bar;
 | |
| 		if (i) {
 | |
| 			Slave->serialNo = ((dword) (Slave->ControllerNumber << 24)) |
 | |
| 				a->xdi_adapter.serialNo;
 | |
| 			Slave->cardType = a->xdi_adapter.cardType;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	  reset contains the base address for the PLX 9054 register set
 | |
| 	*/
 | |
| 	p = DIVA_OS_MEM_ATTACH_RESET(&a->xdi_adapter);
 | |
| 	WRITE_BYTE(&p[PLX9054_INTCSR], 0x00);	/* disable PCI interrupts */
 | |
| 	DIVA_OS_MEM_DETACH_RESET(&a->xdi_adapter, p);
 | |
| 
 | |
| 	/*
 | |
| 	  Set IRQ handler
 | |
| 	*/
 | |
| 	a->xdi_adapter.irq_info.irq_nr = a->resources.pci.irq;
 | |
| 	sprintf(a->xdi_adapter.irq_info.irq_name, "DIVA 4BRI %ld",
 | |
| 		(long) a->xdi_adapter.serialNo);
 | |
| 
 | |
| 	if (diva_os_register_irq(a, a->xdi_adapter.irq_info.irq_nr,
 | |
| 				 a->xdi_adapter.irq_info.irq_name)) {
 | |
| 		diva_4bri_cleanup_slave_adapters(a);
 | |
| 		diva_4bri_cleanup_adapter(a);
 | |
| 		for (i = 1; i < (tasks - 1); i++) {
 | |
| 			diva_os_free(0, adapter_list[i]);
 | |
| 		}
 | |
| 		return (-1);
 | |
| 	}
 | |
| 
 | |
| 	a->xdi_adapter.irq_info.registered = 1;
 | |
| 
 | |
| 	/*
 | |
| 	  Add three slave adapters
 | |
| 	*/
 | |
| 	if (tasks > 1) {
 | |
| 		diva_add_slave_adapter(adapter_list[1]);
 | |
| 		diva_add_slave_adapter(adapter_list[2]);
 | |
| 		diva_add_slave_adapter(adapter_list[3]);
 | |
| 	}
 | |
| 
 | |
| 	diva_log_info("%s IRQ:%d SerNo:%d", a->xdi_adapter.Properties.Name,
 | |
| 		      a->resources.pci.irq, a->xdi_adapter.serialNo);
 | |
| 
 | |
| 	return (0);
 | |
| }
 | |
| 
 | |
| /*
 | |
| **  Cleanup function will be called for master adapter only
 | |
| **  this is guaranteed by design: cleanup callback is set
 | |
| **  by master adapter only
 | |
| */
 | |
| static int diva_4bri_cleanup_adapter(diva_os_xdi_adapter_t *a)
 | |
| {
 | |
| 	int bar;
 | |
| 
 | |
| 	/*
 | |
| 	  Stop adapter if running
 | |
| 	*/
 | |
| 	if (a->xdi_adapter.Initialized) {
 | |
| 		diva_4bri_stop_adapter(a);
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	  Remove IRQ handler
 | |
| 	*/
 | |
| 	if (a->xdi_adapter.irq_info.registered) {
 | |
| 		diva_os_remove_irq(a, a->xdi_adapter.irq_info.irq_nr);
 | |
| 	}
 | |
| 	a->xdi_adapter.irq_info.registered = 0;
 | |
| 
 | |
| 	/*
 | |
| 	  Free DPC's and spin locks on all adapters
 | |
| 	*/
 | |
| 	diva_4bri_cleanup_slave_adapters(a);
 | |
| 
 | |
| 	/*
 | |
| 	  Unmap all BARS
 | |
| 	*/
 | |
| 	for (bar = 0; bar < 4; bar++) {
 | |
| 		if (bar != 1) {
 | |
| 			if (a->resources.pci.bar[bar]
 | |
| 			    && a->resources.pci.addr[bar]) {
 | |
| 				divasa_unmap_pci_bar(a->resources.pci.addr[bar]);
 | |
| 				a->resources.pci.bar[bar] = 0;
 | |
| 				a->resources.pci.addr[bar] = NULL;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	  Unregister I/O
 | |
| 	*/
 | |
| 	if (a->resources.pci.bar[1] && a->resources.pci.addr[1]) {
 | |
| 		diva_os_register_io_port(a, 0, a->resources.pci.bar[1],
 | |
| 					 _4bri_is_rev_2_card(a->
 | |
| 							     CardOrdinal) ?
 | |
| 					 _4bri_v2_bar_length[1] :
 | |
| 					 _4bri_bar_length[1],
 | |
| 					 &a->port_name[0], 1);
 | |
| 		a->resources.pci.bar[1] = 0;
 | |
| 		a->resources.pci.addr[1] = NULL;
 | |
| 	}
 | |
| 
 | |
| 	if (a->slave_list) {
 | |
| 		diva_os_free(0, a->slave_list);
 | |
| 		a->slave_list = NULL;
 | |
| 	}
 | |
| 
 | |
| 	return (0);
 | |
| }
 | |
| 
 | |
| static int _4bri_get_serial_number(diva_os_xdi_adapter_t *a)
 | |
| {
 | |
| 	dword data[64];
 | |
| 	dword serNo;
 | |
| 	word addr, status, i, j;
 | |
| 	byte Bus, Slot;
 | |
| 	void *hdev;
 | |
| 
 | |
| 	Bus = a->resources.pci.bus;
 | |
| 	Slot = a->resources.pci.func;
 | |
| 	hdev = a->resources.pci.hdev;
 | |
| 
 | |
| 	for (i = 0; i < 64; ++i) {
 | |
| 		addr = i * 4;
 | |
| 		for (j = 0; j < 5; ++j) {
 | |
| 			PCIwrite(Bus, Slot, 0x4E, &addr, sizeof(addr),
 | |
| 				 hdev);
 | |
| 			diva_os_wait(1);
 | |
| 			PCIread(Bus, Slot, 0x4E, &status, sizeof(status),
 | |
| 				hdev);
 | |
| 			if (status & 0x8000)
 | |
| 				break;
 | |
| 		}
 | |
| 		if (j >= 5) {
 | |
| 			DBG_ERR(("EEPROM[%d] read failed (0x%x)", i * 4, addr))
 | |
| 				return (0);
 | |
| 		}
 | |
| 		PCIread(Bus, Slot, 0x50, &data[i], sizeof(data[i]), hdev);
 | |
| 	}
 | |
| 	DBG_BLK(((char *) &data[0], sizeof(data)))
 | |
| 
 | |
| 		serNo = data[32];
 | |
| 	if (serNo == 0 || serNo == 0xffffffff)
 | |
| 		serNo = data[63];
 | |
| 
 | |
| 	if (!serNo) {
 | |
| 		DBG_LOG(("W: Serial Number == 0, create one serial number"));
 | |
| 		serNo = a->resources.pci.bar[1] & 0xffff0000;
 | |
| 		serNo |= a->resources.pci.bus << 8;
 | |
| 		serNo |= a->resources.pci.func;
 | |
| 	}
 | |
| 
 | |
| 	a->xdi_adapter.serialNo = serNo;
 | |
| 
 | |
| 	DBG_REG(("Serial No.          : %ld", a->xdi_adapter.serialNo))
 | |
| 
 | |
| 		return (serNo);
 | |
| }
 | |
| 
 | |
| /*
 | |
| **  Release resources of slave adapters
 | |
| */
 | |
| static int diva_4bri_cleanup_slave_adapters(diva_os_xdi_adapter_t *a)
 | |
| {
 | |
| 	diva_os_xdi_adapter_t *adapter_list[4];
 | |
| 	diva_os_xdi_adapter_t *diva_current;
 | |
| 	int i;
 | |
| 
 | |
| 	adapter_list[0] = a;
 | |
| 	adapter_list[1] = a->slave_adapters[0];
 | |
| 	adapter_list[2] = a->slave_adapters[1];
 | |
| 	adapter_list[3] = a->slave_adapters[2];
 | |
| 
 | |
| 	for (i = 0; i < a->xdi_adapter.tasks; i++) {
 | |
| 		diva_current = adapter_list[i];
 | |
| 		if (diva_current) {
 | |
| 			diva_os_destroy_spin_lock(&diva_current->
 | |
| 						  xdi_adapter.
 | |
| 						  isr_spin_lock, "unload");
 | |
| 			diva_os_destroy_spin_lock(&diva_current->
 | |
| 						  xdi_adapter.
 | |
| 						  data_spin_lock,
 | |
| 						  "unload");
 | |
| 
 | |
| 			diva_os_cancel_soft_isr(&diva_current->xdi_adapter.
 | |
| 						req_soft_isr);
 | |
| 			diva_os_cancel_soft_isr(&diva_current->xdi_adapter.
 | |
| 						isr_soft_isr);
 | |
| 
 | |
| 			diva_os_remove_soft_isr(&diva_current->xdi_adapter.
 | |
| 						req_soft_isr);
 | |
| 			diva_current->xdi_adapter.isr_soft_isr.object = NULL;
 | |
| 
 | |
| 			if (diva_current->xdi_adapter.e_tbl) {
 | |
| 				diva_os_free(0,
 | |
| 					     diva_current->xdi_adapter.
 | |
| 					     e_tbl);
 | |
| 			}
 | |
| 			diva_current->xdi_adapter.e_tbl = NULL;
 | |
| 			diva_current->xdi_adapter.e_max = 0;
 | |
| 			diva_current->xdi_adapter.e_count = 0;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return (0);
 | |
| }
 | |
| 
 | |
| static int
 | |
| diva_4bri_cmd_card_proc(struct _diva_os_xdi_adapter *a,
 | |
| 			diva_xdi_um_cfg_cmd_t *cmd, int length)
 | |
| {
 | |
| 	int ret = -1;
 | |
| 
 | |
| 	if (cmd->adapter != a->controller) {
 | |
| 		DBG_ERR(("A: 4bri_cmd, invalid controller=%d != %d",
 | |
| 			 cmd->adapter, a->controller))
 | |
| 			return (-1);
 | |
| 	}
 | |
| 
 | |
| 	switch (cmd->command) {
 | |
| 	case DIVA_XDI_UM_CMD_GET_CARD_ORDINAL:
 | |
| 		a->xdi_mbox.data_length = sizeof(dword);
 | |
| 		a->xdi_mbox.data =
 | |
| 			diva_os_malloc(0, a->xdi_mbox.data_length);
 | |
| 		if (a->xdi_mbox.data) {
 | |
| 			*(dword *) a->xdi_mbox.data =
 | |
| 				(dword) a->CardOrdinal;
 | |
| 			a->xdi_mbox.status = DIVA_XDI_MBOX_BUSY;
 | |
| 			ret = 0;
 | |
| 		}
 | |
| 		break;
 | |
| 
 | |
| 	case DIVA_XDI_UM_CMD_GET_SERIAL_NR:
 | |
| 		a->xdi_mbox.data_length = sizeof(dword);
 | |
| 		a->xdi_mbox.data =
 | |
| 			diva_os_malloc(0, a->xdi_mbox.data_length);
 | |
| 		if (a->xdi_mbox.data) {
 | |
| 			*(dword *) a->xdi_mbox.data =
 | |
| 				(dword) a->xdi_adapter.serialNo;
 | |
| 			a->xdi_mbox.status = DIVA_XDI_MBOX_BUSY;
 | |
| 			ret = 0;
 | |
| 		}
 | |
| 		break;
 | |
| 
 | |
| 	case DIVA_XDI_UM_CMD_GET_PCI_HW_CONFIG:
 | |
| 		if (!a->xdi_adapter.ControllerNumber) {
 | |
| 			/*
 | |
| 			  Only master adapter can access hardware config
 | |
| 			*/
 | |
| 			a->xdi_mbox.data_length = sizeof(dword) * 9;
 | |
| 			a->xdi_mbox.data =
 | |
| 				diva_os_malloc(0, a->xdi_mbox.data_length);
 | |
| 			if (a->xdi_mbox.data) {
 | |
| 				int i;
 | |
| 				dword *data = (dword *) a->xdi_mbox.data;
 | |
| 
 | |
| 				for (i = 0; i < 8; i++) {
 | |
| 					*data++ = a->resources.pci.bar[i];
 | |
| 				}
 | |
| 				*data++ = (dword) a->resources.pci.irq;
 | |
| 				a->xdi_mbox.status = DIVA_XDI_MBOX_BUSY;
 | |
| 				ret = 0;
 | |
| 			}
 | |
| 		}
 | |
| 		break;
 | |
| 
 | |
| 	case DIVA_XDI_UM_CMD_GET_CARD_STATE:
 | |
| 		if (!a->xdi_adapter.ControllerNumber) {
 | |
| 			a->xdi_mbox.data_length = sizeof(dword);
 | |
| 			a->xdi_mbox.data =
 | |
| 				diva_os_malloc(0, a->xdi_mbox.data_length);
 | |
| 			if (a->xdi_mbox.data) {
 | |
| 				dword *data = (dword *) a->xdi_mbox.data;
 | |
| 				if (!a->xdi_adapter.ram
 | |
| 				    || !a->xdi_adapter.reset
 | |
| 				    || !a->xdi_adapter.cfg) {
 | |
| 					*data = 3;
 | |
| 				} else if (a->xdi_adapter.trapped) {
 | |
| 					*data = 2;
 | |
| 				} else if (a->xdi_adapter.Initialized) {
 | |
| 					*data = 1;
 | |
| 				} else {
 | |
| 					*data = 0;
 | |
| 				}
 | |
| 				a->xdi_mbox.status = DIVA_XDI_MBOX_BUSY;
 | |
| 				ret = 0;
 | |
| 			}
 | |
| 		}
 | |
| 		break;
 | |
| 
 | |
| 	case DIVA_XDI_UM_CMD_WRITE_FPGA:
 | |
| 		if (!a->xdi_adapter.ControllerNumber) {
 | |
| 			ret =
 | |
| 				diva_4bri_write_fpga_image(a,
 | |
| 							   (byte *)&cmd[1],
 | |
| 							   cmd->command_data.
 | |
| 							   write_fpga.
 | |
| 							   image_length);
 | |
| 		}
 | |
| 		break;
 | |
| 
 | |
| 	case DIVA_XDI_UM_CMD_RESET_ADAPTER:
 | |
| 		if (!a->xdi_adapter.ControllerNumber) {
 | |
| 			ret = diva_4bri_reset_adapter(&a->xdi_adapter);
 | |
| 		}
 | |
| 		break;
 | |
| 
 | |
| 	case DIVA_XDI_UM_CMD_WRITE_SDRAM_BLOCK:
 | |
| 		if (!a->xdi_adapter.ControllerNumber) {
 | |
| 			ret = diva_4bri_write_sdram_block(&a->xdi_adapter,
 | |
| 							  cmd->
 | |
| 							  command_data.
 | |
| 							  write_sdram.
 | |
| 							  offset,
 | |
| 							  (byte *) &
 | |
| 							  cmd[1],
 | |
| 							  cmd->
 | |
| 							  command_data.
 | |
| 							  write_sdram.
 | |
| 							  length,
 | |
| 							  a->xdi_adapter.
 | |
| 							  MemorySize);
 | |
| 		}
 | |
| 		break;
 | |
| 
 | |
| 	case DIVA_XDI_UM_CMD_START_ADAPTER:
 | |
| 		if (!a->xdi_adapter.ControllerNumber) {
 | |
| 			ret = diva_4bri_start_adapter(&a->xdi_adapter,
 | |
| 						      cmd->command_data.
 | |
| 						      start.offset,
 | |
| 						      cmd->command_data.
 | |
| 						      start.features);
 | |
| 		}
 | |
| 		break;
 | |
| 
 | |
| 	case DIVA_XDI_UM_CMD_SET_PROTOCOL_FEATURES:
 | |
| 		if (!a->xdi_adapter.ControllerNumber) {
 | |
| 			a->xdi_adapter.features =
 | |
| 				cmd->command_data.features.features;
 | |
| 			a->xdi_adapter.a.protocol_capabilities =
 | |
| 				a->xdi_adapter.features;
 | |
| 			DBG_TRC(("Set raw protocol features (%08x)",
 | |
| 				 a->xdi_adapter.features))
 | |
| 				ret = 0;
 | |
| 		}
 | |
| 		break;
 | |
| 
 | |
| 	case DIVA_XDI_UM_CMD_STOP_ADAPTER:
 | |
| 		if (!a->xdi_adapter.ControllerNumber) {
 | |
| 			ret = diva_4bri_stop_adapter(a);
 | |
| 		}
 | |
| 		break;
 | |
| 
 | |
| 	case DIVA_XDI_UM_CMD_READ_XLOG_ENTRY:
 | |
| 		ret = diva_card_read_xlog(a);
 | |
| 		break;
 | |
| 
 | |
| 	case DIVA_XDI_UM_CMD_READ_SDRAM:
 | |
| 		if (!a->xdi_adapter.ControllerNumber
 | |
| 		    && a->xdi_adapter.Address) {
 | |
| 			if (
 | |
| 				(a->xdi_mbox.data_length =
 | |
| 				 cmd->command_data.read_sdram.length)) {
 | |
| 				if (
 | |
| 					(a->xdi_mbox.data_length +
 | |
| 					 cmd->command_data.read_sdram.offset) <
 | |
| 					a->xdi_adapter.MemorySize) {
 | |
| 					a->xdi_mbox.data =
 | |
| 						diva_os_malloc(0,
 | |
| 							       a->xdi_mbox.
 | |
| 							       data_length);
 | |
| 					if (a->xdi_mbox.data) {
 | |
| 						byte __iomem *p = DIVA_OS_MEM_ATTACH_ADDRESS(&a->xdi_adapter);
 | |
| 						byte __iomem *src = p;
 | |
| 						byte *dst = a->xdi_mbox.data;
 | |
| 						dword len = a->xdi_mbox.data_length;
 | |
| 
 | |
| 						src += cmd->command_data.read_sdram.offset;
 | |
| 
 | |
| 						while (len--) {
 | |
| 							*dst++ = READ_BYTE(src++);
 | |
| 						}
 | |
| 						DIVA_OS_MEM_DETACH_ADDRESS(&a->xdi_adapter, p);
 | |
| 						a->xdi_mbox.status = DIVA_XDI_MBOX_BUSY;
 | |
| 						ret = 0;
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		break;
 | |
| 
 | |
| 	default:
 | |
| 		DBG_ERR(("A: A(%d) invalid cmd=%d", a->controller,
 | |
| 			 cmd->command))
 | |
| 			}
 | |
| 
 | |
| 	return (ret);
 | |
| }
 | |
| 
 | |
| void *xdiLoadFile(char *FileName, dword *FileLength,
 | |
| 		  unsigned long lim)
 | |
| {
 | |
| 	void *ret = diva_xdiLoadFileFile;
 | |
| 
 | |
| 	if (FileLength) {
 | |
| 		*FileLength = diva_xdiLoadFileLength;
 | |
| 	}
 | |
| 	diva_xdiLoadFileFile = NULL;
 | |
| 	diva_xdiLoadFileLength = 0;
 | |
| 
 | |
| 	return (ret);
 | |
| }
 | |
| 
 | |
| void diva_os_set_qBri_functions(PISDN_ADAPTER IoAdapter)
 | |
| {
 | |
| }
 | |
| 
 | |
| void diva_os_set_qBri2_functions(PISDN_ADAPTER IoAdapter)
 | |
| {
 | |
| }
 | |
| 
 | |
| static int
 | |
| diva_4bri_write_fpga_image(diva_os_xdi_adapter_t *a, byte *data,
 | |
| 			   dword length)
 | |
| {
 | |
| 	int ret;
 | |
| 
 | |
| 	diva_xdiLoadFileFile = data;
 | |
| 	diva_xdiLoadFileLength = length;
 | |
| 
 | |
| 	ret = qBri_FPGA_download(&a->xdi_adapter);
 | |
| 
 | |
| 	diva_xdiLoadFileFile = NULL;
 | |
| 	diva_xdiLoadFileLength = 0;
 | |
| 
 | |
| 	return (ret ? 0 : -1);
 | |
| }
 | |
| 
 | |
| static int diva_4bri_reset_adapter(PISDN_ADAPTER IoAdapter)
 | |
| {
 | |
| 	PISDN_ADAPTER Slave;
 | |
| 	int i;
 | |
| 
 | |
| 	if (!IoAdapter->Address || !IoAdapter->reset) {
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	if (IoAdapter->Initialized) {
 | |
| 		DBG_ERR(("A: A(%d) can't reset 4BRI adapter - please stop first",
 | |
| 			 IoAdapter->ANum))
 | |
| 			return (-1);
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	  Forget all entities on all adapters
 | |
| 	*/
 | |
| 	for (i = 0; ((i < IoAdapter->tasks) && IoAdapter->QuadroList); i++) {
 | |
| 		Slave = IoAdapter->QuadroList->QuadroAdapter[i];
 | |
| 		Slave->e_count = 0;
 | |
| 		if (Slave->e_tbl) {
 | |
| 			memset(Slave->e_tbl, 0x00,
 | |
| 			       Slave->e_max * sizeof(E_INFO));
 | |
| 		}
 | |
| 		Slave->head = 0;
 | |
| 		Slave->tail = 0;
 | |
| 		Slave->assign = 0;
 | |
| 		Slave->trapped = 0;
 | |
| 
 | |
| 		memset(&Slave->a.IdTable[0], 0x00,
 | |
| 		       sizeof(Slave->a.IdTable));
 | |
| 		memset(&Slave->a.IdTypeTable[0], 0x00,
 | |
| 		       sizeof(Slave->a.IdTypeTable));
 | |
| 		memset(&Slave->a.FlowControlIdTable[0], 0x00,
 | |
| 		       sizeof(Slave->a.FlowControlIdTable));
 | |
| 		memset(&Slave->a.FlowControlSkipTable[0], 0x00,
 | |
| 		       sizeof(Slave->a.FlowControlSkipTable));
 | |
| 		memset(&Slave->a.misc_flags_table[0], 0x00,
 | |
| 		       sizeof(Slave->a.misc_flags_table));
 | |
| 		memset(&Slave->a.rx_stream[0], 0x00,
 | |
| 		       sizeof(Slave->a.rx_stream));
 | |
| 		memset(&Slave->a.tx_stream[0], 0x00,
 | |
| 		       sizeof(Slave->a.tx_stream));
 | |
| 		memset(&Slave->a.tx_pos[0], 0x00, sizeof(Slave->a.tx_pos));
 | |
| 		memset(&Slave->a.rx_pos[0], 0x00, sizeof(Slave->a.rx_pos));
 | |
| 	}
 | |
| 
 | |
| 	return (0);
 | |
| }
 | |
| 
 | |
| 
 | |
| static int
 | |
| diva_4bri_write_sdram_block(PISDN_ADAPTER IoAdapter,
 | |
| 			    dword address,
 | |
| 			    const byte *data, dword length, dword limit)
 | |
| {
 | |
| 	byte __iomem *p = DIVA_OS_MEM_ATTACH_ADDRESS(IoAdapter);
 | |
| 	byte __iomem *mem = p;
 | |
| 
 | |
| 	if (((address + length) >= limit) || !mem) {
 | |
| 		DIVA_OS_MEM_DETACH_ADDRESS(IoAdapter, p);
 | |
| 		DBG_ERR(("A: A(%d) write 4BRI address=0x%08lx",
 | |
| 			 IoAdapter->ANum, address + length))
 | |
| 			return (-1);
 | |
| 	}
 | |
| 	mem += address;
 | |
| 
 | |
| 	while (length--) {
 | |
| 		WRITE_BYTE(mem++, *data++);
 | |
| 	}
 | |
| 
 | |
| 	DIVA_OS_MEM_DETACH_ADDRESS(IoAdapter, p);
 | |
| 	return (0);
 | |
| }
 | |
| 
 | |
| static int
 | |
| diva_4bri_start_adapter(PISDN_ADAPTER IoAdapter,
 | |
| 			dword start_address, dword features)
 | |
| {
 | |
| 	volatile word __iomem *signature;
 | |
| 	int started = 0;
 | |
| 	int i;
 | |
| 	byte __iomem *p;
 | |
| 
 | |
| 	/*
 | |
| 	  start adapter
 | |
| 	*/
 | |
| 	start_qBri_hardware(IoAdapter);
 | |
| 
 | |
| 	p = DIVA_OS_MEM_ATTACH_RAM(IoAdapter);
 | |
| 	/*
 | |
| 	  wait for signature in shared memory (max. 3 seconds)
 | |
| 	*/
 | |
| 	signature = (volatile word __iomem *) (&p[0x1E]);
 | |
| 
 | |
| 	for (i = 0; i < 300; ++i) {
 | |
| 		diva_os_wait(10);
 | |
| 		if (READ_WORD(&signature[0]) == 0x4447) {
 | |
| 			DBG_TRC(("Protocol startup time %d.%02d seconds",
 | |
| 				 (i / 100), (i % 100)))
 | |
| 				started = 1;
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	for (i = 1; i < IoAdapter->tasks; i++) {
 | |
| 		IoAdapter->QuadroList->QuadroAdapter[i]->features =
 | |
| 			IoAdapter->features;
 | |
| 		IoAdapter->QuadroList->QuadroAdapter[i]->a.
 | |
| 			protocol_capabilities = IoAdapter->features;
 | |
| 	}
 | |
| 
 | |
| 	if (!started) {
 | |
| 		DBG_FTL(("%s: Adapter selftest failed, signature=%04x",
 | |
| 			 IoAdapter->Properties.Name,
 | |
| 			 READ_WORD(&signature[0])))
 | |
| 			DIVA_OS_MEM_DETACH_RAM(IoAdapter, p);
 | |
| 		(*(IoAdapter->trapFnc)) (IoAdapter);
 | |
| 		IoAdapter->stop(IoAdapter);
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	DIVA_OS_MEM_DETACH_RAM(IoAdapter, p);
 | |
| 
 | |
| 	for (i = 0; i < IoAdapter->tasks; i++) {
 | |
| 		IoAdapter->QuadroList->QuadroAdapter[i]->Initialized = 1;
 | |
| 		IoAdapter->QuadroList->QuadroAdapter[i]->IrqCount = 0;
 | |
| 	}
 | |
| 
 | |
| 	if (check_qBri_interrupt(IoAdapter)) {
 | |
| 		DBG_ERR(("A: A(%d) interrupt test failed",
 | |
| 			 IoAdapter->ANum))
 | |
| 			for (i = 0; i < IoAdapter->tasks; i++) {
 | |
| 				IoAdapter->QuadroList->QuadroAdapter[i]->Initialized = 0;
 | |
| 			}
 | |
| 		IoAdapter->stop(IoAdapter);
 | |
| 		return (-1);
 | |
| 	}
 | |
| 
 | |
| 	IoAdapter->Properties.Features = (word) features;
 | |
| 	diva_xdi_display_adapter_features(IoAdapter->ANum);
 | |
| 
 | |
| 	for (i = 0; i < IoAdapter->tasks; i++) {
 | |
| 		DBG_LOG(("A(%d) %s adapter successfully started",
 | |
| 			 IoAdapter->QuadroList->QuadroAdapter[i]->ANum,
 | |
| 			 (IoAdapter->tasks == 1) ? "BRI 2.0" : "4BRI"))
 | |
| 			diva_xdi_didd_register_adapter(IoAdapter->QuadroList->QuadroAdapter[i]->ANum);
 | |
| 		IoAdapter->QuadroList->QuadroAdapter[i]->Properties.Features = (word) features;
 | |
| 	}
 | |
| 
 | |
| 	return (0);
 | |
| }
 | |
| 
 | |
| static int check_qBri_interrupt(PISDN_ADAPTER IoAdapter)
 | |
| {
 | |
| #ifdef	SUPPORT_INTERRUPT_TEST_ON_4BRI
 | |
| 	int i;
 | |
| 	ADAPTER *a = &IoAdapter->a;
 | |
| 	byte __iomem *p;
 | |
| 
 | |
| 	IoAdapter->IrqCount = 0;
 | |
| 
 | |
| 	if (IoAdapter->ControllerNumber > 0)
 | |
| 		return (-1);
 | |
| 
 | |
| 	p = DIVA_OS_MEM_ATTACH_RESET(IoAdapter);
 | |
| 	WRITE_BYTE(&p[PLX9054_INTCSR], PLX9054_INT_ENABLE);
 | |
| 	DIVA_OS_MEM_DETACH_RESET(IoAdapter, p);
 | |
| 	/*
 | |
| 	  interrupt test
 | |
| 	*/
 | |
| 	a->ReadyInt = 1;
 | |
| 	a->ram_out(a, &PR_RAM->ReadyInt, 1);
 | |
| 
 | |
| 	for (i = 100; !IoAdapter->IrqCount && (i-- > 0); diva_os_wait(10));
 | |
| 
 | |
| 	return ((IoAdapter->IrqCount > 0) ? 0 : -1);
 | |
| #else
 | |
| 	dword volatile __iomem *qBriIrq;
 | |
| 	byte __iomem *p;
 | |
| 	/*
 | |
| 	  Reset on-board interrupt register
 | |
| 	*/
 | |
| 	IoAdapter->IrqCount = 0;
 | |
| 	p = DIVA_OS_MEM_ATTACH_CTLREG(IoAdapter);
 | |
| 	qBriIrq = (dword volatile __iomem *) (&p[_4bri_is_rev_2_card
 | |
| 						 (IoAdapter->
 | |
| 						  cardType) ? (MQ2_BREG_IRQ_TEST)
 | |
| 						 : (MQ_BREG_IRQ_TEST)]);
 | |
| 
 | |
| 	WRITE_DWORD(qBriIrq, MQ_IRQ_REQ_OFF);
 | |
| 	DIVA_OS_MEM_DETACH_CTLREG(IoAdapter, p);
 | |
| 
 | |
| 	p = DIVA_OS_MEM_ATTACH_RESET(IoAdapter);
 | |
| 	WRITE_BYTE(&p[PLX9054_INTCSR], PLX9054_INT_ENABLE);
 | |
| 	DIVA_OS_MEM_DETACH_RESET(IoAdapter, p);
 | |
| 
 | |
| 	diva_os_wait(100);
 | |
| 
 | |
| 	return (0);
 | |
| #endif				/* SUPPORT_INTERRUPT_TEST_ON_4BRI */
 | |
| }
 | |
| 
 | |
| static void diva_4bri_clear_interrupts(diva_os_xdi_adapter_t *a)
 | |
| {
 | |
| 	PISDN_ADAPTER IoAdapter = &a->xdi_adapter;
 | |
| 
 | |
| 	/*
 | |
| 	  clear any pending interrupt
 | |
| 	*/
 | |
| 	IoAdapter->disIrq(IoAdapter);
 | |
| 
 | |
| 	IoAdapter->tst_irq(&IoAdapter->a);
 | |
| 	IoAdapter->clr_irq(&IoAdapter->a);
 | |
| 	IoAdapter->tst_irq(&IoAdapter->a);
 | |
| 
 | |
| 	/*
 | |
| 	  kill pending dpcs
 | |
| 	*/
 | |
| 	diva_os_cancel_soft_isr(&IoAdapter->req_soft_isr);
 | |
| 	diva_os_cancel_soft_isr(&IoAdapter->isr_soft_isr);
 | |
| }
 | |
| 
 | |
| static int diva_4bri_stop_adapter(diva_os_xdi_adapter_t *a)
 | |
| {
 | |
| 	PISDN_ADAPTER IoAdapter = &a->xdi_adapter;
 | |
| 	int i;
 | |
| 
 | |
| 	if (!IoAdapter->ram) {
 | |
| 		return (-1);
 | |
| 	}
 | |
| 
 | |
| 	if (!IoAdapter->Initialized) {
 | |
| 		DBG_ERR(("A: A(%d) can't stop PRI adapter - not running",
 | |
| 			 IoAdapter->ANum))
 | |
| 			return (-1);	/* nothing to stop */
 | |
| 	}
 | |
| 
 | |
| 	for (i = 0; i < IoAdapter->tasks; i++) {
 | |
| 		IoAdapter->QuadroList->QuadroAdapter[i]->Initialized = 0;
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	  Disconnect Adapters from DIDD
 | |
| 	*/
 | |
| 	for (i = 0; i < IoAdapter->tasks; i++) {
 | |
| 		diva_xdi_didd_remove_adapter(IoAdapter->QuadroList->QuadroAdapter[i]->ANum);
 | |
| 	}
 | |
| 
 | |
| 	i = 100;
 | |
| 
 | |
| 	/*
 | |
| 	  Stop interrupts
 | |
| 	*/
 | |
| 	a->clear_interrupts_proc = diva_4bri_clear_interrupts;
 | |
| 	IoAdapter->a.ReadyInt = 1;
 | |
| 	IoAdapter->a.ram_inc(&IoAdapter->a, &PR_RAM->ReadyInt);
 | |
| 	do {
 | |
| 		diva_os_sleep(10);
 | |
| 	} while (i-- && a->clear_interrupts_proc);
 | |
| 
 | |
| 	if (a->clear_interrupts_proc) {
 | |
| 		diva_4bri_clear_interrupts(a);
 | |
| 		a->clear_interrupts_proc = NULL;
 | |
| 		DBG_ERR(("A: A(%d) no final interrupt from 4BRI adapter",
 | |
| 			 IoAdapter->ANum))
 | |
| 			}
 | |
| 	IoAdapter->a.ReadyInt = 0;
 | |
| 
 | |
| 	/*
 | |
| 	  Stop and reset adapter
 | |
| 	*/
 | |
| 	IoAdapter->stop(IoAdapter);
 | |
| 
 | |
| 	return (0);
 | |
| }
 | 
