mirror of https://github.com/snes9xgit/snes9x.git
1441 lines
45 KiB
C++
1441 lines
45 KiB
C++
// snes_spc 0.9.0. http://www.slack.net/~ant/
|
|
|
|
#include "../../../snes9x.h"
|
|
|
|
#include "SPC_DSP.h"
|
|
|
|
#include "blargg_endian.h"
|
|
#include <string.h>
|
|
|
|
/* Copyright (C) 2007 Shay Green. This module is free software; you
|
|
can redistribute it and/or modify it under the terms of the GNU Lesser
|
|
General Public License as published by the Free Software Foundation; either
|
|
version 2.1 of the License, or (at your option) any later version. This
|
|
module is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
|
FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
|
|
details. You should have received a copy of the GNU Lesser General Public
|
|
License along with this module; if not, write to the Free Software Foundation,
|
|
Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */
|
|
|
|
#include "blargg_source.h"
|
|
|
|
#ifdef BLARGG_ENABLE_OPTIMIZER
|
|
#include BLARGG_ENABLE_OPTIMIZER
|
|
#endif
|
|
|
|
#if INT_MAX < 0x7FFFFFFF
|
|
#error "Requires that int type have at least 32 bits"
|
|
#endif
|
|
|
|
// TODO: add to blargg_endian.h
|
|
#define GET_LE16SA( addr ) ((BOOST::int16_t) GET_LE16( addr ))
|
|
#define GET_LE16A( addr ) GET_LE16( addr )
|
|
#define SET_LE16A( addr, data ) SET_LE16( addr, data )
|
|
|
|
static BOOST::uint8_t const initial_regs [SPC_DSP::register_count] =
|
|
{
|
|
0x45,0x8B,0x5A,0x9A,0xE4,0x82,0x1B,0x78,0x00,0x00,0xAA,0x96,0x89,0x0E,0xE0,0x80,
|
|
0x2A,0x49,0x3D,0xBA,0x14,0xA0,0xAC,0xC5,0x00,0x00,0x51,0xBB,0x9C,0x4E,0x7B,0xFF,
|
|
0xF4,0xFD,0x57,0x32,0x37,0xD9,0x42,0x22,0x00,0x00,0x5B,0x3C,0x9F,0x1B,0x87,0x9A,
|
|
0x6F,0x27,0xAF,0x7B,0xE5,0x68,0x0A,0xD9,0x00,0x00,0x9A,0xC5,0x9C,0x4E,0x7B,0xFF,
|
|
0xEA,0x21,0x78,0x4F,0xDD,0xED,0x24,0x14,0x00,0x00,0x77,0xB1,0xD1,0x36,0xC1,0x67,
|
|
0x52,0x57,0x46,0x3D,0x59,0xF4,0x87,0xA4,0x00,0x00,0x7E,0x44,0x00,0x4E,0x7B,0xFF,
|
|
0x75,0xF5,0x06,0x97,0x10,0xC3,0x24,0xBB,0x00,0x00,0x7B,0x7A,0xE0,0x60,0x12,0x0F,
|
|
0xF7,0x74,0x1C,0xE5,0x39,0x3D,0x73,0xC1,0x00,0x00,0x7A,0xB3,0xFF,0x4E,0x7B,0xFF
|
|
};
|
|
|
|
// if ( io < -32768 ) io = -32768;
|
|
// if ( io > 32767 ) io = 32767;
|
|
#define CLAMP16( io )\
|
|
{\
|
|
if ( (int16_t) io != io )\
|
|
io = (io >> 31) ^ 0x7FFF;\
|
|
}
|
|
|
|
// Access global DSP register
|
|
#define REG(n) m.regs [r_##n]
|
|
#define XREG(n) m.external_regs [r_##n]
|
|
|
|
// Access voice DSP register
|
|
#define VREG(r,n) r [v_##n]
|
|
#define XVREG(r,n) (m.external_regs + (r - m.regs))[v_##n]
|
|
|
|
#define WRITE_SAMPLES( l, r, out ) \
|
|
{\
|
|
out [0] = l;\
|
|
out [1] = r;\
|
|
out += 2;\
|
|
}
|
|
|
|
#define SPC_DSP_OUT_HOOK(l, r) \
|
|
{ \
|
|
resampler->push_sample(l, r); \
|
|
if (Settings.MSU1) \
|
|
S9xMSU1Generate(2); \
|
|
}
|
|
|
|
void SPC_DSP::set_output( Resampler *resampler )
|
|
{
|
|
this->resampler = resampler;
|
|
}
|
|
|
|
void SPC_DSP::set_output( sample_t* out, int size )
|
|
{
|
|
require( (size & 1) == 0 ); // must be even
|
|
m.out_begin = out;
|
|
m.out = out;
|
|
m.out_end = out + size;
|
|
}
|
|
|
|
// Volume registers and efb are signed! Easy to forget int8_t cast.
|
|
// Prefixes are to avoid accidental use of locals with same names.
|
|
|
|
// Gaussian interpolation
|
|
|
|
static short const gauss [512] =
|
|
{
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2,
|
|
2, 2, 3, 3, 3, 3, 3, 4, 4, 4, 4, 4, 5, 5, 5, 5,
|
|
6, 6, 6, 6, 7, 7, 7, 8, 8, 8, 9, 9, 9, 10, 10, 10,
|
|
11, 11, 11, 12, 12, 13, 13, 14, 14, 15, 15, 15, 16, 16, 17, 17,
|
|
18, 19, 19, 20, 20, 21, 21, 22, 23, 23, 24, 24, 25, 26, 27, 27,
|
|
28, 29, 29, 30, 31, 32, 32, 33, 34, 35, 36, 36, 37, 38, 39, 40,
|
|
41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56,
|
|
58, 59, 60, 61, 62, 64, 65, 66, 67, 69, 70, 71, 73, 74, 76, 77,
|
|
78, 80, 81, 83, 84, 86, 87, 89, 90, 92, 94, 95, 97, 99, 100, 102,
|
|
104, 106, 107, 109, 111, 113, 115, 117, 118, 120, 122, 124, 126, 128, 130, 132,
|
|
134, 137, 139, 141, 143, 145, 147, 150, 152, 154, 156, 159, 161, 163, 166, 168,
|
|
171, 173, 175, 178, 180, 183, 186, 188, 191, 193, 196, 199, 201, 204, 207, 210,
|
|
212, 215, 218, 221, 224, 227, 230, 233, 236, 239, 242, 245, 248, 251, 254, 257,
|
|
260, 263, 267, 270, 273, 276, 280, 283, 286, 290, 293, 297, 300, 304, 307, 311,
|
|
314, 318, 321, 325, 328, 332, 336, 339, 343, 347, 351, 354, 358, 362, 366, 370,
|
|
374, 378, 381, 385, 389, 393, 397, 401, 405, 410, 414, 418, 422, 426, 430, 434,
|
|
439, 443, 447, 451, 456, 460, 464, 469, 473, 477, 482, 486, 491, 495, 499, 504,
|
|
508, 513, 517, 522, 527, 531, 536, 540, 545, 550, 554, 559, 563, 568, 573, 577,
|
|
582, 587, 592, 596, 601, 606, 611, 615, 620, 625, 630, 635, 640, 644, 649, 654,
|
|
659, 664, 669, 674, 678, 683, 688, 693, 698, 703, 708, 713, 718, 723, 728, 732,
|
|
737, 742, 747, 752, 757, 762, 767, 772, 777, 782, 787, 792, 797, 802, 806, 811,
|
|
816, 821, 826, 831, 836, 841, 846, 851, 855, 860, 865, 870, 875, 880, 884, 889,
|
|
894, 899, 904, 908, 913, 918, 923, 927, 932, 937, 941, 946, 951, 955, 960, 965,
|
|
969, 974, 978, 983, 988, 992, 997,1001,1005,1010,1014,1019,1023,1027,1032,1036,
|
|
1040,1045,1049,1053,1057,1061,1066,1070,1074,1078,1082,1086,1090,1094,1098,1102,
|
|
1106,1109,1113,1117,1121,1125,1128,1132,1136,1139,1143,1146,1150,1153,1157,1160,
|
|
1164,1167,1170,1174,1177,1180,1183,1186,1190,1193,1196,1199,1202,1205,1207,1210,
|
|
1213,1216,1219,1221,1224,1227,1229,1232,1234,1237,1239,1241,1244,1246,1248,1251,
|
|
1253,1255,1257,1259,1261,1263,1265,1267,1269,1270,1272,1274,1275,1277,1279,1280,
|
|
1282,1283,1284,1286,1287,1288,1290,1291,1292,1293,1294,1295,1296,1297,1297,1298,
|
|
1299,1300,1300,1301,1302,1302,1303,1303,1303,1304,1304,1304,1304,1304,1305,1305,
|
|
};
|
|
|
|
// Interpolation by Mudlord
|
|
static short const cubic [514] =
|
|
{
|
|
0, -4, -8, -12, -16, -20, -23, -27, -30, -34, -37, -41, -44, -47, -50, -53,
|
|
-56, -59, -62, -65, -68, -71, -73, -76, -78, -81, -84, -87, -89, -91, -93, -95,
|
|
-98,-100,-102,-104,-106,-109,-110,-112,-113,-116,-117,-119,-121,-122,-123,-125,
|
|
-126,-128,-129,-131,-132,-134,-134,-136,-136,-138,-138,-140,-141,-141,-142,-143,
|
|
-144,-144,-145,-146,-147,-148,-147,-148,-148,-149,-149,-150,-150,-150,-150,-151,
|
|
-151,-151,-151,-151,-152,-152,-151,-152,-151,-152,-151,-151,-151,-151,-150,-150,
|
|
-150,-149,-149,-149,-149,-148,-147,-147,-146,-146,-145,-145,-144,-144,-143,-142,
|
|
-141,-141,-140,-139,-139,-138,-137,-136,-135,-135,-133,-133,-132,-131,-130,-129,
|
|
-128,-127,-126,-125,-124,-123,-121,-121,-119,-118,-117,-116,-115,-114,-112,-111,
|
|
-110,-109,-107,-106,-105,-104,-102,-102,-100, -99, -97, -97, -95, -94, -92, -91,
|
|
-90, -88, -87, -86, -85, -84, -82, -81, -79, -78, -76, -76, -74, -73, -71, -70,
|
|
-68, -67, -66, -65, -63, -62, -60, -60, -58, -57, -55, -55, -53, -52, -50, -49,
|
|
-48, -46, -45, -44, -43, -42, -40, -39, -38, -37, -36, -35, -34, -32, -31, -30,
|
|
-29, -28, -27, -26, -25, -24, -23, -22, -21, -20, -19, -19, -17, -16, -15, -14,
|
|
-14, -13, -12, -11, -11, -10, -9, -9, -8, -8, -7, -7, -6, -5, -4, -4,
|
|
-3, -3, -3, -2, -2, -2, -1, -1, 0, -1, 0, -1, 0, 0, 0, 0,
|
|
0,
|
|
2048,2048,2048,2048,2047,2047,2046,2045,2043,2042,2041,2039,2037,2035,2033,2031,
|
|
2028,2026,2024,2021,2018,2015,2012,2009,2005,2002,1999,1995,1991,1987,1982,1978,
|
|
1974,1969,1965,1960,1955,1951,1946,1940,1934,1929,1924,1918,1912,1906,1900,1895,
|
|
1888,1882,1875,1869,1862,1856,1849,1842,1835,1828,1821,1814,1806,1799,1791,1783,
|
|
1776,1768,1760,1753,1744,1737,1728,1720,1711,1703,1695,1686,1677,1668,1659,1651,
|
|
1641,1633,1623,1614,1605,1596,1587,1577,1567,1559,1549,1539,1529,1520,1510,1499,
|
|
1490,1480,1470,1460,1450,1440,1430,1420,1408,1398,1389,1378,1367,1357,1346,1336,
|
|
1325,1315,1304,1293,1282,1272,1261,1250,1239,1229,1218,1207,1196,1185,1174,1163,
|
|
1152,1141,1130,1119,1108,1097,1086,1075,1063,1052,1042,1030,1019,1008, 997, 986,
|
|
974, 964, 952, 941, 930, 919, 908, 897, 886, 875, 864, 853, 842, 831, 820, 809,
|
|
798, 787, 776, 765, 754, 744, 733, 722, 711, 700, 690, 679, 668, 658, 647, 637,
|
|
626, 616, 605, 595, 584, 574, 564, 554, 543, 534, 524, 514, 503, 494, 483, 473,
|
|
464, 454, 444, 435, 425, 416, 407, 397, 387, 378, 370, 360, 351, 342, 333, 325,
|
|
315, 307, 298, 290, 281, 273, 265, 256, 248, 241, 233, 225, 216, 209, 201, 193,
|
|
186, 178, 171, 164, 157, 150, 143, 137, 129, 123, 117, 110, 103, 97, 91, 85,
|
|
79, 74, 68, 62, 56, 51, 46, 41, 35, 31, 27, 22, 17, 13, 8, 4,
|
|
0
|
|
};
|
|
|
|
static short const sinc [2048] =
|
|
{
|
|
39, -315, 666, 15642, 666, -315, 39, -38,
|
|
38, -302, 613, 15642, 718, -328, 41, -38,
|
|
36, -288, 561, 15641, 772, -342, 42, -38,
|
|
35, -275, 510, 15639, 826, -355, 44, -38,
|
|
33, -263, 459, 15636, 880, -369, 46, -38,
|
|
32, -250, 408, 15632, 935, -383, 47, -38,
|
|
31, -237, 358, 15628, 990, -396, 49, -38,
|
|
29, -224, 309, 15622, 1046, -410, 51, -38,
|
|
28, -212, 259, 15616, 1103, -425, 53, -38,
|
|
27, -200, 211, 15609, 1159, -439, 54, -38,
|
|
25, -188, 163, 15601, 1216, -453, 56, -38,
|
|
24, -175, 115, 15593, 1274, -467, 58, -38,
|
|
23, -164, 68, 15583, 1332, -482, 60, -38,
|
|
22, -152, 22, 15573, 1391, -496, 62, -37,
|
|
21, -140, -24, 15562, 1450, -511, 64, -37,
|
|
19, -128, -70, 15550, 1509, -526, 66, -37,
|
|
18, -117, -115, 15538, 1569, -540, 68, -37,
|
|
17, -106, -159, 15524, 1629, -555, 70, -37,
|
|
16, -94, -203, 15510, 1690, -570, 72, -36,
|
|
15, -83, -247, 15495, 1751, -585, 74, -36,
|
|
14, -72, -289, 15479, 1813, -600, 76, -36,
|
|
13, -62, -332, 15462, 1875, -616, 79, -36,
|
|
12, -51, -374, 15445, 1937, -631, 81, -35,
|
|
11, -40, -415, 15426, 2000, -646, 83, -35,
|
|
11, -30, -456, 15407, 2063, -662, 85, -35,
|
|
10, -20, -496, 15387, 2127, -677, 88, -34,
|
|
9, -9, -536, 15366, 2191, -693, 90, -34,
|
|
8, 1, -576, 15345, 2256, -708, 92, -34,
|
|
7, 10, -614, 15323, 2321, -724, 95, -33,
|
|
7, 20, -653, 15300, 2386, -740, 97, -33,
|
|
6, 30, -690, 15276, 2451, -755, 99, -33,
|
|
5, 39, -728, 15251, 2517, -771, 102, -32,
|
|
5, 49, -764, 15226, 2584, -787, 104, -32,
|
|
4, 58, -801, 15200, 2651, -803, 107, -32,
|
|
3, 67, -836, 15173, 2718, -819, 109, -31,
|
|
3, 76, -871, 15145, 2785, -835, 112, -31,
|
|
2, 85, -906, 15117, 2853, -851, 115, -30,
|
|
2, 93, -940, 15087, 2921, -867, 117, -30,
|
|
1, 102, -974, 15057, 2990, -883, 120, -29,
|
|
1, 110, -1007, 15027, 3059, -899, 122, -29,
|
|
0, 118, -1039, 14995, 3128, -915, 125, -29,
|
|
0, 127, -1071, 14963, 3198, -931, 128, -28,
|
|
-1, 135, -1103, 14930, 3268, -948, 131, -28,
|
|
-1, 142, -1134, 14896, 3338, -964, 133, -27,
|
|
-1, 150, -1164, 14862, 3409, -980, 136, -27,
|
|
-2, 158, -1194, 14827, 3480, -996, 139, -26,
|
|
-2, 165, -1224, 14791, 3551, -1013, 142, -26,
|
|
-3, 172, -1253, 14754, 3622, -1029, 144, -25,
|
|
-3, 179, -1281, 14717, 3694, -1045, 147, -25,
|
|
-3, 187, -1309, 14679, 3766, -1062, 150, -24,
|
|
-3, 193, -1337, 14640, 3839, -1078, 153, -24,
|
|
-4, 200, -1363, 14601, 3912, -1094, 156, -23,
|
|
-4, 207, -1390, 14561, 3985, -1110, 159, -23,
|
|
-4, 213, -1416, 14520, 4058, -1127, 162, -22,
|
|
-4, 220, -1441, 14479, 4131, -1143, 165, -22,
|
|
-4, 226, -1466, 14437, 4205, -1159, 168, -22,
|
|
-5, 232, -1490, 14394, 4279, -1175, 171, -21,
|
|
-5, 238, -1514, 14350, 4354, -1192, 174, -21,
|
|
-5, 244, -1537, 14306, 4428, -1208, 177, -20,
|
|
-5, 249, -1560, 14261, 4503, -1224, 180, -20,
|
|
-5, 255, -1583, 14216, 4578, -1240, 183, -19,
|
|
-5, 260, -1604, 14169, 4653, -1256, 186, -19,
|
|
-5, 265, -1626, 14123, 4729, -1272, 189, -18,
|
|
-5, 271, -1647, 14075, 4805, -1288, 192, -18,
|
|
-5, 276, -1667, 14027, 4881, -1304, 195, -17,
|
|
-6, 280, -1687, 13978, 4957, -1320, 198, -17,
|
|
-6, 285, -1706, 13929, 5033, -1336, 201, -16,
|
|
-6, 290, -1725, 13879, 5110, -1352, 204, -16,
|
|
-6, 294, -1744, 13829, 5186, -1368, 207, -15,
|
|
-6, 299, -1762, 13777, 5263, -1383, 210, -15,
|
|
-6, 303, -1779, 13726, 5340, -1399, 213, -14,
|
|
-6, 307, -1796, 13673, 5418, -1414, 216, -14,
|
|
-6, 311, -1813, 13620, 5495, -1430, 219, -13,
|
|
-5, 315, -1829, 13567, 5573, -1445, 222, -13,
|
|
-5, 319, -1844, 13512, 5651, -1461, 225, -13,
|
|
-5, 322, -1859, 13458, 5728, -1476, 229, -12,
|
|
-5, 326, -1874, 13402, 5806, -1491, 232, -12,
|
|
-5, 329, -1888, 13347, 5885, -1506, 235, -11,
|
|
-5, 332, -1902, 13290, 5963, -1521, 238, -11,
|
|
-5, 335, -1915, 13233, 6041, -1536, 241, -10,
|
|
-5, 338, -1928, 13176, 6120, -1551, 244, -10,
|
|
-5, 341, -1940, 13118, 6199, -1566, 247, -10,
|
|
-5, 344, -1952, 13059, 6277, -1580, 250, -9,
|
|
-5, 347, -1964, 13000, 6356, -1595, 253, -9,
|
|
-5, 349, -1975, 12940, 6435, -1609, 256, -8,
|
|
-4, 352, -1986, 12880, 6514, -1623, 259, -8,
|
|
-4, 354, -1996, 12819, 6594, -1637, 262, -8,
|
|
-4, 356, -2005, 12758, 6673, -1651, 265, -7,
|
|
-4, 358, -2015, 12696, 6752, -1665, 268, -7,
|
|
-4, 360, -2024, 12634, 6831, -1679, 271, -7,
|
|
-4, 362, -2032, 12572, 6911, -1693, 274, -6,
|
|
-4, 364, -2040, 12509, 6990, -1706, 277, -6,
|
|
-4, 366, -2048, 12445, 7070, -1719, 280, -6,
|
|
-3, 367, -2055, 12381, 7149, -1732, 283, -5,
|
|
-3, 369, -2062, 12316, 7229, -1745, 286, -5,
|
|
-3, 370, -2068, 12251, 7308, -1758, 289, -5,
|
|
-3, 371, -2074, 12186, 7388, -1771, 291, -4,
|
|
-3, 372, -2079, 12120, 7467, -1784, 294, -4,
|
|
-3, 373, -2084, 12054, 7547, -1796, 297, -4,
|
|
-3, 374, -2089, 11987, 7626, -1808, 300, -4,
|
|
-2, 375, -2094, 11920, 7706, -1820, 303, -3,
|
|
-2, 376, -2098, 11852, 7785, -1832, 305, -3,
|
|
-2, 376, -2101, 11785, 7865, -1844, 308, -3,
|
|
-2, 377, -2104, 11716, 7944, -1855, 311, -3,
|
|
-2, 377, -2107, 11647, 8024, -1866, 313, -2,
|
|
-2, 378, -2110, 11578, 8103, -1877, 316, -2,
|
|
-2, 378, -2112, 11509, 8182, -1888, 318, -2,
|
|
-1, 378, -2113, 11439, 8262, -1899, 321, -2,
|
|
-1, 378, -2115, 11369, 8341, -1909, 323, -2,
|
|
-1, 378, -2116, 11298, 8420, -1920, 326, -2,
|
|
-1, 378, -2116, 11227, 8499, -1930, 328, -1,
|
|
-1, 378, -2116, 11156, 8578, -1940, 331, -1,
|
|
-1, 378, -2116, 11084, 8656, -1949, 333, -1,
|
|
-1, 377, -2116, 11012, 8735, -1959, 335, -1,
|
|
-1, 377, -2115, 10940, 8814, -1968, 337, -1,
|
|
-1, 377, -2114, 10867, 8892, -1977, 340, -1,
|
|
-1, 376, -2112, 10795, 8971, -1985, 342, -1,
|
|
0, 375, -2111, 10721, 9049, -1994, 344, -1,
|
|
0, 375, -2108, 10648, 9127, -2002, 346, 0,
|
|
0, 374, -2106, 10574, 9205, -2010, 348, 0,
|
|
0, 373, -2103, 10500, 9283, -2018, 350, 0,
|
|
0, 372, -2100, 10426, 9360, -2025, 352, 0,
|
|
0, 371, -2097, 10351, 9438, -2032, 354, 0,
|
|
0, 370, -2093, 10276, 9515, -2039, 355, 0,
|
|
0, 369, -2089, 10201, 9592, -2046, 357, 0,
|
|
0, 367, -2084, 10126, 9669, -2052, 359, 0,
|
|
0, 366, -2080, 10050, 9745, -2058, 360, 0,
|
|
0, 365, -2075, 9974, 9822, -2064, 362, 0,
|
|
0, 363, -2070, 9898, 9898, -2070, 363, 0,
|
|
0, 362, -2064, 9822, 9974, -2075, 365, 0,
|
|
0, 360, -2058, 9745, 10050, -2080, 366, 0,
|
|
0, 359, -2052, 9669, 10126, -2084, 367, 0,
|
|
0, 357, -2046, 9592, 10201, -2089, 369, 0,
|
|
0, 355, -2039, 9515, 10276, -2093, 370, 0,
|
|
0, 354, -2032, 9438, 10351, -2097, 371, 0,
|
|
0, 352, -2025, 9360, 10426, -2100, 372, 0,
|
|
0, 350, -2018, 9283, 10500, -2103, 373, 0,
|
|
0, 348, -2010, 9205, 10574, -2106, 374, 0,
|
|
0, 346, -2002, 9127, 10648, -2108, 375, 0,
|
|
-1, 344, -1994, 9049, 10721, -2111, 375, 0,
|
|
-1, 342, -1985, 8971, 10795, -2112, 376, -1,
|
|
-1, 340, -1977, 8892, 10867, -2114, 377, -1,
|
|
-1, 337, -1968, 8814, 10940, -2115, 377, -1,
|
|
-1, 335, -1959, 8735, 11012, -2116, 377, -1,
|
|
-1, 333, -1949, 8656, 11084, -2116, 378, -1,
|
|
-1, 331, -1940, 8578, 11156, -2116, 378, -1,
|
|
-1, 328, -1930, 8499, 11227, -2116, 378, -1,
|
|
-2, 326, -1920, 8420, 11298, -2116, 378, -1,
|
|
-2, 323, -1909, 8341, 11369, -2115, 378, -1,
|
|
-2, 321, -1899, 8262, 11439, -2113, 378, -1,
|
|
-2, 318, -1888, 8182, 11509, -2112, 378, -2,
|
|
-2, 316, -1877, 8103, 11578, -2110, 378, -2,
|
|
-2, 313, -1866, 8024, 11647, -2107, 377, -2,
|
|
-3, 311, -1855, 7944, 11716, -2104, 377, -2,
|
|
-3, 308, -1844, 7865, 11785, -2101, 376, -2,
|
|
-3, 305, -1832, 7785, 11852, -2098, 376, -2,
|
|
-3, 303, -1820, 7706, 11920, -2094, 375, -2,
|
|
-4, 300, -1808, 7626, 11987, -2089, 374, -3,
|
|
-4, 297, -1796, 7547, 12054, -2084, 373, -3,
|
|
-4, 294, -1784, 7467, 12120, -2079, 372, -3,
|
|
-4, 291, -1771, 7388, 12186, -2074, 371, -3,
|
|
-5, 289, -1758, 7308, 12251, -2068, 370, -3,
|
|
-5, 286, -1745, 7229, 12316, -2062, 369, -3,
|
|
-5, 283, -1732, 7149, 12381, -2055, 367, -3,
|
|
-6, 280, -1719, 7070, 12445, -2048, 366, -4,
|
|
-6, 277, -1706, 6990, 12509, -2040, 364, -4,
|
|
-6, 274, -1693, 6911, 12572, -2032, 362, -4,
|
|
-7, 271, -1679, 6831, 12634, -2024, 360, -4,
|
|
-7, 268, -1665, 6752, 12696, -2015, 358, -4,
|
|
-7, 265, -1651, 6673, 12758, -2005, 356, -4,
|
|
-8, 262, -1637, 6594, 12819, -1996, 354, -4,
|
|
-8, 259, -1623, 6514, 12880, -1986, 352, -4,
|
|
-8, 256, -1609, 6435, 12940, -1975, 349, -5,
|
|
-9, 253, -1595, 6356, 13000, -1964, 347, -5,
|
|
-9, 250, -1580, 6277, 13059, -1952, 344, -5,
|
|
-10, 247, -1566, 6199, 13118, -1940, 341, -5,
|
|
-10, 244, -1551, 6120, 13176, -1928, 338, -5,
|
|
-10, 241, -1536, 6041, 13233, -1915, 335, -5,
|
|
-11, 238, -1521, 5963, 13290, -1902, 332, -5,
|
|
-11, 235, -1506, 5885, 13347, -1888, 329, -5,
|
|
-12, 232, -1491, 5806, 13402, -1874, 326, -5,
|
|
-12, 229, -1476, 5728, 13458, -1859, 322, -5,
|
|
-13, 225, -1461, 5651, 13512, -1844, 319, -5,
|
|
-13, 222, -1445, 5573, 13567, -1829, 315, -5,
|
|
-13, 219, -1430, 5495, 13620, -1813, 311, -6,
|
|
-14, 216, -1414, 5418, 13673, -1796, 307, -6,
|
|
-14, 213, -1399, 5340, 13726, -1779, 303, -6,
|
|
-15, 210, -1383, 5263, 13777, -1762, 299, -6,
|
|
-15, 207, -1368, 5186, 13829, -1744, 294, -6,
|
|
-16, 204, -1352, 5110, 13879, -1725, 290, -6,
|
|
-16, 201, -1336, 5033, 13929, -1706, 285, -6,
|
|
-17, 198, -1320, 4957, 13978, -1687, 280, -6,
|
|
-17, 195, -1304, 4881, 14027, -1667, 276, -5,
|
|
-18, 192, -1288, 4805, 14075, -1647, 271, -5,
|
|
-18, 189, -1272, 4729, 14123, -1626, 265, -5,
|
|
-19, 186, -1256, 4653, 14169, -1604, 260, -5,
|
|
-19, 183, -1240, 4578, 14216, -1583, 255, -5,
|
|
-20, 180, -1224, 4503, 14261, -1560, 249, -5,
|
|
-20, 177, -1208, 4428, 14306, -1537, 244, -5,
|
|
-21, 174, -1192, 4354, 14350, -1514, 238, -5,
|
|
-21, 171, -1175, 4279, 14394, -1490, 232, -5,
|
|
-22, 168, -1159, 4205, 14437, -1466, 226, -4,
|
|
-22, 165, -1143, 4131, 14479, -1441, 220, -4,
|
|
-22, 162, -1127, 4058, 14520, -1416, 213, -4,
|
|
-23, 159, -1110, 3985, 14561, -1390, 207, -4,
|
|
-23, 156, -1094, 3912, 14601, -1363, 200, -4,
|
|
-24, 153, -1078, 3839, 14640, -1337, 193, -3,
|
|
-24, 150, -1062, 3766, 14679, -1309, 187, -3,
|
|
-25, 147, -1045, 3694, 14717, -1281, 179, -3,
|
|
-25, 144, -1029, 3622, 14754, -1253, 172, -3,
|
|
-26, 142, -1013, 3551, 14791, -1224, 165, -2,
|
|
-26, 139, -996, 3480, 14827, -1194, 158, -2,
|
|
-27, 136, -980, 3409, 14862, -1164, 150, -1,
|
|
-27, 133, -964, 3338, 14896, -1134, 142, -1,
|
|
-28, 131, -948, 3268, 14930, -1103, 135, -1,
|
|
-28, 128, -931, 3198, 14963, -1071, 127, 0,
|
|
-29, 125, -915, 3128, 14995, -1039, 118, 0,
|
|
-29, 122, -899, 3059, 15027, -1007, 110, 1,
|
|
-29, 120, -883, 2990, 15057, -974, 102, 1,
|
|
-30, 117, -867, 2921, 15087, -940, 93, 2,
|
|
-30, 115, -851, 2853, 15117, -906, 85, 2,
|
|
-31, 112, -835, 2785, 15145, -871, 76, 3,
|
|
-31, 109, -819, 2718, 15173, -836, 67, 3,
|
|
-32, 107, -803, 2651, 15200, -801, 58, 4,
|
|
-32, 104, -787, 2584, 15226, -764, 49, 5,
|
|
-32, 102, -771, 2517, 15251, -728, 39, 5,
|
|
-33, 99, -755, 2451, 15276, -690, 30, 6,
|
|
-33, 97, -740, 2386, 15300, -653, 20, 7,
|
|
-33, 95, -724, 2321, 15323, -614, 10, 7,
|
|
-34, 92, -708, 2256, 15345, -576, 1, 8,
|
|
-34, 90, -693, 2191, 15366, -536, -9, 9,
|
|
-34, 88, -677, 2127, 15387, -496, -20, 10,
|
|
-35, 85, -662, 2063, 15407, -456, -30, 11,
|
|
-35, 83, -646, 2000, 15426, -415, -40, 11,
|
|
-35, 81, -631, 1937, 15445, -374, -51, 12,
|
|
-36, 79, -616, 1875, 15462, -332, -62, 13,
|
|
-36, 76, -600, 1813, 15479, -289, -72, 14,
|
|
-36, 74, -585, 1751, 15495, -247, -83, 15,
|
|
-36, 72, -570, 1690, 15510, -203, -94, 16,
|
|
-37, 70, -555, 1629, 15524, -159, -106, 17,
|
|
-37, 68, -540, 1569, 15538, -115, -117, 18,
|
|
-37, 66, -526, 1509, 15550, -70, -128, 19,
|
|
-37, 64, -511, 1450, 15562, -24, -140, 21,
|
|
-37, 62, -496, 1391, 15573, 22, -152, 22,
|
|
-38, 60, -482, 1332, 15583, 68, -164, 23,
|
|
-38, 58, -467, 1274, 15593, 115, -175, 24,
|
|
-38, 56, -453, 1216, 15601, 163, -188, 25,
|
|
-38, 54, -439, 1159, 15609, 211, -200, 27,
|
|
-38, 53, -425, 1103, 15616, 259, -212, 28,
|
|
-38, 51, -410, 1046, 15622, 309, -224, 29,
|
|
-38, 49, -396, 990, 15628, 358, -237, 31,
|
|
-38, 47, -383, 935, 15632, 408, -250, 32,
|
|
-38, 46, -369, 880, 15636, 459, -263, 33,
|
|
-38, 44, -355, 826, 15639, 510, -275, 35,
|
|
-38, 42, -342, 772, 15641, 561, -288, 36,
|
|
-38, 41, -328, 718, 15642, 613, -302, 38,
|
|
};
|
|
|
|
inline int SPC_DSP::interpolate( voice_t const* v )
|
|
{
|
|
int out;
|
|
int const* in = &v->buf [(v->interp_pos >> 12) + v->buf_pos];
|
|
|
|
switch (Settings.InterpolationMethod)
|
|
{
|
|
case 0: // raw
|
|
{
|
|
out = in [0] & ~1;
|
|
break;
|
|
}
|
|
|
|
case 1: // linear interpolation
|
|
{
|
|
int fract = v->interp_pos & 0xFFF;
|
|
out = (0x1000 - fract) * in [0];
|
|
out += fract * in [1];
|
|
out >>= 12;
|
|
|
|
CLAMP16( out );
|
|
break;
|
|
}
|
|
|
|
case 3: // cubic filter
|
|
{
|
|
// Make pointers into cubic based on fractional position between samples
|
|
int offset = v->interp_pos >> 4 & 0xFF;
|
|
short const* fwd = cubic + offset;
|
|
short const* rev = cubic + 256 - offset; // mirror left half of cubic
|
|
|
|
out = fwd [ 0] * in [0];
|
|
out += fwd [257] * in [1];
|
|
out += rev [257] * in [2];
|
|
out += rev [ 0] * in [3];
|
|
out >>= 11;
|
|
|
|
CLAMP16( out );
|
|
break;
|
|
}
|
|
|
|
case 4: // sinc filter
|
|
{
|
|
int offset = (v->interp_pos & 0xFF0) >> 1;
|
|
short const* filt = sinc + offset;
|
|
|
|
out = filt [0] * in [0];
|
|
out += filt [1] * in [1];
|
|
out += filt [2] * in [2];
|
|
out += filt [3] * in [3];
|
|
out += filt [4] * in [4];
|
|
out += filt [5] * in [5];
|
|
out += filt [6] * in [6];
|
|
out += filt [7] * in [7];
|
|
out >>= 14;
|
|
|
|
CLAMP16( out );
|
|
break;
|
|
}
|
|
|
|
default:
|
|
case 2: // Original gaussian filter
|
|
{
|
|
// Make pointers into gaussian based on fractional position between samples
|
|
int offset = v->interp_pos >> 4 & 0xFF;
|
|
short const* fwd = gauss + 255 - offset;
|
|
short const* rev = gauss + offset; // mirror left half of gaussian
|
|
|
|
out = (fwd [ 0] * in [0]) >> 11;
|
|
out += (fwd [256] * in [1]) >> 11;
|
|
out += (rev [256] * in [2]) >> 11;
|
|
out = (int16_t) out;
|
|
out += (rev [ 0] * in [3]) >> 11;
|
|
|
|
CLAMP16( out );
|
|
out &= ~1;
|
|
|
|
break;
|
|
}
|
|
|
|
}
|
|
|
|
return out;
|
|
}
|
|
|
|
//// Counters
|
|
|
|
int const simple_counter_range = 2048 * 5 * 3; // 30720
|
|
|
|
static unsigned const counter_rates [32] =
|
|
{
|
|
simple_counter_range + 1, // never fires
|
|
2048, 1536,
|
|
1280, 1024, 768,
|
|
640, 512, 384,
|
|
320, 256, 192,
|
|
160, 128, 96,
|
|
80, 64, 48,
|
|
40, 32, 24,
|
|
20, 16, 12,
|
|
10, 8, 6,
|
|
5, 4, 3,
|
|
2,
|
|
1
|
|
};
|
|
|
|
static unsigned const counter_offsets [32] =
|
|
{
|
|
1, 0, 1040,
|
|
536, 0, 1040,
|
|
536, 0, 1040,
|
|
536, 0, 1040,
|
|
536, 0, 1040,
|
|
536, 0, 1040,
|
|
536, 0, 1040,
|
|
536, 0, 1040,
|
|
536, 0, 1040,
|
|
536, 0, 1040,
|
|
0,
|
|
0
|
|
};
|
|
|
|
inline void SPC_DSP::init_counter()
|
|
{
|
|
m.counter = 0;
|
|
}
|
|
|
|
inline void SPC_DSP::run_counters()
|
|
{
|
|
if ( --m.counter < 0 )
|
|
m.counter = simple_counter_range - 1;
|
|
}
|
|
|
|
inline unsigned SPC_DSP::read_counter( int rate )
|
|
{
|
|
return ((unsigned) m.counter + counter_offsets [rate]) % counter_rates [rate];
|
|
}
|
|
|
|
|
|
//// Envelope
|
|
|
|
inline void SPC_DSP::run_envelope( voice_t* const v )
|
|
{
|
|
int env = v->env;
|
|
if ( v->env_mode == env_release ) // 60%
|
|
{
|
|
if ( (env -= 0x8) < 0 )
|
|
env = 0;
|
|
v->env = env;
|
|
}
|
|
else
|
|
{
|
|
int rate;
|
|
int env_data = VREG(v->regs,adsr1);
|
|
if ( m.t_adsr0 & 0x80 ) // 99% ADSR
|
|
{
|
|
if ( v->env_mode >= env_decay ) // 99%
|
|
{
|
|
env--;
|
|
env -= env >> 8;
|
|
rate = env_data & 0x1F;
|
|
if ( v->env_mode == env_decay ) // 1%
|
|
rate = (m.t_adsr0 >> 3 & 0x0E) + 0x10;
|
|
}
|
|
else // env_attack
|
|
{
|
|
rate = (m.t_adsr0 & 0x0F) * 2 + 1;
|
|
env += rate < 31 ? 0x20 : 0x400;
|
|
}
|
|
}
|
|
else // GAIN
|
|
{
|
|
int mode;
|
|
env_data = VREG(v->regs,gain);
|
|
mode = env_data >> 5;
|
|
if ( mode < 4 ) // direct
|
|
{
|
|
env = env_data * 0x10;
|
|
rate = 31;
|
|
}
|
|
else
|
|
{
|
|
rate = env_data & 0x1F;
|
|
if ( mode == 4 ) // 4: linear decrease
|
|
{
|
|
env -= 0x20;
|
|
}
|
|
else if ( mode < 6 ) // 5: exponential decrease
|
|
{
|
|
env--;
|
|
env -= env >> 8;
|
|
}
|
|
else // 6,7: linear increase
|
|
{
|
|
env += 0x20;
|
|
if ( mode > 6 && (unsigned) v->hidden_env >= 0x600 )
|
|
env += 0x8 - 0x20; // 7: two-slope linear increase
|
|
}
|
|
}
|
|
}
|
|
|
|
// Sustain level
|
|
if ( (env >> 8) == (env_data >> 5) && v->env_mode == env_decay )
|
|
v->env_mode = env_sustain;
|
|
|
|
v->hidden_env = env;
|
|
|
|
// unsigned cast because linear decrease going negative also triggers this
|
|
if ( (unsigned) env > 0x7FF )
|
|
{
|
|
env = (env < 0 ? 0 : 0x7FF);
|
|
if ( v->env_mode == env_attack )
|
|
v->env_mode = env_decay;
|
|
}
|
|
|
|
if ( !read_counter( rate ) )
|
|
v->env = env; // nothing else is controlled by the counter
|
|
}
|
|
}
|
|
|
|
|
|
//// BRR Decoding
|
|
|
|
inline void SPC_DSP::decode_brr( voice_t* v )
|
|
{
|
|
// Arrange the four input nybbles in 0xABCD order for easy decoding
|
|
int nybbles = m.t_brr_byte * 0x100 + m.ram [(v->brr_addr + v->brr_offset + 1) & 0xFFFF];
|
|
|
|
int const header = m.t_brr_header;
|
|
|
|
// Write to next four samples in circular buffer
|
|
int* pos = &v->buf [v->buf_pos];
|
|
int* end;
|
|
if ( (v->buf_pos += 4) >= brr_buf_size )
|
|
v->buf_pos = 0;
|
|
|
|
// Decode four samples
|
|
for ( end = pos + 4; pos < end; pos++, nybbles <<= 4 )
|
|
{
|
|
// Extract nybble and sign-extend
|
|
int s = (int16_t) nybbles >> 12;
|
|
|
|
// Shift sample based on header
|
|
int const shift = header >> 4;
|
|
if (shift <= 12)
|
|
s = (s << shift) >> 1;
|
|
else
|
|
s &= 0x7ff;
|
|
|
|
// Apply IIR filter (8 is the most commonly used)
|
|
int const filter = header & 0x0C;
|
|
int const p1 = pos [brr_buf_size - 1];
|
|
int const p2 = pos [brr_buf_size - 2] >> 1;
|
|
if ( filter >= 8 )
|
|
{
|
|
s += p1;
|
|
s -= p2;
|
|
if ( filter == 8 ) // s += p1 * 0.953125 - p2 * 0.46875
|
|
{
|
|
s += p2 >> 4;
|
|
s += (p1 * -3) >> 6;
|
|
}
|
|
else // s += p1 * 0.8984375 - p2 * 0.40625
|
|
{
|
|
s += (p1 * -13) >> 7;
|
|
s += (p2 * 3) >> 4;
|
|
}
|
|
}
|
|
else if ( filter ) // s += p1 * 0.46875
|
|
{
|
|
s += p1 >> 1;
|
|
s += (-p1) >> 5;
|
|
}
|
|
|
|
// Adjust and write sample
|
|
CLAMP16( s );
|
|
s = (int16_t) (s * 2);
|
|
pos [brr_buf_size] = pos [0] = s; // second copy simplifies wrap-around
|
|
}
|
|
}
|
|
|
|
|
|
//// Misc
|
|
|
|
#define MISC_CLOCK( n ) inline void SPC_DSP::misc_##n()
|
|
|
|
MISC_CLOCK( 27 )
|
|
{
|
|
m.t_pmon = REG(pmon) & 0xFE; // voice 0 doesn't support PMON
|
|
}
|
|
MISC_CLOCK( 28 )
|
|
{
|
|
m.t_non = REG(non);
|
|
m.t_eon = REG(eon);
|
|
m.t_dir = REG(dir);
|
|
}
|
|
MISC_CLOCK( 29 )
|
|
{
|
|
if ( (m.every_other_sample ^= 1) != 0 )
|
|
m.new_kon &= ~m.kon; // clears KON 63 clocks after it was last read
|
|
}
|
|
MISC_CLOCK( 30 )
|
|
{
|
|
if ( m.every_other_sample )
|
|
{
|
|
m.kon = m.new_kon;
|
|
m.t_koff = REG(koff) | m.mute_mask;
|
|
}
|
|
|
|
run_counters();
|
|
|
|
// Noise
|
|
if ( !read_counter( REG(flg) & 0x1F ) )
|
|
{
|
|
int feedback = (m.noise << 13) ^ (m.noise << 14);
|
|
m.noise = (feedback & 0x4000) ^ (m.noise >> 1);
|
|
}
|
|
}
|
|
|
|
|
|
//// Voices
|
|
|
|
#define VOICE_CLOCK( n ) void SPC_DSP::voice_##n( voice_t* const v )
|
|
|
|
inline VOICE_CLOCK( V1 )
|
|
{
|
|
m.t_dir_addr = (m.t_dir * 0x100 + m.t_srcn * 4) & 0xffff;
|
|
m.t_srcn = VREG(v->regs,srcn);
|
|
}
|
|
inline VOICE_CLOCK( V2 )
|
|
{
|
|
// Read sample pointer (ignored if not needed)
|
|
uint8_t const* entry = &m.ram [m.t_dir_addr];
|
|
if ( !v->kon_delay )
|
|
entry += 2;
|
|
m.t_brr_next_addr = GET_LE16A( entry );
|
|
|
|
m.t_adsr0 = VREG(v->regs,adsr0);
|
|
|
|
// Read pitch, spread over two clocks
|
|
m.t_pitch = VREG(v->regs,pitchl);
|
|
}
|
|
inline VOICE_CLOCK( V3a )
|
|
{
|
|
m.t_pitch += (VREG(v->regs,pitchh) & 0x3F) << 8;
|
|
}
|
|
inline VOICE_CLOCK( V3b )
|
|
{
|
|
// Read BRR header and byte
|
|
m.t_brr_byte = m.ram [(v->brr_addr + v->brr_offset) & 0xFFFF];
|
|
m.t_brr_header = m.ram [v->brr_addr]; // brr_addr doesn't need masking
|
|
}
|
|
|
|
inline VOICE_CLOCK( V3c )
|
|
{
|
|
// Pitch modulation using previous voice's output
|
|
if ( m.t_pmon & v->vbit )
|
|
m.t_pitch += ((m.t_output >> 5) * m.t_pitch) >> 10;
|
|
|
|
if ( v->kon_delay )
|
|
{
|
|
// Get ready to start BRR decoding on next sample
|
|
if ( v->kon_delay == 5 )
|
|
{
|
|
v->brr_addr = m.t_brr_next_addr;
|
|
v->brr_offset = 1;
|
|
v->buf_pos = 0;
|
|
m.t_brr_header = 0; // header is ignored on this sample
|
|
m.kon_check = true;
|
|
|
|
if (take_spc_snapshot)
|
|
{
|
|
take_spc_snapshot = 0;
|
|
if (spc_snapshot_callback)
|
|
spc_snapshot_callback();
|
|
}
|
|
}
|
|
|
|
// Envelope is never run during KON
|
|
v->env = 0;
|
|
v->hidden_env = 0;
|
|
|
|
// Disable BRR decoding until last three samples
|
|
v->interp_pos = 0;
|
|
if ( --v->kon_delay & 3 )
|
|
v->interp_pos = 0x4000;
|
|
|
|
// Pitch is never added during KON
|
|
m.t_pitch = 0;
|
|
}
|
|
|
|
// Gaussian interpolation
|
|
{
|
|
int output = interpolate( v );
|
|
|
|
// Noise
|
|
if ( m.t_non & v->vbit )
|
|
output = (int16_t) (m.noise * 2);
|
|
|
|
// Apply envelope
|
|
m.t_output = (output * v->env) >> 11 & ~1;
|
|
v->t_envx_out = (uint8_t) (v->env >> 4);
|
|
}
|
|
|
|
// Immediate silence due to end of sample or soft reset
|
|
if ( REG(flg) & 0x80 || (m.t_brr_header & 3) == 1 )
|
|
{
|
|
v->env_mode = env_release;
|
|
v->env = 0;
|
|
}
|
|
|
|
if ( m.every_other_sample )
|
|
{
|
|
// KOFF
|
|
if ( m.t_koff & v->vbit )
|
|
v->env_mode = env_release;
|
|
|
|
// KON
|
|
if ( m.kon & v->vbit )
|
|
{
|
|
v->kon_delay = 5;
|
|
v->env_mode = env_attack;
|
|
}
|
|
}
|
|
|
|
// Run envelope for next sample
|
|
if ( !v->kon_delay )
|
|
run_envelope( v );
|
|
}
|
|
|
|
inline void SPC_DSP::voice_output( voice_t const* v, int ch )
|
|
{
|
|
// Apply left/right volume
|
|
int amp = (m.t_output * (int8_t) VREG(v->regs,voll + ch)) >> 7;
|
|
amp *= ((stereo_switch & (1 << (v->voice_number + ch * voice_count))) ? 1 : 0);
|
|
|
|
// Add to output total
|
|
m.t_main_out [ch] += amp;
|
|
CLAMP16( m.t_main_out [ch] );
|
|
|
|
// Optionally add to echo total
|
|
if ( m.t_eon & v->vbit )
|
|
{
|
|
m.t_echo_out [ch] += amp;
|
|
CLAMP16( m.t_echo_out [ch] );
|
|
}
|
|
}
|
|
|
|
inline VOICE_CLOCK( V4 )
|
|
{
|
|
// Decode BRR
|
|
m.t_looped = 0;
|
|
if ( v->interp_pos >= 0x4000 )
|
|
{
|
|
decode_brr( v );
|
|
|
|
if ( (v->brr_offset += 2) >= brr_block_size )
|
|
{
|
|
// Start decoding next BRR block
|
|
assert( v->brr_offset == brr_block_size );
|
|
v->brr_addr = (v->brr_addr + brr_block_size) & 0xFFFF;
|
|
if ( m.t_brr_header & 1 )
|
|
{
|
|
v->brr_addr = m.t_brr_next_addr;
|
|
m.t_looped = v->vbit;
|
|
}
|
|
v->brr_offset = 1;
|
|
}
|
|
}
|
|
|
|
// Apply pitch
|
|
v->interp_pos = (v->interp_pos & 0x3FFF) + m.t_pitch;
|
|
|
|
// Keep from getting too far ahead (when using pitch modulation)
|
|
if ( v->interp_pos > 0x7FFF )
|
|
v->interp_pos = 0x7FFF;
|
|
|
|
// Output left
|
|
voice_output( v, 0 );
|
|
}
|
|
inline VOICE_CLOCK( V5 )
|
|
{
|
|
// Output right
|
|
voice_output( v, 1 );
|
|
|
|
// ENDX, OUTX, and ENVX won't update if you wrote to them 1-2 clocks earlier
|
|
int endx_buf = REG(endx) | m.t_looped;
|
|
|
|
// Clear bit in ENDX if KON just began
|
|
if ( v->kon_delay == 5 )
|
|
endx_buf &= ~v->vbit;
|
|
m.endx_buf = (uint8_t) endx_buf;
|
|
}
|
|
inline VOICE_CLOCK( V6 )
|
|
{
|
|
(void) v; // avoid compiler warning about unused v
|
|
m.outx_buf = (uint8_t) (m.t_output >> 8);
|
|
}
|
|
inline VOICE_CLOCK( V7 )
|
|
{
|
|
// Update ENDX
|
|
REG(endx) = m.endx_buf;
|
|
XREG(endx) = m.endx_buf;
|
|
|
|
m.envx_buf = v->t_envx_out;
|
|
}
|
|
inline VOICE_CLOCK( V8 )
|
|
{
|
|
// Update OUTX
|
|
VREG(v->regs,outx) = m.outx_buf;
|
|
XVREG(v->regs,outx) = m.outx_buf;
|
|
}
|
|
inline VOICE_CLOCK( V9 )
|
|
{
|
|
// Update ENVX
|
|
VREG(v->regs,envx) = m.envx_buf;
|
|
XVREG(v->regs,envx) = m.envx_buf;
|
|
}
|
|
|
|
// Most voices do all these in one clock, so make a handy composite
|
|
inline VOICE_CLOCK( V3 )
|
|
{
|
|
voice_V3a( v );
|
|
voice_V3b( v );
|
|
voice_V3c( v );
|
|
}
|
|
|
|
// Common combinations of voice steps on different voices. This greatly reduces
|
|
// code size and allows everything to be inlined in these functions.
|
|
VOICE_CLOCK(V7_V4_V1) { voice_V7(v); voice_V1(v+3); voice_V4(v+1); }
|
|
VOICE_CLOCK(V8_V5_V2) { voice_V8(v); voice_V5(v+1); voice_V2(v+2); }
|
|
VOICE_CLOCK(V9_V6_V3) { voice_V9(v); voice_V6(v+1); voice_V3(v+2); }
|
|
|
|
|
|
//// Echo
|
|
|
|
// Current echo buffer pointer for left/right channel
|
|
#define ECHO_PTR( ch ) ((Settings.SeparateEchoBuffer) ? (&m.separate_echo_buffer [m.t_echo_ptr + ch * 2]) : (&m.ram [m.t_echo_ptr + ch * 2]))
|
|
|
|
// Sample in echo history buffer, where 0 is the oldest
|
|
#define ECHO_FIR( i ) (m.echo_hist_pos [i])
|
|
|
|
// Calculate FIR point for left/right channel
|
|
#define CALC_FIR( i, ch ) ((ECHO_FIR( i + 1 ) [ch] * (int8_t) REG(fir + i * 0x10)) >> 6)
|
|
|
|
#define ECHO_CLOCK( n ) inline void SPC_DSP::echo_##n()
|
|
|
|
inline void SPC_DSP::echo_read( int ch )
|
|
{
|
|
int s = GET_LE16SA( ECHO_PTR( ch ) );
|
|
// second copy simplifies wrap-around handling
|
|
ECHO_FIR( 0 ) [ch] = ECHO_FIR( 8 ) [ch] = s >> 1;
|
|
}
|
|
|
|
ECHO_CLOCK( 22 )
|
|
{
|
|
// History
|
|
if ( ++m.echo_hist_pos >= &m.echo_hist [echo_hist_size] )
|
|
m.echo_hist_pos = m.echo_hist;
|
|
|
|
m.t_echo_ptr = (m.t_esa * 0x100 + m.echo_offset) & 0xFFFF;
|
|
echo_read( 0 );
|
|
|
|
// FIR (using l and r temporaries below helps compiler optimize)
|
|
int l = CALC_FIR( 0, 0 );
|
|
int r = CALC_FIR( 0, 1 );
|
|
|
|
m.t_echo_in [0] = l;
|
|
m.t_echo_in [1] = r;
|
|
}
|
|
ECHO_CLOCK( 23 )
|
|
{
|
|
int l = CALC_FIR( 1, 0 ) + CALC_FIR( 2, 0 );
|
|
int r = CALC_FIR( 1, 1 ) + CALC_FIR( 2, 1 );
|
|
|
|
m.t_echo_in [0] += l;
|
|
m.t_echo_in [1] += r;
|
|
|
|
echo_read( 1 );
|
|
}
|
|
ECHO_CLOCK( 24 )
|
|
{
|
|
int l = CALC_FIR( 3, 0 ) + CALC_FIR( 4, 0 ) + CALC_FIR( 5, 0 );
|
|
int r = CALC_FIR( 3, 1 ) + CALC_FIR( 4, 1 ) + CALC_FIR( 5, 1 );
|
|
|
|
m.t_echo_in [0] += l;
|
|
m.t_echo_in [1] += r;
|
|
}
|
|
ECHO_CLOCK( 25 )
|
|
{
|
|
int l = m.t_echo_in [0] + CALC_FIR( 6, 0 );
|
|
int r = m.t_echo_in [1] + CALC_FIR( 6, 1 );
|
|
|
|
l = (int16_t) l;
|
|
r = (int16_t) r;
|
|
|
|
l += (int16_t) CALC_FIR( 7, 0 );
|
|
r += (int16_t) CALC_FIR( 7, 1 );
|
|
|
|
CLAMP16( l );
|
|
CLAMP16( r );
|
|
|
|
m.t_echo_in [0] = l & ~1;
|
|
m.t_echo_in [1] = r & ~1;
|
|
}
|
|
inline int SPC_DSP::echo_output( int ch )
|
|
{
|
|
int out = (int16_t) ((m.t_main_out [ch] * (int8_t) REG(mvoll + ch * 0x10)) >> 7) +
|
|
(int16_t) ((m.t_echo_in [ch] * (int8_t) REG(evoll + ch * 0x10)) >> 7);
|
|
CLAMP16( out );
|
|
return out;
|
|
}
|
|
ECHO_CLOCK( 26 )
|
|
{
|
|
// Left output volumes
|
|
// (save sample for next clock so we can output both together)
|
|
m.t_main_out [0] = echo_output( 0 );
|
|
|
|
// Echo feedback
|
|
int l = m.t_echo_out [0] + (int16_t) ((m.t_echo_in [0] * (int8_t) REG(efb)) >> 7);
|
|
int r = m.t_echo_out [1] + (int16_t) ((m.t_echo_in [1] * (int8_t) REG(efb)) >> 7);
|
|
|
|
CLAMP16( l );
|
|
CLAMP16( r );
|
|
|
|
m.t_echo_out [0] = l & ~1;
|
|
m.t_echo_out [1] = r & ~1;
|
|
}
|
|
ECHO_CLOCK( 27 )
|
|
{
|
|
// Output
|
|
int l = m.t_main_out [0];
|
|
int r = echo_output( 1 );
|
|
m.t_main_out [0] = 0;
|
|
m.t_main_out [1] = 0;
|
|
|
|
// TODO: global muting isn't this simple (turns DAC on and off
|
|
// or something, causing small ~37-sample pulse when first muted)
|
|
if ( REG(flg) & 0x40 )
|
|
{
|
|
l = 0;
|
|
r = 0;
|
|
}
|
|
|
|
// Output sample to DAC
|
|
#ifdef SPC_DSP_OUT_HOOK
|
|
SPC_DSP_OUT_HOOK( l, r );
|
|
#else
|
|
sample_t* out = m.out;
|
|
WRITE_SAMPLES( l, r, out );
|
|
m.out = out;
|
|
#endif
|
|
}
|
|
ECHO_CLOCK( 28 )
|
|
{
|
|
m.t_echo_enabled = REG(flg);
|
|
}
|
|
inline void SPC_DSP::echo_write( int ch )
|
|
{
|
|
if ( !(m.t_echo_enabled & 0x20) )
|
|
SET_LE16A( ECHO_PTR( ch ), m.t_echo_out [ch] );
|
|
|
|
m.t_echo_out [ch] = 0;
|
|
}
|
|
ECHO_CLOCK( 29 )
|
|
{
|
|
m.t_esa = REG(esa);
|
|
|
|
if ( !m.echo_offset )
|
|
m.echo_length = (REG(edl) & 0x0F) * 0x800;
|
|
|
|
m.echo_offset += 4;
|
|
if ( m.echo_offset >= m.echo_length )
|
|
m.echo_offset = 0;
|
|
|
|
// Write left echo
|
|
echo_write( 0 );
|
|
|
|
m.t_echo_enabled = REG(flg);
|
|
}
|
|
ECHO_CLOCK( 30 )
|
|
{
|
|
// Write right echo
|
|
echo_write( 1 );
|
|
}
|
|
|
|
|
|
//// Timing
|
|
|
|
// Execute clock for a particular voice
|
|
#define V( clock, voice ) voice_##clock( &m.voices [voice] );
|
|
|
|
/* The most common sequence of clocks uses composite operations
|
|
for efficiency. For example, the following are equivalent to the
|
|
individual steps on the right:
|
|
|
|
V(V7_V4_V1,2) -> V(V7,2) V(V4,3) V(V1,5)
|
|
V(V8_V5_V2,2) -> V(V8,2) V(V5,3) V(V2,4)
|
|
V(V9_V6_V3,2) -> V(V9,2) V(V6,3) V(V3,4) */
|
|
|
|
// Voice 0 1 2 3 4 5 6 7
|
|
#define GEN_DSP_TIMING \
|
|
PHASE( 0) V(V5,0)V(V2,1)\
|
|
PHASE( 1) V(V6,0)V(V3,1)\
|
|
PHASE( 2) V(V7_V4_V1,0)\
|
|
PHASE( 3) V(V8_V5_V2,0)\
|
|
PHASE( 4) V(V9_V6_V3,0)\
|
|
PHASE( 5) V(V7_V4_V1,1)\
|
|
PHASE( 6) V(V8_V5_V2,1)\
|
|
PHASE( 7) V(V9_V6_V3,1)\
|
|
PHASE( 8) V(V7_V4_V1,2)\
|
|
PHASE( 9) V(V8_V5_V2,2)\
|
|
PHASE(10) V(V9_V6_V3,2)\
|
|
PHASE(11) V(V7_V4_V1,3)\
|
|
PHASE(12) V(V8_V5_V2,3)\
|
|
PHASE(13) V(V9_V6_V3,3)\
|
|
PHASE(14) V(V7_V4_V1,4)\
|
|
PHASE(15) V(V8_V5_V2,4)\
|
|
PHASE(16) V(V9_V6_V3,4)\
|
|
PHASE(17) V(V1,0) V(V7,5)V(V4,6)\
|
|
PHASE(18) V(V8_V5_V2,5)\
|
|
PHASE(19) V(V9_V6_V3,5)\
|
|
PHASE(20) V(V1,1) V(V7,6)V(V4,7)\
|
|
PHASE(21) V(V8,6)V(V5,7) V(V2,0) /* t_brr_next_addr order dependency */\
|
|
PHASE(22) V(V3a,0) V(V9,6)V(V6,7) echo_22();\
|
|
PHASE(23) V(V7,7) echo_23();\
|
|
PHASE(24) V(V8,7) echo_24();\
|
|
PHASE(25) V(V3b,0) V(V9,7) echo_25();\
|
|
PHASE(26) echo_26();\
|
|
PHASE(27) misc_27(); echo_27();\
|
|
PHASE(28) misc_28(); echo_28();\
|
|
PHASE(29) misc_29(); echo_29();\
|
|
PHASE(30) misc_30();V(V3c,0) echo_30();\
|
|
PHASE(31) V(V4,0) V(V1,2)\
|
|
|
|
#if !SPC_DSP_CUSTOM_RUN
|
|
|
|
void SPC_DSP::run( int clocks_remain )
|
|
{
|
|
require( clocks_remain > 0 );
|
|
|
|
int const phase = m.phase;
|
|
m.phase = (phase + clocks_remain) & 31;
|
|
switch ( phase )
|
|
{
|
|
loop:
|
|
|
|
#define PHASE( n ) if ( n && !--clocks_remain ) break; /* Fall through */ case n:
|
|
GEN_DSP_TIMING
|
|
#undef PHASE
|
|
|
|
if ( --clocks_remain )
|
|
goto loop;
|
|
}
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
//// Setup
|
|
|
|
void SPC_DSP::init( void* ram_64k )
|
|
{
|
|
m.ram = (uint8_t*) ram_64k;
|
|
mute_voices( 0 );
|
|
disable_surround( false );
|
|
set_output( 0, 0 );
|
|
reset();
|
|
|
|
stereo_switch = 0xffff;
|
|
take_spc_snapshot = 0;
|
|
spc_snapshot_callback = 0;
|
|
|
|
#ifndef NDEBUG
|
|
// be sure this sign-extends
|
|
assert( (int16_t) 0x8000 == -0x8000 );
|
|
|
|
// be sure right shift preserves sign
|
|
assert( (-1 >> 1) == -1 );
|
|
|
|
// check clamp macro
|
|
int i;
|
|
i = +0x8000; CLAMP16( i ); assert( i == +0x7FFF );
|
|
i = -0x8001; CLAMP16( i ); assert( i == -0x8000 );
|
|
|
|
blargg_verify_byte_order();
|
|
#endif
|
|
}
|
|
|
|
void SPC_DSP::soft_reset_common()
|
|
{
|
|
require( m.ram ); // init() must have been called already
|
|
|
|
m.noise = 0x4000;
|
|
m.echo_hist_pos = m.echo_hist;
|
|
m.every_other_sample = 1;
|
|
m.echo_offset = 0;
|
|
m.phase = 0;
|
|
|
|
memset(m.separate_echo_buffer, 0, 0x10000);
|
|
|
|
init_counter();
|
|
|
|
for (int i = 0; i < voice_count; i++)
|
|
m.voices[i].voice_number = i;
|
|
}
|
|
|
|
void SPC_DSP::soft_reset()
|
|
{
|
|
REG(flg) = 0xE0;
|
|
soft_reset_common();
|
|
}
|
|
|
|
void SPC_DSP::load( uint8_t const regs [register_count] )
|
|
{
|
|
memcpy( m.external_regs, regs, sizeof m.regs );
|
|
memset( m.regs, 0, sizeof m.regs);
|
|
m.regs[r_flg] = 0xE0;
|
|
memset( &m.regs [register_count], 0, offsetof (state_t,ram) - register_count );
|
|
|
|
// Internal state
|
|
for ( int i = voice_count; --i >= 0; )
|
|
{
|
|
voice_t* v = &m.voices [i];
|
|
v->brr_offset = 1;
|
|
v->vbit = 1 << i;
|
|
v->regs = &m.regs [i * 0x10];
|
|
}
|
|
m.new_kon = REG(kon);
|
|
m.t_dir = REG(dir);
|
|
m.t_esa = REG(esa);
|
|
|
|
soft_reset_common();
|
|
}
|
|
|
|
void SPC_DSP::reset() { load( initial_regs ); }
|
|
|
|
|
|
//// State save/load
|
|
|
|
#if !SPC_NO_COPY_STATE_FUNCS
|
|
|
|
void SPC_State_Copier::copy( void* state, size_t size )
|
|
{
|
|
func( buf, state, size );
|
|
}
|
|
|
|
int SPC_State_Copier::copy_int( int state, int size )
|
|
{
|
|
BOOST::uint8_t s [2];
|
|
SET_LE16( s, state );
|
|
func( buf, &s, size );
|
|
return GET_LE16( s );
|
|
}
|
|
|
|
void SPC_State_Copier::skip( int count )
|
|
{
|
|
if ( count > 0 )
|
|
{
|
|
char temp [64];
|
|
memset( temp, 0, sizeof temp );
|
|
do
|
|
{
|
|
int n = sizeof temp;
|
|
if ( n > count )
|
|
n = count;
|
|
count -= n;
|
|
func( buf, temp, n );
|
|
}
|
|
while ( count );
|
|
}
|
|
}
|
|
|
|
void SPC_State_Copier::extra()
|
|
{
|
|
int n = 0;
|
|
SPC_State_Copier& copier = *this;
|
|
SPC_COPY( uint8_t, n );
|
|
skip( n );
|
|
}
|
|
|
|
void SPC_DSP::copy_state( unsigned char** io, copy_func_t copy )
|
|
{
|
|
SPC_State_Copier copier( io, copy );
|
|
|
|
// DSP registers
|
|
copier.copy( m.regs, register_count );
|
|
|
|
// Internal state
|
|
|
|
// Voices
|
|
int i;
|
|
for ( i = 0; i < voice_count; i++ )
|
|
{
|
|
voice_t* v = &m.voices [i];
|
|
|
|
// BRR buffer
|
|
int i;
|
|
for ( i = 0; i < brr_buf_size; i++ )
|
|
{
|
|
int s = v->buf [i];
|
|
SPC_COPY( int16_t, s );
|
|
v->buf [i] = v->buf [i + brr_buf_size] = s;
|
|
}
|
|
|
|
SPC_COPY( uint16_t, v->interp_pos );
|
|
SPC_COPY( uint16_t, v->brr_addr );
|
|
SPC_COPY( uint16_t, v->env );
|
|
SPC_COPY( int16_t, v->hidden_env );
|
|
SPC_COPY( uint8_t, v->buf_pos );
|
|
SPC_COPY( uint8_t, v->brr_offset );
|
|
SPC_COPY( uint8_t, v->kon_delay );
|
|
{
|
|
int m = v->env_mode;
|
|
SPC_COPY( uint8_t, m );
|
|
v->env_mode = (enum env_mode_t) m;
|
|
}
|
|
SPC_COPY( uint8_t, v->t_envx_out );
|
|
|
|
copier.extra();
|
|
}
|
|
|
|
// Echo history
|
|
for ( i = 0; i < echo_hist_size; i++ )
|
|
{
|
|
int j;
|
|
for ( j = 0; j < 2; j++ )
|
|
{
|
|
int s = m.echo_hist_pos [i] [j];
|
|
SPC_COPY( int16_t, s );
|
|
m.echo_hist [i] [j] = s; // write back at offset 0
|
|
}
|
|
}
|
|
m.echo_hist_pos = m.echo_hist;
|
|
memcpy( &m.echo_hist [echo_hist_size], m.echo_hist, echo_hist_size * sizeof m.echo_hist [0] );
|
|
|
|
// Misc
|
|
SPC_COPY( uint8_t, m.every_other_sample );
|
|
SPC_COPY( uint8_t, m.kon );
|
|
|
|
SPC_COPY( uint16_t, m.noise );
|
|
SPC_COPY( uint16_t, m.counter );
|
|
SPC_COPY( uint16_t, m.echo_offset );
|
|
SPC_COPY( uint16_t, m.echo_length );
|
|
SPC_COPY( uint8_t, m.phase );
|
|
|
|
SPC_COPY( uint8_t, m.new_kon );
|
|
SPC_COPY( uint8_t, m.endx_buf );
|
|
SPC_COPY( uint8_t, m.envx_buf );
|
|
SPC_COPY( uint8_t, m.outx_buf );
|
|
|
|
SPC_COPY( uint8_t, m.t_pmon );
|
|
SPC_COPY( uint8_t, m.t_non );
|
|
SPC_COPY( uint8_t, m.t_eon );
|
|
SPC_COPY( uint8_t, m.t_dir );
|
|
SPC_COPY( uint8_t, m.t_koff );
|
|
|
|
SPC_COPY( uint16_t, m.t_brr_next_addr );
|
|
SPC_COPY( uint8_t, m.t_adsr0 );
|
|
SPC_COPY( uint8_t, m.t_brr_header );
|
|
SPC_COPY( uint8_t, m.t_brr_byte );
|
|
SPC_COPY( uint8_t, m.t_srcn );
|
|
SPC_COPY( uint8_t, m.t_esa );
|
|
SPC_COPY( uint8_t, m.t_echo_enabled );
|
|
|
|
SPC_COPY( int16_t, m.t_main_out [0] );
|
|
SPC_COPY( int16_t, m.t_main_out [1] );
|
|
SPC_COPY( int16_t, m.t_echo_out [0] );
|
|
SPC_COPY( int16_t, m.t_echo_out [1] );
|
|
SPC_COPY( int16_t, m.t_echo_in [0] );
|
|
SPC_COPY( int16_t, m.t_echo_in [1] );
|
|
|
|
SPC_COPY( uint16_t, m.t_dir_addr );
|
|
SPC_COPY( uint16_t, m.t_pitch );
|
|
SPC_COPY( int16_t, m.t_output );
|
|
SPC_COPY( uint16_t, m.t_echo_ptr );
|
|
SPC_COPY( uint8_t, m.t_looped );
|
|
|
|
copier.copy(m.external_regs, register_count);
|
|
copier.extra();
|
|
}
|
|
#endif
|
|
|
|
|
|
//// Snes9x Accessor
|
|
|
|
void SPC_DSP::set_spc_snapshot_callback( void (*callback) (void) )
|
|
{
|
|
spc_snapshot_callback = callback;
|
|
}
|
|
|
|
void SPC_DSP::dump_spc_snapshot( void )
|
|
{
|
|
take_spc_snapshot = 1;
|
|
}
|
|
|
|
void SPC_DSP::set_stereo_switch( int value )
|
|
{
|
|
stereo_switch = value;
|
|
}
|
|
|
|
SPC_DSP::uint8_t SPC_DSP::reg_value( int ch, int addr )
|
|
{
|
|
return m.voices[ch].regs[addr];
|
|
}
|
|
|
|
int SPC_DSP::envx_value( int ch )
|
|
{
|
|
return m.voices[ch].env;
|
|
}
|