4 * Copyright (C) 2000-2007 Tibor "TS" Schütz
6 * Permission is hereby granted, free of charge, to any person obtaining a copy
7 * of this software and associated documentation files (the "Software"), to deal
8 * in the Software without restriction, including without limitation the rights
9 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10 * copies of the Software, and to permit persons to whom the Software is
11 * furnished to do so, subject to the following conditions:
13 * The above copyright notice and this permission notice shall be included in
14 * all copies or substantial portions of the Software.
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
28 typedef struct _GUSEmuState
30 uint8_t *himemaddr
; /* 1024*1024 bytes used for storing uploaded samples (+32 additional bytes for read padding) */
31 uint8_t *gusdatapos
; /* (gusdataend-gusdata) bytes used for storing emulated GF1/mixer register states (32*32+4 bytes in initial GUSemu32 version) */
34 unsigned int timer1fraction
;
35 unsigned int timer2fraction
;
39 /* ** Callback functions needed: */
40 /* NMI is defined as hwirq=-1 (not supported (yet?)) */
41 /* GUS_irqrequest returns the number of IRQs actually scheduled into the virtual machine */
42 /* Level triggered IRQ simulations normally return 1 */
43 /* Event triggered IRQ simulation can safely ignore GUS_irqclear calls */
44 int GUS_irqrequest(GUSEmuState
*state
, int hwirq
, int num
);/* needed in both mixer and bus emulation functions. */
45 void GUS_irqclear( GUSEmuState
*state
, int hwirq
); /* used by gus_write() only - can be left empty for mixer functions */
46 void GUS_dmarequest(GUSEmuState
*state
); /* used by gus_write() only - can be left empty for mixer functions */
48 /* ** ISA bus interface functions: */
50 /* Port I/O handlers */
51 /* support the following ports: */
52 /* 2x0,2x6,2x8...2xF,3x0...3x7; */
53 /* optional: 388,389 (at least writes should be forwarded or some GUS detection algorithms will fail) */
54 /* data is passed in host byte order */
55 unsigned int gus_read( GUSEmuState
*state
, int port
, int size
);
56 void gus_write(GUSEmuState
*state
, int port
, int size
, unsigned int data
);
57 /* size is given in bytes (1 for byte, 2 for word) */
59 /* DMA data transfer function */
60 /* data pointed to is passed in native x86 order */
61 void gus_dma_transferdata(GUSEmuState
*state
, char *dma_addr
, unsigned int count
, int TC
);
62 /* Called back by GUS_start_DMA as soon as the emulated DMA controller is ready for a transfer to or from GUS */
63 /* (might be immediately if the DMA controller was programmed first) */
64 /* dma_addr is an already translated address directly pointing to the beginning of the memory block */
65 /* do not forget to update DMA states after the call, including the DREQ and TC flags */
66 /* it is possible to break down a single transfer into multiple ones, but take care that: */
67 /* -dma_count is actually count-1 */
68 /* -before and during a transfer, DREQ is set and TC cleared */
69 /* -when calling gus_dma_transferdata(), TC is only set true for call transferring the last byte */
70 /* -after the last transfer, DREQ is cleared and TC is set */
72 /* ** GF1 mixer emulation functions: */
73 /* Usually, gus_irqgen should be called directly after gus_mixvoices if you can meet the recommended ranges. */
74 /* If the interrupts are executed immediately (i.e., are synchronous), it may be useful to break this */
75 /* down into a sequence of gus_mixvoice();gus_irqgen(); calls while mixing an audio block. */
76 /* If the interrupts are asynchronous, it may be needed to use a separate thread mixing into a temporary */
77 /* audio buffer in order to avoid quality loss caused by large numsamples and elapsed_time values. */
79 void gus_mixvoices(GUSEmuState
*state
, unsigned int playback_freq
, unsigned int numsamples
, int16_t *bufferpos
);
80 /* recommended range: 10 < numsamples < 100 */
81 /* lower values may result in increased rounding error, higher values often cause audible timing delays */
83 void gus_irqgen(GUSEmuState
*state
, unsigned int elapsed_time
);
84 /* recommended range: 80us < elapsed_time < max(1000us, numsamples/playback_freq) */
85 /* lower values won´t provide any benefit at all, higher values can cause audible timing delays */
86 /* note: masked timers are also calculated by this function, thus it might be needed even without any IRQs in use! */