[BACK]Return to gazel.c CVS log [TXT][DIR] Up to [Development] / linux-2.6-xfs / drivers / isdn / hisax

File: [Development] / linux-2.6-xfs / drivers / isdn / hisax / gazel.c (download)

Revision 1.1, Tue Dec 30 23:58:53 2003 UTC (13 years, 10 months ago) by cattelan
Branch: MAIN

Initial Import 2.6.0

/* $Id: gazel.c,v 2.11.6.7 2001/09/23 22:24:47 kai Exp $
 *
 * low level stuff for Gazel isdn cards
 *
 * Author       BeWan Systems
 *              based on source code from Karsten Keil
 * Copyright    by BeWan Systems
 * 
 * This software may be used and distributed according to the terms
 * of the GNU General Public License, incorporated herein by reference.
 *
 */

#include <linux/config.h>
#include <linux/init.h>
#include "hisax.h"
#include "isac.h"
#include "hscx.h"
#include "isdnl1.h"
#include "ipac.h"
#include <linux/pci.h>

extern const char *CardType[];
const char *gazel_revision = "$Revision: 2.11.6.7 $";
static spinlock_t gazel_lock = SPIN_LOCK_UNLOCKED;

#define R647      1
#define R685      2
#define R753      3
#define R742      4

#define PLX_CNTRL    0x50	/* registre de controle PLX */
#define RESET_GAZEL  0x4
#define RESET_9050   0x40000000
#define PLX_INCSR    0x4C	/* registre d'IT du 9050 */
#define INT_ISAC_EN  0x8	/* 1 = enable IT isac */
#define INT_ISAC     0x20	/* 1 = IT isac en cours */
#define INT_HSCX_EN  0x1	/* 1 = enable IT hscx */
#define INT_HSCX     0x4	/* 1 = IT hscx en cours */
#define INT_PCI_EN   0x40	/* 1 = enable IT PCI */
#define INT_IPAC_EN  0x3	/* enable IT ipac */


#define byteout(addr,val) outb(val,addr)
#define bytein(addr) inb(addr)

static inline u8
readreg(unsigned int adr, u_short off)
{
	return bytein(adr + off);
}

static inline void
writereg(unsigned int adr, u_short off, u8 data)
{
	byteout(adr + off, data);
}


static inline void
read_fifo(unsigned int adr, u8 * data, int size)
{
	insb(adr, data, size);
}

static void
write_fifo(unsigned int adr, u8 * data, int size)
{
	outsb(adr, data, size);
}

static u8
r685_isac_read(struct IsdnCardState *cs, u8 off)
{
	return readreg(cs->hw.gazel.isac, off);
}

static u8
r647_isac_read(struct IsdnCardState *cs, u8 off)
{
	return readreg(cs->hw.gazel.isac, (off << 8 & 0xf000) | (off & 0xf));
}

static void
r685_isac_write(struct IsdnCardState *cs, u8 off, u8 value)
{
	writereg(cs->hw.gazel.isac, off, value);
}

static void
r647_isac_write(struct IsdnCardState *cs, u8 off, u8 value)
{
	writereg(cs->hw.gazel.isac, (off << 8 & 0xf000) | (off & 0xf), value);
}

static void
isac_read_fifo(struct IsdnCardState *cs, u8 * data, int size)
{
	read_fifo(cs->hw.gazel.isacfifo, data, size);
}

static void
isac_write_fifo(struct IsdnCardState *cs, u8 * data, int size)
{
	write_fifo(cs->hw.gazel.isacfifo, data, size);
}

static struct dc_hw_ops r685_isac_ops = {
	.read_reg   = r685_isac_read,
	.write_reg  = r685_isac_write,
	.read_fifo  = isac_read_fifo,
	.write_fifo = isac_write_fifo,
};

static struct dc_hw_ops r647_isac_ops = {
	.read_reg   = r647_isac_read,
	.write_reg  = r647_isac_write,
	.read_fifo  = isac_read_fifo,
	.write_fifo = isac_write_fifo,
};
  
