isdn source code uses a not-current coding style. Update the coding style used on a per-line basis so that git diff -w shows only elided blank lines at EOF. Done with emacs and some scripts and some typing. Built x86 allyesconfig. No detected change in objdump -d or size. Signed-off-by: Joe Perches <joe@perches.com>
		
			
				
	
	
		
			657 lines
		
	
	
	
		
			17 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			657 lines
		
	
	
	
		
			17 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
/* $Id: diva.c,v 1.21.4.1 2004/05/08 14:33:43 armin Exp $ */
 | 
						|
 | 
						|
#define CARDTYPE_H_WANT_DATA            1
 | 
						|
#define CARDTYPE_H_WANT_IDI_DATA        0
 | 
						|
#define CARDTYPE_H_WANT_RESOURCE_DATA   0
 | 
						|
#define CARDTYPE_H_WANT_FILE_DATA       0
 | 
						|
 | 
						|
#include "platform.h"
 | 
						|
#include "debuglib.h"
 | 
						|
#include "cardtype.h"
 | 
						|
#include "pc.h"
 | 
						|
#include "di_defs.h"
 | 
						|
#include "di.h"
 | 
						|
#include "io.h"
 | 
						|
#include "pc_maint.h"
 | 
						|
#include "xdi_msg.h"
 | 
						|
#include "xdi_adapter.h"
 | 
						|
#include "diva_pci.h"
 | 
						|
#include "diva.h"
 | 
						|
 | 
						|
#ifdef CONFIG_ISDN_DIVAS_PRIPCI
 | 
						|
#include "os_pri.h"
 | 
						|
#endif
 | 
						|
#ifdef CONFIG_ISDN_DIVAS_BRIPCI
 | 
						|
#include "os_bri.h"
 | 
						|
#include "os_4bri.h"
 | 
						|
#endif
 | 
						|
 | 
						|
PISDN_ADAPTER IoAdapters[MAX_ADAPTER];
 | 
						|
extern IDI_CALL Requests[MAX_ADAPTER];
 | 
						|
extern int create_adapter_proc(diva_os_xdi_adapter_t *a);
 | 
						|
extern void remove_adapter_proc(diva_os_xdi_adapter_t *a);
 | 
						|
 | 
						|
#define DivaIdiReqFunc(N)						\
 | 
						|
	static void DivaIdiRequest##N(ENTITY *e)			\
 | 
						|
	{ if (IoAdapters[N]) (*IoAdapters[N]->DIRequest)(IoAdapters[N], e); }
 | 
						|
 | 
						|
/*
 | 
						|
**  Create own 32 Adapters
 | 
						|
*/
 | 
						|
DivaIdiReqFunc(0)
 | 
						|
DivaIdiReqFunc(1)
 | 
						|
DivaIdiReqFunc(2)
 | 
						|
DivaIdiReqFunc(3)
 | 
						|
DivaIdiReqFunc(4)
 | 
						|
DivaIdiReqFunc(5)
 | 
						|
DivaIdiReqFunc(6)
 | 
						|
DivaIdiReqFunc(7)
 | 
						|
DivaIdiReqFunc(8)
 | 
						|
DivaIdiReqFunc(9)
 | 
						|
DivaIdiReqFunc(10)
 | 
						|
DivaIdiReqFunc(11)
 | 
						|
DivaIdiReqFunc(12)
 | 
						|
DivaIdiReqFunc(13)
 | 
						|
DivaIdiReqFunc(14)
 | 
						|
DivaIdiReqFunc(15)
 | 
						|
DivaIdiReqFunc(16)
 | 
						|
DivaIdiReqFunc(17)
 | 
						|
DivaIdiReqFunc(18)
 | 
						|
DivaIdiReqFunc(19)
 | 
						|
DivaIdiReqFunc(20)
 | 
						|
DivaIdiReqFunc(21)
 | 
						|
DivaIdiReqFunc(22)
 | 
						|
DivaIdiReqFunc(23)
 | 
						|
DivaIdiReqFunc(24)
 | 
						|
DivaIdiReqFunc(25)
 | 
						|
DivaIdiReqFunc(26)
 | 
						|
DivaIdiReqFunc(27)
 | 
						|
DivaIdiReqFunc(28)
 | 
						|
DivaIdiReqFunc(29)
 | 
						|
DivaIdiReqFunc(30)
 | 
						|
DivaIdiReqFunc(31)
 | 
						|
 | 
						|
