2539 lines
70 KiB
C
2539 lines
70 KiB
C
/*****************************************************************************
|
|
*
|
|
* Yamaha YM2151 driver (version 2.150 final beta)
|
|
*
|
|
******************************************************************************/
|
|
|
|
#include <stdio.h>
|
|
#include <math.h>
|
|
|
|
#include "mamedef.h"
|
|
#include <stdlib.h>
|
|
#include <memory.h>
|
|
//#include "sndintrf.h"
|
|
//#include "streams.h"
|
|
#include "ym2151.h"
|
|
|
|
#define NULL ((void *)0)
|
|
|
|
|
|
/* undef this to not use MAME timer system */
|
|
//#define USE_MAME_TIMERS
|
|
|
|
/*#define FM_EMU*/
|
|
#ifdef FM_EMU
|
|
#ifdef USE_MAME_TIMERS
|
|
#undef USE_MAME_TIMERS
|
|
#endif
|
|
#endif
|
|
|
|
//static char LOG_CYM_FILE = 0x00;
|
|
//static FILE * cymfile = NULL;
|
|
|
|
|
|
/* struct describing a single operator */
|
|
typedef struct{
|
|
UINT32 phase; /* accumulated operator phase */
|
|
UINT32 freq; /* operator frequency count */
|
|
INT32 dt1; /* current DT1 (detune 1 phase inc/decrement) value */
|
|
UINT32 mul; /* frequency count multiply */
|
|
UINT32 dt1_i; /* DT1 index * 32 */
|
|
UINT32 dt2; /* current DT2 (detune 2) value */
|
|
|
|
signed int *connect; /* operator output 'direction' */
|
|
|
|
/* only M1 (operator 0) is filled with this data: */
|
|
signed int *mem_connect; /* where to put the delayed sample (MEM) */
|
|
INT32 mem_value; /* delayed sample (MEM) value */
|
|
|
|
/* channel specific data; note: each operator number 0 contains channel specific data */
|
|
UINT32 fb_shift; /* feedback shift value for operators 0 in each channel */
|
|
INT32 fb_out_curr; /* operator feedback value (used only by operators 0) */
|
|
INT32 fb_out_prev; /* previous feedback value (used only by operators 0) */
|
|
UINT32 kc; /* channel KC (copied to all operators) */
|
|
UINT32 kc_i; /* just for speedup */
|
|
UINT32 pms; /* channel PMS */
|
|
UINT32 ams; /* channel AMS */
|
|
/* end of channel specific data */
|
|
|
|
UINT32 AMmask; /* LFO Amplitude Modulation enable mask */
|
|
UINT32 state; /* Envelope state: 4-attack(AR) 3-decay(D1R) 2-sustain(D2R) 1-release(RR) 0-off */
|
|
UINT8 eg_sh_ar; /* (attack state) */
|
|
UINT8 eg_sel_ar; /* (attack state) */
|
|
UINT32 tl; /* Total attenuation Level */
|
|
INT32 volume; /* current envelope attenuation level */
|
|
UINT8 eg_sh_d1r; /* (decay state) */
|
|
UINT8 eg_sel_d1r; /* (decay state) */
|
|
UINT32 d1l; /* envelope switches to sustain state after reaching this level */
|
|
UINT8 eg_sh_d2r; /* (sustain state) */
|
|
UINT8 eg_sel_d2r; /* (sustain state) */
|
|
UINT8 eg_sh_rr; /* (release state) */
|
|
UINT8 eg_sel_rr; /* (release state) */
|
|
|
|
UINT32 key; /* 0=last key was KEY OFF, 1=last key was KEY ON */
|
|
|
|
UINT32 ks; /* key scale */
|
|
UINT32 ar; /* attack rate */
|
|
UINT32 d1r; /* decay rate */
|
|
UINT32 d2r; /* sustain rate */
|
|
UINT32 rr; /* release rate */
|
|
|
|
UINT32 reserved0; /**/
|
|
UINT32 reserved1; /**/
|
|
|
|
} YM2151Operator;
|
|
|
|
|
|
typedef struct
|
|
{
|
|
YM2151Operator oper[32]; /* the 32 operators */
|
|
|
|
UINT32 pan[16]; /* channels output masks (0xffffffff = enable) */
|
|
UINT8 Muted[8]; /* used for muting */
|
|
|
|
UINT32 eg_cnt; /* global envelope generator counter */
|
|
UINT32 eg_timer; /* global envelope generator counter works at frequency = chipclock/64/3 */
|
|
UINT32 eg_timer_add; /* step of eg_timer */
|
|
UINT32 eg_timer_overflow; /* envelope generator timer overlfows every 3 samples (on real chip) */
|
|
|
|
UINT32 lfo_phase; /* accumulated LFO phase (0 to 255) */
|
|
UINT32 lfo_timer; /* LFO timer */
|
|
UINT32 lfo_timer_add; /* step of lfo_timer */
|
|
UINT32 lfo_overflow; /* LFO generates new output when lfo_timer reaches this value */
|
|
UINT32 lfo_counter; /* LFO phase increment counter */
|
|
UINT32 lfo_counter_add; /* step of lfo_counter */
|
|
UINT8 lfo_wsel; /* LFO waveform (0-saw, 1-square, 2-triangle, 3-random noise) */
|
|
UINT8 amd; /* LFO Amplitude Modulation Depth */
|
|
INT8 pmd; /* LFO Phase Modulation Depth */
|
|
UINT32 lfa; /* LFO current AM output */
|
|
INT32 lfp; /* LFO current PM output */
|
|
|
|
UINT8 test; /* TEST register */
|
|
UINT8 ct; /* output control pins (bit1-CT2, bit0-CT1) */
|
|
|
|
UINT32 noise; /* noise enable/period register (bit 7 - noise enable, bits 4-0 - noise period */
|
|
UINT32 noise_rng; /* 17 bit noise shift register */
|
|
UINT32 noise_p; /* current noise 'phase'*/
|
|
UINT32 noise_f; /* current noise period */
|
|
|
|
UINT32 csm_req; /* CSM KEY ON / KEY OFF sequence request */
|
|
|
|
UINT32 irq_enable; /* IRQ enable for timer B (bit 3) and timer A (bit 2); bit 7 - CSM mode (keyon to all slots, everytime timer A overflows) */
|
|
UINT32 status; /* chip status (BUSY, IRQ Flags) */
|
|
UINT8 connect[8]; /* channels connections */
|
|
|
|
#ifdef USE_MAME_TIMERS
|
|
/* ASG 980324 -- added for tracking timers */
|
|
emu_timer *timer_A;
|
|
emu_timer *timer_B;
|
|
attotime timer_A_time[1024]; /* timer A times for MAME */
|
|
attotime timer_B_time[256]; /* timer B times for MAME */
|
|
int irqlinestate;
|
|
#else
|
|
UINT8 tim_A; /* timer A enable (0-disabled) */
|
|
UINT8 tim_B; /* timer B enable (0-disabled) */
|
|
INT32 tim_A_val; /* current value of timer A */
|
|
INT32 tim_B_val; /* current value of timer B */
|
|
UINT32 tim_A_tab[1024]; /* timer A deltas */
|
|
UINT32 tim_B_tab[256]; /* timer B deltas */
|
|
#endif
|
|
UINT32 timer_A_index; /* timer A index */
|
|
UINT32 timer_B_index; /* timer B index */
|
|
UINT32 timer_A_index_old; /* timer A previous index */
|
|
UINT32 timer_B_index_old; /* timer B previous index */
|
|
|
|
/* Frequency-deltas to get the closest frequency possible.
|
|
* There are 11 octaves because of DT2 (max 950 cents over base frequency)
|
|
* and LFO phase modulation (max 800 cents below AND over base frequency)
|
|
* Summary: octave explanation
|
|
* 0 note code - LFO PM
|
|
* 1 note code
|
|
* 2 note code
|
|
* 3 note code
|
|
* 4 note code
|
|
* 5 note code
|
|
* 6 note code
|
|
* 7 note code
|
|
* 8 note code
|
|
* 9 note code + DT2 + LFO PM
|
|
* 10 note code + DT2 + LFO PM
|
|
*/
|
|
UINT32 freq[11*768]; /* 11 octaves, 768 'cents' per octave */
|
|
|
|
/* Frequency deltas for DT1. These deltas alter operator frequency
|
|
* after it has been taken from frequency-deltas table.
|
|
*/
|
|
INT32 dt1_freq[8*32]; /* 8 DT1 levels, 32 KC values */
|
|
|
|
UINT32 noise_tab[32]; /* 17bit Noise Generator periods */
|
|
|
|
//void (*irqhandler)(const device_config *device, int irq); /* IRQ function handler */
|
|
//write8_device_func porthandler; /* port write function handler */
|
|
|
|
//const device_config *device;
|
|
unsigned int clock; /* chip clock in Hz (passed from 2151intf.c) */
|
|
unsigned int sampfreq; /* sampling frequency in Hz (passed from 2151intf.c) */
|
|
} YM2151;
|
|
|
|
|
|
#define FREQ_SH 16 /* 16.16 fixed point (frequency calculations) */
|
|
#define EG_SH 16 /* 16.16 fixed point (envelope generator timing) */
|
|
#define LFO_SH 10 /* 22.10 fixed point (LFO calculations) */
|
|
#define TIMER_SH 16 /* 16.16 fixed point (timers calculations) */
|
|
|
|
#define FREQ_MASK ((1<<FREQ_SH)-1)
|
|
|
|
#define ENV_BITS 10
|
|
#define ENV_LEN (1<<ENV_BITS)
|
|
#define ENV_STEP (128.0/ENV_LEN)
|
|
|
|
#define MAX_ATT_INDEX (ENV_LEN-1) /* 1023 */
|
|
#define MIN_ATT_INDEX (0) /* 0 */
|
|
|
|
#define EG_ATT 4
|
|
#define EG_DEC 3
|
|
#define EG_SUS 2
|
|
#define EG_REL 1
|
|
#define EG_OFF 0
|
|
|
|
#define SIN_BITS 10
|
|
#define SIN_LEN (1<<SIN_BITS)
|
|
#define SIN_MASK (SIN_LEN-1)
|
|
|
|
#define TL_RES_LEN (256) /* 8 bits addressing (real chip) */
|
|
|
|
|
|
#if (SAMPLE_BITS==16)
|
|
#define FINAL_SH (0)
|
|
#define MAXOUT (+32767)
|
|
#define MINOUT (-32768)
|
|
#else
|
|
#define FINAL_SH (8)
|
|
#define MAXOUT (+127)
|
|
#define MINOUT (-128)
|
|
#endif
|
|
|
|
|
|
/* TL_TAB_LEN is calculated as:
|
|
* 13 - sinus amplitude bits (Y axis)
|
|
* 2 - sinus sign bit (Y axis)
|
|
* TL_RES_LEN - sinus resolution (X axis)
|
|
*/
|
|
#define TL_TAB_LEN (13*2*TL_RES_LEN)
|
|
static signed int tl_tab[TL_TAB_LEN];
|
|
|
|
#define ENV_QUIET (TL_TAB_LEN>>3)
|
|
|
|
/* sin waveform table in 'decibel' scale */
|
|
static unsigned int sin_tab[SIN_LEN];
|
|
|
|
|
|
/* translate from D1L to volume index (16 D1L levels) */
|
|
static UINT32 d1l_tab[16];
|
|
|
|
|
|
#define RATE_STEPS (8)
|
|
static const UINT8 eg_inc[19*RATE_STEPS]={
|
|
|
|
/*cycle:0 1 2 3 4 5 6 7*/
|
|
|
|
/* 0 */ 0,1, 0,1, 0,1, 0,1, /* rates 00..11 0 (increment by 0 or 1) */
|
|
/* 1 */ 0,1, 0,1, 1,1, 0,1, /* rates 00..11 1 */
|
|
/* 2 */ 0,1, 1,1, 0,1, 1,1, /* rates 00..11 2 */
|
|
/* 3 */ 0,1, 1,1, 1,1, 1,1, /* rates 00..11 3 */
|
|
|
|
/* 4 */ 1,1, 1,1, 1,1, 1,1, /* rate 12 0 (increment by 1) */
|
|
/* 5 */ 1,1, 1,2, 1,1, 1,2, /* rate 12 1 */
|
|
/* 6 */ 1,2, 1,2, 1,2, 1,2, /* rate 12 2 */
|
|
/* 7 */ 1,2, 2,2, 1,2, 2,2, /* rate 12 3 */
|
|
|
|
/* 8 */ 2,2, 2,2, 2,2, 2,2, /* rate 13 0 (increment by 2) */
|
|
/* 9 */ 2,2, 2,4, 2,2, 2,4, /* rate 13 1 */
|
|
/*10 */ 2,4, 2,4, 2,4, 2,4, /* rate 13 2 */
|
|
/*11 */ 2,4, 4,4, 2,4, 4,4, /* rate 13 3 */
|
|
|
|
/*12 */ 4,4, 4,4, 4,4, 4,4, /* rate 14 0 (increment by 4) */
|
|
/*13 */ 4,4, 4,8, 4,4, 4,8, /* rate 14 1 */
|
|
/*14 */ 4,8, 4,8, 4,8, 4,8, /* rate 14 2 */
|
|
/*15 */ 4,8, 8,8, 4,8, 8,8, /* rate 14 3 */
|
|
|
|
/*16 */ 8,8, 8,8, 8,8, 8,8, /* rates 15 0, 15 1, 15 2, 15 3 (increment by 8) */
|
|
/*17 */ 16,16,16,16,16,16,16,16, /* rates 15 2, 15 3 for attack */
|
|
/*18 */ 0,0, 0,0, 0,0, 0,0, /* infinity rates for attack and decay(s) */
|
|
};
|
|
|
|
|
|
#define O(a) (a*RATE_STEPS)
|
|
|
|
/*note that there is no O(17) in this table - it's directly in the code */
|
|
static const UINT8 eg_rate_select[32+64+32]={ /* Envelope Generator rates (32 + 64 rates + 32 RKS) */
|
|
/* 32 dummy (infinite time) rates */
|
|
O(18),O(18),O(18),O(18),O(18),O(18),O(18),O(18),
|
|
O(18),O(18),O(18),O(18),O(18),O(18),O(18),O(18),
|
|
O(18),O(18),O(18),O(18),O(18),O(18),O(18),O(18),
|
|
O(18),O(18),O(18),O(18),O(18),O(18),O(18),O(18),
|
|
|
|
/* rates 00-11 */
|
|
O( 0),O( 1),O( 2),O( 3),
|
|
O( 0),O( 1),O( 2),O( 3),
|
|
O( 0),O( 1),O( 2),O( 3),
|
|
O( 0),O( 1),O( 2),O( 3),
|
|
O( 0),O( 1),O( 2),O( 3),
|
|
O( 0),O( 1),O( 2),O( 3),
|
|
O( 0),O( 1),O( 2),O( 3),
|
|
O( 0),O( 1),O( 2),O( 3),
|
|
O( 0),O( 1),O( 2),O( 3),
|
|
O( 0),O( 1),O( 2),O( 3),
|
|
O( 0),O( 1),O( 2),O( 3),
|
|
O( 0),O( 1),O( 2),O( 3),
|
|
|
|
/* rate 12 */
|
|
O( 4),O( 5),O( 6),O( 7),
|
|
|
|
/* rate 13 */
|
|
O( 8),O( 9),O(10),O(11),
|
|
|
|
/* rate 14 */
|
|
O(12),O(13),O(14),O(15),
|
|
|
|
/* rate 15 */
|
|
O(16),O(16),O(16),O(16),
|
|
|
|
/* 32 dummy rates (same as 15 3) */
|
|
O(16),O(16),O(16),O(16),O(16),O(16),O(16),O(16),
|
|
O(16),O(16),O(16),O(16),O(16),O(16),O(16),O(16),
|
|
O(16),O(16),O(16),O(16),O(16),O(16),O(16),O(16),
|
|
O(16),O(16),O(16),O(16),O(16),O(16),O(16),O(16)
|
|
|
|
};
|
|
#undef O
|
|
|
|
/*rate 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15*/
|
|
/*shift 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0, 0, 0, 0, 0 */
|
|
/*mask 2047, 1023, 511, 255, 127, 63, 31, 15, 7, 3, 1, 0, 0, 0, 0, 0 */
|
|
|
|
#define O(a) (a*1)
|
|
static const UINT8 eg_rate_shift[32+64+32]={ /* Envelope Generator counter shifts (32 + 64 rates + 32 RKS) */
|
|
/* 32 infinite time rates */
|
|
O(0),O(0),O(0),O(0),O(0),O(0),O(0),O(0),
|
|
O(0),O(0),O(0),O(0),O(0),O(0),O(0),O(0),
|
|
O(0),O(0),O(0),O(0),O(0),O(0),O(0),O(0),
|
|
O(0),O(0),O(0),O(0),O(0),O(0),O(0),O(0),
|
|
|
|
|
|
/* rates 00-11 */
|
|
O(11),O(11),O(11),O(11),
|
|
O(10),O(10),O(10),O(10),
|
|
O( 9),O( 9),O( 9),O( 9),
|
|
O( 8),O( 8),O( 8),O( 8),
|
|
O( 7),O( 7),O( 7),O( 7),
|
|
O( 6),O( 6),O( 6),O( 6),
|
|
O( 5),O( 5),O( 5),O( 5),
|
|
O( 4),O( 4),O( 4),O( 4),
|
|
O( 3),O( 3),O( 3),O( 3),
|
|
O( 2),O( 2),O( 2),O( 2),
|
|
O( 1),O( 1),O( 1),O( 1),
|
|
O( 0),O( 0),O( 0),O( 0),
|
|
|
|
/* rate 12 */
|
|
O( 0),O( 0),O( 0),O( 0),
|
|
|
|
/* rate 13 */
|
|
O( 0),O( 0),O( 0),O( 0),
|
|
|
|
/* rate 14 */
|
|
O( 0),O( 0),O( 0),O( 0),
|
|
|
|
/* rate 15 */
|
|
O( 0),O( 0),O( 0),O( 0),
|
|
|
|
/* 32 dummy rates (same as 15 3) */
|
|
O( 0),O( 0),O( 0),O( 0),O( 0),O( 0),O( 0),O( 0),
|
|
O( 0),O( 0),O( 0),O( 0),O( 0),O( 0),O( 0),O( 0),
|
|
O( 0),O( 0),O( 0),O( 0),O( 0),O( 0),O( 0),O( 0),
|
|
O( 0),O( 0),O( 0),O( 0),O( 0),O( 0),O( 0),O( 0)
|
|
|
|
};
|
|
#undef O
|
|
|
|
/* DT2 defines offset in cents from base note
|
|
*
|
|
* This table defines offset in frequency-deltas table.
|
|
* User's Manual page 22
|
|
*
|
|
* Values below were calculated using formula: value = orig.val / 1.5625
|
|
*
|
|
* DT2=0 DT2=1 DT2=2 DT2=3
|
|
* 0 600 781 950
|
|
*/
|
|
static const UINT32 dt2_tab[4] = { 0, 384, 500, 608 };
|
|
|
|
/* DT1 defines offset in Hertz from base note
|
|
* This table is converted while initialization...
|
|
* Detune table shown in YM2151 User's Manual is wrong (verified on the real chip)
|
|
*/
|
|
|
|
static const UINT8 dt1_tab[4*32] = { /* 4*32 DT1 values */
|
|
/* DT1=0 */
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
|
|
/* DT1=1 */
|
|
0, 0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2,
|
|
2, 3, 3, 3, 4, 4, 4, 5, 5, 6, 6, 7, 8, 8, 8, 8,
|
|
|
|
/* DT1=2 */
|
|
1, 1, 1, 1, 2, 2, 2, 2, 2, 3, 3, 3, 4, 4, 4, 5,
|
|
5, 6, 6, 7, 8, 8, 9,10,11,12,13,14,16,16,16,16,
|
|
|
|
/* DT1=3 */
|
|
2, 2, 2, 2, 2, 3, 3, 3, 4, 4, 4, 5, 5, 6, 6, 7,
|
|
8, 8, 9,10,11,12,13,14,16,17,19,20,22,22,22,22
|
|
};
|
|
|
|
static const UINT16 phaseinc_rom[768]={
|
|
1299,1300,1301,1302,1303,1304,1305,1306,1308,1309,1310,1311,1313,1314,1315,1316,
|
|
1318,1319,1320,1321,1322,1323,1324,1325,1327,1328,1329,1330,1332,1333,1334,1335,
|
|
1337,1338,1339,1340,1341,1342,1343,1344,1346,1347,1348,1349,1351,1352,1353,1354,
|
|
1356,1357,1358,1359,1361,1362,1363,1364,1366,1367,1368,1369,1371,1372,1373,1374,
|
|
1376,1377,1378,1379,1381,1382,1383,1384,1386,1387,1388,1389,1391,1392,1393,1394,
|
|
1396,1397,1398,1399,1401,1402,1403,1404,1406,1407,1408,1409,1411,1412,1413,1414,
|
|
1416,1417,1418,1419,1421,1422,1423,1424,1426,1427,1429,1430,1431,1432,1434,1435,
|
|
1437,1438,1439,1440,1442,1443,1444,1445,1447,1448,1449,1450,1452,1453,1454,1455,
|
|
1458,1459,1460,1461,1463,1464,1465,1466,1468,1469,1471,1472,1473,1474,1476,1477,
|
|
1479,1480,1481,1482,1484,1485,1486,1487,1489,1490,1492,1493,1494,1495,1497,1498,
|
|
1501,1502,1503,1504,1506,1507,1509,1510,1512,1513,1514,1515,1517,1518,1520,1521,
|
|
1523,1524,1525,1526,1528,1529,1531,1532,1534,1535,1536,1537,1539,1540,1542,1543,
|
|
1545,1546,1547,1548,1550,1551,1553,1554,1556,1557,1558,1559,1561,1562,1564,1565,
|
|
1567,1568,1569,1570,1572,1573,1575,1576,1578,1579,1580,1581,1583,1584,1586,1587,
|
|
1590,1591,1592,1593,1595,1596,1598,1599,1601,1602,1604,1605,1607,1608,1609,1610,
|
|
1613,1614,1615,1616,1618,1619,1621,1622,1624,1625,1627,1628,1630,1631,1632,1633,
|
|
1637,1638,1639,1640,1642,1643,1645,1646,1648,1649,1651,1652,1654,1655,1656,1657,
|
|
1660,1661,1663,1664,1666,1667,1669,1670,1672,1673,1675,1676,1678,1679,1681,1682,
|
|
1685,1686,1688,1689,1691,1692,1694,1695,1697,1698,1700,1701,1703,1704,1706,1707,
|
|
1709,1710,1712,1713,1715,1716,1718,1719,1721,1722,1724,1725,1727,1728,1730,1731,
|
|
1734,1735,1737,1738,1740,1741,1743,1744,1746,1748,1749,1751,1752,1754,1755,1757,
|
|
1759,1760,1762,1763,1765,1766,1768,1769,1771,1773,1774,1776,1777,1779,1780,1782,
|
|
1785,1786,1788,1789,1791,1793,1794,1796,1798,1799,1801,1802,1804,1806,1807,1809,
|
|
1811,1812,1814,1815,1817,1819,1820,1822,1824,1825,1827,1828,1830,1832,1833,1835,
|
|
1837,1838,1840,1841,1843,1845,1846,1848,1850,1851,1853,1854,1856,1858,1859,1861,
|
|
1864,1865,1867,1868,1870,1872,1873,1875,1877,1879,1880,1882,1884,1885,1887,1888,
|
|
1891,1892,1894,1895,1897,1899,1900,1902,1904,1906,1907,1909,1911,1912,1914,1915,
|
|
1918,1919,1921,1923,1925,1926,1928,1930,1932,1933,1935,1937,1939,1940,1942,1944,
|
|
1946,1947,1949,1951,1953,1954,1956,1958,1960,1961,1963,1965,1967,1968,1970,1972,
|
|
1975,1976,1978,1980,1982,1983,1985,1987,1989,1990,1992,1994,1996,1997,1999,2001,
|
|
2003,2004,2006,2008,2010,2011,2013,2015,2017,2019,2021,2022,2024,2026,2028,2029,
|
|
2032,2033,2035,2037,2039,2041,2043,2044,2047,2048,2050,2052,2054,2056,2058,2059,
|
|
2062,2063,2065,2067,2069,2071,2073,2074,2077,2078,2080,2082,2084,2086,2088,2089,
|
|
2092,2093,2095,2097,2099,2101,2103,2104,2107,2108,2110,2112,2114,2116,2118,2119,
|
|
2122,2123,2125,2127,2129,2131,2133,2134,2137,2139,2141,2142,2145,2146,2148,2150,
|
|
2153,2154,2156,2158,2160,2162,2164,2165,2168,2170,2172,2173,2176,2177,2179,2181,
|
|
2185,2186,2188,2190,2192,2194,2196,2197,2200,2202,2204,2205,2208,2209,2211,2213,
|
|
2216,2218,2220,2222,2223,2226,2227,2230,2232,2234,2236,2238,2239,2242,2243,2246,
|
|
2249,2251,2253,2255,2256,2259,2260,2263,2265,2267,2269,2271,2272,2275,2276,2279,
|
|
2281,2283,2285,2287,2288,2291,2292,2295,2297,2299,2301,2303,2304,2307,2308,2311,
|
|
2315,2317,2319,2321,2322,2325,2326,2329,2331,2333,2335,2337,2338,2341,2342,2345,
|
|
2348,2350,2352,2354,2355,2358,2359,2362,2364,2366,2368,2370,2371,2374,2375,2378,
|
|
2382,2384,2386,2388,2389,2392,2393,2396,2398,2400,2402,2404,2407,2410,2411,2414,
|
|
2417,2419,2421,2423,2424,2427,2428,2431,2433,2435,2437,2439,2442,2445,2446,2449,
|
|
2452,2454,2456,2458,2459,2462,2463,2466,2468,2470,2472,2474,2477,2480,2481,2484,
|
|
2488,2490,2492,2494,2495,2498,2499,2502,2504,2506,2508,2510,2513,2516,2517,2520,
|
|
2524,2526,2528,2530,2531,2534,2535,2538,2540,2542,2544,2546,2549,2552,2553,2556,
|
|
2561,2563,2565,2567,2568,2571,2572,2575,2577,2579,2581,2583,2586,2589,2590,2593
|
|
};
|
|
|
|
|
|
/*
|
|
Noise LFO waveform.
|
|
|
|
Here are just 256 samples out of much longer data.
|
|
|
|
It does NOT repeat every 256 samples on real chip and I wasnt able to find
|
|
the point where it repeats (even in strings as long as 131072 samples).
|
|
|
|
I only put it here because its better than nothing and perhaps
|
|
someone might be able to figure out the real algorithm.
|
|
|
|
|
|
Note that (due to the way the LFO output is calculated) it is quite
|
|
possible that two values: 0x80 and 0x00 might be wrong in this table.
|
|
To be exact:
|
|
some 0x80 could be 0x81 as well as some 0x00 could be 0x01.
|
|
*/
|
|
|
|
static const UINT8 lfo_noise_waveform[256] = {
|
|
0xFF,0xEE,0xD3,0x80,0x58,0xDA,0x7F,0x94,0x9E,0xE3,0xFA,0x00,0x4D,0xFA,0xFF,0x6A,
|
|
0x7A,0xDE,0x49,0xF6,0x00,0x33,0xBB,0x63,0x91,0x60,0x51,0xFF,0x00,0xD8,0x7F,0xDE,
|
|
0xDC,0x73,0x21,0x85,0xB2,0x9C,0x5D,0x24,0xCD,0x91,0x9E,0x76,0x7F,0x20,0xFB,0xF3,
|
|
0x00,0xA6,0x3E,0x42,0x27,0x69,0xAE,0x33,0x45,0x44,0x11,0x41,0x72,0x73,0xDF,0xA2,
|
|
|
|
0x32,0xBD,0x7E,0xA8,0x13,0xEB,0xD3,0x15,0xDD,0xFB,0xC9,0x9D,0x61,0x2F,0xBE,0x9D,
|
|
0x23,0x65,0x51,0x6A,0x84,0xF9,0xC9,0xD7,0x23,0xBF,0x65,0x19,0xDC,0x03,0xF3,0x24,
|
|
0x33,0xB6,0x1E,0x57,0x5C,0xAC,0x25,0x89,0x4D,0xC5,0x9C,0x99,0x15,0x07,0xCF,0xBA,
|
|
0xC5,0x9B,0x15,0x4D,0x8D,0x2A,0x1E,0x1F,0xEA,0x2B,0x2F,0x64,0xA9,0x50,0x3D,0xAB,
|
|
|
|
0x50,0x77,0xE9,0xC0,0xAC,0x6D,0x3F,0xCA,0xCF,0x71,0x7D,0x80,0xA6,0xFD,0xFF,0xB5,
|
|
0xBD,0x6F,0x24,0x7B,0x00,0x99,0x5D,0xB1,0x48,0xB0,0x28,0x7F,0x80,0xEC,0xBF,0x6F,
|
|
0x6E,0x39,0x90,0x42,0xD9,0x4E,0x2E,0x12,0x66,0xC8,0xCF,0x3B,0x3F,0x10,0x7D,0x79,
|
|
0x00,0xD3,0x1F,0x21,0x93,0x34,0xD7,0x19,0x22,0xA2,0x08,0x20,0xB9,0xB9,0xEF,0x51,
|
|
|
|
0x99,0xDE,0xBF,0xD4,0x09,0x75,0xE9,0x8A,0xEE,0xFD,0xE4,0x4E,0x30,0x17,0xDF,0xCE,
|
|
0x11,0xB2,0x28,0x35,0xC2,0x7C,0x64,0xEB,0x91,0x5F,0x32,0x0C,0x6E,0x00,0xF9,0x92,
|
|
0x19,0xDB,0x8F,0xAB,0xAE,0xD6,0x12,0xC4,0x26,0x62,0xCE,0xCC,0x0A,0x03,0xE7,0xDD,
|
|
0xE2,0x4D,0x8A,0xA6,0x46,0x95,0x0F,0x8F,0xF5,0x15,0x97,0x32,0xD4,0x28,0x1E,0x55
|
|
};
|
|
|
|
|
|
|
|
/* these variables stay here for speedup purposes only */
|
|
static YM2151 * PSG;
|
|
static signed int chanout[8];
|
|
static signed int m2,c1,c2; /* Phase Modulation input for operators 2,3,4 */
|
|
static signed int mem; /* one sample delay memory */
|
|
|
|
|
|
/* save output as raw 16-bit sample */
|
|
// #define SAVE_SAMPLE
|
|
// #define SAVE_SEPARATE_CHANNELS
|
|
#if defined SAVE_SAMPLE || defined SAVE_SEPARATE_CHANNELS
|
|
#include <stdio.h>
|
|
|
|
static FILE *sample[9];
|
|
#endif
|
|
|
|
|
|
|
|
|
|
static void init_tables(void)
|
|
{
|
|
signed int i,x,n;
|
|
double o,m;
|
|
|
|
for (x=0; x<TL_RES_LEN; x++)
|
|
{
|
|
m = (1<<16) / pow(2, (x+1) * (ENV_STEP/4.0) / 8.0);
|
|
m = floor(m);
|
|
|
|
/* we never reach (1<<16) here due to the (x+1) */
|
|
/* result fits within 16 bits at maximum */
|
|
|
|
n = (int)m; /* 16 bits here */
|
|
n >>= 4; /* 12 bits here */
|
|
if (n&1) /* round to closest */
|
|
n = (n>>1)+1;
|
|
else
|
|
n = n>>1;
|
|
/* 11 bits here (rounded) */
|
|
n <<= 2; /* 13 bits here (as in real chip) */
|
|
tl_tab[ x*2 + 0 ] = n;
|
|
tl_tab[ x*2 + 1 ] = -tl_tab[ x*2 + 0 ];
|
|
|
|
for (i=1; i<13; i++)
|
|
{
|
|
tl_tab[ x*2+0 + i*2*TL_RES_LEN ] = tl_tab[ x*2+0 ]>>i;
|
|
tl_tab[ x*2+1 + i*2*TL_RES_LEN ] = -tl_tab[ x*2+0 + i*2*TL_RES_LEN ];
|
|
}
|
|
#if 0
|
|
logerror("tl %04i", x*2);
|
|
for (i=0; i<13; i++)
|
|
logerror(", [%02i] %4i", i*2, tl_tab[ x*2 /*+1*/ + i*2*TL_RES_LEN ]);
|
|
logerror("\n");
|
|
#endif
|
|
}
|
|
/*logerror("TL_TAB_LEN = %i (%i bytes)\n",TL_TAB_LEN, (int)sizeof(tl_tab));*/
|
|
/*logerror("ENV_QUIET= %i\n",ENV_QUIET );*/
|
|
|
|
|
|
for (i=0; i<SIN_LEN; i++)
|
|
{
|
|
/* non-standard sinus */
|
|
m = sin( ((i*2)+1) * M_PI / SIN_LEN ); /* verified on the real chip */
|
|
|
|
/* we never reach zero here due to ((i*2)+1) */
|
|
|
|
if (m>0.0)
|
|
o = 8*log(1.0/m)/log(2.0); /* convert to 'decibels' */
|
|
else
|
|
o = 8*log(-1.0/m)/log(2.0); /* convert to 'decibels' */
|
|
|
|
o = o / (ENV_STEP/4);
|
|
|
|
n = (int)(2.0*o);
|
|
if (n&1) /* round to closest */
|
|
n = (n>>1)+1;
|
|
else
|
|
n = n>>1;
|
|
|
|
sin_tab[ i ] = n*2 + (m>=0.0? 0: 1 );
|
|
/*logerror("sin [0x%4x]= %4i (tl_tab value=%8x)\n", i, sin_tab[i],tl_tab[sin_tab[i]]);*/
|
|
}
|
|
|
|
|
|
/* calculate d1l_tab table */
|
|
for (i=0; i<16; i++)
|
|
{
|
|
m = (i!=15 ? i : i+16) * (4.0/ENV_STEP); /* every 3 'dB' except for all bits = 1 = 45+48 'dB' */
|
|
d1l_tab[i] = m;
|
|
/*logerror("d1l_tab[%02x]=%08x\n",i,d1l_tab[i] );*/
|
|
}
|
|
|
|
#ifdef SAVE_SAMPLE
|
|
sample[8]=fopen("sampsum.pcm","wb");
|
|
#endif
|
|
#ifdef SAVE_SEPARATE_CHANNELS
|
|
sample[0]=fopen("samp0.pcm","wb");
|
|
sample[1]=fopen("samp1.pcm","wb");
|
|
sample[2]=fopen("samp2.pcm","wb");
|
|
sample[3]=fopen("samp3.pcm","wb");
|
|
sample[4]=fopen("samp4.pcm","wb");
|
|
sample[5]=fopen("samp5.pcm","wb");
|
|
sample[6]=fopen("samp6.pcm","wb");
|
|
sample[7]=fopen("samp7.pcm","wb");
|
|
#endif
|
|
}
|
|
|
|
|
|
static void init_chip_tables(YM2151 *chip)
|
|
{
|
|
int i,j;
|
|
double mult,phaseinc,Hz;
|
|
double scaler;
|
|
//attotime pom;
|
|
double pom;
|
|
|
|
scaler = ( (double)chip->clock / 64.0 ) / ( (double)chip->sampfreq );
|
|
/*logerror("scaler = %20.15f\n", scaler);*/
|
|
|
|
|
|
/* this loop calculates Hertz values for notes from c-0 to b-7 */
|
|
/* including 64 'cents' (100/64 that is 1.5625 of real cent) per note */
|
|
/* i*100/64/1200 is equal to i/768 */
|
|
|
|
/* real chip works with 10 bits fixed point values (10.10) */
|
|
mult = (1<<(FREQ_SH-10)); /* -10 because phaseinc_rom table values are already in 10.10 format */
|
|
|
|
for (i=0; i<768; i++)
|
|
{
|
|
/* 3.4375 Hz is note A; C# is 4 semitones higher */
|
|
Hz = 1000;
|
|
#if 0
|
|
/* Hz is close, but not perfect */
|
|
//Hz = scaler * 3.4375 * pow (2, (i + 4 * 64 ) / 768.0 );
|
|
/* calculate phase increment */
|
|
phaseinc = (Hz*SIN_LEN) / (double)chip->sampfreq;
|
|
#endif
|
|
|
|
phaseinc = phaseinc_rom[i]; /* real chip phase increment */
|
|
phaseinc *= scaler; /* adjust */
|
|
|
|
|
|
/* octave 2 - reference octave */
|
|
chip->freq[ 768+2*768+i ] = ((int)(phaseinc*mult)) & 0xffffffc0; /* adjust to X.10 fixed point */
|
|
/* octave 0 and octave 1 */
|
|
for (j=0; j<2; j++)
|
|
{
|
|
chip->freq[768 + j*768 + i] = (chip->freq[ 768+2*768+i ] >> (2-j) ) & 0xffffffc0; /* adjust to X.10 fixed point */
|
|
}
|
|
/* octave 3 to 7 */
|
|
for (j=3; j<8; j++)
|
|
{
|
|
chip->freq[768 + j*768 + i] = chip->freq[ 768+2*768+i ] << (j-2);
|
|
}
|
|
|
|
#if 0
|
|
pom = (double)chip->freq[ 768+2*768+i ] / ((double)(1<<FREQ_SH));
|
|
pom = pom * (double)chip->sampfreq / (double)SIN_LEN;
|
|
logerror("1freq[%4i][%08x]= real %20.15f Hz emul %20.15f Hz\n", i, chip->freq[ 768+2*768+i ], Hz, pom);
|
|
#endif
|
|
}
|
|
|
|
/* octave -1 (all equal to: oct 0, _KC_00_, _KF_00_) */
|
|
for (i=0; i<768; i++)
|
|
{
|
|
chip->freq[ 0*768 + i ] = chip->freq[1*768+0];
|
|
}
|
|
|
|
/* octave 8 and 9 (all equal to: oct 7, _KC_14_, _KF_63_) */
|
|
for (j=8; j<10; j++)
|
|
{
|
|
for (i=0; i<768; i++)
|
|
{
|
|
chip->freq[768+ j*768 + i ] = chip->freq[768 + 8*768 -1];
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
for (i=0; i<11*768; i++)
|
|
{
|
|
pom = (double)chip->freq[i] / ((double)(1<<FREQ_SH));
|
|
pom = pom * (double)chip->sampfreq / (double)SIN_LEN;
|
|
logerror("freq[%4i][%08x]= emul %20.15f Hz\n", i, chip->freq[i], pom);
|
|
}
|
|
#endif
|
|
|
|
mult = (1<<FREQ_SH);
|
|
for (j=0; j<4; j++)
|
|
{
|
|
for (i=0; i<32; i++)
|
|
{
|
|
Hz = ( (double)dt1_tab[j*32+i] * ((double)chip->clock/64.0) ) / (double)(1<<20);
|
|
|
|
/*calculate phase increment*/
|
|
phaseinc = (Hz*SIN_LEN) / (double)chip->sampfreq;
|
|
|
|
/*positive and negative values*/
|
|
chip->dt1_freq[ (j+0)*32 + i ] = phaseinc * mult;
|
|
chip->dt1_freq[ (j+4)*32 + i ] = -chip->dt1_freq[ (j+0)*32 + i ];
|
|
|
|
#if 0
|
|
{
|
|
int x = j*32 + i;
|
|
pom = (double)chip->dt1_freq[x] / mult;
|
|
pom = pom * (double)chip->sampfreq / (double)SIN_LEN;
|
|
logerror("DT1(%03i)[%02i %02i][%08x]= real %19.15f Hz emul %19.15f Hz\n",
|
|
x, j, i, chip->dt1_freq[x], Hz, pom);
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
|
|
|
|
/* calculate timers' deltas */
|
|
/* User's Manual pages 15,16 */
|
|
mult = (1<<TIMER_SH);
|
|
for (i=0; i<1024; i++)
|
|
{
|
|
/* ASG 980324: changed to compute both tim_A_tab and timer_A_time */
|
|
//pom= attotime_mul(ATTOTIME_IN_HZ(chip->clock), 64 * (1024 - i));
|
|
pom = ((double)64 * (1024 - i) / chip->clock);
|
|
#ifdef USE_MAME_TIMERS
|
|
chip->timer_A_time[i] = pom;
|
|
#else
|
|
//chip->tim_A_tab[i] = attotime_to_double(pom) * (double)chip->sampfreq * mult; /* number of samples that timer period takes (fixed point) */
|
|
chip->tim_A_tab[i] = pom * (double)chip->sampfreq * mult; /* number of samples that timer period takes (fixed point) */
|
|
#endif
|
|
}
|
|
for (i=0; i<256; i++)
|
|
{
|
|
/* ASG 980324: changed to compute both tim_B_tab and timer_B_time */
|
|
//pom= attotime_mul(ATTOTIME_IN_HZ(chip->clock), 1024 * (256 - i));
|
|
pom = ((double)1024 * (256 - i) / chip->clock);
|
|
#ifdef USE_MAME_TIMERS
|
|
chip->timer_B_time[i] = pom;
|
|
#else
|
|
//chip->tim_B_tab[i] = attotime_to_double(pom) * (double)chip->sampfreq * mult; /* number of samples that timer period takes (fixed point) */
|
|
chip->tim_B_tab[i] = pom * (double)chip->sampfreq * mult; /* number of samples that timer period takes (fixed point) */
|
|
#endif
|
|
}
|
|
|
|
/* calculate noise periods table */
|
|
scaler = ( (double)chip->clock / 64.0 ) / ( (double)chip->sampfreq );
|
|
for (i=0; i<32; i++)
|
|
{
|
|
j = (i!=31 ? i : 30); /* rate 30 and 31 are the same */
|
|
j = 32-j;
|
|
j = (65536.0 / (double)(j*32.0)); /* number of samples per one shift of the shift register */
|
|
/*chip->noise_tab[i] = j * 64;*/ /* number of chip clock cycles per one shift */
|
|
chip->noise_tab[i] = j * 64 * scaler;
|
|
/*logerror("noise_tab[%02x]=%08x\n", i, chip->noise_tab[i]);*/
|
|
}
|
|
}
|
|
|
|
#define KEY_ON(op, key_set){ \
|
|
if (!(op)->key) \
|
|
{ \
|
|
(op)->phase = 0; /* clear phase */ \
|
|
(op)->state = EG_ATT; /* KEY ON = attack */ \
|
|
(op)->volume += (~(op)->volume * \
|
|
(eg_inc[(op)->eg_sel_ar + ((PSG->eg_cnt>>(op)->eg_sh_ar)&7)]) \
|
|
) >>4; \
|
|
if ((op)->volume <= MIN_ATT_INDEX) \
|
|
{ \
|
|
(op)->volume = MIN_ATT_INDEX; \
|
|
(op)->state = EG_DEC; \
|
|
} \
|
|
} \
|
|
(op)->key |= key_set; \
|
|
}
|
|
|
|
#define KEY_OFF(op, key_clr){ \
|
|
if ((op)->key) \
|
|
{ \
|
|
(op)->key &= key_clr; \
|
|
if (!(op)->key) \
|
|
{ \
|
|
if ((op)->state>EG_REL) \
|
|
(op)->state = EG_REL;/* KEY OFF = release */\
|
|
} \
|
|
} \
|
|
}
|
|
|
|
INLINE void envelope_KONKOFF(YM2151Operator * op, int v)
|
|
{
|
|
if (v&0x08) /* M1 */
|
|
KEY_ON (op+0, 1)
|
|
else
|
|
KEY_OFF(op+0,~1)
|
|
|
|
if (v&0x20) /* M2 */
|
|
KEY_ON (op+1, 1)
|
|
else
|
|
KEY_OFF(op+1,~1)
|
|
|
|
if (v&0x10) /* C1 */
|
|
KEY_ON (op+2, 1)
|
|
else
|
|
KEY_OFF(op+2,~1)
|
|
|
|
if (v&0x40) /* C2 */
|
|
KEY_ON (op+3, 1)
|
|
else
|
|
KEY_OFF(op+3,~1)
|
|
}
|
|
|
|
|
|
#ifdef USE_MAME_TIMERS
|
|
|
|
/*static TIMER_CALLBACK( irqAon_callback )
|
|
{
|
|
YM2151 *chip = (YM2151 *)ptr;
|
|
int oldstate = chip->irqlinestate;
|
|
|
|
chip->irqlinestate |= 1;
|
|
|
|
if (oldstate == 0 && chip->irqhandler) (*chip->irqhandler)(chip->device, 1);
|
|
}
|
|
|
|
static TIMER_CALLBACK( irqBon_callback )
|
|
{
|
|
YM2151 *chip = (YM2151 *)ptr;
|
|
int oldstate = chip->irqlinestate;
|
|
|
|
chip->irqlinestate |= 2;
|
|
|
|
if (oldstate == 0 && chip->irqhandler) (*chip->irqhandler)(chip->device, 1);
|
|
}
|
|
|
|
static TIMER_CALLBACK( irqAoff_callback )
|
|
{
|
|
YM2151 *chip = (YM2151 *)ptr;
|
|
int oldstate = chip->irqlinestate;
|
|
|
|
chip->irqlinestate &= ~1;
|
|
|
|
if (oldstate == 1 && chip->irqhandler) (*chip->irqhandler)(chip->device, 0);
|
|
}
|
|
|
|
static TIMER_CALLBACK( irqBoff_callback )
|
|
{
|
|
YM2151 *chip = (YM2151 *)ptr;
|
|
int oldstate = chip->irqlinestate;
|
|
|
|
chip->irqlinestate &= ~2;
|
|
|
|
if (oldstate == 2 && chip->irqhandler) (*chip->irqhandler)(chip->device, 0);
|
|
}
|
|
|
|
static TIMER_CALLBACK( timer_callback_a )
|
|
{
|
|
YM2151 *chip = (YM2151 *)ptr;
|
|
timer_adjust_oneshot(chip->timer_A, chip->timer_A_time[ chip->timer_A_index ], 0);
|
|
chip->timer_A_index_old = chip->timer_A_index;
|
|
if (chip->irq_enable & 0x04)
|
|
{
|
|
chip->status |= 1;
|
|
timer_set(machine, attotime_zero,chip,0,irqAon_callback);
|
|
}
|
|
if (chip->irq_enable & 0x80)
|
|
chip->csm_req = 2; // request KEY ON / KEY OFF sequence
|
|
}
|
|
static TIMER_CALLBACK( timer_callback_b )
|
|
{
|
|
YM2151 *chip = (YM2151 *)ptr;
|
|
timer_adjust_oneshot(chip->timer_B, chip->timer_B_time[ chip->timer_B_index ], 0);
|
|
chip->timer_B_index_old = chip->timer_B_index;
|
|
if (chip->irq_enable & 0x08)
|
|
{
|
|
chip->status |= 2;
|
|
timer_set(machine, attotime_zero,chip,0,irqBon_callback);
|
|
}
|
|
}*/
|
|
#if 0
|
|
static TIMER_CALLBACK( timer_callback_chip_busy )
|
|
{
|
|
YM2151 *chip = (YM2151 *)ptr;
|
|
chip->status &= 0x7f; /* reset busy flag */
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
|
INLINE void set_connect( YM2151Operator *om1, int cha, int v)
|
|
{
|
|
YM2151Operator *om2 = om1+1;
|
|
YM2151Operator *oc1 = om1+2;
|
|
|
|
/* set connect algorithm */
|
|
|
|
/* MEM is simply one sample delay */
|
|
|
|
switch( v&7 )
|
|
{
|
|
case 0:
|
|
/* M1---C1---MEM---M2---C2---OUT */
|
|
om1->connect = &c1;
|
|
oc1->connect = &mem;
|
|
om2->connect = &c2;
|
|
om1->mem_connect = &m2;
|
|
break;
|
|
|
|
case 1:
|
|
/* M1------+-MEM---M2---C2---OUT */
|
|
/* C1-+ */
|
|
om1->connect = &mem;
|
|
oc1->connect = &mem;
|
|
om2->connect = &c2;
|
|
om1->mem_connect = &m2;
|
|
break;
|
|
|
|
case 2:
|
|
/* M1-----------------+-C2---OUT */
|
|
/* C1---MEM---M2-+ */
|
|
om1->connect = &c2;
|
|
oc1->connect = &mem;
|
|
om2->connect = &c2;
|
|
om1->mem_connect = &m2;
|
|
break;
|
|
|
|
case 3:
|
|
/* M1---C1---MEM------+-C2---OUT */
|
|
/* M2-+ */
|
|
om1->connect = &c1;
|
|
oc1->connect = &mem;
|
|
om2->connect = &c2;
|
|
om1->mem_connect = &c2;
|
|
break;
|
|
|
|
case 4:
|
|
/* M1---C1-+-OUT */
|
|
/* M2---C2-+ */
|
|
/* MEM: not used */
|
|
om1->connect = &c1;
|
|
oc1->connect = &chanout[cha];
|
|
om2->connect = &c2;
|
|
om1->mem_connect = &mem; /* store it anywhere where it will not be used */
|
|
break;
|
|
|
|
case 5:
|
|
/* +----C1----+ */
|
|
/* M1-+-MEM---M2-+-OUT */
|
|
/* +----C2----+ */
|
|
om1->connect = 0; /* special mark */
|
|
oc1->connect = &chanout[cha];
|
|
om2->connect = &chanout[cha];
|
|
om1->mem_connect = &m2;
|
|
break;
|
|
|
|
case 6:
|
|
/* M1---C1-+ */
|
|
/* M2-+-OUT */
|
|
/* C2-+ */
|
|
/* MEM: not used */
|
|
om1->connect = &c1;
|
|
oc1->connect = &chanout[cha];
|
|
om2->connect = &chanout[cha];
|
|
om1->mem_connect = &mem; /* store it anywhere where it will not be used */
|
|
break;
|
|
|
|
case 7:
|
|
/* M1-+ */
|
|
/* C1-+-OUT */
|
|
/* M2-+ */
|
|
/* C2-+ */
|
|
/* MEM: not used*/
|
|
om1->connect = &chanout[cha];
|
|
oc1->connect = &chanout[cha];
|
|
om2->connect = &chanout[cha];
|
|
om1->mem_connect = &mem; /* store it anywhere where it will not be used */
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
INLINE void refresh_EG(YM2151Operator * op)
|
|
{
|
|
UINT32 kc;
|
|
UINT32 v;
|
|
|
|
kc = op->kc;
|
|
|
|
/* v = 32 + 2*RATE + RKS = max 126 */
|
|
|
|
v = kc >> op->ks;
|
|
if ((op->ar+v) < 32+62)
|
|
{
|
|
op->eg_sh_ar = eg_rate_shift [op->ar + v ];
|
|
op->eg_sel_ar = eg_rate_select[op->ar + v ];
|
|
}
|
|
else
|
|
{
|
|
op->eg_sh_ar = 0;
|
|
op->eg_sel_ar = 17*RATE_STEPS;
|
|
}
|
|
op->eg_sh_d1r = eg_rate_shift [op->d1r + v];
|
|
op->eg_sel_d1r= eg_rate_select[op->d1r + v];
|
|
op->eg_sh_d2r = eg_rate_shift [op->d2r + v];
|
|
op->eg_sel_d2r= eg_rate_select[op->d2r + v];
|
|
op->eg_sh_rr = eg_rate_shift [op->rr + v];
|
|
op->eg_sel_rr = eg_rate_select[op->rr + v];
|
|
|
|
|
|
op+=1;
|
|
|
|
v = kc >> op->ks;
|
|
if ((op->ar+v) < 32+62)
|
|
{
|
|
op->eg_sh_ar = eg_rate_shift [op->ar + v ];
|
|
op->eg_sel_ar = eg_rate_select[op->ar + v ];
|
|
}
|
|
else
|
|
{
|
|
op->eg_sh_ar = 0;
|
|
op->eg_sel_ar = 17*RATE_STEPS;
|
|
}
|
|
op->eg_sh_d1r = eg_rate_shift [op->d1r + v];
|
|
op->eg_sel_d1r= eg_rate_select[op->d1r + v];
|
|
op->eg_sh_d2r = eg_rate_shift [op->d2r + v];
|
|
op->eg_sel_d2r= eg_rate_select[op->d2r + v];
|
|
op->eg_sh_rr = eg_rate_shift [op->rr + v];
|
|
op->eg_sel_rr = eg_rate_select[op->rr + v];
|
|
|
|
op+=1;
|
|
|
|
v = kc >> op->ks;
|
|
if ((op->ar+v) < 32+62)
|
|
{
|
|
op->eg_sh_ar = eg_rate_shift [op->ar + v ];
|
|
op->eg_sel_ar = eg_rate_select[op->ar + v ];
|
|
}
|
|
else
|
|
{
|
|
op->eg_sh_ar = 0;
|
|
op->eg_sel_ar = 17*RATE_STEPS;
|
|
}
|
|
op->eg_sh_d1r = eg_rate_shift [op->d1r + v];
|
|
op->eg_sel_d1r= eg_rate_select[op->d1r + v];
|
|
op->eg_sh_d2r = eg_rate_shift [op->d2r + v];
|
|
op->eg_sel_d2r= eg_rate_select[op->d2r + v];
|
|
op->eg_sh_rr = eg_rate_shift [op->rr + v];
|
|
op->eg_sel_rr = eg_rate_select[op->rr + v];
|
|
|
|
op+=1;
|
|
|
|
v = kc >> op->ks;
|
|
if ((op->ar+v) < 32+62)
|
|
{
|
|
op->eg_sh_ar = eg_rate_shift [op->ar + v ];
|
|
op->eg_sel_ar = eg_rate_select[op->ar + v ];
|
|
}
|
|
else
|
|
{
|
|
op->eg_sh_ar = 0;
|
|
op->eg_sel_ar = 17*RATE_STEPS;
|
|
}
|
|
op->eg_sh_d1r = eg_rate_shift [op->d1r + v];
|
|
op->eg_sel_d1r= eg_rate_select[op->d1r + v];
|
|
op->eg_sh_d2r = eg_rate_shift [op->d2r + v];
|
|
op->eg_sel_d2r= eg_rate_select[op->d2r + v];
|
|
op->eg_sh_rr = eg_rate_shift [op->rr + v];
|
|
op->eg_sel_rr = eg_rate_select[op->rr + v];
|
|
}
|
|
|
|
|
|
/* write a register on YM2151 chip number 'n' */
|
|
void ym2151_write_reg(void *_chip, int r, int v)
|
|
{
|
|
YM2151 *chip = (YM2151 *)_chip;
|
|
YM2151Operator *op = &chip->oper[ (r&0x07)*4+((r&0x18)>>3) ];
|
|
|
|
/* adjust bus to 8 bits */
|
|
r &= 0xff;
|
|
v &= 0xff;
|
|
|
|
#if 0
|
|
/* There is no info on what YM2151 really does when busy flag is set */
|
|
if ( chip->status & 0x80 ) return;
|
|
timer_set ( attotime_mul(ATTOTIME_IN_HZ(chip->clock), 64), chip, 0, timer_callback_chip_busy);
|
|
chip->status |= 0x80; /* set busy flag for 64 chip clock cycles */
|
|
#endif
|
|
|
|
/*if (LOG_CYM_FILE && (cymfile) && (r!=0) )
|
|
{
|
|
fputc( (unsigned char)r, cymfile );
|
|
fputc( (unsigned char)v, cymfile );
|
|
}*/
|
|
|
|
|
|
switch(r & 0xe0){
|
|
case 0x00:
|
|
switch(r){
|
|
case 0x01: /* LFO reset(bit 1), Test Register (other bits) */
|
|
chip->test = v;
|
|
if (v&2) chip->lfo_phase = 0;
|
|
break;
|
|
|
|
case 0x08:
|
|
PSG = chip; /* PSG is used in KEY_ON macro */
|
|
envelope_KONKOFF(&chip->oper[ (v&7)*4 ], v );
|
|
break;
|
|
|
|
case 0x0f: /* noise mode enable, noise period */
|
|
chip->noise = v;
|
|
chip->noise_f = chip->noise_tab[ v & 0x1f ];
|
|
break;
|
|
|
|
case 0x10: /* timer A hi */
|
|
chip->timer_A_index = (chip->timer_A_index & 0x003) | (v<<2);
|
|
break;
|
|
|
|
case 0x11: /* timer A low */
|
|
chip->timer_A_index = (chip->timer_A_index & 0x3fc) | (v & 3);
|
|
break;
|
|
|
|
case 0x12: /* timer B */
|
|
chip->timer_B_index = v;
|
|
break;
|
|
|
|
case 0x14: /* CSM, irq flag reset, irq enable, timer start/stop */
|
|
|
|
chip->irq_enable = v; /* bit 3-timer B, bit 2-timer A, bit 7 - CSM */
|
|
|
|
if (v&0x10) /* reset timer A irq flag */
|
|
{
|
|
#ifdef USE_MAME_TIMERS
|
|
chip->status &= ~1;
|
|
timer_set(chip->device->machine, attotime_zero,chip,0,irqAoff_callback);
|
|
#else
|
|
int oldstate = chip->status & 3;
|
|
chip->status &= ~1;
|
|
//if ((oldstate==1) && (chip->irqhandler)) (*chip->irqhandler)(chip->device, 0);
|
|
#endif
|
|
}
|
|
|
|
if (v&0x20) /* reset timer B irq flag */
|
|
{
|
|
#ifdef USE_MAME_TIMERS
|
|
chip->status &= ~2;
|
|
timer_set(chip->device->machine, attotime_zero,chip,0,irqBoff_callback);
|
|
#else
|
|
int oldstate = chip->status & 3;
|
|
chip->status &= ~2;
|
|
//if ((oldstate==2) && (chip->irqhandler)) (*chip->irqhandler)(chip->device, 0);
|
|
#endif
|
|
}
|
|
|
|
if (v&0x02){ /* load and start timer B */
|
|
#ifdef USE_MAME_TIMERS
|
|
/* ASG 980324: added a real timer */
|
|
/* start timer _only_ if it wasn't already started (it will reload time value next round) */
|
|
if (!timer_enable(chip->timer_B, 1))
|
|
{
|
|
timer_adjust_oneshot(chip->timer_B, chip->timer_B_time[ chip->timer_B_index ], 0);
|
|
chip->timer_B_index_old = chip->timer_B_index;
|
|
}
|
|
#else
|
|
if (!chip->tim_B)
|
|
{
|
|
chip->tim_B = 1;
|
|
chip->tim_B_val = chip->tim_B_tab[ chip->timer_B_index ];
|
|
}
|
|
#endif
|
|
}else{ /* stop timer B */
|
|
#ifdef USE_MAME_TIMERS
|
|
/* ASG 980324: added a real timer */
|
|
timer_enable(chip->timer_B, 0);
|
|
#else
|
|
chip->tim_B = 0;
|
|
#endif
|
|
}
|
|
|
|
if (v&0x01){ /* load and start timer A */
|
|
#ifdef USE_MAME_TIMERS
|
|
/* ASG 980324: added a real timer */
|
|
/* start timer _only_ if it wasn't already started (it will reload time value next round) */
|
|
if (!timer_enable(chip->timer_A, 1))
|
|
{
|
|
timer_adjust_oneshot(chip->timer_A, chip->timer_A_time[ chip->timer_A_index ], 0);
|
|
chip->timer_A_index_old = chip->timer_A_index;
|
|
}
|
|
#else
|
|
if (!chip->tim_A)
|
|
{
|
|
chip->tim_A = 1;
|
|
chip->tim_A_val = chip->tim_A_tab[ chip->timer_A_index ];
|
|
}
|
|
#endif
|
|
}else{ /* stop timer A */
|
|
#ifdef USE_MAME_TIMERS
|
|
/* ASG 980324: added a real timer */
|
|
timer_enable(chip->timer_A, 0);
|
|
#else
|
|
chip->tim_A = 0;
|
|
#endif
|
|
}
|
|
break;
|
|
|
|
case 0x18: /* LFO frequency */
|
|
{
|
|
chip->lfo_overflow = ( 1 << ((15-(v>>4))+3) ) * (1<<LFO_SH);
|
|
chip->lfo_counter_add = 0x10 + (v & 0x0f);
|
|
}
|
|
break;
|
|
|
|
case 0x19: /* PMD (bit 7==1) or AMD (bit 7==0) */
|
|
if (v&0x80)
|
|
chip->pmd = v & 0x7f;
|
|
else
|
|
chip->amd = v & 0x7f;
|
|
break;
|
|
|
|
case 0x1b: /* CT2, CT1, LFO waveform */
|
|
chip->ct = v >> 6;
|
|
chip->lfo_wsel = v & 3;
|
|
//if (chip->porthandler) (*chip->porthandler)(chip->device, 0 , chip->ct );
|
|
break;
|
|
|
|
default:
|
|
#ifdef _DEBUG
|
|
//logerror("YM2151 Write %02x to undocumented register #%02x\n",v,r);
|
|
#endif
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case 0x20:
|
|
op = &chip->oper[ (r&7) * 4 ];
|
|
switch(r & 0x18){
|
|
case 0x00: /* RL enable, Feedback, Connection */
|
|
op->fb_shift = ((v>>3)&7) ? ((v>>3)&7)+6:0;
|
|
chip->pan[ (r&7)*2 ] = (v & 0x40) ? ~0 : 0;
|
|
chip->pan[ (r&7)*2 +1 ] = (v & 0x80) ? ~0 : 0;
|
|
chip->connect[r&7] = v&7;
|
|
set_connect(op, r&7, v&7);
|
|
break;
|
|
|
|
case 0x08: /* Key Code */
|
|
v &= 0x7f;
|
|
if (v != op->kc)
|
|
{
|
|
UINT32 kc, kc_channel;
|
|
|
|
kc_channel = (v - (v>>2))*64;
|
|
kc_channel += 768;
|
|
kc_channel |= (op->kc_i & 63);
|
|
|
|
(op+0)->kc = v;
|
|
(op+0)->kc_i = kc_channel;
|
|
(op+1)->kc = v;
|
|
(op+1)->kc_i = kc_channel;
|
|
(op+2)->kc = v;
|
|
(op+2)->kc_i = kc_channel;
|
|
(op+3)->kc = v;
|
|
(op+3)->kc_i = kc_channel;
|
|
|
|
kc = v>>2;
|
|
|
|
(op+0)->dt1 = chip->dt1_freq[ (op+0)->dt1_i + kc ];
|
|
(op+0)->freq = ( (chip->freq[ kc_channel + (op+0)->dt2 ] + (op+0)->dt1) * (op+0)->mul ) >> 1;
|
|
|
|
(op+1)->dt1 = chip->dt1_freq[ (op+1)->dt1_i + kc ];
|
|
(op+1)->freq = ( (chip->freq[ kc_channel + (op+1)->dt2 ] + (op+1)->dt1) * (op+1)->mul ) >> 1;
|
|
|
|
(op+2)->dt1 = chip->dt1_freq[ (op+2)->dt1_i + kc ];
|
|
(op+2)->freq = ( (chip->freq[ kc_channel + (op+2)->dt2 ] + (op+2)->dt1) * (op+2)->mul ) >> 1;
|
|
|
|
(op+3)->dt1 = chip->dt1_freq[ (op+3)->dt1_i + kc ];
|
|
(op+3)->freq = ( (chip->freq[ kc_channel + (op+3)->dt2 ] + (op+3)->dt1) * (op+3)->mul ) >> 1;
|
|
|
|
refresh_EG( op );
|
|
}
|
|
break;
|
|
|
|
case 0x10: /* Key Fraction */
|
|
v >>= 2;
|
|
if (v != (op->kc_i & 63))
|
|
{
|
|
UINT32 kc_channel;
|
|
|
|
kc_channel = v;
|
|
kc_channel |= (op->kc_i & ~63);
|
|
|
|
(op+0)->kc_i = kc_channel;
|
|
(op+1)->kc_i = kc_channel;
|
|
(op+2)->kc_i = kc_channel;
|
|
(op+3)->kc_i = kc_channel;
|
|
|
|
(op+0)->freq = ( (chip->freq[ kc_channel + (op+0)->dt2 ] + (op+0)->dt1) * (op+0)->mul ) >> 1;
|
|
(op+1)->freq = ( (chip->freq[ kc_channel + (op+1)->dt2 ] + (op+1)->dt1) * (op+1)->mul ) >> 1;
|
|
(op+2)->freq = ( (chip->freq[ kc_channel + (op+2)->dt2 ] + (op+2)->dt1) * (op+2)->mul ) >> 1;
|
|
(op+3)->freq = ( (chip->freq[ kc_channel + (op+3)->dt2 ] + (op+3)->dt1) * (op+3)->mul ) >> 1;
|
|
}
|
|
break;
|
|
|
|
case 0x18: /* PMS, AMS */
|
|
op->pms = (v>>4) & 7;
|
|
op->ams = (v & 3);
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case 0x40: /* DT1, MUL */
|
|
{
|
|
UINT32 olddt1_i = op->dt1_i;
|
|
UINT32 oldmul = op->mul;
|
|
|
|
op->dt1_i = (v&0x70)<<1;
|
|
op->mul = (v&0x0f) ? (v&0x0f)<<1: 1;
|
|
|
|
if (olddt1_i != op->dt1_i)
|
|
op->dt1 = chip->dt1_freq[ op->dt1_i + (op->kc>>2) ];
|
|
|
|
if ( (olddt1_i != op->dt1_i) || (oldmul != op->mul) )
|
|
op->freq = ( (chip->freq[ op->kc_i + op->dt2 ] + op->dt1) * op->mul ) >> 1;
|
|
}
|
|
break;
|
|
|
|
case 0x60: /* TL */
|
|
op->tl = (v&0x7f)<<(ENV_BITS-7); /* 7bit TL */
|
|
break;
|
|
|
|
case 0x80: /* KS, AR */
|
|
{
|
|
UINT32 oldks = op->ks;
|
|
UINT32 oldar = op->ar;
|
|
|
|
op->ks = 5-(v>>6);
|
|
op->ar = (v&0x1f) ? 32 + ((v&0x1f)<<1) : 0;
|
|
|
|
if ( (op->ar != oldar) || (op->ks != oldks) )
|
|
{
|
|
if ((op->ar + (op->kc>>op->ks)) < 32+62)
|
|
{
|
|
op->eg_sh_ar = eg_rate_shift [op->ar + (op->kc>>op->ks) ];
|
|
op->eg_sel_ar = eg_rate_select[op->ar + (op->kc>>op->ks) ];
|
|
}
|
|
else
|
|
{
|
|
op->eg_sh_ar = 0;
|
|
op->eg_sel_ar = 17*RATE_STEPS;
|
|
}
|
|
}
|
|
|
|
if (op->ks != oldks)
|
|
{
|
|
op->eg_sh_d1r = eg_rate_shift [op->d1r + (op->kc>>op->ks) ];
|
|
op->eg_sel_d1r= eg_rate_select[op->d1r + (op->kc>>op->ks) ];
|
|
op->eg_sh_d2r = eg_rate_shift [op->d2r + (op->kc>>op->ks) ];
|
|
op->eg_sel_d2r= eg_rate_select[op->d2r + (op->kc>>op->ks) ];
|
|
op->eg_sh_rr = eg_rate_shift [op->rr + (op->kc>>op->ks) ];
|
|
op->eg_sel_rr = eg_rate_select[op->rr + (op->kc>>op->ks) ];
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 0xa0: /* LFO AM enable, D1R */
|
|
op->AMmask = (v&0x80) ? ~0 : 0;
|
|
op->d1r = (v&0x1f) ? 32 + ((v&0x1f)<<1) : 0;
|
|
op->eg_sh_d1r = eg_rate_shift [op->d1r + (op->kc>>op->ks) ];
|
|
op->eg_sel_d1r= eg_rate_select[op->d1r + (op->kc>>op->ks) ];
|
|
break;
|
|
|
|
case 0xc0: /* DT2, D2R */
|
|
{
|
|
UINT32 olddt2 = op->dt2;
|
|
op->dt2 = dt2_tab[ v>>6 ];
|
|
if (op->dt2 != olddt2)
|
|
op->freq = ( (chip->freq[ op->kc_i + op->dt2 ] + op->dt1) * op->mul ) >> 1;
|
|
}
|
|
op->d2r = (v&0x1f) ? 32 + ((v&0x1f)<<1) : 0;
|
|
op->eg_sh_d2r = eg_rate_shift [op->d2r + (op->kc>>op->ks) ];
|
|
op->eg_sel_d2r= eg_rate_select[op->d2r + (op->kc>>op->ks) ];
|
|
break;
|
|
|
|
case 0xe0: /* D1L, RR */
|
|
op->d1l = d1l_tab[ v>>4 ];
|
|
op->rr = 34 + ((v&0x0f)<<2);
|
|
op->eg_sh_rr = eg_rate_shift [op->rr + (op->kc>>op->ks) ];
|
|
op->eg_sel_rr = eg_rate_select[op->rr + (op->kc>>op->ks) ];
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
/*static TIMER_CALLBACK( cymfile_callback )
|
|
{
|
|
if (cymfile)
|
|
fputc( (unsigned char)0, cymfile );
|
|
}*/
|
|
|
|
|
|
int ym2151_read_status( void *_chip )
|
|
{
|
|
YM2151 *chip = (YM2151 *)_chip;
|
|
return chip->status;
|
|
}
|
|
|
|
|
|
|
|
//#ifdef USE_MAME_TIMERS
|
|
#if 1 // disabled for now due to crashing with winalloc.c (ERROR_NOT_ENOUGH_MEMORY)
|
|
/*
|
|
* state save support for MAME
|
|
*/
|
|
//STATE_POSTLOAD( ym2151_postload )
|
|
/*void ym2151_postload(void *param)
|
|
{
|
|
YM2151 *YM2151_chip = (YM2151 *)param;
|
|
int j;
|
|
|
|
for (j=0; j<8; j++)
|
|
set_connect(&YM2151_chip->oper[j*4], j, YM2151_chip->connect[j]);
|
|
}
|
|
|
|
static void ym2151_state_save_register( YM2151 *chip, const device_config *device )
|
|
{
|
|
int j;
|
|
|
|
// save all 32 operators of chip #i
|
|
for (j=0; j<32; j++)
|
|
{
|
|
YM2151Operator *op;
|
|
|
|
op = &chip->oper[(j&7)*4+(j>>3)];
|
|
|
|
state_save_register_device_item(device, j, op->phase);
|
|
state_save_register_device_item(device, j, op->freq);
|
|
state_save_register_device_item(device, j, op->dt1);
|
|
state_save_register_device_item(device, j, op->mul);
|
|
state_save_register_device_item(device, j, op->dt1_i);
|
|
state_save_register_device_item(device, j, op->dt2);
|
|
// operators connection is saved in chip data block
|
|
state_save_register_device_item(device, j, op->mem_value);
|
|
|
|
state_save_register_device_item(device, j, op->fb_shift);
|
|
state_save_register_device_item(device, j, op->fb_out_curr);
|
|
state_save_register_device_item(device, j, op->fb_out_prev);
|
|
state_save_register_device_item(device, j, op->kc);
|
|
state_save_register_device_item(device, j, op->kc_i);
|
|
state_save_register_device_item(device, j, op->pms);
|
|
state_save_register_device_item(device, j, op->ams);
|
|
state_save_register_device_item(device, j, op->AMmask);
|
|
|
|
state_save_register_device_item(device, j, op->state);
|
|
state_save_register_device_item(device, j, op->eg_sh_ar);
|
|
state_save_register_device_item(device, j, op->eg_sel_ar);
|
|
state_save_register_device_item(device, j, op->tl);
|
|
state_save_register_device_item(device, j, op->volume);
|
|
state_save_register_device_item(device, j, op->eg_sh_d1r);
|
|
state_save_register_device_item(device, j, op->eg_sel_d1r);
|
|
state_save_register_device_item(device, j, op->d1l);
|
|
state_save_register_device_item(device, j, op->eg_sh_d2r);
|
|
state_save_register_device_item(device, j, op->eg_sel_d2r);
|
|
state_save_register_device_item(device, j, op->eg_sh_rr);
|
|
state_save_register_device_item(device, j, op->eg_sel_rr);
|
|
|
|
state_save_register_device_item(device, j, op->key);
|
|
state_save_register_device_item(device, j, op->ks);
|
|
state_save_register_device_item(device, j, op->ar);
|
|
state_save_register_device_item(device, j, op->d1r);
|
|
state_save_register_device_item(device, j, op->d2r);
|
|
state_save_register_device_item(device, j, op->rr);
|
|
|
|
state_save_register_device_item(device, j, op->reserved0);
|
|
state_save_register_device_item(device, j, op->reserved1);
|
|
}
|
|
|
|
state_save_register_device_item_array(device, 0, chip->pan);
|
|
|
|
state_save_register_device_item(device, 0, chip->eg_cnt);
|
|
state_save_register_device_item(device, 0, chip->eg_timer);
|
|
state_save_register_device_item(device, 0, chip->eg_timer_add);
|
|
state_save_register_device_item(device, 0, chip->eg_timer_overflow);
|
|
|
|
state_save_register_device_item(device, 0, chip->lfo_phase);
|
|
state_save_register_device_item(device, 0, chip->lfo_timer);
|
|
state_save_register_device_item(device, 0, chip->lfo_timer_add);
|
|
state_save_register_device_item(device, 0, chip->lfo_overflow);
|
|
state_save_register_device_item(device, 0, chip->lfo_counter);
|
|
state_save_register_device_item(device, 0, chip->lfo_counter_add);
|
|
state_save_register_device_item(device, 0, chip->lfo_wsel);
|
|
state_save_register_device_item(device, 0, chip->amd);
|
|
state_save_register_device_item(device, 0, chip->pmd);
|
|
state_save_register_device_item(device, 0, chip->lfa);
|
|
state_save_register_device_item(device, 0, chip->lfp);
|
|
|
|
state_save_register_device_item(device, 0, chip->test);
|
|
state_save_register_device_item(device, 0, chip->ct);
|
|
|
|
state_save_register_device_item(device, 0, chip->noise);
|
|
state_save_register_device_item(device, 0, chip->noise_rng);
|
|
state_save_register_device_item(device, 0, chip->noise_p);
|
|
state_save_register_device_item(device, 0, chip->noise_f);
|
|
|
|
state_save_register_device_item(device, 0, chip->csm_req);
|
|
state_save_register_device_item(device, 0, chip->irq_enable);
|
|
state_save_register_device_item(device, 0, chip->status);
|
|
|
|
state_save_register_device_item(device, 0, chip->timer_A_index);
|
|
state_save_register_device_item(device, 0, chip->timer_B_index);
|
|
state_save_register_device_item(device, 0, chip->timer_A_index_old);
|
|
state_save_register_device_item(device, 0, chip->timer_B_index_old);
|
|
|
|
#ifdef USE_MAME_TIMERS
|
|
state_save_register_device_item(device, 0, chip->irqlinestate);
|
|
#endif
|
|
|
|
state_save_register_device_item_array(device, 0, chip->connect);
|
|
|
|
state_save_register_postload(device->machine, ym2151_postload, chip);
|
|
}*/
|
|
#else
|
|
/*STATE_POSTLOAD( ym2151_postload )
|
|
{
|
|
}
|
|
|
|
static void ym2151_state_save_register( YM2151 *chip, const device_config *device )
|
|
{
|
|
}*/
|
|
#endif
|
|
|
|
|
|
/*
|
|
* Initialize YM2151 emulator(s).
|
|
*
|
|
* 'num' is the number of virtual YM2151's to allocate
|
|
* 'clock' is the chip clock in Hz
|
|
* 'rate' is sampling rate
|
|
*/
|
|
void * ym2151_init(int clock, int rate)
|
|
{
|
|
YM2151 *PSG;
|
|
int chn;
|
|
|
|
PSG = (YM2151 *)malloc(sizeof(YM2151));
|
|
if (PSG == NULL)
|
|
return NULL;
|
|
|
|
memset(PSG, 0, sizeof(YM2151));
|
|
|
|
//ym2151_state_save_register( PSG, device );
|
|
|
|
init_tables();
|
|
|
|
//PSG->device = device;
|
|
PSG->clock = clock;
|
|
/*rate = clock/64;*/
|
|
PSG->sampfreq = rate ? rate : 44100; /* avoid division by 0 in init_chip_tables() */
|
|
//PSG->irqhandler = NULL; /* interrupt handler */
|
|
//PSG->porthandler = NULL; /* port write handler */
|
|
init_chip_tables( PSG );
|
|
|
|
PSG->lfo_timer_add = (1<<LFO_SH) * (clock/64.0) / PSG->sampfreq;
|
|
|
|
PSG->eg_timer_add = (1<<EG_SH) * (clock/64.0) / PSG->sampfreq;
|
|
PSG->eg_timer_overflow = ( 3 ) * (1<<EG_SH);
|
|
/*logerror("YM2151[init] eg_timer_add=%8x eg_timer_overflow=%8x\n", PSG->eg_timer_add, PSG->eg_timer_overflow);*/
|
|
|
|
#ifdef USE_MAME_TIMERS
|
|
/* this must be done _before_ a call to ym2151_reset_chip() */
|
|
PSG->timer_A = timer_alloc(device->machine, timer_callback_a, PSG);
|
|
PSG->timer_B = timer_alloc(device->machine, timer_callback_b, PSG);
|
|
#else
|
|
PSG->tim_A = 0;
|
|
PSG->tim_B = 0;
|
|
#endif
|
|
for (chn = 0; chn < 8; chn ++)
|
|
PSG->Muted[chn] = 0x00;
|
|
//ym2151_reset_chip(PSG);
|
|
/*logerror("YM2151[init] clock=%i sampfreq=%i\n", PSG->clock, PSG->sampfreq);*/
|
|
|
|
/*LOG_CYM_FILE = (options_get_int(mame_options(), OPTION_VGMWRITE) > 0x00);
|
|
if (LOG_CYM_FILE)
|
|
{
|
|
cymfile = fopen("2151_.cym","wb");
|
|
if (cymfile)
|
|
timer_pulse ( device->machine, ATTOTIME_IN_HZ(60), NULL, 0, cymfile_callback); //110 Hz pulse timer
|
|
else
|
|
logerror("Could not create file 2151_.cym\n");
|
|
}*/
|
|
|
|
return PSG;
|
|
}
|
|
|
|
|
|
|
|
void ym2151_shutdown(void *_chip)
|
|
{
|
|
YM2151 *chip = (YM2151 *)_chip;
|
|
|
|
free (chip);
|
|
|
|
/*if (cymfile)
|
|
fclose (cymfile);
|
|
cymfile = NULL;*/
|
|
|
|
#ifdef SAVE_SAMPLE
|
|
fclose(sample[8]);
|
|
#endif
|
|
#ifdef SAVE_SEPARATE_CHANNELS
|
|
fclose(sample[0]);
|
|
fclose(sample[1]);
|
|
fclose(sample[2]);
|
|
fclose(sample[3]);
|
|
fclose(sample[4]);
|
|
fclose(sample[5]);
|
|
fclose(sample[6]);
|
|
fclose(sample[7]);
|
|
#endif
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Reset chip number 'n'.
|
|
*/
|
|
void ym2151_reset_chip(void *_chip)
|
|
{
|
|
int i;
|
|
YM2151 *chip = (YM2151 *)_chip;
|
|
|
|
|
|
/* initialize hardware registers */
|
|
for (i=0; i<32; i++)
|
|
{
|
|
memset(&chip->oper[i],'\0',sizeof(YM2151Operator));
|
|
chip->oper[i].volume = MAX_ATT_INDEX;
|
|
chip->oper[i].kc_i = 768; /* min kc_i value */
|
|
}
|
|
|
|
chip->eg_timer = 0;
|
|
chip->eg_cnt = 0;
|
|
|
|
chip->lfo_timer = 0;
|
|
chip->lfo_counter= 0;
|
|
chip->lfo_phase = 0;
|
|
chip->lfo_wsel = 0;
|
|
chip->pmd = 0;
|
|
chip->amd = 0;
|
|
chip->lfa = 0;
|
|
chip->lfp = 0;
|
|
|
|
chip->test= 0;
|
|
|
|
chip->irq_enable = 0;
|
|
#ifdef USE_MAME_TIMERS
|
|
/* ASG 980324 -- reset the timers before writing to the registers */
|
|
timer_enable(chip->timer_A, 0);
|
|
timer_enable(chip->timer_B, 0);
|
|
#else
|
|
chip->tim_A = 0;
|
|
chip->tim_B = 0;
|
|
chip->tim_A_val = 0;
|
|
chip->tim_B_val = 0;
|
|
#endif
|
|
chip->timer_A_index = 0;
|
|
chip->timer_B_index = 0;
|
|
chip->timer_A_index_old = 0;
|
|
chip->timer_B_index_old = 0;
|
|
|
|
chip->noise = 0;
|
|
chip->noise_rng = 0;
|
|
chip->noise_p = 0;
|
|
chip->noise_f = chip->noise_tab[0];
|
|
|
|
chip->csm_req = 0;
|
|
chip->status = 0;
|
|
|
|
ym2151_write_reg(chip, 0x1b, 0); /* only because of CT1, CT2 output pins */
|
|
ym2151_write_reg(chip, 0x18, 0); /* set LFO frequency */
|
|
for (i=0x20; i<0x100; i++) /* set the operators */
|
|
{
|
|
ym2151_write_reg(chip, i, 0);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
INLINE signed int op_calc(YM2151Operator * OP, unsigned int env, signed int pm)
|
|
{
|
|
UINT32 p;
|
|
|
|
|
|
p = (env<<3) + sin_tab[ ( ((signed int)((OP->phase & ~FREQ_MASK) + (pm<<15))) >> FREQ_SH ) & SIN_MASK ];
|
|
|
|
if (p >= TL_TAB_LEN)
|
|
return 0;
|
|
|
|
return tl_tab[p];
|
|
}
|
|
|
|
INLINE signed int op_calc1(YM2151Operator * OP, unsigned int env, signed int pm)
|
|
{
|
|
UINT32 p;
|
|
INT32 i;
|
|
|
|
|
|
i = (OP->phase & ~FREQ_MASK) + pm;
|
|
|
|
/*logerror("i=%08x (i>>16)&511=%8i phase=%i [pm=%08x] ",i, (i>>16)&511, OP->phase>>FREQ_SH, pm);*/
|
|
|
|
p = (env<<3) + sin_tab[ (i>>FREQ_SH) & SIN_MASK];
|
|
|
|
/*logerror("(p&255=%i p>>8=%i) out= %i\n", p&255,p>>8, tl_tab[p&255]>>(p>>8) );*/
|
|
|
|
if (p >= TL_TAB_LEN)
|
|
return 0;
|
|
|
|
return tl_tab[p];
|
|
}
|
|
|
|
|
|
|
|
#define volume_calc(OP) ((OP)->tl + ((UINT32)(OP)->volume) + (AM & (OP)->AMmask))
|
|
|
|
INLINE void chan_calc(unsigned int chan)
|
|
{
|
|
YM2151Operator *op;
|
|
unsigned int env;
|
|
UINT32 AM = 0;
|
|
|
|
if (PSG->Muted[chan])
|
|
return;
|
|
|
|
m2 = c1 = c2 = mem = 0;
|
|
op = &PSG->oper[chan*4]; /* M1 */
|
|
|
|
*op->mem_connect = op->mem_value; /* restore delayed sample (MEM) value to m2 or c2 */
|
|
|
|
if (op->ams)
|
|
AM = PSG->lfa << (op->ams-1);
|
|
env = volume_calc(op);
|
|
{
|
|
INT32 out = op->fb_out_prev + op->fb_out_curr;
|
|
op->fb_out_prev = op->fb_out_curr;
|
|
|
|
if (!op->connect)
|
|
/* algorithm 5 */
|
|
mem = c1 = c2 = op->fb_out_prev;
|
|
else
|
|
/* other algorithms */
|
|
*op->connect = op->fb_out_prev;
|
|
|
|
op->fb_out_curr = 0;
|
|
if (env < ENV_QUIET)
|
|
{
|
|
if (!op->fb_shift)
|
|
out=0;
|
|
op->fb_out_curr = op_calc1(op, env, (out<<op->fb_shift) );
|
|
}
|
|
}
|
|
|
|
env = volume_calc(op+1); /* M2 */
|
|
if (env < ENV_QUIET)
|
|
*(op+1)->connect += op_calc(op+1, env, m2);
|
|
|
|
env = volume_calc(op+2); /* C1 */
|
|
if (env < ENV_QUIET)
|
|
*(op+2)->connect += op_calc(op+2, env, c1);
|
|
|
|
env = volume_calc(op+3); /* C2 */
|
|
if (env < ENV_QUIET)
|
|
chanout[chan] += op_calc(op+3, env, c2);
|
|
if (chanout[chan] > +8192) chanout[chan] = +8192;
|
|
else if (chanout[chan] < -8192) chanout[chan] = -8192;
|
|
|
|
/* M1 */
|
|
op->mem_value = mem;
|
|
}
|
|
INLINE void chan7_calc(void)
|
|
{
|
|
YM2151Operator *op;
|
|
unsigned int env;
|
|
UINT32 AM = 0;
|
|
|
|
if (PSG->Muted[7])
|
|
return;
|
|
|
|
m2 = c1 = c2 = mem = 0;
|
|
op = &PSG->oper[7*4]; /* M1 */
|
|
|
|
*op->mem_connect = op->mem_value; /* restore delayed sample (MEM) value to m2 or c2 */
|
|
|
|
if (op->ams)
|
|
AM = PSG->lfa << (op->ams-1);
|
|
env = volume_calc(op);
|
|
{
|
|
INT32 out = op->fb_out_prev + op->fb_out_curr;
|
|
op->fb_out_prev = op->fb_out_curr;
|
|
|
|
if (!op->connect)
|
|
/* algorithm 5 */
|
|
mem = c1 = c2 = op->fb_out_prev;
|
|
else
|
|
/* other algorithms */
|
|
*op->connect = op->fb_out_prev;
|
|
|
|
op->fb_out_curr = 0;
|
|
if (env < ENV_QUIET)
|
|
{
|
|
if (!op->fb_shift)
|
|
out=0;
|
|
op->fb_out_curr = op_calc1(op, env, (out<<op->fb_shift) );
|
|
}
|
|
}
|
|
|
|
env = volume_calc(op+1); /* M2 */
|
|
if (env < ENV_QUIET)
|
|
*(op+1)->connect += op_calc(op+1, env, m2);
|
|
|
|
env = volume_calc(op+2); /* C1 */
|
|
if (env < ENV_QUIET)
|
|
*(op+2)->connect += op_calc(op+2, env, c1);
|
|
|
|
env = volume_calc(op+3); /* C2 */
|
|
if (PSG->noise & 0x80)
|
|
{
|
|
INT32 noiseout;
|
|
|
|
noiseout = 0;
|
|
if (env < 0x3ff)
|
|
noiseout = (env ^ 0x3ff) * 2; /* range of the YM2151 noise output is -2044 to 2040 */
|
|
chanout[7] += ((PSG->noise_rng&0x10000) ? noiseout: -noiseout); /* bit 16 -> output */
|
|
}
|
|
else
|
|
{
|
|
if (env < ENV_QUIET)
|
|
chanout[7] += op_calc(op+3, env, c2);
|
|
}
|
|
if (chanout[7] > +8192) chanout[7] = +8192;
|
|
else if (chanout[7] < -8192) chanout[7] = -8192;
|
|
/* M1 */
|
|
op->mem_value = mem;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
The 'rate' is calculated from following formula (example on decay rate):
|
|
rks = notecode after key scaling (a value from 0 to 31)
|
|
DR = value written to the chip register
|
|
rate = 2*DR + rks; (max rate = 2*31+31 = 93)
|
|
Four MSBs of the 'rate' above are the 'main' rate (from 00 to 15)
|
|
Two LSBs of the 'rate' above are the value 'x' (the shape type).
|
|
(eg. '11 2' means that 'rate' is 11*4+2=46)
|
|
|
|
NOTE: A 'sample' in the description below is actually 3 output samples,
|
|
thats because the Envelope Generator clock is equal to internal_clock/3.
|
|
|
|
Single '-' (minus) character in the diagrams below represents one sample
|
|
on the output; this is for rates 11 x (11 0, 11 1, 11 2 and 11 3)
|
|
|
|
these 'main' rates:
|
|
00 x: single '-' = 2048 samples; (ie. level can change every 2048 samples)
|
|
01 x: single '-' = 1024 samples;
|
|
02 x: single '-' = 512 samples;
|
|
03 x: single '-' = 256 samples;
|
|
04 x: single '-' = 128 samples;
|
|
05 x: single '-' = 64 samples;
|
|
06 x: single '-' = 32 samples;
|
|
07 x: single '-' = 16 samples;
|
|
08 x: single '-' = 8 samples;
|
|
09 x: single '-' = 4 samples;
|
|
10 x: single '-' = 2 samples;
|
|
11 x: single '-' = 1 sample; (ie. level can change every 1 sample)
|
|
|
|
Shapes for rates 11 x look like this:
|
|
rate: step:
|
|
11 0 01234567
|
|
|
|
level:
|
|
0 --
|
|
1 --
|
|
2 --
|
|
3 --
|
|
|
|
rate: step:
|
|
11 1 01234567
|
|
|
|
level:
|
|
0 --
|
|
1 --
|
|
2 -
|
|
3 -
|
|
4 --
|
|
|
|
rate: step:
|
|
11 2 01234567
|
|
|
|
level:
|
|
0 --
|
|
1 -
|
|
2 -
|
|
3 --
|
|
4 -
|
|
5 -
|
|
|
|
rate: step:
|
|
11 3 01234567
|
|
|
|
level:
|
|
0 --
|
|
1 -
|
|
2 -
|
|
3 -
|
|
4 -
|
|
5 -
|
|
6 -
|
|
|
|
|
|
For rates 12 x, 13 x, 14 x and 15 x output level changes on every
|
|
sample - this means that the waveform looks like this: (but the level
|
|
changes by different values on different steps)
|
|
12 3 01234567
|
|
|
|
0 -
|
|
2 -
|
|
4 -
|
|
8 -
|
|
10 -
|
|
12 -
|
|
14 -
|
|
18 -
|
|
20 -
|
|
|
|
Notes about the timing:
|
|
----------------------
|
|
|
|
1. Synchronism
|
|
|
|
Output level of each two (or more) voices running at the same 'main' rate
|
|
(eg 11 0 and 11 1 in the diagram below) will always be changing in sync,
|
|
even if there're started with some delay.
|
|
|
|
Note that, in the diagram below, the decay phase in channel 0 starts at
|
|
sample #2, while in channel 1 it starts at sample #6. Anyway, both channels
|
|
will always change their levels at exactly the same (following) samples.
|
|
|
|
(S - start point of this channel, A-attack phase, D-decay phase):
|
|
|
|
step:
|
|
01234567012345670123456
|
|
|
|
channel 0:
|
|
--
|
|
| --
|
|
| -
|
|
| -
|
|
| --
|
|
| --
|
|
| --
|
|
| -
|
|
| -
|
|
| --
|
|
AADDDDDDDDDDDDDDDD
|
|
S
|
|
|
|
01234567012345670123456
|
|
channel 1:
|
|
-
|
|
| -
|
|
| --
|
|
| --
|
|
| --
|
|
| -
|
|
| -
|
|
| --
|
|
| --
|
|
| --
|
|
AADDDDDDDDDDDDDDDD
|
|
S
|
|
01234567012345670123456
|
|
|
|
|
|
2. Shifted (delayed) synchronism
|
|
|
|
Output of each two (or more) voices running at different 'main' rate
|
|
(9 1, 10 1 and 11 1 in the diagrams below) will always be changing
|
|
in 'delayed-sync' (even if there're started with some delay as in "1.")
|
|
|
|
Note that the shapes are delayed by exactly one sample per one 'main' rate
|
|
increment. (Normally one would expect them to start at the same samples.)
|
|
|
|
See diagram below (* - start point of the shape).
|
|
|
|
cycle:
|
|
0123456701234567012345670123456701234567012345670123456701234567
|
|
|
|
rate 09 1
|
|
*-------
|
|
--------
|
|
----
|
|
----
|
|
--------
|
|
*-------
|
|
| --------
|
|
| ----
|
|
| ----
|
|
| --------
|
|
rate 10 1 |
|
|
-- |
|
|
*--- |
|
|
---- |
|
|
-- |
|
|
-- |
|
|
---- |
|
|
*--- |
|
|
| ---- |
|
|
| -- | | <- one step (two samples) delay between 9 1 and 10 1
|
|
| -- | |
|
|
| ----|
|
|
| *---
|
|
| ----
|
|
| --
|
|
| --
|
|
| ----
|
|
rate 11 1 |
|
|
- |
|
|
-- |
|
|
*- |
|
|
-- |
|
|
- |
|
|
- |
|
|
-- |
|
|
*- |
|
|
-- |
|
|
- || <- one step (one sample) delay between 10 1 and 11 1
|
|
- ||
|
|
--|
|
|
*-
|
|
--
|
|
-
|
|
-
|
|
--
|
|
*-
|
|
--
|
|
-
|
|
-
|
|
--
|
|
*/
|
|
|
|
INLINE void advance_eg(void)
|
|
{
|
|
YM2151Operator *op;
|
|
unsigned int i;
|
|
|
|
|
|
|
|
PSG->eg_timer += PSG->eg_timer_add;
|
|
|
|
while (PSG->eg_timer >= PSG->eg_timer_overflow)
|
|
{
|
|
PSG->eg_timer -= PSG->eg_timer_overflow;
|
|
|
|
PSG->eg_cnt++;
|
|
|
|
/* envelope generator */
|
|
op = &PSG->oper[0]; /* CH 0 M1 */
|
|
i = 32;
|
|
do
|
|
{
|
|
switch(op->state)
|
|
{
|
|
case EG_ATT: /* attack phase */
|
|
if ( !(PSG->eg_cnt & ((1<<op->eg_sh_ar)-1) ) )
|
|
{
|
|
op->volume += (~op->volume *
|
|
(eg_inc[op->eg_sel_ar + ((PSG->eg_cnt>>op->eg_sh_ar)&7)])
|
|
) >>4;
|
|
|
|
if (op->volume <= MIN_ATT_INDEX)
|
|
{
|
|
op->volume = MIN_ATT_INDEX;
|
|
op->state = EG_DEC;
|
|
}
|
|
|
|
}
|
|
break;
|
|
|
|
case EG_DEC: /* decay phase */
|
|
if ( !(PSG->eg_cnt & ((1<<op->eg_sh_d1r)-1) ) )
|
|
{
|
|
op->volume += eg_inc[op->eg_sel_d1r + ((PSG->eg_cnt>>op->eg_sh_d1r)&7)];
|
|
|
|
if ( op->volume >= op->d1l )
|
|
op->state = EG_SUS;
|
|
|
|
}
|
|
break;
|
|
|
|
case EG_SUS: /* sustain phase */
|
|
if ( !(PSG->eg_cnt & ((1<<op->eg_sh_d2r)-1) ) )
|
|
{
|
|
op->volume += eg_inc[op->eg_sel_d2r + ((PSG->eg_cnt>>op->eg_sh_d2r)&7)];
|
|
|
|
if ( op->volume >= MAX_ATT_INDEX )
|
|
{
|
|
op->volume = MAX_ATT_INDEX;
|
|
op->state = EG_OFF;
|
|
}
|
|
|
|
}
|
|
break;
|
|
|
|
case EG_REL: /* release phase */
|
|
if ( !(PSG->eg_cnt & ((1<<op->eg_sh_rr)-1) ) )
|
|
{
|
|
op->volume += eg_inc[op->eg_sel_rr + ((PSG->eg_cnt>>op->eg_sh_rr)&7)];
|
|
|
|
if ( op->volume >= MAX_ATT_INDEX )
|
|
{
|
|
op->volume = MAX_ATT_INDEX;
|
|
op->state = EG_OFF;
|
|
}
|
|
|
|
}
|
|
break;
|
|
}
|
|
op++;
|
|
i--;
|
|
}while (i);
|
|
}
|
|
}
|
|
|
|
|
|
INLINE void advance(void)
|
|
{
|
|
YM2151Operator *op;
|
|
unsigned int i;
|
|
int a,p;
|
|
|
|
/* LFO */
|
|
if (PSG->test&2)
|
|
PSG->lfo_phase = 0;
|
|
else
|
|
{
|
|
PSG->lfo_timer += PSG->lfo_timer_add;
|
|
if (PSG->lfo_timer >= PSG->lfo_overflow)
|
|
{
|
|
PSG->lfo_timer -= PSG->lfo_overflow;
|
|
PSG->lfo_counter += PSG->lfo_counter_add;
|
|
PSG->lfo_phase += (PSG->lfo_counter>>4);
|
|
PSG->lfo_phase &= 255;
|
|
PSG->lfo_counter &= 15;
|
|
}
|
|
}
|
|
|
|
i = PSG->lfo_phase;
|
|
/* calculate LFO AM and PM waveform value (all verified on real chip, except for noise algorithm which is impossible to analyse)*/
|
|
switch (PSG->lfo_wsel)
|
|
{
|
|
case 0:
|
|
/* saw */
|
|
/* AM: 255 down to 0 */
|
|
/* PM: 0 to 127, -127 to 0 (at PMD=127: LFP = 0 to 126, -126 to 0) */
|
|
a = 255 - i;
|
|
if (i<128)
|
|
p = i;
|
|
else
|
|
p = i - 255;
|
|
break;
|
|
case 1:
|
|
/* square */
|
|
/* AM: 255, 0 */
|
|
/* PM: 128,-128 (LFP = exactly +PMD, -PMD) */
|
|
if (i<128){
|
|
a = 255;
|
|
p = 128;
|
|
}else{
|
|
a = 0;
|
|
p = -128;
|
|
}
|
|
break;
|
|
case 2:
|
|
/* triangle */
|
|
/* AM: 255 down to 1 step -2; 0 up to 254 step +2 */
|
|
/* PM: 0 to 126 step +2, 127 to 1 step -2, 0 to -126 step -2, -127 to -1 step +2*/
|
|
if (i<128)
|
|
a = 255 - (i*2);
|
|
else
|
|
a = (i*2) - 256;
|
|
|
|
if (i<64) /* i = 0..63 */
|
|
p = i*2; /* 0 to 126 step +2 */
|
|
else if (i<128) /* i = 64..127 */
|
|
p = 255 - i*2; /* 127 to 1 step -2 */
|
|
else if (i<192) /* i = 128..191 */
|
|
p = 256 - i*2; /* 0 to -126 step -2*/
|
|
else /* i = 192..255 */
|
|
p = i*2 - 511; /*-127 to -1 step +2*/
|
|
break;
|
|
case 3:
|
|
default: /*keep the compiler happy*/
|
|
/* random */
|
|
/* the real algorithm is unknown !!!
|
|
We just use a snapshot of data from real chip */
|
|
|
|
/* AM: range 0 to 255 */
|
|
/* PM: range -128 to 127 */
|
|
|
|
a = lfo_noise_waveform[i];
|
|
p = a-128;
|
|
break;
|
|
}
|
|
PSG->lfa = a * PSG->amd / 128;
|
|
PSG->lfp = p * PSG->pmd / 128;
|
|
|
|
|
|
/* The Noise Generator of the YM2151 is 17-bit shift register.
|
|
* Input to the bit16 is negated (bit0 XOR bit3) (EXNOR).
|
|
* Output of the register is negated (bit0 XOR bit3).
|
|
* Simply use bit16 as the noise output.
|
|
*/
|
|
PSG->noise_p += PSG->noise_f;
|
|
i = (PSG->noise_p>>16); /* number of events (shifts of the shift register) */
|
|
PSG->noise_p &= 0xffff;
|
|
while (i)
|
|
{
|
|
UINT32 j;
|
|
j = ( (PSG->noise_rng ^ (PSG->noise_rng>>3) ) & 1) ^ 1;
|
|
PSG->noise_rng = (j<<16) | (PSG->noise_rng>>1);
|
|
i--;
|
|
}
|
|
|
|
|
|
/* phase generator */
|
|
op = &PSG->oper[0]; /* CH 0 M1 */
|
|
i = 8;
|
|
do
|
|
{
|
|
if (op->pms) /* only when phase modulation from LFO is enabled for this channel */
|
|
{
|
|
INT32 mod_ind = PSG->lfp; /* -128..+127 (8bits signed) */
|
|
if (op->pms < 6)
|
|
mod_ind >>= (6 - op->pms);
|
|
else
|
|
mod_ind <<= (op->pms - 5);
|
|
|
|
if (mod_ind)
|
|
{
|
|
UINT32 kc_channel = op->kc_i + mod_ind;
|
|
(op+0)->phase += ( (PSG->freq[ kc_channel + (op+0)->dt2 ] + (op+0)->dt1) * (op+0)->mul ) >> 1;
|
|
(op+1)->phase += ( (PSG->freq[ kc_channel + (op+1)->dt2 ] + (op+1)->dt1) * (op+1)->mul ) >> 1;
|
|
(op+2)->phase += ( (PSG->freq[ kc_channel + (op+2)->dt2 ] + (op+2)->dt1) * (op+2)->mul ) >> 1;
|
|
(op+3)->phase += ( (PSG->freq[ kc_channel + (op+3)->dt2 ] + (op+3)->dt1) * (op+3)->mul ) >> 1;
|
|
}
|
|
else /* phase modulation from LFO is equal to zero */
|
|
{
|
|
(op+0)->phase += (op+0)->freq;
|
|
(op+1)->phase += (op+1)->freq;
|
|
(op+2)->phase += (op+2)->freq;
|
|
(op+3)->phase += (op+3)->freq;
|
|
}
|
|
}
|
|
else /* phase modulation from LFO is disabled */
|
|
{
|
|
(op+0)->phase += (op+0)->freq;
|
|
(op+1)->phase += (op+1)->freq;
|
|
(op+2)->phase += (op+2)->freq;
|
|
(op+3)->phase += (op+3)->freq;
|
|
}
|
|
|
|
op+=4;
|
|
i--;
|
|
}while (i);
|
|
|
|
|
|
/* CSM is calculated *after* the phase generator calculations (verified on real chip)
|
|
* CSM keyon line seems to be ORed with the KO line inside of the chip.
|
|
* The result is that it only works when KO (register 0x08) is off, ie. 0
|
|
*
|
|
* Interesting effect is that when timer A is set to 1023, the KEY ON happens
|
|
* on every sample, so there is no KEY OFF at all - the result is that
|
|
* the sound played is the same as after normal KEY ON.
|
|
*/
|
|
|
|
if (PSG->csm_req) /* CSM KEYON/KEYOFF seqeunce request */
|
|
{
|
|
if (PSG->csm_req==2) /* KEY ON */
|
|
{
|
|
op = &PSG->oper[0]; /* CH 0 M1 */
|
|
i = 32;
|
|
do
|
|
{
|
|
KEY_ON(op, 2);
|
|
op++;
|
|
i--;
|
|
}while (i);
|
|
PSG->csm_req = 1;
|
|
}
|
|
else /* KEY OFF */
|
|
{
|
|
op = &PSG->oper[0]; /* CH 0 M1 */
|
|
i = 32;
|
|
do
|
|
{
|
|
KEY_OFF(op,~2);
|
|
op++;
|
|
i--;
|
|
}while (i);
|
|
PSG->csm_req = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
INLINE signed int acc_calc(signed int value)
|
|
{
|
|
if (value>=0)
|
|
{
|
|
if (value < 0x0200)
|
|
return (value & ~0);
|
|
if (value < 0x0400)
|
|
return (value & ~1);
|
|
if (value < 0x0800)
|
|
return (value & ~3);
|
|
if (value < 0x1000)
|
|
return (value & ~7);
|
|
if (value < 0x2000)
|
|
return (value & ~15);
|
|
if (value < 0x4000)
|
|
return (value & ~31);
|
|
return (value & ~63);
|
|
}
|
|
/*else value < 0*/
|
|
if (value > -0x0200)
|
|
return (~abs(value) & ~0);
|
|
if (value > -0x0400)
|
|
return (~abs(value) & ~1);
|
|
if (value > -0x0800)
|
|
return (~abs(value) & ~3);
|
|
if (value > -0x1000)
|
|
return (~abs(value) & ~7);
|
|
if (value > -0x2000)
|
|
return (~abs(value) & ~15);
|
|
if (value > -0x4000)
|
|
return (~abs(value) & ~31);
|
|
return (~abs(value) & ~63);
|
|
}
|
|
#endif
|
|
|
|
/* first macro saves left and right channels to mono file */
|
|
/* second macro saves left and right channels to stereo file */
|
|
#if 0 /*MONO*/
|
|
#ifdef SAVE_SEPARATE_CHANNELS
|
|
#define SAVE_SINGLE_CHANNEL(j) \
|
|
{ signed int pom= -(chanout[j] & PSG->pan[j*2]); \
|
|
if (pom > 32767) pom = 32767; else if (pom < -32768) pom = -32768; \
|
|
fputc((unsigned short)pom&0xff,sample[j]); \
|
|
fputc(((unsigned short)pom>>8)&0xff,sample[j]); \
|
|
}
|
|
#else
|
|
#define SAVE_SINGLE_CHANNEL(j)
|
|
#endif
|
|
#else /*STEREO*/
|
|
#ifdef SAVE_SEPARATE_CHANNELS
|
|
#define SAVE_SINGLE_CHANNEL(j) \
|
|
{ signed int pom = -(chanout[j] & PSG->pan[j*2]); \
|
|
if (pom > 32767) pom = 32767; else if (pom < -32768) pom = -32768; \
|
|
fputc((unsigned short)pom&0xff,sample[j]); \
|
|
fputc(((unsigned short)pom>>8)&0xff,sample[j]); \
|
|
pom = -(chanout[j] & PSG->pan[j*2+1]); \
|
|
if (pom > 32767) pom = 32767; else if (pom < -32768) pom = -32768; \
|
|
fputc((unsigned short)pom&0xff,sample[j]); \
|
|
fputc(((unsigned short)pom>>8)&0xff,sample[j]); \
|
|
}
|
|
#else
|
|
#define SAVE_SINGLE_CHANNEL(j)
|
|
#endif
|
|
#endif
|
|
|
|
/* first macro saves left and right channels to mono file */
|
|
/* second macro saves left and right channels to stereo file */
|
|
#if 1 /*MONO*/
|
|
#ifdef SAVE_SAMPLE
|
|
#define SAVE_ALL_CHANNELS \
|
|
{ signed int pom = outl; \
|
|
/*pom = acc_calc(pom);*/ \
|
|
/*fprintf(sample[8]," %i\n",pom);*/ \
|
|
fputc((unsigned short)pom&0xff,sample[8]); \
|
|
fputc(((unsigned short)pom>>8)&0xff,sample[8]); \
|
|
}
|
|
#else
|
|
#define SAVE_ALL_CHANNELS
|
|
#endif
|
|
#else /*STEREO*/
|
|
#ifdef SAVE_SAMPLE
|
|
#define SAVE_ALL_CHANNELS \
|
|
{ signed int pom = outl; \
|
|
fputc((unsigned short)pom&0xff,sample[8]); \
|
|
fputc(((unsigned short)pom>>8)&0xff,sample[8]); \
|
|
pom = outr; \
|
|
fputc((unsigned short)pom&0xff,sample[8]); \
|
|
fputc(((unsigned short)pom>>8)&0xff,sample[8]); \
|
|
}
|
|
#else
|
|
#define SAVE_ALL_CHANNELS
|
|
#endif
|
|
#endif
|
|
|
|
|
|
/* Generate samples for one of the YM2151's
|
|
*
|
|
* 'num' is the number of virtual YM2151
|
|
* '**buffers' is table of pointers to the buffers: left and right
|
|
* 'length' is the number of samples that should be generated
|
|
*/
|
|
void ym2151_update_one(void *chip, SAMP **buffers, int length)
|
|
{
|
|
int i, chn;
|
|
signed int outl,outr;
|
|
SAMP *bufL, *bufR;
|
|
|
|
bufL = buffers[0];
|
|
bufR = buffers[1];
|
|
|
|
PSG = (YM2151 *)chip;
|
|
|
|
#ifdef USE_MAME_TIMERS
|
|
/* ASG 980324 - handled by real timers now */
|
|
#else
|
|
if (PSG->tim_B)
|
|
{
|
|
PSG->tim_B_val -= ( length << TIMER_SH );
|
|
if (PSG->tim_B_val<=0)
|
|
{
|
|
PSG->tim_B_val += PSG->tim_B_tab[ PSG->timer_B_index ];
|
|
if ( PSG->irq_enable & 0x08 )
|
|
{
|
|
int oldstate = PSG->status & 3;
|
|
PSG->status |= 2;
|
|
//if ((!oldstate) && (PSG->irqhandler)) (*PSG->irqhandler)(chip->device, 1);
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
for (i=0; i<length; i++)
|
|
{
|
|
advance_eg();
|
|
|
|
chanout[0] = 0;
|
|
chanout[1] = 0;
|
|
chanout[2] = 0;
|
|
chanout[3] = 0;
|
|
chanout[4] = 0;
|
|
chanout[5] = 0;
|
|
chanout[6] = 0;
|
|
chanout[7] = 0;
|
|
|
|
chan_calc(0);
|
|
SAVE_SINGLE_CHANNEL(0)
|
|
chan_calc(1);
|
|
SAVE_SINGLE_CHANNEL(1)
|
|
chan_calc(2);
|
|
SAVE_SINGLE_CHANNEL(2)
|
|
chan_calc(3);
|
|
SAVE_SINGLE_CHANNEL(3)
|
|
chan_calc(4);
|
|
SAVE_SINGLE_CHANNEL(4)
|
|
chan_calc(5);
|
|
SAVE_SINGLE_CHANNEL(5)
|
|
chan_calc(6);
|
|
SAVE_SINGLE_CHANNEL(6)
|
|
chan7_calc();
|
|
SAVE_SINGLE_CHANNEL(7)
|
|
|
|
outl = chanout[0] & PSG->pan[0];
|
|
outr = chanout[0] & PSG->pan[1];
|
|
outl += (chanout[1] & PSG->pan[2]);
|
|
outr += (chanout[1] & PSG->pan[3]);
|
|
outl += (chanout[2] & PSG->pan[4]);
|
|
outr += (chanout[2] & PSG->pan[5]);
|
|
outl += (chanout[3] & PSG->pan[6]);
|
|
outr += (chanout[3] & PSG->pan[7]);
|
|
outl += (chanout[4] & PSG->pan[8]);
|
|
outr += (chanout[4] & PSG->pan[9]);
|
|
outl += (chanout[5] & PSG->pan[10]);
|
|
outr += (chanout[5] & PSG->pan[11]);
|
|
outl += (chanout[6] & PSG->pan[12]);
|
|
outr += (chanout[6] & PSG->pan[13]);
|
|
outl += (chanout[7] & PSG->pan[14]);
|
|
outr += (chanout[7] & PSG->pan[15]);
|
|
|
|
outl >>= FINAL_SH;
|
|
outr >>= FINAL_SH;
|
|
//if (outl > MAXOUT) outl = MAXOUT;
|
|
// else if (outl < MINOUT) outl = MINOUT;
|
|
//if (outr > MAXOUT) outr = MAXOUT;
|
|
// else if (outr < MINOUT) outr = MINOUT;
|
|
((SAMP*)bufL)[i] = (SAMP)outl;
|
|
((SAMP*)bufR)[i] = (SAMP)outr;
|
|
|
|
SAVE_ALL_CHANNELS
|
|
|
|
#ifdef USE_MAME_TIMERS
|
|
/* ASG 980324 - handled by real timers now */
|
|
#else
|
|
/* calculate timer A */
|
|
if (PSG->tim_A)
|
|
{
|
|
PSG->tim_A_val -= ( 1 << TIMER_SH );
|
|
if (PSG->tim_A_val <= 0)
|
|
{
|
|
PSG->tim_A_val += PSG->tim_A_tab[ PSG->timer_A_index ];
|
|
if (PSG->irq_enable & 0x04)
|
|
{
|
|
int oldstate = PSG->status & 3;
|
|
PSG->status |= 1;
|
|
//if ((!oldstate) && (PSG->irqhandler)) (*PSG->irqhandler)(chip->device, 1);
|
|
}
|
|
if (PSG->irq_enable & 0x80)
|
|
PSG->csm_req = 2; /* request KEY ON / KEY OFF sequence */
|
|
}
|
|
}
|
|
#endif
|
|
advance();
|
|
}
|
|
}
|
|
|
|
/*void ym2151_set_irq_handler(void *chip, void(*handler)(int irq))
|
|
{
|
|
YM2151 *PSG = (YM2151 *)chip;
|
|
PSG->irqhandler = handler;
|
|
}
|
|
|
|
void ym2151_set_port_write_handler(void *chip, write8_device_func handler)
|
|
{
|
|
YM2151 *PSG = (YM2151 *)chip;
|
|
PSG->porthandler = handler;
|
|
}*/
|
|
|
|
void ym2151_set_mutemask(void *chip, UINT32 MuteMask)
|
|
{
|
|
YM2151 *PSG = (YM2151 *)chip;
|
|
UINT8 CurChn;
|
|
|
|
for (CurChn = 0; CurChn < 8; CurChn ++)
|
|
PSG->Muted[CurChn] = (MuteMask >> CurChn) & 0x01;
|
|
|
|
return;
|
|
}
|
|
|