static u8
r685_hscx_read(struct IsdnCardState *cs, int hscx, u8 off)
{
	return readreg(cs->hw.gazel.hscx[hscx], off);
}

static u8
r647_hscx_read(struct IsdnCardState *cs, int hscx, u8 off)
{
	return readreg(cs->hw.gazel.hscx[hscx],
		       (off << 8 & 0xf000) | (off & 0xf));
}

static void
r685_hscx_write(struct IsdnCardState *cs, int hscx, u8 off, u8 value)
{
	writereg(cs->hw.gazel.hscx[hscx], off, value);
}

static void
r647_hscx_write(struct IsdnCardState *cs, int hscx, u8 off, u8 value)
{
	writereg(cs->hw.gazel.hscx[hscx],
		 (off << 8 & 0xf000) | (off & 0xf), value);
}

static void
hscx_read_fifo(struct IsdnCardState *cs, int hscx, u8 * data, int size)
{
	read_fifo(cs->hw.gazel.hscxfifo[hscx], data, size);
}

static void
hscx_write_fifo(struct IsdnCardState *cs, int hscx, u8 * data, int size)
{
	write_fifo(cs->hw.gazel.hscxfifo[hscx], data, size);
}

static struct bc_hw_ops r685_hscx_ops = {
	.read_reg   = r685_hscx_read,
	.write_reg  = r685_hscx_write,
	.read_fifo  = hscx_read_fifo,
	.write_fifo = hscx_write_fifo,
};

static struct bc_hw_ops r647_hscx_ops = {
	.read_reg   = r647_hscx_read,
	.write_reg  = r647_hscx_write,
	.read_fifo  = hscx_read_fifo,
	.write_fifo = hscx_write_fifo,
};

static inline u8
ipac_read(struct IsdnCardState *cs, u_short off)
{
	register u8 ret;
	unsigned long flags;

	spin_lock_irqsave(&gazel_lock, flags);
	byteout(cs->hw.gazel.ipac, off);
	ret = bytein(cs->hw.gazel.ipac + 4);
	spin_unlock_irqrestore(&gazel_lock, flags);
	return ret;
}

static inline void
ipac_write(struct IsdnCardState *cs, u_short off, u8 data)
{
	unsigned long flags;

	spin_lock_irqsave(&gazel_lock, flags);
	byteout(cs->hw.gazel.ipac, off);
	byteout(cs->hw.gazel.ipac + 4, data);
	spin_unlock_irqrestore(&gazel_lock, flags);
}


static inline void
ipac_readfifo(struct IsdnCardState *cs, u8 off, u8 * data, int size)
{
	byteout(cs->hw.gazel.ipac, off);
	insb(cs->hw.gazel.ipac + 4, data, size);
}

static inline void
ipac_writefifo(struct IsdnCardState *cs, u8 off, u8 * data, int size)
{
	byteout(cs->hw.gazel.ipac, off);
	outsb(cs->hw.gazel.ipac + 4, data, size);
}

/* This will generate ipac_dc_ops and ipac_bc_ops using the functions
 * above */

BUILD_IPAC_OPS(ipac);

static int
r647_reset(struct IsdnCardState *cs)
{
	writereg(cs->hw.gazel.cfg_reg, 0, 0);
	HZDELAY(10);
	writereg(cs->hw.gazel.cfg_reg, 0, 1);
	HZDELAY(2);
	return 0;
}

static int
r685_reset(struct IsdnCardState *cs)
{
	unsigned long plxcntrl, addr = cs->hw.gazel.cfg_reg;

	plxcntrl = inl(addr + PLX_CNTRL);
	plxcntrl |= (RESET_9050 + RESET_GAZEL);
	outl(plxcntrl, addr + PLX_CNTRL);
	plxcntrl &= ~(RESET_9050 + RESET_GAZEL);
	HZDELAY(4);
	outl(plxcntrl, addr + PLX_CNTRL);
	HZDELAY(10);
	outb(INT_ISAC_EN + INT_HSCX_EN + INT_PCI_EN, addr + PLX_INCSR);
	return 0;
}