/*
 | 
						|
**  LOCALS
 | 
						|
*/
 | 
						|
static LIST_HEAD(adapter_queue);
 | 
						|
 | 
						|
typedef struct _diva_get_xlog {
 | 
						|
	word command;
 | 
						|
	byte req;
 | 
						|
	byte rc;
 | 
						|
	byte data[sizeof(struct mi_pc_maint)];
 | 
						|
} diva_get_xlog_t;
 | 
						|
 | 
						|
typedef struct _diva_supported_cards_info {
 | 
						|
	int CardOrdinal;
 | 
						|
	diva_init_card_proc_t init_card;
 | 
						|
} diva_supported_cards_info_t;
 | 
						|
 | 
						|
static diva_supported_cards_info_t divas_supported_cards[] = {
 | 
						|
#ifdef CONFIG_ISDN_DIVAS_PRIPCI
 | 
						|
	/*
 | 
						|
	  PRI Cards
 | 
						|
	*/
 | 
						|
	{CARDTYPE_DIVASRV_P_30M_PCI, diva_pri_init_card},
 | 
						|
	/*
 | 
						|
	  PRI Rev.2 Cards
 | 
						|
	*/
 | 
						|
	{CARDTYPE_DIVASRV_P_30M_V2_PCI, diva_pri_init_card},
 | 
						|
	/*
 | 
						|
	  PRI Rev.2 VoIP Cards
 | 
						|
	*/
 | 
						|
	{CARDTYPE_DIVASRV_VOICE_P_30M_V2_PCI, diva_pri_init_card},
 | 
						|
#endif
 | 
						|
#ifdef CONFIG_ISDN_DIVAS_BRIPCI
 | 
						|
	/*
 | 
						|
	  4BRI Rev 1 Cards
 | 
						|
	*/
 | 
						|
	{CARDTYPE_DIVASRV_Q_8M_PCI, diva_4bri_init_card},
 | 
						|
	{CARDTYPE_DIVASRV_VOICE_Q_8M_PCI, diva_4bri_init_card},
 | 
						|
	/*
 | 
						|
	  4BRI Rev 2 Cards
 | 
						|
	*/
 | 
						|
	{CARDTYPE_DIVASRV_Q_8M_V2_PCI, diva_4bri_init_card},
 | 
						|
	{CARDTYPE_DIVASRV_VOICE_Q_8M_V2_PCI, diva_4bri_init_card},
 | 
						|
	/*
 | 
						|
	  4BRI Based BRI Rev 2 Cards
 | 
						|
	*/
 | 
						|
	{CARDTYPE_DIVASRV_B_2M_V2_PCI, diva_4bri_init_card},
 | 
						|
	{CARDTYPE_DIVASRV_B_2F_PCI, diva_4bri_init_card},
 | 
						|
	{CARDTYPE_DIVASRV_VOICE_B_2M_V2_PCI, diva_4bri_init_card},
 | 
						|
	/*
 | 
						|
	  BRI
 | 
						|
	*/
 | 
						|
	{CARDTYPE_MAESTRA_PCI, diva_bri_init_card},
 | 
						|
#endif
 | 
						|
 | 
						|
	/*
 | 
						|
	  EOL
 | 
						|
	*/
 | 
						|
	{-1}
 | 
						|
};
 | 
						|
 | 
						|
static void diva_init_request_array(void);
 | 
						|
static void *divas_create_pci_card(int handle, void *pci_dev_handle);
 | 
						|
 | 
						|
static diva_os_spin_lock_t adapter_lock;
 | 
						|
 | 
						|
