forked from vampirefrog/fmtoy
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathfmtoy_ym2612.c
121 lines (108 loc) · 4.65 KB
/
fmtoy_ym2612.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
#include "fmtoy.h"
#include "fmtoy_ym2612.h"
#include "libvgm/emu/SoundEmu.h"
#include "libvgm/emu/SoundDevs.h"
static int fmtoy_ym2612_init(struct fmtoy *fmtoy, int clock, int sample_rate, struct fmtoy_channel *channel) {
channel->chip->clock = clock;
DEV_GEN_CFG devCfg = {
.emuCore = 0,
.srMode = DEVRI_SRMODE_NATIVE,
.flags = 0x00,
.clock = clock,
.smplRate = sample_rate,
};
DEV_INFO *devinf = malloc(sizeof(DEV_INFO));
if(!devinf) return -1;
channel->chip->data = devinf;
if(SndEmu_Start(DEVID_YM2612, &devCfg, devinf))
return -2;
DEVFUNC_WRITE_A8D8 writefn;
SndEmu_GetDeviceFunc(devinf->devDef, RWF_REGISTER | RWF_WRITE, DEVRW_A8D8, 0, (void**)&writefn);
// Enable 6 channel mode
writefn(devinf->dataPtr, 0, 0x29);
writefn(devinf->dataPtr, 1, 0x80);
return 0;
}
static int fmtoy_ym2612_destroy(struct fmtoy *fmtoy, struct fmtoy_channel *channel) {
SndEmu_Stop(channel->chip->data);
free(channel->chip->data);
return 0;
}
static void fmtoy_ym2612_program_change(struct fmtoy *fmtoy, uint8_t program, struct fmtoy_channel *channel) {
DEVFUNC_WRITE_A8D8 writefn;
DEV_INFO *devinf = channel->chip->data;
SndEmu_GetDeviceFunc(devinf->devDef, RWF_REGISTER | RWF_WRITE, DEVRW_A8D8, 0, (void**)&writefn);
struct opn_voice *v = &fmtoy->opn_voices[program];
for(int i = 0; i < 6; i++) {
int base = i < 3 ? 0 : 2;
int c = i % 3;
writefn(devinf->dataPtr, base+0, 0xb0 + c);
writefn(devinf->dataPtr, base+1, v->fb_con);
writefn(devinf->dataPtr, base+0, 0xb4 + c);
writefn(devinf->dataPtr, base+1, v->lr_ams_pms);
for(int j = 0; j < 4; j++) {
struct opn_voice_operator *op = &v->operators[j];
writefn(devinf->dataPtr, base+0, 0x30 + c + j * 4);
writefn(devinf->dataPtr, base+1, op->dt_mul);
writefn(devinf->dataPtr, base+0, 0x40 + c + j * 4);
writefn(devinf->dataPtr, base+1, op->tl);
writefn(devinf->dataPtr, base+0, 0x50 + c + j * 4);
writefn(devinf->dataPtr, base+1, op->ks_ar);
writefn(devinf->dataPtr, base+0, 0x60 + c + j * 4);
writefn(devinf->dataPtr, base+1, op->am_dr);
writefn(devinf->dataPtr, base+0, 0x70 + c + j * 4);
writefn(devinf->dataPtr, base+1, op->sr);
writefn(devinf->dataPtr, base+0, 0x80 + c + j * 4);
writefn(devinf->dataPtr, base+1, op->sl_rr);
writefn(devinf->dataPtr, base+0, 0x90 + c + j * 4);
writefn(devinf->dataPtr, base+1, op->ssg_eg);
}
}
}
static void fmtoy_ym2612_set_pitch(struct fmtoy *fmtoy, int chip_channel, float pitch, struct fmtoy_channel *channel) {
DEVFUNC_WRITE_A8D8 writefn;
DEV_INFO *devinf = channel->chip->data;
SndEmu_GetDeviceFunc(devinf->devDef, RWF_REGISTER | RWF_WRITE, DEVRW_A8D8, 0, (void**)&writefn);
int block_fnum = opnx_pitch_to_block_fnum(pitch, channel->chip->clock);
int base = chip_channel < 3 ? 0 : 2;
chip_channel = chip_channel % 3;
writefn(devinf->dataPtr, base+0, 0xa4 + chip_channel);
writefn(devinf->dataPtr, base+1, block_fnum >> 8);
writefn(devinf->dataPtr, base+0, 0xa0 + chip_channel);
writefn(devinf->dataPtr, base+1, block_fnum & 0xff);
}
static void fmtoy_ym2612_pitch_bend(struct fmtoy *fmtoy, uint8_t chip_channel, float pitch, struct fmtoy_channel *channel) {
fmtoy_ym2612_set_pitch(fmtoy, chip_channel, pitch, channel);
}
static void fmtoy_ym2612_note_on(struct fmtoy *fmtoy, uint8_t chip_channel, float pitch, uint8_t velocity, struct fmtoy_channel *channel) {
DEVFUNC_WRITE_A8D8 writefn;
DEV_INFO *devinf = channel->chip->data;
SndEmu_GetDeviceFunc(devinf->devDef, RWF_REGISTER | RWF_WRITE, DEVRW_A8D8, 0, (void**)&writefn);
fmtoy_ym2612_set_pitch(fmtoy, chip_channel, pitch, channel);
chip_channel = chip_channel < 3 ? chip_channel : (chip_channel + 1);
writefn(devinf->dataPtr, 0, 0x28);
writefn(devinf->dataPtr, 1, 0xf0 + chip_channel);
}
static void fmtoy_ym2612_note_off(struct fmtoy *fmtoy, uint8_t chip_channel, uint8_t velocity, struct fmtoy_channel *channel) {
DEVFUNC_WRITE_A8D8 writefn;
DEV_INFO *devinf = channel->chip->data;
SndEmu_GetDeviceFunc(devinf->devDef, RWF_REGISTER | RWF_WRITE, DEVRW_A8D8, 0, (void**)&writefn);
chip_channel = chip_channel < 3 ? chip_channel : chip_channel + 1;
writefn(devinf->dataPtr, 0, 0x28);
writefn(devinf->dataPtr, 1, chip_channel);
}
static void fmtoy_ym2612_render(struct fmtoy *fmtoy, stream_sample_t **buffers, int num_samples, struct fmtoy_channel *channel) {
DEV_INFO *devinf = channel->chip->data;
devinf->devDef->Update(devinf->dataPtr, num_samples, buffers);
}
struct fmtoy_chip fmtoy_chip_ym2612 = {
.name = "YM2612",
.init = fmtoy_ym2612_init,
.destroy = fmtoy_ym2612_destroy,
.program_change = fmtoy_ym2612_program_change,
.pitch_bend = fmtoy_ym2612_pitch_bend,
.note_on = fmtoy_ym2612_note_on,
.note_off = fmtoy_ym2612_note_off,
.render = fmtoy_ym2612_render,
.max_poliphony = 6,
};