static int
r753_reset(struct IsdnCardState *cs)
{
	unsigned long plxcntrl, addr = cs->hw.gazel.cfg_reg;

	if (test_bit(FLG_BUGGY_PLX9050, &cs->HW_Flags))
		/* we can't read, assume the default */
		plxcntrl = 0x18784db6;
	else
		plxcntrl = inl(addr + PLX_CNTRL);
	plxcntrl |= (RESET_9050 + RESET_GAZEL);
	outl(plxcntrl, addr + PLX_CNTRL);
	ipac_write(cs, IPAC_POTA2, 0x20);
	HZDELAY(4);
	plxcntrl &= ~(RESET_9050 + RESET_GAZEL);
	outl(plxcntrl, addr + PLX_CNTRL);
	HZDELAY(10);
	ipac_write(cs, IPAC_POTA2, 0x00);
	ipac_write(cs, IPAC_ACFG, 0xff);
	ipac_write(cs, IPAC_AOE, 0x0);
	ipac_write(cs, IPAC_MASK, 0xff);
	ipac_write(cs, IPAC_CONF, 0x1);
	outb(INT_IPAC_EN + INT_PCI_EN, addr + PLX_INCSR);
	ipac_write(cs, IPAC_MASK, 0xc0);
	return 0;
}

static int
r742_reset(struct IsdnCardState *cs)
{
	ipac_write(cs, IPAC_POTA2, 0x20);
	HZDELAY(4);
	ipac_write(cs, IPAC_POTA2, 0x00);
	ipac_write(cs, IPAC_ACFG, 0xff);
	ipac_write(cs, IPAC_AOE, 0x0);
	ipac_write(cs, IPAC_MASK, 0xff);
	ipac_write(cs, IPAC_CONF, 0x1);
	ipac_write(cs, IPAC_MASK, 0xc0);
	return 0;
}

static void
gazel_init(struct IsdnCardState *cs)
{
	int i;

	for (i = 0; i < 2; i++) {
		cs->bcs[i].hw.hscx.tsaxr0 = 0x1f;
		cs->bcs[i].hw.hscx.tsaxr1 = 0x23;
	}
	inithscxisac(cs);
}

static struct card_ops r647_ops = {
	.init     = gazel_init,
	.reset    = r647_reset,
	.release  = hisax_release_resources,
	.irq_func = hscxisac_irq,
};

static struct card_ops r685_ops = {
	.init     = gazel_init,
	.reset    = r685_reset,
	.release  = hisax_release_resources,
	.irq_func = hscxisac_irq,
};

static struct card_ops r742_ops = {
	.init     = ipac_init,
	.reset    = r742_reset,
	.release  = hisax_release_resources,
	.irq_func = ipac_irq,
};

static struct card_ops r753_ops = {
	.init     = ipac_init,
	.reset    = r753_reset,
	.release  = hisax_release_resources,
	.irq_func = ipac_irq,
};

