mirror of
https://github.com/rdolbeau/NuBusFPGA.git
synced 2025-01-05 09:31:31 +00:00
209 lines
6.7 KiB
C
209 lines
6.7 KiB
C
#include "NuBusFPGADrvr.h"
|
|
|
|
#include "ROMDefs.h"
|
|
|
|
typedef void(*vblproto)(short);
|
|
|
|
/* how do I tell it to only modify D0 & A1 ? */
|
|
/* the ABI allows to modify D0-D2 & A0-A1 (caller save) */
|
|
/* currently this is clobbering A0 yet seems to work ... */
|
|
/* 'Devices' p1-37 says of interrupt handler "preserving all registers other than D0 through D3 and A0 through A3"
|
|
* but that for NuBus you see SIntInstall which says (2-70)
|
|
* "routine must preserve the contents of all registers except A1 and D0"
|
|
* yet the interrupt handler for the video card driver example in DCDMF3 p589 clobbers A0
|
|
*/
|
|
#pragma parameter __D0 fbIrq(__A1)
|
|
__attribute__ ((section (".text.fbdriver"))) short fbIrq(const long sqParameter) {
|
|
register unsigned long p_D1 asm("d1"), p_D2 asm("d2");
|
|
unsigned int irq;
|
|
short ret;
|
|
asm volatile("" : "+d" (p_D1), "+d" (p_D2));
|
|
ret = 0;
|
|
irq = (*((volatile unsigned int*)(sqParameter+GOBOFB_BASE+GOBOFB_INTR_CLEAR)));
|
|
if (irq & 1) {
|
|
vblproto myVbl = *(vblproto**)0x0d28;
|
|
*((volatile unsigned int*)(sqParameter+GOBOFB_BASE+GOBOFB_INTR_CLEAR)) = 0;
|
|
myVbl((sqParameter>>24)&0xf); // cleaner to use dStore->slot ? but require more code...
|
|
ret = 1;
|
|
}
|
|
asm volatile("" : : "d" (p_D1), "d" (p_D2));
|
|
return ret;
|
|
}
|
|
|
|
#pragma parameter __D0 cNuBusFPGAOpen(__A0, __A1)
|
|
OSErr cNuBusFPGAOpen(IOParamPtr pb, /* DCtlPtr */ AuxDCEPtr dce)
|
|
{
|
|
OSErr ret = noErr;
|
|
/* write_reg(dce, GOBOFB_DEBUG, 0xBEEF0000); */
|
|
/* write_reg(dce, GOBOFB_DEBUG, (unsigned long)dce->dCtlDevBase); */
|
|
|
|
if (dce->dCtlStorage == nil)
|
|
{
|
|
int i;
|
|
/* set up flags in the device control entry */
|
|
/* dce->dCtlFlags |= (dCtlEnableMask | dStatEnableMask | dWritEnableMask |
|
|
dReadEnableMask | dNeedLockMask | dRAMBasedMask ); */
|
|
|
|
/* initialize dCtlStorage */
|
|
ReserveMemSys(sizeof(NuBusFPGADriverGlobals));
|
|
dce->dCtlStorage = NewHandleSysClear(sizeof(NuBusFPGADriverGlobals));
|
|
if (dce->dCtlStorage == nil)
|
|
return(openErr);
|
|
HLock(dce->dCtlStorage);
|
|
NuBusFPGADriverGlobalsHdl dStoreHdl = (NuBusFPGADriverGlobalsHdl)dce->dCtlStorage;
|
|
NuBusFPGADriverGlobalsPtr dStore = *dStoreHdl;
|
|
/* (*dStore)->dce = dce; */
|
|
|
|
/* for (i = 0 ; i < 256 ; i++) { */
|
|
/* dStore->shadowClut[i*3+0] = i; */
|
|
/* dStore->shadowClut[i*3+1] = i; */
|
|
/* dStore->shadowClut[i*3+2] = i; */
|
|
/* } */
|
|
|
|
dStore->gray = 0;
|
|
dStore->irqen = 0;
|
|
dStore->slot = dce->dCtlSlot;
|
|
|
|
/* initialize DRAM controller */
|
|
//sdram_init(0xF0000000 | (((unsigned long)dStore->slot) << 24));
|
|
|
|
/* Get the HW setting for native resolution */
|
|
dStore->hres[0] = __builtin_bswap32((unsigned int)read_reg(dce, GOBOFB_HRES)); // fixme: endianness
|
|
dStore->vres[0] = __builtin_bswap32((unsigned int)read_reg(dce, GOBOFB_VRES)); // fixme: endianness
|
|
|
|
SlotIntQElement *siqel = (SlotIntQElement *)NewPtrSysClear(sizeof(SlotIntQElement));
|
|
if (siqel == NULL) {
|
|
return openErr;
|
|
}
|
|
siqel->sqType = sIQType;
|
|
siqel->sqPrio = 8;
|
|
//siqel->sqAddr = interruptRoutine;
|
|
/* not sure how to get the proper result in C... */
|
|
/* SlotIntServiceProcPtr sqAddr; */
|
|
/* asm("lea %%pc@(interruptRoutine),%0\n" : "=a"(sqAddr)); */
|
|
/* siqel->sqAddr = sqAddr; */
|
|
/* siqel->sqParm = (long)dce->dCtlDevBase; */
|
|
|
|
/* not sure how to get the proper result in C... */
|
|
/* ... from ~mac68k, you need option "-mpcrel", and it works */
|
|
/* SlotIntServiceProcPtr sqAddr; */
|
|
/* asm("lea %%pc@(fbIrq),%0\n" : "=a"(sqAddr)); */
|
|
siqel->sqAddr = fbIrq;
|
|
/* siqel->sqParm = (long)dce; */
|
|
siqel->sqParm = (long)dce->dCtlDevBase;
|
|
dStore->siqel = siqel;
|
|
|
|
dStore->curPage = 0;
|
|
dStore->curMode = nativeVidMode;
|
|
dStore->curDepth = kDepthMode1;
|
|
|
|
{
|
|
OSErr err = noErr;
|
|
SpBlock spb;
|
|
UInt8 max = nativeVidMode;
|
|
|
|
spb.spParamData = 1<<fall|1<<foneslot;
|
|
spb.spCategory = catDisplay;
|
|
spb.spCType = typeVideo;
|
|
spb.spDrvrSW = drSwApple;
|
|
spb.spDrvrHW = 0xBEEF;
|
|
spb.spTBMask = 0; /* match everything above */
|
|
spb.spSlot = dce->dCtlSlot;
|
|
spb.spID = nativeVidMode;
|
|
spb.spExtDev = 0;
|
|
err = SGetTypeSRsrc(&spb);
|
|
while ((err == noErr) &&
|
|
(spb.spSlot == dce->dCtlSlot) &&
|
|
(((UInt8)spb.spID) > (UInt8)0x80) &&
|
|
(((UInt8)spb.spID) < (UInt8)0x90)) {
|
|
/* write_reg(dce, GOBOFB_DEBUG, 0xBEEF0020); */
|
|
/* write_reg(dce, GOBOFB_DEBUG, spb.spID); */
|
|
/* write_reg(dce, GOBOFB_DEBUG, err); */
|
|
|
|
if (((UInt8)spb.spID) == max) // should not happen
|
|
err = smNoMoresRsrcs;
|
|
if (((UInt8)spb.spID) > max)
|
|
max = spb.spID;
|
|
err = SGetTypeSRsrc(&spb);
|
|
}
|
|
dStore->maxMode = max;
|
|
}
|
|
/* write_reg(dce, GOBOFB_DEBUG, 0xBEEF0000); */
|
|
/* write_reg(dce, GOBOFB_DEBUG, dStore->maxMode); */
|
|
{
|
|
OSErr err = noErr;
|
|
SpBlock spb;
|
|
/* check for resolution */
|
|
UInt8 id;
|
|
for (id = nativeVidMode; id <= dStore->maxMode ; id ++) {
|
|
/* try every resource, enabled or not */
|
|
spb.spParamData = 1<<fall; /* wants disabled */
|
|
spb.spCategory = catDisplay;
|
|
spb.spCType = typeVideo;
|
|
spb.spDrvrSW = drSwApple;
|
|
spb.spDrvrHW = 0xBEEF;
|
|
spb.spTBMask = 0;
|
|
spb.spSlot = dce->dCtlSlot;
|
|
spb.spID = id;
|
|
spb.spExtDev = 0;
|
|
err = SGetSRsrc(&spb);
|
|
|
|
if (err == noErr) {
|
|
spb.spID = kDepthMode1;
|
|
err = SFindStruct(&spb); /* that will give us the Parms block ... */
|
|
|
|
if (err == noErr) {
|
|
/* take the Parms pointer, add the offset to the Modes block and then skip the block size at the beginning to get the structure pointer ... */
|
|
const unsigned long offset = *(unsigned long*)spb.spsPointer & 0x00FFFFFF;
|
|
VPBlockPtr vpblock = (VPBlockPtr)(spb.spsPointer + offset + sizeof(long));
|
|
UInt8 idx = id - nativeVidMode;
|
|
dStore->hres[idx] = vpblock->vpBounds.right;
|
|
dStore->vres[idx] = vpblock->vpBounds.bottom;
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
linearGamma(dStore);
|
|
|
|
/* now check the content of PRAM */
|
|
if (0) {
|
|
SpBlock spb;
|
|
NuBusFPGAPramRecord pram;
|
|
OSErr err;
|
|
spb.spSlot = dce->dCtlSlot;
|
|
spb.spResult = (UInt32)&pram;
|
|
err = SReadPRAMRec(&spb);
|
|
if (err == noErr) {
|
|
err = reconfHW(dce, pram.mode, pram.depth, pram.page);
|
|
}
|
|
}
|
|
|
|
write_reg(dce, GOBOFB_VIDEOCTRL, 1);
|
|
|
|
ret = changeIRQ(dce, 1, openErr);
|
|
}
|
|
|
|
return noErr;
|
|
}
|
|
|
|
#pragma parameter __D0 cNuBusFPGAClose(__A0, __A1)
|
|
OSErr cNuBusFPGAClose(IOParamPtr pb, /* DCtlPtr */ AuxDCEPtr dce)
|
|
{
|
|
OSErr ret = noErr;
|
|
/* write_reg(dce, GOBOFB_DEBUG, 0xBEEF0003); */
|
|
/* write_reg(dce, GOBOFB_DEBUG, 0x0000DEAD); */
|
|
asm(".word 0xfe16\n");
|
|
if (dce->dCtlStorage != nil)
|
|
{
|
|
ret = changeIRQ(dce, 0, openErr);
|
|
write_reg(dce, GOBOFB_VIDEOCTRL, 0);
|
|
DisposePtr((Ptr)(*(NuBusFPGADriverGlobalsHdl)dce->dCtlStorage)->siqel);
|
|
DisposeHandle(dce->dCtlStorage);
|
|
dce->dCtlStorage = nil;
|
|
}
|
|
return ret;
|
|
}
|
|
|