support OpenPIC variant found in PowerMac G5s
from Phileas Fogg
This commit is contained in:
parent
ddb0cef57c
commit
b762205fe6
@ -1,4 +1,4 @@
|
|||||||
# $NetBSD: files.macppc,v 1.99 2012/01/24 04:33:11 macallan Exp $
|
# $NetBSD: files.macppc,v 1.100 2013/04/18 16:42:46 macallan Exp $
|
||||||
#
|
#
|
||||||
# macppc-specific configuration info
|
# macppc-specific configuration info
|
||||||
|
|
||||||
@ -25,13 +25,16 @@ file arch/macppc/dev/dbdma.c
|
|||||||
# interrupt controllers
|
# interrupt controllers
|
||||||
define pic_ohare
|
define pic_ohare
|
||||||
define pic_heathrow
|
define pic_heathrow
|
||||||
|
define pic_u3_ht
|
||||||
|
|
||||||
defflag opt_interrupt.h PIC_OHARE: pic_ohare
|
defflag opt_interrupt.h PIC_OHARE: pic_ohare
|
||||||
defflag opt_interrupt.h OHARE_DEBUG
|
defflag opt_interrupt.h OHARE_DEBUG
|
||||||
defflag opt_interrupt.h PIC_HEATHROW: pic_heathrow
|
defflag opt_interrupt.h PIC_HEATHROW: pic_heathrow
|
||||||
|
defflag opt_interrupt.h PIC_U3_HT: pic_u3_ht
|
||||||
|
|
||||||
file arch/macppc/macppc/pic_ohare.c pic_ohare needs-flag
|
file arch/macppc/macppc/pic_ohare.c pic_ohare needs-flag
|
||||||
file arch/macppc/macppc/pic_heathrow.c pic_heathrow needs-flag
|
file arch/macppc/macppc/pic_heathrow.c pic_heathrow needs-flag
|
||||||
|
file arch/macppc/macppc/pic_u3_ht.c pic_u3_ht needs-flag
|
||||||
|
|
||||||
# IPI interface
|
# IPI interface
|
||||||
define ipi_hammerhead
|
define ipi_hammerhead
|
||||||
@ -108,7 +111,7 @@ device grackle: pcibus, pic_heathrow
|
|||||||
attach grackle at mainbus
|
attach grackle at mainbus
|
||||||
file arch/macppc/pci/grackle.c grackle
|
file arch/macppc/pci/grackle.c grackle
|
||||||
|
|
||||||
device uninorth: pcibus, pic_openpic, ipi_openpic
|
device uninorth: pcibus, pic_openpic, ipi_openpic, pic_u3_ht
|
||||||
attach uninorth at mainbus
|
attach uninorth at mainbus
|
||||||
file arch/macppc/pci/uninorth.c uninorth
|
file arch/macppc/pci/uninorth.c uninorth
|
||||||
|
|
||||||
@ -121,7 +124,7 @@ file arch/macppc/dev/uni-n.c uni_n
|
|||||||
include "dev/usb/files.usb"
|
include "dev/usb/files.usb"
|
||||||
include "dev/bluetooth/files.bluetooth"
|
include "dev/bluetooth/files.bluetooth"
|
||||||
|
|
||||||
device ibmcpc: pcibus, pic_openpic, ipi_openpic
|
device ibmcpc: pcibus, pic_openpic, ipi_openpic, pic_u3_ht
|
||||||
attach ibmcpc at mainbus
|
attach ibmcpc at mainbus
|
||||||
file arch/macppc/pci/u3.c ibmcpc
|
file arch/macppc/pci/u3.c ibmcpc
|
||||||
|
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
/* $NetBSD: intr.h,v 1.30 2011/06/17 23:36:17 matt Exp $ */
|
/* $NetBSD: intr.h,v 1.31 2013/04/18 16:42:46 macallan Exp $ */
|
||||||
|
|
||||||
/*-
|
/*-
|
||||||
* Copyright (c) 1998 The NetBSD Foundation, Inc.
|
* Copyright (c) 1998 The NetBSD Foundation, Inc.
|
||||||
@ -52,6 +52,7 @@ struct cpu_info;
|
|||||||
/* probe for a PIC and set it up, return TRUE on success */
|
/* probe for a PIC and set it up, return TRUE on success */
|
||||||
int init_ohare(void);
|
int init_ohare(void);
|
||||||
int init_heathrow(void);
|
int init_heathrow(void);
|
||||||
|
int init_u3_ht(void);
|
||||||
int init_grandcentral(void);
|
int init_grandcentral(void);
|
||||||
void setup_hammerhead_ipi(void);
|
void setup_hammerhead_ipi(void);
|
||||||
#endif
|
#endif
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
/* $NetBSD: interrupts.c,v 1.4 2010/11/14 03:57:17 uebayasi Exp $ */
|
/* $NetBSD: interrupts.c,v 1.5 2013/04/18 16:42:46 macallan Exp $ */
|
||||||
|
|
||||||
/*-
|
/*-
|
||||||
* Copyright (c) 2007 Michael Lorenz
|
* Copyright (c) 2007 Michael Lorenz
|
||||||
@ -27,7 +27,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include <sys/cdefs.h>
|
#include <sys/cdefs.h>
|
||||||
__KERNEL_RCSID(0, "$NetBSD: interrupts.c,v 1.4 2010/11/14 03:57:17 uebayasi Exp $");
|
__KERNEL_RCSID(0, "$NetBSD: interrupts.c,v 1.5 2013/04/18 16:42:46 macallan Exp $");
|
||||||
|
|
||||||
#include "opt_multiprocessor.h"
|
#include "opt_multiprocessor.h"
|
||||||
|
|
||||||
@ -47,6 +47,7 @@ __KERNEL_RCSID(0, "$NetBSD: interrupts.c,v 1.4 2010/11/14 03:57:17 uebayasi Exp
|
|||||||
#include "pic_openpic.h"
|
#include "pic_openpic.h"
|
||||||
#include "pic_ohare.h"
|
#include "pic_ohare.h"
|
||||||
#include "pic_heathrow.h"
|
#include "pic_heathrow.h"
|
||||||
|
#include "pic_u3_ht.h"
|
||||||
#include "opt_ipi.h"
|
#include "opt_ipi.h"
|
||||||
#include "ipi_openpic.h"
|
#include "ipi_openpic.h"
|
||||||
#include "ipi_hammerhead.h"
|
#include "ipi_hammerhead.h"
|
||||||
@ -119,6 +120,10 @@ init_interrupt(void)
|
|||||||
if (init_heathrow())
|
if (init_heathrow())
|
||||||
goto done;
|
goto done;
|
||||||
#endif
|
#endif
|
||||||
|
#if NPIC_U3_HT > 0
|
||||||
|
if (init_u3_ht())
|
||||||
|
goto done;
|
||||||
|
#endif
|
||||||
#if NPIC_OPENPIC > 0
|
#if NPIC_OPENPIC > 0
|
||||||
if (init_openpic(0))
|
if (init_openpic(0))
|
||||||
goto done;
|
goto done;
|
||||||
|
522
sys/arch/macppc/macppc/pic_u3_ht.c
Normal file
522
sys/arch/macppc/macppc/pic_u3_ht.c
Normal file
@ -0,0 +1,522 @@
|
|||||||
|
/*-
|
||||||
|
* Copyright (c) 2013 Phileas Fogg
|
||||||
|
* All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions
|
||||||
|
* are met:
|
||||||
|
* 1. Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
|
||||||
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
||||||
|
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||||
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
|
||||||
|
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
||||||
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
||||||
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
||||||
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
||||||
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
||||||
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
||||||
|
* POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <sys/cdefs.h>
|
||||||
|
|
||||||
|
#include "opt_openpic.h"
|
||||||
|
#include "opt_interrupt.h"
|
||||||
|
|
||||||
|
#include <sys/param.h>
|
||||||
|
#include <sys/kmem.h>
|
||||||
|
#include <sys/kernel.h>
|
||||||
|
|
||||||
|
#include <machine/pio.h>
|
||||||
|
#include <powerpc/openpic.h>
|
||||||
|
|
||||||
|
#include <dev/ofw/openfirm.h>
|
||||||
|
#include <dev/ofw/ofw_pci.h>
|
||||||
|
#include <dev/pci/pcireg.h>
|
||||||
|
|
||||||
|
#include <machine/autoconf.h>
|
||||||
|
#include <arch/powerpc/pic/picvar.h>
|
||||||
|
|
||||||
|
struct u3_ht_irqmap {
|
||||||
|
int im_index;
|
||||||
|
int im_level;
|
||||||
|
volatile uint8_t *im_base;
|
||||||
|
volatile uint8_t *im_apple_base;
|
||||||
|
uint32_t im_data;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct u3_ht_ops {
|
||||||
|
struct pic_ops pic;
|
||||||
|
|
||||||
|
volatile uint8_t *ht_base;
|
||||||
|
|
||||||
|
struct u3_ht_irqmap ht_irqmap[128];
|
||||||
|
|
||||||
|
uint32_t (*ht_read)(struct u3_ht_ops *, u_int);
|
||||||
|
void (*ht_write)(struct u3_ht_ops *, u_int, uint32_t);
|
||||||
|
};
|
||||||
|
|
||||||
|
#define u3_ht_read(ptr,reg) (ptr)->ht_read(ptr, reg)
|
||||||
|
#define u3_ht_write(ptr,reg,val) (ptr)->ht_write(ptr, reg, val)
|
||||||
|
|
||||||
|
static struct u3_ht_ops *setup_u3_ht(uint32_t, uint32_t, int);
|
||||||
|
static int setup_u3_ht_workarounds(struct u3_ht_ops *);
|
||||||
|
|
||||||
|
static void u3_ht_enable_irq(struct pic_ops *, int, int);
|
||||||
|
static void u3_ht_disable_irq(struct pic_ops *, int);
|
||||||
|
static int u3_ht_get_irq(struct pic_ops *, int);
|
||||||
|
static void u3_ht_ack_irq(struct pic_ops *, int);
|
||||||
|
static void u3_ht_establish_irq(struct pic_ops *, int, int, int);
|
||||||
|
static void u3_ht_finish_setup(struct pic_ops *);
|
||||||
|
|
||||||
|
static int u3_ht_is_ht_irq(struct u3_ht_ops *, int);
|
||||||
|
static void u3_ht_establish_ht_irq(struct u3_ht_ops *, int, int);
|
||||||
|
static void u3_ht_enable_ht_irq(struct u3_ht_ops *, int);
|
||||||
|
static void u3_ht_ack_ht_irq(struct u3_ht_ops *, int);
|
||||||
|
|
||||||
|
static void u3_ht_set_priority(struct u3_ht_ops *, int, int);
|
||||||
|
static int u3_ht_read_irq(struct u3_ht_ops *, int);
|
||||||
|
static void u3_ht_eoi(struct u3_ht_ops *, int);
|
||||||
|
|
||||||
|
static uint32_t u3_ht_read_be(struct u3_ht_ops *, u_int);
|
||||||
|
static void u3_ht_write_be(struct u3_ht_ops *, u_int, uint32_t);
|
||||||
|
static uint32_t u3_ht_read_le(struct u3_ht_ops *, u_int);
|
||||||
|
static void u3_ht_write_le(struct u3_ht_ops *, u_int, uint32_t);
|
||||||
|
|
||||||
|
static const char *u3_compat[] = {
|
||||||
|
"u3",
|
||||||
|
NULL
|
||||||
|
};
|
||||||
|
|
||||||
|
const char *pic_compat[] = {
|
||||||
|
"chrp,open-pic",
|
||||||
|
"open-pic",
|
||||||
|
"openpic",
|
||||||
|
NULL
|
||||||
|
};
|
||||||
|
|
||||||
|
int init_u3_ht(void)
|
||||||
|
{
|
||||||
|
int u4, pic;
|
||||||
|
uint32_t reg[2];
|
||||||
|
uint32_t base, len, tmp;
|
||||||
|
int bigendian;
|
||||||
|
volatile uint8_t *unin_reg;
|
||||||
|
|
||||||
|
u4 = OF_finddevice("/u4");
|
||||||
|
if (u4 == -1)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
if (of_compatible(u4, u3_compat) == -1)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
pic = OF_child(u4);
|
||||||
|
while ((pic != 0) && (of_compatible(pic, pic_compat) == -1))
|
||||||
|
pic = OF_peer(pic);
|
||||||
|
|
||||||
|
if ((pic == -1) || (pic == 0))
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
if (OF_getprop(u4, "reg", reg, sizeof(reg)) != 8)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
base = reg[1];
|
||||||
|
|
||||||
|
/* Enable and reset PIC */
|
||||||
|
|
||||||
|
unin_reg = mapiodev(base, PAGE_SIZE, false);
|
||||||
|
KASSERT(unin_reg != NULL);
|
||||||
|
tmp = in32(unin_reg + 0xe0);
|
||||||
|
tmp |= 0x06;
|
||||||
|
out32(unin_reg + 0xe0, tmp);
|
||||||
|
|
||||||
|
bigendian = 0;
|
||||||
|
if (OF_getprop(pic, "big-endian", reg, 4) > -1)
|
||||||
|
bigendian = 1;
|
||||||
|
|
||||||
|
if (OF_getprop(pic, "reg", reg, 8) != 8)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
base = reg[0];
|
||||||
|
len = reg[1];
|
||||||
|
|
||||||
|
aprint_normal("found U3/U4 HT PIC at %08x\n", base);
|
||||||
|
|
||||||
|
setup_u3_ht(base, len, bigendian);
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct u3_ht_ops *
|
||||||
|
setup_u3_ht(uint32_t addr, uint32_t len, int bigendian)
|
||||||
|
{
|
||||||
|
struct u3_ht_ops *u3_ht;
|
||||||
|
struct pic_ops *pic;
|
||||||
|
int irq;
|
||||||
|
u_int x;
|
||||||
|
|
||||||
|
u3_ht = kmem_alloc(sizeof(struct u3_ht_ops), KM_SLEEP);
|
||||||
|
KASSERT(u3_ht != NULL);
|
||||||
|
bzero(u3_ht, sizeof(struct u3_ht_ops));
|
||||||
|
pic = &u3_ht->pic;
|
||||||
|
|
||||||
|
u3_ht->ht_base = mapiodev(addr, len, false);
|
||||||
|
KASSERT(u3_ht->ht_base != NULL);
|
||||||
|
|
||||||
|
if (bigendian) {
|
||||||
|
u3_ht->ht_read = u3_ht_read_be;
|
||||||
|
u3_ht->ht_write = u3_ht_write_be;
|
||||||
|
} else {
|
||||||
|
u3_ht->ht_read = u3_ht_read_le;
|
||||||
|
u3_ht->ht_write = u3_ht_write_le;
|
||||||
|
}
|
||||||
|
|
||||||
|
setup_u3_ht_workarounds(u3_ht);
|
||||||
|
|
||||||
|
/* Reset PIC */
|
||||||
|
|
||||||
|
x = u3_ht_read(u3_ht, OPENPIC_CONFIG);
|
||||||
|
u3_ht_write(u3_ht, OPENPIC_CONFIG, x | OPENPIC_CONFIG_RESET);
|
||||||
|
do {
|
||||||
|
x = u3_ht_read(u3_ht, OPENPIC_CONFIG);
|
||||||
|
} while (x & OPENPIC_CONFIG_RESET);
|
||||||
|
|
||||||
|
x = u3_ht_read(u3_ht, OPENPIC_FEATURE);
|
||||||
|
|
||||||
|
aprint_normal("OpenPIC Version 1.%d: "
|
||||||
|
"Supports %d CPUs and %d interrupt sources.\n",
|
||||||
|
x & 0xff, ((x & 0x1f00) >> 8) + 1, ((x & 0x07ff0000) >> 16) + 1);
|
||||||
|
|
||||||
|
pic->pic_numintrs = ((x & 0x07ff0000) >> 16) + 1;
|
||||||
|
pic->pic_cookie = (void *) addr;
|
||||||
|
pic->pic_enable_irq = u3_ht_enable_irq;
|
||||||
|
pic->pic_reenable_irq = u3_ht_enable_irq;
|
||||||
|
pic->pic_disable_irq = u3_ht_disable_irq;
|
||||||
|
pic->pic_get_irq = u3_ht_get_irq;
|
||||||
|
pic->pic_ack_irq = u3_ht_ack_irq;
|
||||||
|
pic->pic_establish_irq = u3_ht_establish_irq;
|
||||||
|
pic->pic_finish_setup = u3_ht_finish_setup;
|
||||||
|
strcpy(pic->pic_name, "openpic");
|
||||||
|
pic_add(pic);
|
||||||
|
|
||||||
|
u3_ht_set_priority(u3_ht, 0, 15);
|
||||||
|
|
||||||
|
for (irq = 0; irq < pic->pic_numintrs; irq++) {
|
||||||
|
x = irq;
|
||||||
|
x |= OPENPIC_IMASK;
|
||||||
|
x |= OPENPIC_POLARITY_NEGATIVE;
|
||||||
|
x |= OPENPIC_SENSE_LEVEL;
|
||||||
|
x |= 8 << OPENPIC_PRIORITY_SHIFT;
|
||||||
|
u3_ht_write(u3_ht, OPENPIC_SRC_VECTOR(irq), x);
|
||||||
|
u3_ht_write(u3_ht, OPENPIC_IDEST(irq), 1 << 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
x = u3_ht_read(u3_ht, OPENPIC_CONFIG);
|
||||||
|
x |= OPENPIC_CONFIG_8259_PASSTHRU_DISABLE;
|
||||||
|
u3_ht_write(u3_ht, OPENPIC_CONFIG, x);
|
||||||
|
|
||||||
|
u3_ht_write(u3_ht, OPENPIC_SPURIOUS_VECTOR, 0xff);
|
||||||
|
|
||||||
|
u3_ht_set_priority(u3_ht, 0, 0);
|
||||||
|
|
||||||
|
for (irq = 0; irq < pic->pic_numintrs; irq++) {
|
||||||
|
u3_ht_read_irq(u3_ht, 0);
|
||||||
|
u3_ht_eoi(u3_ht, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
return u3_ht;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
setup_u3_ht_workarounds(struct u3_ht_ops *u3_ht)
|
||||||
|
{
|
||||||
|
struct u3_ht_irqmap *irqmap = u3_ht->ht_irqmap;
|
||||||
|
int parent, child;
|
||||||
|
uint32_t reg[5], tmp;
|
||||||
|
uint8_t pos;
|
||||||
|
uint16_t cr, ht_cr;
|
||||||
|
int nirq, irq, i;
|
||||||
|
volatile uint8_t *ht_reg, *dev_reg, *base;
|
||||||
|
|
||||||
|
parent = OF_finddevice("/ht");
|
||||||
|
if (parent == -1)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
if (OF_getprop(parent, "reg", reg, 12) != 12)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
ht_reg = mapiodev(reg[1], reg[2], false);
|
||||||
|
KASSERT(ht_reg != NULL);
|
||||||
|
|
||||||
|
for (child = OF_child(parent); child != 0; child = OF_peer(child)) {
|
||||||
|
if (OF_getprop(child, "reg", reg, 4) != 4)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
dev_reg = ht_reg + (reg[0] & (OFW_PCI_PHYS_HI_DEVICEMASK |
|
||||||
|
OFW_PCI_PHYS_HI_FUNCTIONMASK));
|
||||||
|
|
||||||
|
tmp = in32rb(dev_reg + PCI_COMMAND_STATUS_REG);
|
||||||
|
if ((tmp & PCI_STATUS_CAPLIST_SUPPORT) == 0)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
for (pos = in8rb(dev_reg + PCI_CAPLISTPTR_REG);
|
||||||
|
pos != 0; pos = in8rb(dev_reg + pos + 0x01)) {
|
||||||
|
cr = in16rb(dev_reg + pos);
|
||||||
|
if (PCI_CAPLIST_CAP(cr) != 0x08)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
ht_cr = in16rb(dev_reg + pos + 0x02);
|
||||||
|
if ((ht_cr & 0xf800) == 0x8000)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pos == 0)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
base = dev_reg + pos;
|
||||||
|
|
||||||
|
out8rb(base + 0x02, 0x01);
|
||||||
|
nirq = in32rb(base + 0x04);
|
||||||
|
nirq = (nirq >> 16) & 0xff;
|
||||||
|
|
||||||
|
for (i = 0; i <= nirq; i++) {
|
||||||
|
out8rb(base + 0x02, 0x10 + (i << 1));
|
||||||
|
tmp = in32rb(base + 0x04);
|
||||||
|
irq = (tmp >> 16) & 0xff;
|
||||||
|
tmp |= 0x01;
|
||||||
|
out32rb(base + 0x04, tmp);
|
||||||
|
|
||||||
|
irqmap[irq].im_index = i;
|
||||||
|
irqmap[irq].im_level = 0;
|
||||||
|
irqmap[irq].im_base = base;
|
||||||
|
|
||||||
|
tmp = in32rb(dev_reg + PCI_ID_REG);
|
||||||
|
if (PCI_VENDOR(tmp) == 0x106b)
|
||||||
|
irqmap[irq].im_apple_base = dev_reg + 0x60;
|
||||||
|
else
|
||||||
|
irqmap[irq].im_apple_base = NULL;
|
||||||
|
|
||||||
|
out8rb(base + 0x02, 0x11 + (i << 1));
|
||||||
|
irqmap[irq].im_data = in32rb(base + 0x04);
|
||||||
|
irqmap[irq].im_data |= (1 << 31);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
u3_ht_enable_irq(struct pic_ops *pic, int irq, int type)
|
||||||
|
{
|
||||||
|
struct u3_ht_ops *u3_ht = (struct u3_ht_ops *)pic;
|
||||||
|
u_int x;
|
||||||
|
|
||||||
|
x = u3_ht_read(u3_ht, OPENPIC_SRC_VECTOR(irq));
|
||||||
|
x &= ~OPENPIC_IMASK;
|
||||||
|
u3_ht_write(u3_ht, OPENPIC_SRC_VECTOR(irq), x);
|
||||||
|
|
||||||
|
if (u3_ht_is_ht_irq(u3_ht, irq))
|
||||||
|
u3_ht_enable_ht_irq(u3_ht, irq);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
u3_ht_disable_irq(struct pic_ops *pic, int irq)
|
||||||
|
{
|
||||||
|
struct u3_ht_ops *u3_ht = (struct u3_ht_ops *)pic;
|
||||||
|
u_int x;
|
||||||
|
|
||||||
|
x = u3_ht_read(u3_ht, OPENPIC_SRC_VECTOR(irq));
|
||||||
|
x |= OPENPIC_IMASK;
|
||||||
|
u3_ht_write(u3_ht, OPENPIC_SRC_VECTOR(irq), x);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
u3_ht_get_irq(struct pic_ops *pic, int mode)
|
||||||
|
{
|
||||||
|
struct u3_ht_ops *u3_ht = (struct u3_ht_ops *)pic;
|
||||||
|
|
||||||
|
return u3_ht_read_irq(u3_ht, curcpu()->ci_index);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
u3_ht_ack_irq(struct pic_ops *pic, int irq)
|
||||||
|
{
|
||||||
|
struct u3_ht_ops *u3_ht = (struct u3_ht_ops *)pic;
|
||||||
|
|
||||||
|
if (u3_ht_is_ht_irq(u3_ht, irq))
|
||||||
|
u3_ht_ack_ht_irq(u3_ht, irq);
|
||||||
|
|
||||||
|
u3_ht_eoi(u3_ht, curcpu()->ci_index);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
u3_ht_establish_irq(struct pic_ops *pic, int irq, int type, int pri)
|
||||||
|
{
|
||||||
|
struct u3_ht_ops *u3_ht = (struct u3_ht_ops *)pic;
|
||||||
|
int realpri = max(1, min(15, pri));
|
||||||
|
uint32_t x;
|
||||||
|
|
||||||
|
x = irq;
|
||||||
|
x |= OPENPIC_IMASK;
|
||||||
|
|
||||||
|
if (u3_ht_is_ht_irq(u3_ht, irq)) {
|
||||||
|
x |= OPENPIC_POLARITY_POSITIVE |
|
||||||
|
OPENPIC_SENSE_EDGE;
|
||||||
|
} else {
|
||||||
|
if (irq == 0 || type == IST_EDGE_RISING || type == IST_LEVEL_HIGH)
|
||||||
|
x |= OPENPIC_POLARITY_POSITIVE;
|
||||||
|
else
|
||||||
|
x |= OPENPIC_POLARITY_NEGATIVE;
|
||||||
|
|
||||||
|
if (type == IST_EDGE_FALLING || type == IST_EDGE_RISING)
|
||||||
|
x |= OPENPIC_SENSE_EDGE;
|
||||||
|
else
|
||||||
|
x |= OPENPIC_SENSE_LEVEL;
|
||||||
|
}
|
||||||
|
|
||||||
|
x |= realpri << OPENPIC_PRIORITY_SHIFT;
|
||||||
|
u3_ht_write(u3_ht, OPENPIC_SRC_VECTOR(irq), x);
|
||||||
|
|
||||||
|
if (u3_ht_is_ht_irq(u3_ht, irq))
|
||||||
|
u3_ht_establish_ht_irq(u3_ht, irq, type);
|
||||||
|
|
||||||
|
aprint_debug("%s: setting IRQ %d to priority %d\n", __func__, irq,
|
||||||
|
realpri);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
u3_ht_finish_setup(struct pic_ops *pic)
|
||||||
|
{
|
||||||
|
struct u3_ht_ops *u3_ht = (struct u3_ht_ops *)pic;
|
||||||
|
uint32_t cpumask = 0;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
#ifdef OPENPIC_DISTRIBUTE
|
||||||
|
for (i = 0; i < ncpu; i++)
|
||||||
|
cpumask |= (1 << cpu_info[i].ci_cpuid);
|
||||||
|
#else
|
||||||
|
cpumask = 1;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
for (i = 0; i < pic->pic_numintrs; i++)
|
||||||
|
u3_ht_write(u3_ht, OPENPIC_IDEST(i), cpumask);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
u3_ht_is_ht_irq(struct u3_ht_ops *u3_ht, int irq)
|
||||||
|
{
|
||||||
|
return (irq < 128) && (u3_ht->ht_irqmap[irq].im_base != NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
u3_ht_establish_ht_irq(struct u3_ht_ops *u3_ht, int irq, int type)
|
||||||
|
{
|
||||||
|
struct u3_ht_irqmap *irqmap = &u3_ht->ht_irqmap[irq];
|
||||||
|
u_int x;
|
||||||
|
|
||||||
|
out8rb(irqmap->im_base + 0x02, 0x10 + (irqmap->im_index << 1));
|
||||||
|
|
||||||
|
x = in32rb(irqmap->im_base + 0x04);
|
||||||
|
x &= ~0x23;
|
||||||
|
|
||||||
|
if (type == IST_LEVEL_HIGH || type == IST_LEVEL_LOW) {
|
||||||
|
irqmap->im_level = 1;
|
||||||
|
x |= 0x22;
|
||||||
|
} else {
|
||||||
|
irqmap->im_level = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
out32rb(irqmap->im_base + 0x04, x);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
u3_ht_enable_ht_irq(struct u3_ht_ops *u3_ht, int irq)
|
||||||
|
{
|
||||||
|
struct u3_ht_irqmap *irqmap = &u3_ht->ht_irqmap[irq];
|
||||||
|
u_int x;
|
||||||
|
|
||||||
|
out8rb(irqmap->im_base + 0x02, 0x10 + (irqmap->im_index << 1));
|
||||||
|
x = in32rb(irqmap->im_base + 0x04);
|
||||||
|
x &= ~0x01;
|
||||||
|
out32rb(irqmap->im_base + 0x04, x);
|
||||||
|
|
||||||
|
u3_ht_ack_ht_irq(u3_ht, irq);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
u3_ht_ack_ht_irq(struct u3_ht_ops *u3_ht, int irq)
|
||||||
|
{
|
||||||
|
struct u3_ht_irqmap *irqmap = &u3_ht->ht_irqmap[irq];
|
||||||
|
|
||||||
|
if (irqmap->im_level != 0) {
|
||||||
|
if (irqmap->im_apple_base != NULL) {
|
||||||
|
out32rb(irqmap->im_apple_base + ((irqmap->im_index >> 3) & ~0x03),
|
||||||
|
1 << (irqmap->im_index & 0x1f));
|
||||||
|
} else {
|
||||||
|
out8rb(irqmap->im_base + 0x02, 0x11 + (irqmap->im_index << 1));
|
||||||
|
out32rb(irqmap->im_base + 0x04, irqmap->im_data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
u3_ht_set_priority(struct u3_ht_ops *u3_ht, int cpu, int pri)
|
||||||
|
{
|
||||||
|
u_int x;
|
||||||
|
|
||||||
|
x = u3_ht_read(u3_ht, OPENPIC_CPU_PRIORITY(cpu));
|
||||||
|
x &= ~OPENPIC_CPU_PRIORITY_MASK;
|
||||||
|
x |= pri;
|
||||||
|
u3_ht_write(u3_ht, OPENPIC_CPU_PRIORITY(cpu), x);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
u3_ht_read_irq(struct u3_ht_ops *u3_ht, int cpu)
|
||||||
|
{
|
||||||
|
return u3_ht_read(u3_ht, OPENPIC_IACK(cpu)) & OPENPIC_VECTOR_MASK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
u3_ht_eoi(struct u3_ht_ops *u3_ht, int cpu)
|
||||||
|
{
|
||||||
|
u3_ht_write(u3_ht, OPENPIC_EOI(cpu), 0);
|
||||||
|
u3_ht_read(u3_ht, OPENPIC_EOI(cpu));
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint32_t
|
||||||
|
u3_ht_read_be(struct u3_ht_ops *u3_ht, u_int reg)
|
||||||
|
{
|
||||||
|
volatile uint8_t *addr = u3_ht->ht_base + reg;
|
||||||
|
|
||||||
|
return in32(addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
u3_ht_write_be(struct u3_ht_ops *u3_ht, u_int reg, uint32_t val)
|
||||||
|
{
|
||||||
|
volatile uint8_t *addr = u3_ht->ht_base + reg;
|
||||||
|
|
||||||
|
out32(addr, val);
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint32_t
|
||||||
|
u3_ht_read_le(struct u3_ht_ops *u3_ht, u_int reg)
|
||||||
|
{
|
||||||
|
volatile uint8_t *addr = u3_ht->ht_base + reg;
|
||||||
|
|
||||||
|
return in32rb(addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
u3_ht_write_le(struct u3_ht_ops *u3_ht, u_int reg, uint32_t val)
|
||||||
|
{
|
||||||
|
volatile uint8_t *addr = u3_ht->ht_base + reg;
|
||||||
|
|
||||||
|
out32rb(addr, val);
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user