static int __init
gazel647_probe(struct IsdnCardState *cs, struct IsdnCard *card)
{
	int i, base;

	cs->subtyp = R647;
	cs->irq = card->para[0];
	cs->hw.gazel.cfg_reg = card->para[1] + 0xC000;

	printk(KERN_INFO "Gazel: Card ISA R647/R648 found\n");
	cs->dc.isac.adf2 = 0x87;
	printk(KERN_INFO "Gazel: config irq:%d isac:0x%X  cfg:0x%X\n",
	       cs->irq, cs->hw.gazel.isac, cs->hw.gazel.cfg_reg);
	printk(KERN_INFO
	       "Gazel: hscx A:0x%X  hscx B:0x%X\n",
	       cs->hw.gazel.hscx[0], cs->hw.gazel.hscx[1]);

	cs->hw.gazel.isac = card->para[1] + 0x8000;
	cs->hw.gazel.hscx[0] = card->para[1];
	cs->hw.gazel.hscx[1] = card->para[1] + 0x4000;
	cs->hw.gazel.isacfifo = cs->hw.gazel.isac;
	cs->hw.gazel.hscxfifo[0] = cs->hw.gazel.hscx[0];
	cs->hw.gazel.hscxfifo[1] = cs->hw.gazel.hscx[1];

	base = cs->hw.gazel.hscx[0];
	for (i = 0; i < 0xc000; i += 0x1000) {
		if (!request_io(&cs->rs, base + i, 16, "gazel"))
			goto err;
	}
	if (!request_io(&cs->rs, 0xc000 + base, 1, "gazel"))
		goto err;

	cs->card_ops = &r647_ops;
	if (hscxisac_setup(cs, &r647_isac_ops, &r647_hscx_ops))
		goto err;

	cs->card_ops->reset(cs);
	return 0;
 err:
	hisax_release_resources(cs);
	return -EBUSY;
}

static int __init
gazel742_probe(struct IsdnCardState *cs, struct IsdnCard *card)
{
	cs->subtyp = R742;
	cs->irq = card->para[0];
	cs->hw.gazel.cfg_reg = card->para[1] + 0xC000;

	printk(KERN_INFO "Gazel: Card ISA R742 found\n");
	printk(KERN_INFO "Gazel: config irq:%d ipac:0x%X\n",
	       cs->irq, cs->hw.gazel.ipac);

	if (!request_io(&cs->rs, cs->hw.gazel.ipac, 0x8, "gazel"))
		goto err;

	cs->card_ops = &r742_ops;
	if (ipac_setup(cs, &ipac_dc_ops, &ipac_bc_ops))
		goto err;

	cs->card_ops->reset(cs);
	return 0;
 err:
	hisax_release_resources(cs);
	return -EBUSY;
}

static int __init
gazel685_probe(struct IsdnCardState *cs, struct pci_dev *pdev)
{
	if (pci_enable_device(pdev))
		goto err;
			
	cs->subtyp               = R685;
	cs->irq                  = pdev->irq;
	cs->irq_flags           |= SA_SHIRQ;
	cs->hw.gazel.cfg_reg     = pci_resource_start(pdev, 1);
	cs->hw.gazel.isac        = pci_resource_start(pdev, 2) + 0x80;
	cs->hw.gazel.hscx[0]     = pci_resource_start(pdev, 2);
	cs->hw.gazel.hscx[1]     = pci_resource_start(pdev, 2) + 0x40;
	cs->hw.gazel.isacfifo    = cs->hw.gazel.isac;
	cs->hw.gazel.hscxfifo[0] = cs->hw.gazel.hscx[0];
	cs->hw.gazel.hscxfifo[1] = cs->hw.gazel.hscx[1];
	cs->dc.isac.adf2 = 0x87;

	if (!request_io(&cs->rs, cs->hw.gazel.hscx[0], 0x100, "gazel"))
		goto err;
	if (!request_io(&cs->rs, cs->hw.gazel.cfg_reg, 0x80, "gazel"))
		goto err;

	printk(KERN_INFO "Gazel: Card PCI R685 found\n");
	printk(KERN_INFO "Gazel: config irq:%d isac:0x%X  cfg:0x%X\n",
	       cs->irq, cs->hw.gazel.isac, cs->hw.gazel.cfg_reg);
	printk(KERN_INFO "Gazel: hscx A:0x%X  hscx B:0x%X\n",
	       cs->hw.gazel.hscx[0], cs->hw.gazel.hscx[1]);

	cs->card_ops  = &r685_ops;
	if (hscxisac_setup(cs, &r685_isac_ops, &r685_hscx_ops))
		goto err;

	cs->card_ops->reset(cs);
	return 0;
 err:
	hisax_release_resources(cs);
	return -EBUSY;
}

