Quick overview of the SparcSoftcard resources
The SparcSoftcard provides to each of the machine's 3 processors (CP, IOP and Sparc) two basic resources:
Local memory: Each processor as its own map and is using its own virtual address space or set of virtual address spaces.
Local peripherals: All low level actions on the SparcSoftcard hardware are performed via those peripherals (ie: enabling the Sparc cache...). The VM maps entries can be read or written this way.
Local Memory
Can be 8, 16, 32 or 64 MegaBytes.
- In the current software implementation a part of this memory is used as memory extension for the Cedar system running on the CP. This Cedar memory is defined as follow:
cedarMemoryStartByteAddress:
CARD32 ~ 0H;
Real address of the Cedar extension memory inside the SparcSoftcard.
cedarMemoryByteSize:
CARD32 ~ 200000H;
Size of the Cedar extension memory.
- The remaining of the memory can be used as Sparc memory. A small amount of it is still visible from the CP and IOP. It is called Backdoor and is used as communication area. Any page of this Backdoor can be set to point anywhere in the Sparc memory. This feature will allow in the future direct IO communications.
Local peripherals
Every peripheral in the SparcSoftcard can be reached by any of the machine's 3 processors. They have all equal rights. The exclusion is only enforced by the arbiter to the internal Softcard Bus.
For the CP (6085 mesa processor) and the IOP all the peripherals look as memory mapped IOs.
For the Sparc all the peripherals are found in an alternate address space (special ASI)
Some Base values.
Cedar side
cedarMemoryExtensionBaseByte:
CARD32 ~ 0180000H; -- 1.5 Meg
Base of the MemoryExtension address in 6085 real address.
cedarMemoryExtensionBaseWord16: CARD32 ~ cedarMemoryExtensionBaseByte / 2;
cedarIOBaseByte:
CARD32 ~ 0380000H; -- 3.5 Meg
Base of the memory mapped IO in 6085 real address
cedarIOBaseWord16: CARD32 ~ cedarIOBaseByte / 2;
cedarBackDoorBaseByte:
CARD32 ~ 03C0000H; -- 3.75 Meg
Base of the Backdoor memory in 6085 real address
cedarBackDoorBaseWord16: CARD32 ~ cedarBackDoorBaseByte / 2;
cedarMemoryEndByte:
CARD32 ~ 0400000H; -- 4.0 Meg
End of 6085 real address
cedarMemoryEndWord16: CARD32 ~ cedarMemoryEndByte / 2;
cedarMemoryExtensionSizeWord16:
CARD32 ~ cedarIOBaseWord16 - cedarMemoryExtensionBaseWord16;
Size of the MemoryExtension.
cedarBackDoorSizeWord16:
CARD32 ~ cedarMemoryEndWord16 - cedarBackDoorBaseWord16;
Size of the Backdoor memory.
Sparc side
sparcIOASI: CARD16 ~ 080H;
sparcIOBaseByte : CARD32 ~ 0380000H;
sparcMapBaseByte : CARD32 ~ 0H;
Map Description
The softcard internal bus carries Virtual addresses. Each bus master has one or more virtual address space. The map translates all the virtual addresses into real addresses. => The mapping mechanism is uniform for everybody.
The map is dual ported.
During regular memory accesses it translates the addresses.
Each entry can be read or written during IO accesses.
In this mode the amount of IO space needed to access all the entries is too big for the CP or IOP. A mapPage register as been implemented and holds the high order bits of the map entry address, thus "folding" the map and reducing the amount of space needed. This map register is unique. Care must be taken to avoid access to the "folded" version of the map by two processors at the same time.
The Sparc's IO space is big enough to hold the "unfolded" version of the map. For the sake of decoding simplicity the "folded" version is also available to the Sparc.
Map page size.
softcardPageSizeByte:
CARD16 ~ 02000H;
This size was chosen to be compatible with Sun implementations of the Sparc. Keep in mind that this is 16 times bigger than the current 6085 or dorado page size.
softcardPageSizeWord16:
CARD16 ~ softcardPageSizeByte / 2;
This size was chosen to be compatible with Sun implementations of the Sparc. Keep in mind that this is 16 times bigger than the current 6085 or dorado page size.
map (Unfolded version of the map)
map is a array of records. Each record contains one entry.
mapBaseByteOffset: CARD32 ~ 0H;
mapBaseByte: CARD32 ~ mapBaseByteOffset + sparcMapBaseByte;
mapWindowIndexRegister
When the folded version of the map is accessed mapWindowIndexRegister holds the 5 most significant bits of the map entry number. At any given moment only a "window" of 2048 entries can be reached.
mapPageRegister is Write only
mapWindowIndexRegisterByteOffset: CARD32 ~ 04406H;
mapWindowIndexRegisterWord16: CARD32 ~ cedarIOBaseWord16 + mapWindowIndexRegisterByteOffset / 2;
mapWindow (Folded version of the map)
mapWindow is a window [2048 entry] onto the sparc softcard map. The window is controlled via the mapPageRegister.
mapWindowBaseByteOffset: CARD32 ~ 08000H;
mapWindowBaseWord16: CARD32 ~ cedarIOBaseWord16 + mapWindowBaseByteOffset / 2;
mapEntry
Each map entry is composed of 2 non contiguous 16 bits words.
mapEntryHighByteOffset: NAT ~ 6;
mapEntryHighWord16Offset: NAT ~ mapEntryHighByteOffset / 2;
mapEntryLowByteOffset: NAT ~ 0EH;
mapEntryLowWord16Offset: NAT ~ mapEntryLowByteOffset / 2;
mapEntryBytes: NAT ~ BYTES[MapEntry];
mapEntryWord16: NAT ~ SIZE[MapEntry];
MapEntry: TYPE ~ ARRAY [0..8) OF CARD16; -- 10H bytes in size!
Some usefull constants needed during the initialisation of the Cedar memory extension
To set IOP map entries iOPMapWindowIndex has to be stored in mapWindowIndexRegister.
To set CP map entries cPMapWindowIndex has to be stored in mapWindowIndexRegister.
iOPMapWindowIndex: BYTE ~ 014h;
cPMapWindowIndex: BYTE ~ 015h;
iopMap:
CARD32 ~ mapWindowBaseWord16;
providing mapPageRegister is set to daybreakIOPMapPage.
cpMap:
CARD32 ~ mapWindowBaseWord16;
providing mapPageRegister is set to daybreakCPMapPage.
iopMapHigh: CARD32 ~ iopMap + mapEntryHighWord16Offset;
iopMapLow: CARD32 ~ iopMap + mapEntryLowWord16Offset;
cpMapHigh: CARD32 ~ cpMap + mapEntryHighWord16Offset;
cpMapLow: CARD32 ~ cpMap + mapEntryLowWord16Offset;
The high part contains flags (mostly) and has to be set to mapEntryHighPattern.
The low part contains the page number (mostly) {limited to [0..0400H for 8 MB memory} and has to be set to mapEntryLowPattern + the required physical page number.
mapEntryHighPattern: NAT ~ 0;
mapEntryLowPattern: NAT ~ 0;
Task Register
The task register holds a byte identifying the current user process running on the Sparc. This identifier is used inside the cache to discriminate data. It is used also in the map.
tRegisterByteOffset: CARD32 ~ 04C06H;
tRegisterWord16: CARD32 ~ cedarIOBaseWord16 + tRegisterByteOffset / 2;
tRegisterAlternateWord16: CARD32 ~ cedarIOBaseWord16 + tRegisterByteOffset / 2 + alternatePathByteOffset / 2;
During power up of the machine this register is also used to test the presence of the Softcard.
magic: BYTE ~ 05Ah; -- Arbitrary constant
magicNot: BYTE ~ 0A5h; -- Another arbitrary constant
alternatePathByteOffset:
CARD32 ~ 10000H;
Due to decoding, the peripheral registers of the softcard appear at two different locations
Control and Status bits
Command bits appear as one bit read/write registers. They are set or reset individualy by a write at their address. Their state can be read at any time.
commandBitMask: CARD16 ~ 01H;
A command bit is the low order bit of the word(LSB)
Polarity: TYPE = {ActiveLow, ActiveHigh};
Sparc Reset
sparcResetPolarity: Polarity ~ ActiveLow;
sparcResetRegisterByteOffset: CARD32 ~ 04636H;
sparcResetRegisterWord16: CARD32 ~ cedarIOBaseWord16 + sparcResetRegisterByteOffset/2;
Sparc Cache Enable
sparcCacheEnablePolarity: Polarity ~ ActiveLow;
sparcCacheEnableRegisterByteOffset: CARD32 ~ 04A0EH;
sparcCacheEnableRegisterByte: CARD32 ~ sparcIOBaseByte + sparcCacheEnableRegisterByteOffset;
sparcCacheEnableRegisterWord16: CARD32 ~ cedarIOBaseWord16 + sparcCacheEnableRegisterByteOffset / 2;
DMA Enable
dMAActivePolarity: Polarity ~ ActiveHigh;
dMAActiveRegisterByteOffset: CARD32 ~ 04826H;
dMAActiveRegisterByte: CARD32 ~ sparcIOBaseByte + dMAActiveRegisterByteOffset;
dMAActiveRegisterWord16: CARD32 ~ cedarIOBaseWord16 + dMAActiveRegisterByteOffset/2;
DMA Mode
dMAMode0Polarity: Polarity ~ ActiveHigh;
dMAMode0RegisterByteOffset: CARD32 ~ 0482EH;
dMAMode0RegisterByte: CARD32 ~ sparcIOBaseByte + dMAMode0RegisterByteOffset;
dMAMode0RegisterWord16: CARD32 ~ cedarIOBaseWord16 + dMAMode0RegisterByteOffset/2;
dMAMode1Polarity: Polarity ~ ActiveHigh;
dMAMode1RegisterByteOffset: CARD32 ~ 04836H;
dMAMode1RegisterByte: CARD32 ~ sparcIOBaseByte + dMAMode1RegisterByteOffset;
dMAMode1RegisterWord16: CARD32 ~ cedarIOBaseWord16 + dMAMode1RegisterByteOffset/2;
DMA Address registers
The starting address of the DMA is held is this pair of registers. They are Write only.
Each time one of those registers is written the new value is loaded into the DMA address counter.
If DMA mode is display the content of those registers is loaded into the DMA address counter during vertical retrace.
dMAAddressRegisterHighByteOffset: CARD32 ~ 04006H;
dMAAddressRegisterHighByte: CARD32 ~ sparcIOBaseByte + dMAAddressRegisterHighByteOffset;
dMAAddressRegisterHighWord16: CARD32 ~ cedarIOBaseWord16 + dMAAddressRegisterHighByteOffset / 2;
dMAAddressRegisterLowByteOffset: CARD32 ~ 04206H;
dMAAddressRegisterLowByte: CARD32 ~ sparcIOBaseByte + dMAAddressRegisterLowByteOffset;
dMAAddressRegisterLowWord16: CARD32 ~ cedarIOBaseWord16 + dMAAddressRegisterLowByteOffset / 2;
Softcard memory backdoor addresses
softcardMemoryBackDoorByteOffset: CARD32 ~ 040000H;
softcardMemoryBackDoorByteAddress: CARD32 ~ cedarIOBaseByte + softcardMemoryBackDoorByteOffset;
softcardMemoryBackDoorWord16Address: CARD32 ~ softcardMemoryBackDoorByteAddress / 2;
softcardMemoryBackDoorStartingPage: NAT ~ softcardMemoryBackDoorByteAddress / softcardPageSizeByte;
}.