;/* timersoftint.c - Execute me to compile me with SAS C 5.10 LC -b1 -d0 -cfistq -v -y -j73 timersoftint.c Blink FROM LIB:c.o,timersoftint.o TO timersoftint LIBRARY LIB:LC.lib,LIB:Amiga.lib quit ; */ /* timersoftint.c - Timer device software interrupt message port example. */ #include <exec/memory.h> #include <exec/interrupts.h> #include <devices/timer.h> #include <dos/dos.h> #include <clib/exec_protos.h> #include <clib/dos_protos.h> #include <clib/alib_protos.h> #include <stdio.h> #ifdef LATTICE int CXBRK(void) { return(0); } /* Disable Lattice CTRL/C handling */ void chkabort(void) { return; } /* really */ #endif #define MICRO_DELAY 1000 #define OFF 0 #define ON 1 #define STOPPED 2 struct TSIData { ULONG tsi_Counter; ULONG tsi_Flag; struct MsgPort *tsi_Port; }; struct TSIData *tsidata; void tsoftcode(void); /* Prototype for our software interrupt code */ void main(void) { struct MsgPort *port; struct Interrupt *softint; struct timerequest *tr; ULONG endcount; /* Allocate message port, data & interrupt structures. Don't use CreatePort() */ /* or CreateMsgPort() since they allocate a signal (don't need that) for a */ /* PA_SIGNAL type port. We need PA_SOFTINT. */ if (tsidata = AllocMem(sizeof(struct TSIData), MEMF_PUBLIC|MEMF_CLEAR)) { if(port = AllocMem(sizeof(struct MsgPort), MEMF_PUBLIC|MEMF_CLEAR)) { NewList(&(port->mp_MsgList)); /* Initialize message list */ if (softint = AllocMem(sizeof(struct Interrupt), MEMF_PUBLIC|MEMF_CLEAR)) { /* Set up the (software)interrupt structure. Note that this task runs at */ /* priority 0. Software interrupts may only be priority -32, -16, 0, +16, */ /* +32. Also not that the correct node type for a software interrupt is */ /* NT_INTERRUPT. (NT_SOFTINT is an internal Exec flag). This is the same */ /* setup as that for a software interrupt which you Cause(). If our */ /* interrupt code was in assembler, you could initialize is_Data here to */ /* contain a pointer to shared data structures. An assembler software */ /* interrupt routine would receive the is_Data in A1. */ softint->is_Code = tsoftcode; /* The software interrupt routine */ softint->is_Data = tsidata; softint->is_Node.ln_Pri = 0; port->mp_Node.ln_Type = NT_MSGPORT; /* Set up the PA_SOFTINT message port */ port->mp_Flags = PA_SOFTINT; /* (no need to make this port public). */ port->mp_SigTask = (struct Task *)softint; /* pointer to interrupt structure */ /* Allocate timerequest */ if (tr = (struct timerequest *) CreateExtIO(port, sizeof(struct timerequest))) { /* Open timer.device. NULL is success. */ if (!(OpenDevice("timer.device", UNIT_MICROHZ, (struct IORequest *)tr, 0))) { tsidata->tsi_Flag = ON; /* Init data structure to share globally. */ tsidata->tsi_Port = port; /* Send of the first timerequest to start. IMPORTANT: Do NOT */ /* BeginIO() to any device other than audio or timer from */ /* within a software or hardware interrupt. The BeginIO() code */ /* may allocate memory, wait or perform other functions which */ /* are illegal or dangerous during interrupts. */ printf("starting softint. CTRL-C to break...\n"); tr->tr_node.io_command = tr_addrequest; /* initial iorequest to start */ tr->tr_time.tv_micro = MICRO_DELAY; /* software interrupt. */ BeginIO((struct IORequest *)tr); Wait(SIGBREAKF_CTRL_C); endcount = tsidata->tsi_Counter; printf("timer softint counted %ld milliseconds.\n", endcount); printf("Stopping timer...\n"); tsidata->tsi_Flag = OFF; while (tsidata->tsi_Flag != STOPPED) Delay(10); CloseDevice((struct IORequest *)tr); } else printf("couldn't open timer.device\n"); DeleteExtIO(tr); } else printf("couldn't create timerequest\n"); FreeMem(softint, sizeof(struct Interrupt)); } FreeMem(port, sizeof(struct MsgPort)); } FreeMem(tsidata, sizeof(struct TSIData)); } } void tsoftcode(void) { struct timerequest *tr; /* Remove the message from the port. */ tr = (struct timerequest *)GetMsg(tsidata->tsi_Port); /* Keep on going if main() hasn't set flag to OFF. */ if ((tr) && (tsidata->tsi_Flag == ON)) { /* increment counter and re-send timerequest--IMPORTANT: This */ /* self-perpetuating technique of calling BeginIO() during a software */ /* interrupt may only be used with the audio and timer device. */ tsidata->tsi_Counter++; tr->tr_node.io_command = tr_addrequest; tr->tr_time.tv_micro = MICRO_DELAY; BeginIO((struct IORequest *)tr); } /* Tell main() we're out of here. */ else tsidata->tsi_Flag = STOPPED; }