static int diva_find_free_adapters(int base, int nr)
 | 
						|
{
 | 
						|
	int i;
 | 
						|
 | 
						|
	for (i = 0; i < nr; i++) {
 | 
						|
		if (IoAdapters[base + i]) {
 | 
						|
			return (-1);
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return (0);
 | 
						|
}
 | 
						|
 | 
						|
static diva_os_xdi_adapter_t *diva_q_get_next(struct list_head *what)
 | 
						|
{
 | 
						|
	diva_os_xdi_adapter_t *a = NULL;
 | 
						|
 | 
						|
	if (what && (what->next != &adapter_queue))
 | 
						|
		a = list_entry(what->next, diva_os_xdi_adapter_t, link);
 | 
						|
 | 
						|
	return (a);
 | 
						|
}
 | 
						|
 | 
						|
/* --------------------------------------------------------------------------
 | 
						|
   Add card to the card list
 | 
						|
   -------------------------------------------------------------------------- */
 | 
						|
void *diva_driver_add_card(void *pdev, unsigned long CardOrdinal)
 | 
						|
{
 | 
						|
	diva_os_spin_lock_magic_t old_irql;
 | 
						|
	diva_os_xdi_adapter_t *pdiva, *pa;
 | 
						|
	int i, j, max, nr;
 | 
						|
 | 
						|
	for (i = 0; divas_supported_cards[i].CardOrdinal != -1; i++) {
 | 
						|
		if (divas_supported_cards[i].CardOrdinal == CardOrdinal) {
 | 
						|
			if (!(pdiva = divas_create_pci_card(i, pdev))) {
 | 
						|
				return NULL;
 | 
						|
			}
 | 
						|
			switch (CardOrdinal) {
 | 
						|
			case CARDTYPE_DIVASRV_Q_8M_PCI:
 | 
						|
			case CARDTYPE_DIVASRV_VOICE_Q_8M_PCI:
 | 
						|
			case CARDTYPE_DIVASRV_Q_8M_V2_PCI:
 | 
						|
			case CARDTYPE_DIVASRV_VOICE_Q_8M_V2_PCI:
 | 
						|
				max = MAX_ADAPTER - 4;
 | 
						|
				nr = 4;
 | 
						|
				break;
 | 
						|
 | 
						|
			default:
 | 
						|
				max = MAX_ADAPTER;
 | 
						|
				nr = 1;
 | 
						|
			}
 | 
						|
 | 
						|
			diva_os_enter_spin_lock(&adapter_lock, &old_irql, "add card");
 | 
						|
 | 
						|
			for (i = 0; i < max; i++) {
 | 
						|
				if (!diva_find_free_adapters(i, nr)) {
 | 
						|
					pdiva->controller = i + 1;
 | 
						|
					pdiva->xdi_adapter.ANum = pdiva->controller;
 | 
						|
					IoAdapters[i] = &pdiva->xdi_adapter;
 | 
						|
					diva_os_leave_spin_lock(&adapter_lock, &old_irql, "add card");
 | 
						|
					create_adapter_proc(pdiva);	/* add adapter to proc file system */
 | 
						|
 | 
						|
					DBG_LOG(("add %s:%d",
 | 
						|
						 CardProperties
 | 
						|
						 [CardOrdinal].Name,
 | 
						|
						 pdiva->controller))
 | 
						|
 | 
						|
						diva_os_enter_spin_lock(&adapter_lock, &old_irql, "add card");
 | 
						|
					pa = pdiva;
 | 
						|
					for (j = 1; j < nr; j++) {	/* slave adapters, if any */
 | 
						|
						pa = diva_q_get_next(&pa->link);
 | 
						|
						if (pa && !pa->interface.cleanup_adapter_proc) {
 | 
						|
							pa->controller = i + 1 + j;
 | 
						|
							pa->xdi_adapter.ANum = pa->controller;
 | 
						|
							IoAdapters[i + j] = &pa->xdi_adapter;
 | 
						|
							diva_os_leave_spin_lock(&adapter_lock, &old_irql, "add card");
 | 
						|
							DBG_LOG(("add slave adapter (%d)",
 | 
						|
								 pa->controller))
 | 
						|
								create_adapter_proc(pa);	/* add adapter to proc file system */
 | 
						|
							diva_os_enter_spin_lock(&adapter_lock, &old_irql, "add card");
 | 
						|
						} else {
 | 
						|
							DBG_ERR(("slave adapter problem"))
 | 
						|
								break;
 | 
						|
						}
 | 
						|
					}
 | 
						|
 | 
						|
					diva_os_leave_spin_lock(&adapter_lock, &old_irql, "add card");
 | 
						|
					return (pdiva);
 | 
						|
				}
 | 
						|
			}
 | 
						|
 | 
						|
			diva_os_leave_spin_lock(&adapter_lock, &old_irql, "add card");
 | 
						|
 | 
						|
			/*
 | 
						|
			  Not able to add adapter - remove it and return error
 | 
						|
			*/
 | 
						|
			DBG_ERR(("can not alloc request array"))
 | 
						|
				diva_driver_remove_card(pdiva);
 | 
						|
 | 
						|
			return NULL;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return NULL;
 | 
						|
}
 | 
						|
 | 
						|
/* --------------------------------------------------------------------------
 | 
						|
   Called on driver load, MAIN, main, DriverEntry
 | 
						|
   -------------------------------------------------------------------------- */
 | 
						|
int divasa_xdi_driver_entry(void)
 | 
						|
{
 | 
						|
	diva_os_initialize_spin_lock(&adapter_lock, "adapter");
 | 
						|
	memset(&IoAdapters[0], 0x00, sizeof(IoAdapters));
 | 
						|
	diva_init_request_array();
 | 
						|
 | 
						|
	return (0);
 | 
						|
}
 | 
						|
 | 
						|
/* --------------------------------------------------------------------------
 | 
						|
   Remove adapter from list
 | 
						|
   -------------------------------------------------------------------------- */
 | 
						|
static diva_os_xdi_adapter_t *get_and_remove_from_queue(void)
 | 
						|
{
 | 
						|
	diva_os_spin_lock_magic_t old_irql;
 | 
						|
	diva_os_xdi_adapter_t *a = NULL;
 | 
						|
 | 
						|
	diva_os_enter_spin_lock(&adapter_lock, &old_irql, "driver_unload");
 | 
						|
 | 
						|
	if (!list_empty(&adapter_queue)) {
 | 
						|
		a = list_entry(adapter_queue.next, diva_os_xdi_adapter_t, link);
 | 
						|
		list_del(adapter_queue.next);
 | 
						|
	}
 | 
						|
 | 
						|
	diva_os_leave_spin_lock(&adapter_lock, &old_irql, "driver_unload");
 | 
						|
	return (a);
 | 
						|
}
 | 
						|
 | 
						|
/* --------------------------------------------------------------------------
 | 
						|
   Remove card from the card list
 | 
						|
   -------------------------------------------------------------------------- */
 | 
						|
void diva_driver_remove_card(void *pdiva)
 | 
						|
{
 | 
						|
	diva_os_spin_lock_magic_t old_irql;
 | 
						|
	diva_os_xdi_adapter_t *a[4];
 | 
						|
	diva_os_xdi_adapter_t *pa;
 | 
						|
	int i;
 | 
						|
 | 
						|
	pa = a[0] = (diva_os_xdi_adapter_t *) pdiva;
 | 
						|
	a[1] = a[2] = a[3] = NULL;
 | 
						|
 | 
						|
	diva_os_enter_spin_lock(&adapter_lock, &old_irql, "remode adapter");
 | 
						|
 | 
						|
	for (i = 1; i < 4; i++) {
 | 
						|
		if ((pa = diva_q_get_next(&pa->link))
 | 
						|
		    && !pa->interface.cleanup_adapter_proc) {
 | 
						|
			a[i] = pa;
 | 
						|
		} else {
 | 
						|
			break;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	for (i = 0; ((i < 4) && a[i]); i++) {
 | 
						|
		list_del(&a[i]->link);
 | 
						|
	}
 | 
						|
 | 
						|
	diva_os_leave_spin_lock(&adapter_lock, &old_irql, "driver_unload");
 | 
						|
 | 
						|
	(*(a[0]->interface.cleanup_adapter_proc)) (a[0]);
 | 
						|
 | 
						|
	for (i = 0; i < 4; i++) {
 | 
						|
		if (a[i]) {
 | 
						|
			if (a[i]->controller) {
 | 
						|
				DBG_LOG(("remove adapter (%d)",
 | 
						|
					 a[i]->controller)) IoAdapters[a[i]->controller - 1] = NULL;
 | 
						|
				remove_adapter_proc(a[i]);
 | 
						|
			}
 | 
						|
			diva_os_free(0, a[i]);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/* --------------------------------------------------------------------------
 | 
						|
   Create diva PCI adapter and init internal adapter structures
 | 
						|
   -------------------------------------------------------------------------- */
 | 
						|
static void *divas_create_pci_card(int handle, void *pci_dev_handle)
 | 
						|
{
 | 
						|
	diva_supported_cards_info_t *pI = &divas_supported_cards[handle];
 | 
						|
	diva_os_spin_lock_magic_t old_irql;
 | 
						|
	diva_os_xdi_adapter_t *a;
 | 
						|
 | 
						|
	DBG_LOG(("found %d-%s", pI->CardOrdinal, CardProperties[pI->CardOrdinal].Name))
 | 
						|
 | 
						|
		if (!(a = (diva_os_xdi_adapter_t *) diva_os_malloc(0, sizeof(*a)))) {
 | 
						|
			DBG_ERR(("A: can't alloc adapter"));
 | 
						|
			return NULL;
 | 
						|
		}
 | 
						|
 | 
						|
	memset(a, 0x00, sizeof(*a));
 | 
						|
 | 
						|
	a->CardIndex = handle;
 | 
						|
	a->CardOrdinal = pI->CardOrdinal;
 | 
						|
	a->Bus = DIVAS_XDI_ADAPTER_BUS_PCI;
 | 
						|
	a->xdi_adapter.cardType = a->CardOrdinal;
 | 
						|
	a->resources.pci.bus = diva_os_get_pci_bus(pci_dev_handle);
 | 
						|
	a->resources.pci.func = diva_os_get_pci_func(pci_dev_handle);
 | 
						|
	a->resources.pci.hdev = pci_dev_handle;
 | 
						|
 | 
						|
	/*
 | 
						|
	  Add master adapter first, so slave adapters will receive higher
 | 
						|
	  numbers as master adapter
 | 
						|
	*/
 | 
						|
	diva_os_enter_spin_lock(&adapter_lock, &old_irql, "found_pci_card");
 | 
						|
	list_add_tail(&a->link, &adapter_queue);
 | 
						|
	diva_os_leave_spin_lock(&adapter_lock, &old_irql, "found_pci_card");
 | 
						|
 | 
						|
	if ((*(pI->init_card)) (a)) {
 | 
						|
		diva_os_enter_spin_lock(&adapter_lock, &old_irql, "found_pci_card");
 | 
						|
		list_del(&a->link);
 | 
						|
		diva_os_leave_spin_lock(&adapter_lock, &old_irql, "found_pci_card");
 | 
						|
		diva_os_free(0, a);
 | 
						|
		DBG_ERR(("A: can't get adapter resources"));
 | 
						|
		return NULL;
 | 
						|
	}
 | 
						|
 | 
						|
	return (a);
 | 
						|
}
 | 
						|
 | 
						|
/* --------------------------------------------------------------------------
 | 
						|
   Called on driver unload FINIT, finit, Unload
 | 
						|
   -------------------------------------------------------------------------- */
 | 
						|
void divasa_xdi_driver_unload(void)
 | 
						|
{
 | 
						|
	diva_os_xdi_adapter_t *a;
 | 
						|
 | 
						|
	while ((a = get_and_remove_from_queue())) {
 | 
						|
		if (a->interface.cleanup_adapter_proc) {
 | 
						|
			(*(a->interface.cleanup_adapter_proc)) (a);
 | 
						|
		}
 | 
						|
		if (a->controller) {
 | 
						|
			IoAdapters[a->controller - 1] = NULL;
 | 
						|
			remove_adapter_proc(a);
 | 
						|
		}
 | 
						|
		diva_os_free(0, a);
 | 
						|
	}
 | 
						|
	diva_os_destroy_spin_lock(&adapter_lock, "adapter");
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
**  Receive and process command from user mode utility
 | 
						|
*/
 | 
						|
void *diva_xdi_open_adapter(void *os_handle, const void __user *src,
 | 
						|
			    int length,
 | 
						|
			    divas_xdi_copy_from_user_fn_t cp_fn)
 | 
						|
{
 | 
						|
	diva_xdi_um_cfg_cmd_t msg;
 | 
						|
	diva_os_xdi_adapter_t *a = NULL;
 | 
						|
	diva_os_spin_lock_magic_t old_irql;
 | 
						|
	struct list_head *tmp;
 | 
						|
 | 
						|
	if (length < sizeof(diva_xdi_um_cfg_cmd_t)) {
 | 
						|
		DBG_ERR(("A: A(?) open, msg too small (%d < %d)",
 | 
						|
			 length, sizeof(diva_xdi_um_cfg_cmd_t)))
 | 
						|
			return NULL;
 | 
						|
	}
 | 
						|
	if ((*cp_fn) (os_handle, &msg, src, sizeof(msg)) <= 0) {
 | 
						|
		DBG_ERR(("A: A(?) open, write error"))
 | 
						|
			return NULL;
 | 
						|
	}
 | 
						|
	diva_os_enter_spin_lock(&adapter_lock, &old_irql, "open_adapter");
 | 
						|
	list_for_each(tmp, &adapter_queue) {
 | 
						|
		a = list_entry(tmp, diva_os_xdi_adapter_t, link);
 | 
						|
		if (a->controller == (int)msg.adapter)
 | 
						|
			break;
 | 
						|
		a = NULL;
 | 
						|
	}
 | 
						|
	diva_os_leave_spin_lock(&adapter_lock, &old_irql, "open_adapter");
 | 
						|
 | 
						|
	if (!a) {
 | 
						|
		DBG_ERR(("A: A(%d) open, adapter not found", msg.adapter))
 | 
						|
			}
 | 
						|
 | 
						|
	return (a);
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
**  Easy cleanup mailbox status
 | 
						|
*/
 | 
						|
void diva_xdi_close_adapter(void *adapter, void *os_handle)
 | 
						|
{
 | 
						|
	diva_os_xdi_adapter_t *a = (diva_os_xdi_adapter_t *) adapter;
 | 
						|
 | 
						|
	a->xdi_mbox.status &= ~DIVA_XDI_MBOX_BUSY;
 | 
						|
	if (a->xdi_mbox.data) {
 | 
						|
		diva_os_free(0, a->xdi_mbox.data);
 | 
						|
		a->xdi_mbox.data = NULL;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
int
 | 
						|
diva_xdi_write(void *adapter, void *os_handle, const void __user *src,
 | 
						|
	       int length, divas_xdi_copy_from_user_fn_t cp_fn)
 | 
						|
{
 | 
						|
	diva_os_xdi_adapter_t *a = (diva_os_xdi_adapter_t *) adapter;
 | 
						|
	void *data;
 | 
						|
 | 
						|
	if (a->xdi_mbox.status & DIVA_XDI_MBOX_BUSY) {
 | 
						|
		DBG_ERR(("A: A(%d) write, mbox busy", a->controller))
 | 
						|
			return (-1);
 | 
						|
	}
 | 
						|
 | 
						|
	if (length < sizeof(diva_xdi_um_cfg_cmd_t)) {
 | 
						|
		DBG_ERR(("A: A(%d) write, message too small (%d < %d)",
 | 
						|
			 a->controller, length,
 | 
						|
			 sizeof(diva_xdi_um_cfg_cmd_t)))
 | 
						|
			return (-3);
 | 
						|
	}
 | 
						|
 | 
						|
	if (!(data = diva_os_malloc(0, length))) {
 | 
						|
		DBG_ERR(("A: A(%d) write, ENOMEM", a->controller))
 | 
						|
			return (-2);
 | 
						|
	}
 | 
						|
 | 
						|
	length = (*cp_fn) (os_handle, data, src, length);
 | 
						|
	if (length > 0) {
 | 
						|
		if ((*(a->interface.cmd_proc))
 | 
						|
		    (a, (diva_xdi_um_cfg_cmd_t *) data, length)) {
 | 
						|
			length = -3;
 | 
						|
		}
 | 
						|
	} else {
 | 
						|
		DBG_ERR(("A: A(%d) write error (%d)", a->controller,
 | 
						|
			 length))
 | 
						|
			}
 | 
						|
 | 
						|
	diva_os_free(0, data);
 | 
						|
 | 
						|
	return (length);
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
**  Write answers to user mode utility, if any
 | 
						|
*/
 | 
						|
int
 | 
						|
diva_xdi_read(void *adapter, void *os_handle, void __user *dst,
 | 
						|
	      int max_length, divas_xdi_copy_to_user_fn_t cp_fn)
 | 
						|
{
 | 
						|
	diva_os_xdi_adapter_t *a = (diva_os_xdi_adapter_t *) adapter;
 | 
						|
	int ret;
 | 
						|
 | 
						|
	if (!(a->xdi_mbox.status & DIVA_XDI_MBOX_BUSY)) {
 | 
						|
		DBG_ERR(("A: A(%d) rx mbox empty", a->controller))
 | 
						|
			return (-1);
 | 
						|
	}
 | 
						|
	if (!a->xdi_mbox.data) {
 | 
						|
		a->xdi_mbox.status &= ~DIVA_XDI_MBOX_BUSY;
 | 
						|
		DBG_ERR(("A: A(%d) rx ENOMEM", a->controller))
 | 
						|
			return (-2);
 | 
						|
	}
 | 
						|
 | 
						|
	if (max_length < a->xdi_mbox.data_length) {
 | 
						|
		DBG_ERR(("A: A(%d) rx buffer too short(%d < %d)",
 | 
						|
			 a->controller, max_length,
 | 
						|
			 a->xdi_mbox.data_length))
 | 
						|
			return (-3);
 | 
						|
	}
 | 
						|
 | 
						|
	ret = (*cp_fn) (os_handle, dst, a->xdi_mbox.data,
 | 
						|
			a->xdi_mbox.data_length);
 | 
						|
	if (ret > 0) {
 | 
						|
		diva_os_free(0, a->xdi_mbox.data);
 | 
						|
		a->xdi_mbox.data = NULL;
 | 
						|
		a->xdi_mbox.status &= ~DIVA_XDI_MBOX_BUSY;
 | 
						|
	}
 | 
						|
 | 
						|
	return (ret);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
irqreturn_t diva_os_irq_wrapper(int irq, void *context)
 | 
						|
{
 | 
						|
	diva_os_xdi_adapter_t *a = context;
 | 
						|
	diva_xdi_clear_interrupts_proc_t clear_int_proc;
 | 
						|
 | 
						|
	if (!a || !a->xdi_adapter.diva_isr_handler)
 | 
						|
		return IRQ_NONE;
 | 
						|
 | 
						|
	if ((clear_int_proc = a->clear_interrupts_proc)) {
 | 
						|
		(*clear_int_proc) (a);
 | 
						|
		a->clear_interrupts_proc = NULL;
 | 
						|
		return IRQ_HANDLED;
 | 
						|
	}
 | 
						|
 | 
						|
	(*(a->xdi_adapter.diva_isr_handler)) (&a->xdi_adapter);
 | 
						|
	return IRQ_HANDLED;
 | 
						|
}
 | 
						|
 | 
						|
static void diva_init_request_array(void)
 | 
						|
{
 | 
						|
	Requests[0] = DivaIdiRequest0;
 | 
						|
	Requests[1] = DivaIdiRequest1;
 | 
						|
	Requests[2] = DivaIdiRequest2;
 | 
						|
	Requests[3] = DivaIdiRequest3;
 | 
						|
	Requests[4] = DivaIdiRequest4;
 | 
						|
	Requests[5] = DivaIdiRequest5;
 | 
						|
	Requests[6] = DivaIdiRequest6;
 | 
						|
	Requests[7] = DivaIdiRequest7;
 | 
						|
	Requests[8] = DivaIdiRequest8;
 | 
						|
	Requests[9] = DivaIdiRequest9;
 | 
						|
	Requests[10] = DivaIdiRequest10;
 | 
						|
	Requests[11] = DivaIdiRequest11;
 | 
						|
	Requests[12] = DivaIdiRequest12;
 | 
						|
	Requests[13] = DivaIdiRequest13;
 | 
						|
	Requests[14] = DivaIdiRequest14;
 | 
						|
	Requests[15] = DivaIdiRequest15;
 | 
						|
	Requests[16] = DivaIdiRequest16;
 | 
						|
	Requests[17] = DivaIdiRequest17;
 | 
						|
	Requests[18] = DivaIdiRequest18;
 | 
						|
	Requests[19] = DivaIdiRequest19;
 | 
						|
	Requests[20] = DivaIdiRequest20;
 | 
						|
	Requests[21] = DivaIdiRequest21;
 | 
						|
	Requests[22] = DivaIdiRequest22;
 | 
						|
	Requests[23] = DivaIdiRequest23;
 | 
						|
	Requests[24] = DivaIdiRequest24;
 | 
						|
	Requests[25] = DivaIdiRequest25;
 | 
						|
	Requests[26] = DivaIdiRequest26;
 | 
						|
	Requests[27] = DivaIdiRequest27;
 | 
						|
	Requests[28] = DivaIdiRequest28;
 | 
						|
	Requests[29] = DivaIdiRequest29;
 | 
						|
	Requests[30] = DivaIdiRequest30;
 | 
						|
	Requests[31] = DivaIdiRequest31;
 | 
						|
}
 | 
						|
 | 
						|
void diva_xdi_display_adapter_features(int card)
 | 
						|
{
 | 
						|
	dword features;
 | 
						|
	if (!card || ((card - 1) >= MAX_ADAPTER) || !IoAdapters[card - 1]) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	card--;
 | 
						|
	features = IoAdapters[card]->Properties.Features;
 | 
						|
 | 
						|
	DBG_LOG(("FEATURES FOR ADAPTER: %d", card + 1))
 | 
						|
		DBG_LOG((" DI_FAX3          :  %s",
 | 
						|
			 (features & DI_FAX3) ? "Y" : "N"))
 | 
						|
		DBG_LOG((" DI_MODEM         :  %s",
 | 
						|
			 (features & DI_MODEM) ? "Y" : "N"))
 | 
						|
		DBG_LOG((" DI_POST          :  %s",
 | 
						|
			 (features & DI_POST) ? "Y" : "N"))
 | 
						|
		DBG_LOG((" DI_V110          :  %s",
 | 
						|
			 (features & DI_V110) ? "Y" : "N"))
 | 
						|
		DBG_LOG((" DI_V120          :  %s",
 | 
						|
			 (features & DI_V120) ? "Y" : "N"))
 | 
						|
		DBG_LOG((" DI_POTS          :  %s",
 | 
						|
			 (features & DI_POTS) ? "Y" : "N"))
 | 
						|
		DBG_LOG((" DI_CODEC         :  %s",
 | 
						|
			 (features & DI_CODEC) ? "Y" : "N"))
 | 
						|
		DBG_LOG((" DI_MANAGE        :  %s",
 | 
						|
			 (features & DI_MANAGE) ? "Y" : "N"))
 | 
						|
		DBG_LOG((" DI_V_42          :  %s",
 | 
						|
			 (features & DI_V_42) ? "Y" : "N"))
 | 
						|
		DBG_LOG((" DI_EXTD_FAX      :  %s",
 | 
						|
			 (features & DI_EXTD_FAX) ? "Y" : "N"))
 | 
						|
		DBG_LOG((" DI_AT_PARSER     :  %s",
 | 
						|
			 (features & DI_AT_PARSER) ? "Y" : "N"))
 | 
						|
		DBG_LOG((" DI_VOICE_OVER_IP :  %s",
 | 
						|
			 (features & DI_VOICE_OVER_IP) ? "Y" : "N"))
 | 
						|
		}
 | 
						|
 | 
						|
void diva_add_slave_adapter(diva_os_xdi_adapter_t *a)
 | 
						|
{
 | 
						|
	diva_os_spin_lock_magic_t old_irql;
 | 
						|
 | 
						|
	diva_os_enter_spin_lock(&adapter_lock, &old_irql, "add_slave");
 | 
						|
	list_add_tail(&a->link, &adapter_queue);
 | 
						|
	diva_os_leave_spin_lock(&adapter_lock, &old_irql, "add_slave");
 | 
						|
}
 | 
						|
 | 
						|
int diva_card_read_xlog(diva_os_xdi_adapter_t *a)
 | 
						|
{
 | 
						|
	diva_get_xlog_t *req;
 | 
						|
	byte *data;
 | 
						|
 | 
						|
	if (!a->xdi_adapter.Initialized || !a->xdi_adapter.DIRequest) {
 | 
						|
		return (-1);
 | 
						|
	}
 | 
						|
	if (!(data = diva_os_malloc(0, sizeof(struct mi_pc_maint)))) {
 | 
						|
		return (-1);
 | 
						|
	}
 | 
						|
	memset(data, 0x00, sizeof(struct mi_pc_maint));
 | 
						|
 | 
						|
	if (!(req = diva_os_malloc(0, sizeof(*req)))) {
 | 
						|
		diva_os_free(0, data);
 | 
						|
		return (-1);
 | 
						|
	}
 | 
						|
	req->command = 0x0400;
 | 
						|
	req->req = LOG;
 | 
						|
	req->rc = 0x00;
 | 
						|
 | 
						|
	(*(a->xdi_adapter.DIRequest)) (&a->xdi_adapter, (ENTITY *) req);
 | 
						|
 | 
						|
	if (!req->rc || req->req) {
 | 
						|
		diva_os_free(0, data);
 | 
						|
		diva_os_free(0, req);
 | 
						|
		return (-1);
 | 
						|
	}
 | 
						|
 | 
						|
	memcpy(data, &req->req, sizeof(struct mi_pc_maint));
 | 
						|
 | 
						|
	diva_os_free(0, req);
 | 
						|
 | 
						|
	a->xdi_mbox.data_length = sizeof(struct mi_pc_maint);
 | 
						|
	a->xdi_mbox.data = data;
 | 
						|
	a->xdi_mbox.status = DIVA_XDI_MBOX_BUSY;
 | 
						|
 | 
						|
	return (0);
 | 
						|
}
 | 
						|
 | 
						|
void xdiFreeFile(void *handle)
 | 
						|
{
 | 
						|
}
 |