static int __init
gazel753_probe(struct IsdnCardState *cs, struct pci_dev *pdev)
{
	u8 pci_rev;

	if (pci_enable_device(pdev))
		goto err;
			
	cs->subtyp           = R753;
	cs->irq              = pdev->irq;
	cs->irq_flags       |= SA_SHIRQ;
	cs->hw.gazel.cfg_reg = pci_resource_start(pdev, 1);
	cs->hw.gazel.ipac    = pci_resource_start(pdev, 2);

	if (!request_io(&cs->rs, cs->hw.gazel.ipac, 0x8, "gazel"))
		goto err;
	if (!request_io(&cs->rs, cs->hw.gazel.cfg_reg, 0x80, "gazel"))
		goto err;

	printk(KERN_INFO "Gazel: Card PCI R753 found\n");
	printk(KERN_INFO "Gazel: config irq:%d ipac:0x%X  cfg:0x%X\n",
	       cs->irq, cs->hw.gazel.ipac, cs->hw.gazel.cfg_reg);
	/* 
	 * Erratum for PLX9050, revision 1:
	 * If bit 7 of BAR 0/1 is set, local config registers
	 * can not be read (write is okay)
	 */
	if (cs->hw.gazel.cfg_reg & 0x80) {
		pci_read_config_byte(pdev, PCI_REVISION_ID, &pci_rev);
		if (pci_rev == 1) {
			printk(KERN_INFO "Gazel: PLX9050 rev1 workaround "
			       "activated\n");
			__set_bit(FLG_BUGGY_PLX9050, &cs->HW_Flags);
		}
	}
	cs->card_ops = &r753_ops;
	if (ipac_setup(cs, &ipac_dc_ops, &ipac_bc_ops))
		goto err;

	cs->card_ops->reset(cs);
	return 0;
 err:
	hisax_release_resources(cs);
	return -EBUSY;
}

static struct pci_dev *dev_tel __initdata = NULL;
static u16 __initdata dev_id = PCI_DEVICE_ID_PLX_R685;

int __init
setup_gazel(struct IsdnCard *card)
{
	char tmp[64];

	strcpy(tmp, gazel_revision);
	printk(KERN_INFO "Gazel: Driver Revision %s\n", HiSax_getrev(tmp));

	if (card->para[0]) {
		printk(KERN_INFO "Gazel: ISA card automatic recognition\n");
		// we got an irq parameter, assume it is an ISA card
		// R742 decodes address even in not started...
		// R647 returns FF if not present or not started
		// eventually needs improvment
		card->cs->hw.gazel.ipac = card->para[1];
		if (ipac_read(card->cs, IPAC_ID) == 1) {
			if (gazel742_probe(card->cs, card))
				return 0;
		} else {
			if (gazel647_probe(card->cs, card))
				return 0;
		}
		return 1;
	}

	for (;;) {
		dev_tel = pci_find_device(PCI_VENDOR_ID_PLX, dev_id, dev_tel);
		if (dev_tel) {
			switch (dev_id) {
			case PCI_DEVICE_ID_PLX_R685:
				if (gazel685_probe(card->cs, dev_tel))
					return 0;
				return 1;
			case PCI_DEVICE_ID_PLX_R753:
			case PCI_DEVICE_ID_PLX_DJINN_ITOO:
				if (gazel753_probe(card->cs, dev_tel))
					return 0;
				return 1;
			}
		}
		switch (dev_id) {
		case PCI_DEVICE_ID_PLX_R685:
			dev_id = PCI_DEVICE_ID_PLX_R753;
		case PCI_DEVICE_ID_PLX_R753:
			dev_id = PCI_DEVICE_ID_PLX_DJINN_ITOO;
		default:
			break;
		}
	}
	printk(KERN_WARNING "Gazel: No PCI card found\n");
	return 0;
}