#include #include #include #include "psg.h" #include "scheduler.h" const uint16_t frequencyCodes[8][12] = { // C, Cis, D, Dis, E, F, Fis, G, Gis, A, Ais, H { 06535, 06234, 05747, 05474, 05233, 05002, 04563, 04353, 04153, 03762, 03600, 03424 }, // Octave 1 { 03256, 03116, 02764, 02636, 02515, 02401, 02271, 02165, 02065, 01771, 01700, 01612 }, // Octave 2 { 01527, 01447, 01372, 01317, 01247, 01201, 01135, 01073, 01033, 00774, 00740, 00705 }, // Octave 3 { 00654, 00624, 00575, 00550, 00523, 00500, 00456, 00435, 00415, 00376, 00360, 00342 }, // Octave 4 { 00326, 00312, 00276, 00264, 00252, 00240, 00227, 00217, 00207, 00177, 00170, 00161 }, // Octave 5 { 00153, 00145, 00137, 00132, 00125, 00120, 00114, 00107, 00103, 00100, 00074, 00071 }, // Octave 6 { 00065, 00062, 00060, 00055, 00052, 00050, 00046, 00044, 00042, 00040, 00036, 00034 }, // Octave 7 { 00033, 00031, 00030, 00026, 00025, 00024, 00023, 00022, 00021, 00020, 00017, 00016 } // Octave 8 }; #define ADDR_DATA_REG P2OUT #define BUS_CTRL_REG P1OUT #define BC1 BIT3 #define BDIR BIT1 #define _CS0 BIT2 #define _CS1 BIT0 #define R0 0 #define CHANNEL_A_TONE_PERIOD_FINE_REG R0 #define R1 1 #define CHANNEL_A_TONE_PERIOD_COARSE_REG R1 #define R2 2 #define CHANNEL_B_TONE_PERIOD_FINE_REG R2 #define R3 3 #define CHANNEL_B_TONE_PERIOD_COARSE_REG R3 #define R4 4 #define CHANNEL_C_TONE_PERIOD_FINE_REG R4 #define R5 5 #define CHANNEL_C_TONE_PERIOD_COARSE_REG R5 #define R6 6 #define NOISE_PERIOD_REG R6 #define R7 7 #define _ENABLE_REG R7 #define R10 010 #define CHANNEL_A_AMPLITUDE_REG R10 #define R11 011 #define CHANNEL_B_AMPLITUDE_REG R11 #define R12 012 #define CHANNEL_C_AMPLITUDE_REG R12 #define R13 013 #define ENVELOPE_PERIOD_FINE_REG R13 #define R14 014 #define ENVELOPE_PERIOD_COARSE_REG R13 #define R15 015 #define ENVELOPE_SHAPE_REG R15 uint8_t psgShadowRegisters[2][14]; inline static void BUS_OP_NACT() { BUS_CTRL_REG &= ~(BDIR | BC1); } inline static void BUS_OP_INTAK() { BUS_CTRL_REG |= BDIR | BC1; } inline static void BUS_OP_DWS() { BUS_CTRL_REG |= BDIR; BUS_CTRL_REG &= ~BC1; } inline static void BUS_OP_CS0_ENABLE() { BUS_CTRL_REG &= ~_CS0; } inline static void BUS_OP_CS0_DISABLE() { BUS_CTRL_REG |= _CS0; } inline static void BUS_OP_CS1_ENABLE() { BUS_CTRL_REG &= ~_CS1; } inline static void BUS_OP_CS1_DISABLE() { BUS_CTRL_REG |= _CS1; } static uint8_t psgReadShadow(uint8_t chip, uint8_t address) { return psgShadowRegisters[chip][address]; } static void psgWrite(uint8_t chip, uint8_t address, uint8_t data) { psgShadowRegisters[chip][address] = data; // according to "State Timing" (p. 15) of datasheet if (chip == 0) { BUS_OP_CS0_ENABLE(); } else { BUS_OP_CS1_ENABLE(); } // put bus into inactive state BUS_OP_NACT(); // put address on bus ADDR_DATA_REG = address; // address latch mode BUS_OP_INTAK(); // latch address BUS_OP_NACT(); // put data on bus ADDR_DATA_REG = data; // set write to psg BUS_OP_DWS(); // set inactive again BUS_OP_NACT(); if (chip == 0) { BUS_OP_CS0_DISABLE(); } else { BUS_OP_CS1_DISABLE(); } } static void psgWriteFrequency(uint8_t chip, uint8_t channel, uint16_t frequencyCode) { psgWrite(chip, CHANNEL_A_TONE_PERIOD_FINE_REG + (channel * 2), (frequencyCode & 0x00ff)); psgWrite(chip, CHANNEL_A_TONE_PERIOD_COARSE_REG + (channel * 2), ((frequencyCode >> 8) & 0x000f)); } void psgPlayTone(uint8_t chip, uint8_t channel, uint8_t volume, t_octave octave, t_note note) { if (note == e_Pause) { psgWrite(chip, _ENABLE_REG, psgReadShadow(chip, _ENABLE_REG) | (1 << channel)); } else { psgWrite(chip, _ENABLE_REG, psgReadShadow(chip, _ENABLE_REG) & ~(1 << channel)); psgAmplitude(chip, channel, volume); psgWriteFrequency(chip, channel, frequencyCodes[octave][note]); } } void psgAmplitude(uint8_t chip, uint8_t channel, uint8_t volume) { psgWrite(chip, CHANNEL_A_AMPLITUDE_REG + channel, volume); } void psgInit() { // address/data bus P2DIR = 0xff; P2SEL = 0; P2SEL2 = 0; // bus control lines // BIT3: BC1 // BIT1: BDIR // BIT0: _CS1 // BIT2: _CS0 P1DIR |= BIT0 | BIT1 | BIT2 | BIT3 ; // put bus into inactive state BUS_CTRL_REG &= ~(BDIR | BC1); // disable everything psgWrite(0, _ENABLE_REG, 0xff); psgWrite(1, _ENABLE_REG, 0xff); }