mirror of
https://github.com/phabrics/Run-Sun3-SunOS-4.1.1.git
synced 2026-04-29 11:02:59 -04:00
1128 lines
34 KiB
C
1128 lines
34 KiB
C
/* $Id: sun-mie.c,v 1.4 2010/06/05 13:57:27 fredette Exp $ */
|
|
|
|
/* bus/multibus/sun_mie.c - implementation of the Sun Intel Ethernet Multibus emulation: */
|
|
|
|
/*
|
|
* Copyright (c) 2003 Matt Fredette
|
|
* 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.
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
* must display the following acknowledgement:
|
|
* This product includes software developed by Matt Fredette.
|
|
* 4. The name of the author may not be used to endorse or promote products
|
|
* derived from this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 AUTHOR 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 <tme/common.h>
|
|
_TME_RCSID("$Id: sun-mie.c,v 1.4 2010/06/05 13:57:27 fredette Exp $");
|
|
|
|
/* includes: */
|
|
#include <tme/element.h>
|
|
#undef TME_BUS_VERSION
|
|
#define TME_BUS_VERSION TME_X_VERSION(0, 0)
|
|
#include <tme/generic/bus.h>
|
|
#undef TME_I825X6_VERSION
|
|
#define TME_I825X6_VERSION TME_X_VERSION(0, 0)
|
|
#include <tme/ic/i825x6.h>
|
|
|
|
/* macros: */
|
|
|
|
/* the amount of memory on the board: */
|
|
#define TME_SUN_MIE_MEMSIZE (256 * 1024)
|
|
|
|
/* the board page size: */
|
|
#define TME_SUN_MIE_PAGESIZE (1024)
|
|
|
|
/* the number of page map entries: */
|
|
#define TME_SUN_MIE_PGMAP_COUNT (1024)
|
|
|
|
/* the number of active TLB entries we can have per page map entry: */
|
|
#define TME_SUN_MIE_PGMAP_TLBS (4)
|
|
|
|
/* register offsets and sizes: */
|
|
#define TME_SUN_MIE_REG_PGMAP (0)
|
|
#define TME_SUN_MIE_SIZ_PGMAP (TME_SUN_MIE_PGMAP_COUNT * sizeof(tme_uint16_t))
|
|
#define TME_SUN_MIE_REG_PROM (TME_SUN_MIE_REG_PGMAP + TME_SUN_MIE_SIZ_PGMAP)
|
|
#define TME_SUN_MIE_SIZ_PROM (32 * sizeof(tme_uint16_t))
|
|
#define TME_SUN_MIE_REG_CSR (TME_SUN_MIE_REG_PROM + TME_SUN_MIE_SIZ_PROM)
|
|
#define TME_SUN_MIE_SIZ_CSR (sizeof(tme_uint16_t))
|
|
#define TME_SUN_MIE_REG_PCR (TME_SUN_MIE_REG_CSR + TME_SUN_MIE_SIZ_CSR + sizeof(tme_uint16_t))
|
|
#define TME_SUN_MIE_SIZ_PCR (sizeof(tme_uint16_t))
|
|
#define TME_SUN_MIE_REG_PE_ALO (TME_SUN_MIE_REG_PCR + TME_SUN_MIE_SIZ_PCR)
|
|
#define TME_SUN_MIE_SIZ_PE_ALO (sizeof(tme_uint16_t))
|
|
#define TME_SUN_MIE_SIZ_REGS (TME_SUN_MIE_REG_PE_ALO + TME_SUN_MIE_SIZ_PE_ALO)
|
|
|
|
/* the bits in a pagemap entry: */
|
|
#define TME_SUN_MIE_PGMAP_SWAB (0x8000)
|
|
/* 0x4000 unused */
|
|
#define TME_SUN_MIE_PGMAP_P2MEM (0x2000)
|
|
/* 0x1000 unused */
|
|
#define TME_SUN_MIE_PGMAP_PFNUM (0x0fff)
|
|
|
|
/* the bits in the Control/Status Register: */
|
|
#define TME_SUN_MIE_CSR_RESET (0x8000)
|
|
#define TME_SUN_MIE_CSR_NOLOOP (0x4000)
|
|
#define TME_SUN_MIE_CSR_CA (0x2000)
|
|
#define TME_SUN_MIE_CSR_IE (0x1000)
|
|
#define TME_SUN_MIE_CSR_PIE (0x0800)
|
|
/* 0x0400 unused */
|
|
#define TME_SUN_MIE_CSR_PE (0x0200)
|
|
#define TME_SUN_MIE_CSR_INTR (0x0100)
|
|
/* 0x0080 unused */
|
|
/* 0x0040 unused */
|
|
#define TME_SUN_MIE_CSR_P2MEM (0x0020)
|
|
#define TME_SUN_MIE_CSR_BIGRAM (0x0010)
|
|
#define TME_SUN_MIE_CSR_MPMHI (0x000f)
|
|
#define TME_SUN_MIE_CSR_READONLY (0x0400 \
|
|
| TME_SUN_MIE_CSR_PE \
|
|
| TME_SUN_MIE_CSR_INTR \
|
|
| 0x0080 \
|
|
| 0x0040 \
|
|
| TME_SUN_MIE_CSR_BIGRAM\
|
|
| TME_SUN_MIE_CSR_MPMHI)
|
|
|
|
/* the bits in the Parity Control Register: */
|
|
#define TME_SUN_MIE_PCR_PEACK (0x0100)
|
|
#define TME_SUN_MIE_PCR_PESRC (0x0080)
|
|
#define TME_SUN_MIE_PCR_PEBYTE (0x0040)
|
|
/* 0x0020 unused */
|
|
/* 0x0010 unused */
|
|
#define TME_SUN_MIE_PCR_PE_AHI (0x000f)
|
|
#define TME_SUN_MIE_PCR_READONLY (~TME_SUN_MIE_PCR_PEACK)
|
|
|
|
/* the size of the memory port: */
|
|
#define TME_SUN_MIE_SIZ_MBM (0x10000)
|
|
|
|
/* these get and put the CSR: */
|
|
#define TME_SUN_MIE_CSR_GET(sun_mie) \
|
|
tme_betoh_u16(*((tme_uint16_t *) &(sun_mie)->tme_sun_mie_regs[TME_SUN_MIE_REG_CSR]))
|
|
#define TME_SUN_MIE_CSR_PUT(sun_mie, csr) \
|
|
(*((tme_uint16_t *) &(sun_mie)->tme_sun_mie_regs[TME_SUN_MIE_REG_CSR]) = tme_htobe_u16(csr))
|
|
|
|
/* these get and put the PCR: */
|
|
#define TME_SUN_MIE_PCR_GET(sun_mie) \
|
|
tme_betoh_u16(*((tme_uint16_t *) &(sun_mie)->tme_sun_mie_regs[TME_SUN_MIE_REG_PCR]))
|
|
#define TME_SUN_MIE_PCR_PUT(sun_mie, pcr) \
|
|
(*((tme_uint16_t *) &(sun_mie)->tme_sun_mie_regs[TME_SUN_MIE_REG_PCR]) = tme_htobe_u16(pcr))
|
|
|
|
/* the callout flags: */
|
|
#define TME_SUN_MIE_CALLOUT_CHECK (0)
|
|
#define TME_SUN_MIE_CALLOUT_RUNNING TME_BIT(0)
|
|
#define TME_SUN_MIE_CALLOUTS_MASK (-2)
|
|
#define TME_SUN_MIE_CALLOUT_SIGNALS TME_BIT(1)
|
|
#define TME_SUN_MIE_CALLOUT_INT TME_BIT(2)
|
|
|
|
/* structures: */
|
|
|
|
/* the card: */
|
|
struct tme_sun_mie {
|
|
|
|
/* backpointer to our element: */
|
|
struct tme_element *tme_sun_mie_element;
|
|
|
|
/* the mutex protecting the card: */
|
|
tme_mutex_t tme_sun_mie_mutex;
|
|
|
|
/* the rwlock protecting the card: */
|
|
tme_rwlock_t tme_sun_mie_rwlock;
|
|
|
|
/* the bus connection for the card's registers: */
|
|
struct tme_bus_connection *tme_sun_mie_conn_regs;
|
|
|
|
/* the bus connection for the card's memory: */
|
|
struct tme_bus_connection *tme_sun_mie_conn_memory;
|
|
|
|
/* the bus connection for the card's i825x6 chip: */
|
|
struct tme_bus_connection *tme_sun_mie_conn_i825x6;
|
|
|
|
/* the callout flags: */
|
|
int tme_sun_mie_callout_flags;
|
|
|
|
int tme_sun_mie_ca_callout_delay;
|
|
int tme_sun_mie_ca_callout_pending;
|
|
int tme_sun_mie_pending_deassert_intr;
|
|
|
|
/* if our interrupt line is currently asserted: */
|
|
int tme_sun_mie_int_asserted;
|
|
|
|
/* it's easiest to just model the board registers as a chunk of memory: */
|
|
tme_uint8_t tme_sun_mie_regs[TME_SUN_MIE_SIZ_REGS];
|
|
|
|
/* the board memory really is a chunk of memory: */
|
|
tme_uint8_t tme_sun_mie_memory[TME_SUN_MIE_MEMSIZE];
|
|
|
|
/* the active TLB entries for each page map entry on the board: */
|
|
struct tme_token *tme_sun_mie_tlb_tokens[TME_SUN_MIE_PGMAP_COUNT * TME_SUN_MIE_PGMAP_TLBS];
|
|
unsigned int tme_sun_mie_tlb_head[TME_SUN_MIE_PGMAP_COUNT];
|
|
|
|
/* the i825x6 image of the CSR: */
|
|
tme_uint16_t tme_sun_mie_csr_i825x6;
|
|
|
|
#ifndef TME_NO_LOG
|
|
tme_uint16_t tme_sun_mie_last_log_csr;
|
|
#endif /* !TME_NO_LOG */
|
|
};
|
|
|
|
/* a sun_mie internal bus connection: */
|
|
struct tme_sun_mie_connection {
|
|
|
|
/* the external bus connection: */
|
|
struct tme_bus_connection tme_sun_mie_connection;
|
|
|
|
/* this is nonzero if a TME_CONNECTION_BUS_GENERIC chip connection
|
|
is for the registers: */
|
|
tme_uint8_t tme_sun_mie_connection_regs;
|
|
|
|
/* if this connection is for the memory port, this is the high
|
|
nibble (A19..A16) of that port's connection: */
|
|
tme_uint8_t tme_sun_mie_connection_mpmhi;
|
|
};
|
|
|
|
/* globals: */
|
|
|
|
/* our bus signals sets: */
|
|
static const struct tme_bus_signals _tme_sun_mie_bus_signals_generic = TME_BUS_SIGNALS_GENERIC;
|
|
static const struct tme_bus_signals _tme_sun_mie_bus_signals_i825x6 = TME_BUS_SIGNALS_I825X6;
|
|
|
|
extern int printf(const char *format, ...);
|
|
|
|
/* the sun_mie callout function. it must be called with the mutex locked: */
|
|
static void
|
|
_tme_sun_mie_callout(struct tme_sun_mie *sun_mie, int new_callouts)
|
|
{
|
|
struct tme_bus_connection *conn_i825x6;
|
|
struct tme_bus_connection *conn_bus;
|
|
tme_uint16_t csr, csr_diff;
|
|
unsigned int signal, level;
|
|
int callouts, later_callouts;
|
|
int rc;
|
|
int int_asserted;
|
|
|
|
/* add in any new callouts: */
|
|
sun_mie->tme_sun_mie_callout_flags |= new_callouts;
|
|
|
|
/* if this function is already running in another thread, simply
|
|
return now. the other thread will do our work: */
|
|
if (sun_mie->tme_sun_mie_callout_flags & TME_SUN_MIE_CALLOUT_RUNNING) {
|
|
return;
|
|
}
|
|
|
|
/* callouts are now running: */
|
|
sun_mie->tme_sun_mie_callout_flags |= TME_SUN_MIE_CALLOUT_RUNNING;
|
|
|
|
/* assume that we won't need any later callouts: */
|
|
later_callouts = 0;
|
|
|
|
/* loop while callouts are needed: */
|
|
for (; (callouts = sun_mie->tme_sun_mie_callout_flags) & TME_SUN_MIE_CALLOUTS_MASK; ) {
|
|
|
|
/* clear the needed callouts: */
|
|
sun_mie->tme_sun_mie_callout_flags = callouts & ~TME_SUN_MIE_CALLOUTS_MASK;
|
|
callouts &= TME_SUN_MIE_CALLOUTS_MASK;
|
|
|
|
/* if we need to call out one or more signals to the i825x6: */
|
|
if (callouts & TME_SUN_MIE_CALLOUT_SIGNALS) {
|
|
|
|
/* get the current CSR value: */
|
|
csr = TME_SUN_MIE_CSR_GET(sun_mie);
|
|
|
|
/* get the next signal to call out to the i825x6: */
|
|
csr_diff = ((csr
|
|
^ sun_mie->tme_sun_mie_csr_i825x6)
|
|
& (TME_SUN_MIE_CSR_RESET
|
|
| TME_SUN_MIE_CSR_NOLOOP
|
|
| TME_SUN_MIE_CSR_CA));
|
|
csr_diff = (csr_diff ^ (csr_diff - 1)) & csr_diff;
|
|
|
|
/* if there is a signal to call out: */
|
|
if (csr_diff != 0) {
|
|
|
|
/* assume that if the signal's bit is set in the CSR, it will
|
|
be asserted: */
|
|
level = csr & csr_diff;
|
|
|
|
/* assume that we're calling out an i825x6 signal: */
|
|
signal = (_tme_sun_mie_bus_signals_generic.tme_bus_signals_first
|
|
+ TME_BUS_SIGNAL_X(_tme_sun_mie_bus_signals_generic.tme_bus_signals_count));
|
|
|
|
/* dispatch on the CSR bit: */
|
|
switch (csr_diff) {
|
|
default:
|
|
assert (FALSE);
|
|
case TME_SUN_MIE_CSR_RESET:
|
|
signal = TME_BUS_SIGNAL_RESET;
|
|
break;
|
|
case TME_SUN_MIE_CSR_NOLOOP:
|
|
signal += TME_I825X6_SIGNAL_LOOP;
|
|
level = !level;
|
|
break;
|
|
case TME_SUN_MIE_CSR_CA:
|
|
signal += TME_I825X6_SIGNAL_CA;
|
|
break;
|
|
}
|
|
|
|
/* create a real signal level value: */
|
|
level = (level
|
|
? TME_BUS_SIGNAL_LEVEL_ASSERTED
|
|
: TME_BUS_SIGNAL_LEVEL_NEGATED);
|
|
|
|
/* get this card's i825x6 connection: */
|
|
conn_i825x6 = sun_mie->tme_sun_mie_conn_i825x6;
|
|
|
|
/* unlock the mutex: */
|
|
tme_mutex_unlock(&sun_mie->tme_sun_mie_mutex);
|
|
|
|
#if 0
|
|
printf("callout conn_i825x6->tme_bus_signal; signal %x, level %x; %x\n",
|
|
(int)signal, (int)level, (int)(signal | level));
|
|
#endif
|
|
|
|
if (signal)
|
|
/* do the callout: */
|
|
rc = (conn_i825x6 != NULL
|
|
? ((*conn_i825x6->tme_bus_signal)
|
|
(conn_i825x6,
|
|
signal | level))
|
|
: TME_OK);
|
|
|
|
/* lock the mutex: */
|
|
tme_mutex_lock(&sun_mie->tme_sun_mie_mutex);
|
|
|
|
/* if the callout was unsuccessful, remember that at some later
|
|
time this callout should be attempted again: */
|
|
if (rc != TME_OK) {
|
|
later_callouts |= TME_SUN_MIE_CALLOUT_SIGNALS;
|
|
}
|
|
|
|
/* otherwise, the callout was successful: */
|
|
else {
|
|
|
|
/* update the i825x6 image of the CSR: */
|
|
sun_mie->tme_sun_mie_csr_i825x6 =
|
|
((sun_mie->tme_sun_mie_csr_i825x6 & ~csr_diff)
|
|
| (csr & csr_diff));
|
|
|
|
/* there may be more signals to call out, so attempt this
|
|
callout again now: */
|
|
sun_mie->tme_sun_mie_callout_flags |= TME_SUN_MIE_CALLOUT_SIGNALS;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* if we need to call out a possible change to our interrupt
|
|
signal: */
|
|
if (callouts & TME_SUN_MIE_CALLOUT_INT) {
|
|
|
|
/* get the current CSR value: */
|
|
csr = TME_SUN_MIE_CSR_GET(sun_mie);
|
|
|
|
/* see if the interrupt signal should be asserted or negated: */
|
|
int_asserted = ((csr & (TME_SUN_MIE_CSR_IE
|
|
| TME_SUN_MIE_CSR_INTR))
|
|
== (TME_SUN_MIE_CSR_IE
|
|
| TME_SUN_MIE_CSR_INTR));
|
|
|
|
/* if the interrupt signal doesn't already have the right state: */
|
|
if (!int_asserted != !sun_mie->tme_sun_mie_int_asserted) {
|
|
|
|
/* get our bus connection: */
|
|
conn_bus = sun_mie->tme_sun_mie_conn_regs;
|
|
|
|
/* unlock our mutex: */
|
|
tme_mutex_unlock(&sun_mie->tme_sun_mie_mutex);
|
|
|
|
/* call out the bus interrupt signal edge: */
|
|
rc = (conn_bus != NULL
|
|
? ((*conn_bus->tme_bus_signal)
|
|
(conn_bus,
|
|
TME_BUS_SIGNAL_INT_UNSPEC
|
|
| (int_asserted
|
|
? TME_BUS_SIGNAL_LEVEL_ASSERTED
|
|
: TME_BUS_SIGNAL_LEVEL_NEGATED)))
|
|
: TME_OK);
|
|
|
|
/* lock our mutex: */
|
|
tme_mutex_lock(&sun_mie->tme_sun_mie_mutex);
|
|
|
|
/* if this callout was successful, note the new state of the
|
|
interrupt signal: */
|
|
if (rc == TME_OK) {
|
|
#if 0
|
|
printf("IE assert int %x, csr %x\n", (int)int_asserted, (int)csr);
|
|
#endif
|
|
sun_mie->tme_sun_mie_int_asserted = int_asserted;
|
|
}
|
|
|
|
/* otherwise, remember that at some later time this callout
|
|
should be attempted again: */
|
|
else {
|
|
later_callouts |= TME_SUN_MIE_CALLOUT_INT;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* put in any later callouts, and clear that callouts are running: */
|
|
sun_mie->tme_sun_mie_callout_flags = later_callouts;
|
|
}
|
|
|
|
/* the sun_mie bus cycle handler for the board memory: */
|
|
static int
|
|
_tme_sun_mie_bus_cycle(void *_sun_mie,
|
|
struct tme_bus_cycle *cycle_init)
|
|
{
|
|
struct tme_sun_mie *sun_mie;
|
|
|
|
/* recover our data structure: */
|
|
sun_mie = (struct tme_sun_mie *) _sun_mie;
|
|
|
|
/* run the cycle: */
|
|
tme_bus_cycle_xfer_memory(cycle_init,
|
|
sun_mie->tme_sun_mie_memory,
|
|
TME_SUN_MIE_MEMSIZE - 1);
|
|
|
|
/* no faults: */
|
|
return (TME_OK);
|
|
}
|
|
|
|
/* the sun_mie bus cycle handler for the board registers: */
|
|
static int
|
|
_tme_sun_mie_bus_cycle_regs(void *_sun_mie,
|
|
struct tme_bus_cycle *cycle_init)
|
|
{
|
|
struct tme_sun_mie *sun_mie;
|
|
unsigned int pgmap_i;
|
|
unsigned int pgmap_j;
|
|
unsigned int tlb_i;
|
|
unsigned int tlb_j;
|
|
tme_uint16_t csr_old, csr_new, csr_diff;
|
|
tme_uint16_t pcr_old, pcr_new, pcr_diff;
|
|
int new_callouts;
|
|
|
|
#if 0
|
|
printf("mie_bus_cycle_regs; type %x, addr %x, size %d, %02x %02x\n",
|
|
(int)cycle_init->tme_bus_cycle_type,
|
|
(int)cycle_init->tme_bus_cycle_address,
|
|
(int)cycle_init->tme_bus_cycle_size,
|
|
(int)cycle_init->tme_bus_cycle_buffer[0],
|
|
(int)cycle_init->tme_bus_cycle_buffer[1]);
|
|
#endif
|
|
|
|
/* recover our data structure: */
|
|
sun_mie = (struct tme_sun_mie *) _sun_mie;
|
|
|
|
/* assume we won't need any new callouts: */
|
|
new_callouts = 0;
|
|
|
|
/* lock the mutex: */
|
|
tme_mutex_lock(&sun_mie->tme_sun_mie_mutex);
|
|
|
|
/* if this is a write cycle and the address falls within one or more
|
|
page map entries, invalidate the TLBs associated with those
|
|
entries: */
|
|
if ((cycle_init->tme_bus_cycle_type
|
|
& TME_BUS_CYCLE_WRITE)
|
|
&& (TME_SUN_MIE_REG_PGMAP
|
|
<= cycle_init->tme_bus_cycle_address)
|
|
&& (cycle_init->tme_bus_cycle_address
|
|
< (TME_SUN_MIE_REG_PGMAP
|
|
+ TME_SUN_MIE_SIZ_PGMAP))) {
|
|
|
|
/* get the range of page map entries: */
|
|
pgmap_i
|
|
= (cycle_init->tme_bus_cycle_address
|
|
/ sizeof(tme_uint16_t));
|
|
pgmap_j
|
|
= ((cycle_init->tme_bus_cycle_address
|
|
+ cycle_init->tme_bus_cycle_size
|
|
+ sizeof(tme_uint16_t)
|
|
- 1)
|
|
/ sizeof(tme_uint16_t));
|
|
pgmap_j = TME_MIN(pgmap_j, TME_SUN_MIE_PGMAP_COUNT);
|
|
|
|
/* get the range of TLB handles: */
|
|
tlb_i = pgmap_i * TME_SUN_MIE_PGMAP_TLBS;
|
|
tlb_j = pgmap_j * TME_SUN_MIE_PGMAP_TLBS;
|
|
|
|
/* invalidate the TLB entries: */
|
|
for (; tlb_i < tlb_j; tlb_i++) {
|
|
if (sun_mie->tme_sun_mie_tlb_tokens[tlb_i] != NULL) {
|
|
tme_token_invalidate(sun_mie->tme_sun_mie_tlb_tokens[tlb_i]);
|
|
sun_mie->tme_sun_mie_tlb_tokens[tlb_i] = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* get the previous CSR and PCR values: */
|
|
csr_old = TME_SUN_MIE_CSR_GET(sun_mie);
|
|
pcr_old = TME_SUN_MIE_PCR_GET(sun_mie);
|
|
|
|
/* unless this address falls within the PROM, run the cycle: */
|
|
if ((cycle_init->tme_bus_cycle_address
|
|
< TME_SUN_MIE_REG_PROM)
|
|
|| (cycle_init->tme_bus_cycle_address
|
|
>= (TME_SUN_MIE_REG_PROM
|
|
+ TME_SUN_MIE_SIZ_PROM))) {
|
|
tme_bus_cycle_xfer_memory(cycle_init,
|
|
sun_mie->tme_sun_mie_regs,
|
|
TME_SUN_MIE_SIZ_REGS - 1);
|
|
}
|
|
|
|
/* get the current CSR and PCR values, and put back any bits that
|
|
software can't change: */
|
|
csr_new = ((TME_SUN_MIE_CSR_GET(sun_mie)
|
|
& ~TME_SUN_MIE_CSR_READONLY)
|
|
| (csr_old
|
|
& TME_SUN_MIE_CSR_READONLY));
|
|
#if 0
|
|
printf("csr_new %x, csr_old %x, csr %x\n",
|
|
(int)csr_new, (int)csr_old, (int)TME_SUN_MIE_CSR_GET(sun_mie));
|
|
#endif
|
|
TME_SUN_MIE_CSR_PUT(sun_mie, csr_new);
|
|
pcr_new = ((TME_SUN_MIE_PCR_GET(sun_mie)
|
|
& ~TME_SUN_MIE_PCR_READONLY)
|
|
| (pcr_old
|
|
& TME_SUN_MIE_PCR_READONLY));
|
|
TME_SUN_MIE_PCR_PUT(sun_mie, pcr_new);
|
|
|
|
/* get the sets of CSR and PCR bits that have changed: */
|
|
csr_diff = (csr_old ^ csr_new);
|
|
pcr_diff = (pcr_old ^ pcr_new);
|
|
|
|
/* if this is a RESET, NOLOOP or CA change, possibly call out the
|
|
appropriate signal change to the i825x6: */
|
|
if (csr_diff & (TME_SUN_MIE_CSR_RESET
|
|
| TME_SUN_MIE_CSR_NOLOOP
|
|
| TME_SUN_MIE_CSR_CA)) {
|
|
new_callouts |= TME_SUN_MIE_CALLOUT_SIGNALS;
|
|
}
|
|
|
|
/* if this is an interrupt mask change, possibly call out an
|
|
interrupt signal change to the bus: */
|
|
if (csr_diff & TME_SUN_MIE_CSR_IE) {
|
|
new_callouts |= TME_SUN_MIE_CALLOUT_INT;
|
|
}
|
|
|
|
/* abort on any attempt to enable the P2 bus: */
|
|
if (csr_new & TME_SUN_MIE_CSR_P2MEM) {
|
|
abort ();
|
|
}
|
|
|
|
/* if this is acknowledging a parity error: */
|
|
if (pcr_diff & TME_SUN_MIE_PCR_PEACK) {
|
|
/* nothing to do */
|
|
}
|
|
|
|
#if 0
|
|
printf("IE now csr %x, tme_sun_mie_int_asserted %x\n",
|
|
(int)csr_new, (int)sun_mie->tme_sun_mie_int_asserted);
|
|
#endif
|
|
|
|
#ifndef TME_NO_LOG
|
|
if (csr_new != sun_mie->tme_sun_mie_last_log_csr) {
|
|
sun_mie->tme_sun_mie_last_log_csr = csr_new;
|
|
tme_log(&sun_mie->tme_sun_mie_element->tme_element_log_handle,
|
|
1000, TME_OK,
|
|
(&sun_mie->tme_sun_mie_element->tme_element_log_handle,
|
|
"csr now 0x%04x",
|
|
csr_new));
|
|
}
|
|
#endif /* !TME_NO_LOG */
|
|
|
|
#if 1
|
|
if (cycle_init->tme_bus_cycle_type & TME_BUS_CYCLE_WRITE)
|
|
if (sun_mie->tme_sun_mie_pending_deassert_intr) {
|
|
sun_mie->tme_sun_mie_pending_deassert_intr = 0;
|
|
/* printf("turning off pending intr\n");
|
|
csr_new &= ~TME_SUN_MIE_CSR_INTR;
|
|
TME_SUN_MIE_CSR_PUT(sun_mie, csr_new); */
|
|
}
|
|
#endif
|
|
|
|
/* make any new callouts: */
|
|
_tme_sun_mie_callout(sun_mie, new_callouts);
|
|
|
|
/* unlock the mutex: */
|
|
tme_mutex_unlock(&sun_mie->tme_sun_mie_mutex);
|
|
|
|
/* no faults: */
|
|
return (TME_OK);
|
|
}
|
|
|
|
/* the sun_mie bus signal handler: */
|
|
static int
|
|
_tme_sun_mie_bus_signal(struct tme_bus_connection *conn_bus,
|
|
unsigned int signal)
|
|
{
|
|
struct tme_sun_mie *sun_mie;
|
|
|
|
#if 0
|
|
printf("_tme_sun_mie_bus_signal() %x, index %d, count %d\n",
|
|
(int)signal,
|
|
(int)TME_BUS_SIGNAL_INDEX(signal),
|
|
(int)_tme_sun_mie_bus_signals_generic.tme_bus_signals_count);
|
|
#endif
|
|
|
|
/* return now if this is not a generic bus signal: */
|
|
if (TME_BUS_SIGNAL_INDEX(signal)
|
|
> _tme_sun_mie_bus_signals_generic.tme_bus_signals_count) {
|
|
return (TME_OK);
|
|
}
|
|
|
|
/* recover our data structures: */
|
|
sun_mie = conn_bus->tme_bus_connection.tme_connection_element->tme_element_private;
|
|
|
|
#if 0
|
|
printf("_tme_sun_mie_bus_signal() doing it %x\n", (int)signal);
|
|
#endif
|
|
|
|
#if 1
|
|
if (TME_BUS_SIGNAL_WHICH(signal) == TME_BUS_SIGNAL_INT_UNSPEC)
|
|
{
|
|
tme_uint16_t csr;
|
|
unsigned int level;
|
|
csr = TME_SUN_MIE_CSR_GET(sun_mie);
|
|
level = signal & TME_BUS_SIGNAL_LEVEL_MASK;
|
|
|
|
if (level == TME_BUS_SIGNAL_LEVEL_ASSERTED) {
|
|
// printf("assert csr intr!\n");
|
|
csr |= TME_SUN_MIE_CSR_INTR;
|
|
TME_SUN_MIE_CSR_PUT(sun_mie, csr);
|
|
}
|
|
if (level == TME_BUS_SIGNAL_LEVEL_NEGATED) {
|
|
// printf("deassert csr intr!\n");
|
|
// csr &= ~TME_SUN_MIE_CSR_INTR;
|
|
// TME_SUN_MIE_CSR_PUT(sun_mie, csr);
|
|
sun_mie->tme_sun_mie_pending_deassert_intr = 1;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/* since this function is currently only given to the i825x6,
|
|
just copy its signal through to the Multibus: */
|
|
conn_bus = sun_mie->tme_sun_mie_conn_regs;
|
|
return ((*conn_bus->tme_bus_signal)(conn_bus, signal));
|
|
}
|
|
|
|
/* the sun_mie bus signals adder for the i825x6: */
|
|
static int
|
|
_tme_sun_mie_bus_signals_add(struct tme_bus_connection *conn_bus,
|
|
struct tme_bus_signals *bus_signals_caller)
|
|
{
|
|
const struct tme_bus_signals *bus_signals;
|
|
tme_uint32_t signal_first;
|
|
|
|
/* we only support the generic and i825x6 bus signals: */
|
|
switch (bus_signals_caller->tme_bus_signals_id) {
|
|
case TME_BUS_SIGNALS_ID_GENERIC:
|
|
bus_signals = &_tme_sun_mie_bus_signals_generic;
|
|
signal_first = _tme_sun_mie_bus_signals_generic.tme_bus_signals_first;
|
|
break;
|
|
case TME_BUS_SIGNALS_ID_I825X6:
|
|
bus_signals = &_tme_sun_mie_bus_signals_i825x6;
|
|
signal_first = (_tme_sun_mie_bus_signals_generic.tme_bus_signals_first
|
|
+ TME_BUS_SIGNAL_X(_tme_sun_mie_bus_signals_generic.tme_bus_signals_count));
|
|
break;
|
|
default:
|
|
return (ENOENT);
|
|
}
|
|
|
|
/* XXX we should check versions here: */
|
|
*bus_signals_caller = *bus_signals;
|
|
bus_signals_caller->tme_bus_signals_first = signal_first;
|
|
return (TME_OK);
|
|
}
|
|
|
|
/* the sun_mie TLB adder for the i825x6: */
|
|
static int
|
|
_tme_sun_mie_tlb_set_add(struct tme_bus_connection *conn_bus,
|
|
struct tme_bus_tlb_set_info *tlb_set_info)
|
|
{
|
|
|
|
/* if this TLB set provides a bus context register: */
|
|
if (tlb_set_info->tme_bus_tlb_set_info_bus_context != NULL) {
|
|
|
|
/* this bus only has one context: */
|
|
(*tlb_set_info->tme_bus_tlb_set_info_bus_context) = 0;
|
|
tlb_set_info->tme_bus_tlb_set_info_bus_context_max = 0;
|
|
}
|
|
|
|
return (TME_OK);
|
|
}
|
|
|
|
/* the sun_mie TLB filler for the board memory: */
|
|
static int
|
|
_tme_sun_mie_tlb_fill(struct tme_bus_connection *conn_bus,
|
|
struct tme_bus_tlb *tlb,
|
|
tme_bus_addr_t address_wider,
|
|
unsigned int cycles)
|
|
{
|
|
struct tme_sun_mie *sun_mie;
|
|
tme_bus_addr32_t address;
|
|
unsigned int pgmap_i;
|
|
unsigned int tlb_i;
|
|
tme_uint16_t pgmap;
|
|
|
|
/* recover our data structures: */
|
|
sun_mie = conn_bus->tme_bus_connection.tme_connection_element->tme_element_private;
|
|
|
|
/* get the normal-width address: */
|
|
address = address_wider;
|
|
assert (address == address_wider);
|
|
|
|
/* the address must be within range: */
|
|
assert(address <= 0xffffff);
|
|
|
|
/* mask the address with the board page size: */
|
|
address &= -TME_SUN_MIE_PAGESIZE;
|
|
|
|
/* lock our mutex: */
|
|
tme_mutex_lock(&sun_mie->tme_sun_mie_mutex);
|
|
|
|
/* get the pagemap entry: */
|
|
pgmap_i = (address / TME_SUN_MIE_PAGESIZE) & (TME_SUN_MIE_PGMAP_COUNT - 1);
|
|
pgmap = tme_betoh_u16(((tme_uint16_t *) &sun_mie->tme_sun_mie_regs[TME_SUN_MIE_REG_PGMAP])[pgmap_i]);
|
|
|
|
/* update the head pointer for this page map entry's active TLB
|
|
entry list: */
|
|
tlb_i = sun_mie->tme_sun_mie_tlb_head[pgmap_i];
|
|
if (++tlb_i == TME_SUN_MIE_PGMAP_TLBS) {
|
|
tlb_i = 0;
|
|
}
|
|
sun_mie->tme_sun_mie_tlb_head[pgmap_i] = tlb_i;
|
|
tlb_i += pgmap_i * TME_SUN_MIE_PGMAP_TLBS;
|
|
|
|
/* if the new head pointer already has a TLB entry, and it doesn't
|
|
happen to be the same one that we're filling now, invalidate it: */
|
|
if (sun_mie->tme_sun_mie_tlb_tokens[tlb_i] != NULL
|
|
&& sun_mie->tme_sun_mie_tlb_tokens[tlb_i] != tlb->tme_bus_tlb_token) {
|
|
tme_token_invalidate(sun_mie->tme_sun_mie_tlb_tokens[tlb_i]);
|
|
}
|
|
|
|
/* initialize the TLB entry: */
|
|
tme_bus_tlb_initialize(tlb);
|
|
|
|
/* this TLB entry covers this range: */
|
|
tlb->tme_bus_tlb_addr_first = address;
|
|
tlb->tme_bus_tlb_addr_last = address | (TME_SUN_MIE_PAGESIZE - 1);
|
|
|
|
/* allow reading and writing: */
|
|
tlb->tme_bus_tlb_cycles_ok = TME_BUS_CYCLE_READ | TME_BUS_CYCLE_WRITE;
|
|
|
|
/* our bus cycle handler: */
|
|
tlb->tme_bus_tlb_cycle_private = sun_mie;
|
|
tlb->tme_bus_tlb_cycle = _tme_sun_mie_bus_cycle;
|
|
|
|
/* this TLB entry allows fast reading and writing: */
|
|
tlb->tme_bus_tlb_emulator_off_write =
|
|
(&sun_mie->tme_sun_mie_memory[((pgmap & TME_SUN_MIE_PGMAP_PFNUM)
|
|
* TME_SUN_MIE_PAGESIZE)]
|
|
- address);
|
|
tlb->tme_bus_tlb_emulator_off_read =
|
|
tlb->tme_bus_tlb_emulator_off_write;
|
|
|
|
#if 0
|
|
printf("_tme_sun_mie_tlb_fill() tlb first %08x, last %08x, off_write %08x\n",
|
|
(int)tlb->tme_bus_tlb_addr_first,
|
|
(int)tlb->tme_bus_tlb_addr_last,
|
|
(int)tlb->tme_bus_tlb_emulator_off_write);
|
|
#endif
|
|
|
|
/* add this TLB entry to the active list: */
|
|
sun_mie->tme_sun_mie_tlb_tokens[tlb_i] =
|
|
tlb->tme_bus_tlb_token;
|
|
|
|
/* unlock our mutex: */
|
|
tme_mutex_unlock(&sun_mie->tme_sun_mie_mutex);
|
|
|
|
return (TME_OK);
|
|
}
|
|
|
|
/* the sun_mie TLB filler for the board registers: */
|
|
static int
|
|
_tme_sun_mie_tlb_fill_regs(struct tme_bus_connection *conn_bus,
|
|
struct tme_bus_tlb *tlb,
|
|
tme_bus_addr_t address_wider,
|
|
unsigned int cycles)
|
|
{
|
|
struct tme_sun_mie *sun_mie;
|
|
tme_bus_addr32_t address;
|
|
|
|
/* recover our data structures: */
|
|
sun_mie = conn_bus->tme_bus_connection.tme_connection_element->tme_element_private;
|
|
|
|
/* get the normal-width address: */
|
|
address = address_wider;
|
|
assert (address == address_wider);
|
|
|
|
/* the address must be within range: */
|
|
assert(address < TME_SUN_MIE_SIZ_REGS);
|
|
|
|
/* initialize the TLB entry: */
|
|
tme_bus_tlb_initialize(tlb);
|
|
|
|
/* if the address falls in the page map: */
|
|
if (TME_SUN_MIE_REG_PGMAP <= address
|
|
&& address < (TME_SUN_MIE_REG_PGMAP
|
|
+ TME_SUN_MIE_SIZ_PGMAP)) {
|
|
|
|
/* this TLB entry covers this range: */
|
|
tlb->tme_bus_tlb_addr_first = TME_SUN_MIE_REG_PGMAP;
|
|
tlb->tme_bus_tlb_addr_last = (TME_SUN_MIE_REG_PGMAP + TME_SUN_MIE_SIZ_PGMAP - 1);
|
|
}
|
|
|
|
/* if this address falls in the PROM: */
|
|
else if (TME_SUN_MIE_REG_PROM <= address
|
|
&& address < (TME_SUN_MIE_REG_PROM
|
|
+ TME_SUN_MIE_SIZ_PROM)) {
|
|
|
|
|
|
/* this TLB entry covers this range: */
|
|
tlb->tme_bus_tlb_addr_first = TME_SUN_MIE_REG_PROM;
|
|
tlb->tme_bus_tlb_addr_last = TME_SUN_MIE_REG_PROM + TME_SUN_MIE_SIZ_PROM - 1;
|
|
}
|
|
|
|
/* if this address falls in the CSR: */
|
|
else if (TME_SUN_MIE_REG_CSR <= address
|
|
&& address < (TME_SUN_MIE_REG_CSR
|
|
+ TME_SUN_MIE_SIZ_CSR)) {
|
|
|
|
|
|
/* this TLB entry covers this range: */
|
|
tlb->tme_bus_tlb_addr_first = TME_SUN_MIE_REG_CSR;
|
|
tlb->tme_bus_tlb_addr_last = TME_SUN_MIE_REG_CSR + TME_SUN_MIE_SIZ_CSR - 1;
|
|
}
|
|
|
|
/* otherwise, this address must fall in the unused hole or in the
|
|
parity registers: */
|
|
else {
|
|
assert (address >= (TME_SUN_MIE_REG_CSR
|
|
+ TME_SUN_MIE_SIZ_CSR));
|
|
|
|
/* this TLB entry covers this range: */
|
|
tlb->tme_bus_tlb_addr_first = (TME_SUN_MIE_REG_CSR + TME_SUN_MIE_SIZ_CSR);
|
|
tlb->tme_bus_tlb_addr_last = (TME_SUN_MIE_REG_PE_ALO + TME_SUN_MIE_SIZ_PE_ALO - 1);
|
|
}
|
|
|
|
/* all address ranges allow fast reading: */
|
|
tlb->tme_bus_tlb_emulator_off_read = &sun_mie->tme_sun_mie_regs[0];
|
|
tlb->tme_bus_tlb_rwlock = &sun_mie->tme_sun_mie_rwlock;
|
|
|
|
#if 0
|
|
/* XXX when debugging, nothing is fast readable or writable: */
|
|
tlb->tme_bus_tlb_emulator_off_read = TME_EMULATOR_OFF_UNDEF;
|
|
tlb->tme_bus_tlb_emulator_off_write = TME_EMULATOR_OFF_UNDEF;
|
|
#endif
|
|
|
|
/* allow reading and writing: */
|
|
tlb->tme_bus_tlb_cycles_ok = TME_BUS_CYCLE_READ | TME_BUS_CYCLE_WRITE;
|
|
|
|
/* our bus cycle handler: */
|
|
tlb->tme_bus_tlb_cycle_private = sun_mie;
|
|
tlb->tme_bus_tlb_cycle = _tme_sun_mie_bus_cycle_regs;
|
|
|
|
return (TME_OK);
|
|
}
|
|
|
|
/* this scores a new connection: */
|
|
static int
|
|
_tme_sun_mie_connection_score(struct tme_connection *conn, unsigned int *_score)
|
|
{
|
|
struct tme_sun_mie *sun_mie;
|
|
struct tme_sun_mie_connection *conn_sun_mie;
|
|
|
|
/* both sides must be generic bus connections: */
|
|
assert(conn->tme_connection_type == TME_CONNECTION_BUS_GENERIC);
|
|
assert(conn->tme_connection_other->tme_connection_type
|
|
== conn->tme_connection_type);
|
|
|
|
/* recover our data structures: */
|
|
sun_mie = conn->tme_connection_element->tme_element_private;
|
|
conn_sun_mie = (struct tme_sun_mie_connection *)conn;
|
|
|
|
/* this is a generic bus connection, so just score it nonzero and
|
|
return. note that there's no good way to differentiate a
|
|
connection to a bus from a connection to just another chip, so we
|
|
always return a nonzero score here: */
|
|
*_score = 1;
|
|
return (TME_OK);
|
|
}
|
|
|
|
/* this makes a new connection: */
|
|
static int
|
|
_tme_sun_mie_connection_make(struct tme_connection *conn, unsigned int state)
|
|
{
|
|
struct tme_sun_mie *sun_mie;
|
|
struct tme_sun_mie_connection *conn_sun_mie;
|
|
struct tme_bus_connection *conn_bus;
|
|
tme_uint16_t csr;
|
|
|
|
/* both sides must be generic bus connections: */
|
|
assert(conn->tme_connection_type == TME_CONNECTION_BUS_GENERIC);
|
|
assert(conn->tme_connection_other->tme_connection_type
|
|
== conn->tme_connection_type);
|
|
|
|
/* recover our data structures: */
|
|
sun_mie = conn->tme_connection_element->tme_element_private;
|
|
conn_sun_mie = (struct tme_sun_mie_connection *)conn;
|
|
conn_bus = &conn_sun_mie->tme_sun_mie_connection;
|
|
|
|
/* we're always set up to answer calls across the connection, so we
|
|
only have to do work when the connection has gone full, namely
|
|
taking the other side of the connection: */
|
|
if (state == TME_CONNECTION_FULL) {
|
|
|
|
/* lock our mutex: */
|
|
tme_mutex_lock(&sun_mie->tme_sun_mie_mutex);
|
|
|
|
/* save our connection: */
|
|
if (conn_bus->tme_bus_signals_add != NULL) {
|
|
sun_mie->tme_sun_mie_conn_i825x6 = (struct tme_bus_connection *) conn->tme_connection_other;
|
|
}
|
|
else if (conn_sun_mie->tme_sun_mie_connection_regs) {
|
|
sun_mie->tme_sun_mie_conn_regs = (struct tme_bus_connection *) conn->tme_connection_other;
|
|
}
|
|
else {
|
|
sun_mie->tme_sun_mie_conn_memory = (struct tme_bus_connection *) conn->tme_connection_other;
|
|
csr = TME_SUN_MIE_CSR_GET(sun_mie);
|
|
csr &= ~TME_SUN_MIE_CSR_MPMHI;
|
|
csr |= conn_sun_mie->tme_sun_mie_connection_mpmhi;
|
|
TME_SUN_MIE_CSR_PUT(sun_mie, csr);
|
|
}
|
|
|
|
/* unlock our mutex: */
|
|
tme_mutex_unlock(&sun_mie->tme_sun_mie_mutex);
|
|
}
|
|
|
|
return (TME_OK);
|
|
}
|
|
|
|
/* this breaks a connection: */
|
|
static int
|
|
_tme_sun_mie_connection_break(struct tme_connection *conn, unsigned int state)
|
|
{
|
|
abort();
|
|
}
|
|
|
|
/* this makes a new connection side for a sun_mie: */
|
|
static int
|
|
_tme_sun_mie_connections_new(struct tme_element *element,
|
|
const char * const *args,
|
|
struct tme_connection **_conns,
|
|
char **_output)
|
|
{
|
|
struct tme_sun_mie *sun_mie;
|
|
struct tme_sun_mie_connection *conn_sun_mie;
|
|
struct tme_bus_connection *conn_bus;
|
|
struct tme_connection *conn;
|
|
unsigned int i825x6;
|
|
tme_uint8_t regs;
|
|
tme_bus_addr_t mpmhi;
|
|
int usage;
|
|
int rc;
|
|
|
|
/* recover our data structure: */
|
|
sun_mie = (struct tme_sun_mie *) element->tme_element_private;
|
|
|
|
/* we don't bother locking the mutex simply to check if connections
|
|
already exist: */
|
|
|
|
/* check our arguments: */
|
|
usage = FALSE;
|
|
rc = 0;
|
|
i825x6 = FALSE;
|
|
regs = FALSE;
|
|
mpmhi = 0;
|
|
|
|
/* if this connection is for the registers: */
|
|
if (TME_ARG_IS(args[1], "csr")) {
|
|
|
|
/* if we already have a register connection, complain: */
|
|
if (sun_mie->tme_sun_mie_conn_regs != NULL) {
|
|
rc = EEXIST;
|
|
}
|
|
|
|
/* otherwise, make the new connection: */
|
|
else {
|
|
regs = TRUE;
|
|
}
|
|
}
|
|
|
|
/* else, if this connection is for the memory: */
|
|
else if (TME_ARG_IS(args[1], "memory")) {
|
|
|
|
/* if we already have a memory connection, complain: */
|
|
if (sun_mie->tme_sun_mie_conn_memory != NULL) {
|
|
rc = EEXIST;
|
|
}
|
|
|
|
/* otherwise, check the value after "memory". it must be the low
|
|
20 bits of the bus address of the board's memory; in our csr we
|
|
have to report the most significant nibble (A19-A16, as A15..A0
|
|
are expected to be zero) to software so it can find that
|
|
memory: */
|
|
else {
|
|
mpmhi = tme_bus_addr_parse_any(args[2], &usage);
|
|
if (!usage
|
|
&& ((mpmhi > 0xfffff)
|
|
|| (mpmhi & (TME_SUN_MIE_SIZ_MBM - 1)))) {
|
|
tme_output_append_error(_output,
|
|
"%s %s, ",
|
|
args[2],
|
|
_(" is not a 20-bit address with A15..A0 zero"));
|
|
usage = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* else, the connection must be for the i825x6: */
|
|
else if (args[1] == NULL) {
|
|
|
|
/* if we already have an i825x6 connection, complain: */
|
|
if (sun_mie->tme_sun_mie_conn_i825x6 != NULL) {
|
|
rc = EEXIST;
|
|
}
|
|
|
|
/* otherwise, make the new conection: */
|
|
else {
|
|
i825x6 = TRUE;
|
|
}
|
|
}
|
|
|
|
/* otherwise, this is a bad argument: */
|
|
else {
|
|
tme_output_append_error(_output,
|
|
"%s %s, ",
|
|
args[1],
|
|
_("unexpected"));
|
|
usage = TRUE;
|
|
}
|
|
|
|
if (usage) {
|
|
tme_output_append_error(_output,
|
|
"%s %s [ csr | memory %s ]",
|
|
_("usage:"),
|
|
args[0],
|
|
_("BUS-ADDRESS"));
|
|
rc = EINVAL;
|
|
}
|
|
|
|
if (rc) {
|
|
return (rc);
|
|
}
|
|
|
|
/* make a new connection: */
|
|
conn_sun_mie = tme_new0(struct tme_sun_mie_connection, 1);
|
|
conn_bus = &conn_sun_mie->tme_sun_mie_connection;
|
|
conn = &conn_bus->tme_bus_connection;
|
|
|
|
/* fill in the generic connection: */
|
|
conn->tme_connection_next = *_conns;
|
|
conn->tme_connection_type = TME_CONNECTION_BUS_GENERIC;
|
|
conn->tme_connection_score = _tme_sun_mie_connection_score;
|
|
conn->tme_connection_make = _tme_sun_mie_connection_make;
|
|
conn->tme_connection_break = _tme_sun_mie_connection_break;
|
|
|
|
/* fill in the generic bus connection: */
|
|
conn_bus->tme_bus_subregions.tme_bus_subregion_address_first = 0;
|
|
conn_bus->tme_bus_subregions.tme_bus_subregion_next = NULL;
|
|
if (i825x6) {
|
|
conn_bus->tme_bus_subregions.tme_bus_subregion_address_last = 0xffffff;
|
|
conn_bus->tme_bus_signals_add = _tme_sun_mie_bus_signals_add;
|
|
conn_bus->tme_bus_signal = _tme_sun_mie_bus_signal;
|
|
conn_bus->tme_bus_tlb_set_add = _tme_sun_mie_tlb_set_add;
|
|
conn_bus->tme_bus_tlb_fill = _tme_sun_mie_tlb_fill;
|
|
}
|
|
else if (regs) {
|
|
conn_bus->tme_bus_subregions.tme_bus_subregion_address_last = TME_SUN_MIE_SIZ_REGS - 1;
|
|
conn_bus->tme_bus_tlb_fill = _tme_sun_mie_tlb_fill_regs;
|
|
}
|
|
else {
|
|
conn_bus->tme_bus_subregions.tme_bus_subregion_address_last = TME_SUN_MIE_SIZ_MBM - 1;
|
|
conn_bus->tme_bus_tlb_fill = _tme_sun_mie_tlb_fill;
|
|
}
|
|
|
|
/* fill in the internal information: */
|
|
conn_sun_mie->tme_sun_mie_connection_regs = regs;
|
|
conn_sun_mie->tme_sun_mie_connection_mpmhi = (mpmhi >> 16);
|
|
|
|
/* return the connection side possibility: */
|
|
*_conns = conn;
|
|
return (TME_OK);
|
|
}
|
|
|
|
/* the new sun_mie function: */
|
|
TME_ELEMENT_SUB_NEW_DECL(tme_bus_multibus,sun_mie) {
|
|
struct tme_sun_mie *sun_mie;
|
|
int arg_i;
|
|
int usage;
|
|
|
|
/* check our arguments: */
|
|
usage = 0;
|
|
arg_i = 1;
|
|
for (;;) {
|
|
|
|
if (0) {
|
|
}
|
|
|
|
/* if we ran out of arguments: */
|
|
else if (args[arg_i] == NULL) {
|
|
|
|
break;
|
|
}
|
|
|
|
/* otherwise this is a bad argument: */
|
|
else {
|
|
tme_output_append_error(_output,
|
|
"%s %s, ",
|
|
args[arg_i],
|
|
_("unexpected"));
|
|
usage = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (usage) {
|
|
tme_output_append_error(_output,
|
|
"%s %s",
|
|
_("usage:"),
|
|
args[0]);
|
|
return (EINVAL);
|
|
}
|
|
|
|
/* start the sun_mie structure: */
|
|
sun_mie = tme_new0(struct tme_sun_mie, 1);
|
|
sun_mie->tme_sun_mie_element = element;
|
|
TME_SUN_MIE_CSR_PUT(sun_mie,
|
|
(TME_SUN_MIE_CSR_NOLOOP
|
|
| TME_SUN_MIE_CSR_BIGRAM));
|
|
tme_mutex_init(&sun_mie->tme_sun_mie_mutex);
|
|
tme_rwlock_init(&sun_mie->tme_sun_mie_rwlock);
|
|
|
|
/* fill the element: */
|
|
element->tme_element_private = sun_mie;
|
|
element->tme_element_connections_new = _tme_sun_mie_connections_new;
|
|
|
|
return (TME_OK);
|
|
}
|