dingusppc/devices/sound/awacs.h

194 lines
5.4 KiB
C
Raw Normal View History

2020-03-15 16:24:40 +00:00
/*
DingusPPC - The Experimental PowerPC Macintosh emulator
2024-04-07 16:37:03 +00:00
Copyright (C) 2018-24 divingkatae and maximum
2020-03-15 16:24:40 +00:00
(theweirdo) spatium
(Contact divingkatae#1017 or powermax#2286 on Discord for more info)
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program 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 General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
/** AWAC sound devices family definitions.
2022-08-14 20:49:24 +00:00
Audio Waveform Amplifier and Converters (AWACs) is a family of custom audio
2020-03-15 16:24:40 +00:00
chips used in Power Macintosh.
*/
#ifndef AWAC_H
#define AWAC_H
2023-02-11 13:51:03 +00:00
#include <devices/common/hwcomponent.h>
#include <devices/common/i2c/i2c.h>
2020-05-12 18:55:45 +00:00
#include <cinttypes>
#include <memory>
2023-02-11 13:51:03 +00:00
#include <string>
2020-03-15 16:24:40 +00:00
class DmaOutChannel;
2023-11-22 04:38:56 +00:00
class DmaInChannel;
class SoundServer;
/** Base class for the AWACs codecs. */
2023-02-11 13:51:03 +00:00
class AwacsBase : public HWComponent {
public:
2023-02-11 13:51:03 +00:00
AwacsBase(std::string name);
2022-07-17 03:50:32 +00:00
~AwacsBase() = default;
void set_dma_out(DmaOutChannel *dma_out_ch) {
this->dma_out_ch = dma_out_ch;
};
2023-11-22 04:38:56 +00:00
void set_dma_in(DmaInChannel *dma_in_ch) {
this->dma_in_ch = dma_in_ch;
};
void set_sample_rate(int sr_id);
2023-02-11 13:51:03 +00:00
void dma_out_start();
void dma_out_stop();
void dma_out_pause();
2023-11-22 04:38:56 +00:00
void dma_in_start();
void dma_in_stop();
void dma_in_pause();
void dma_in_data();
protected:
SoundServer *snd_server; // SoundServer instance pointer
DmaOutChannel *dma_out_ch; // DMA output channel instance pointer
2023-11-22 04:38:56 +00:00
DmaInChannel *dma_in_ch; // DMA input channel instance pointer
uint32_t dma_in_timer_id = 0;
int *sr_table; // pointer to the table of supported sample rates
int max_sr_id; // maximum value for sample rate ID
bool out_stream_ready = false;
bool out_stream_running = false;
int cur_sample_rate = -1;
int out_sample_rate = -1;
};
/** AWACs PDM-style sound codec. */
class AwacDevicePdm : public AwacsBase {
public:
AwacDevicePdm();
~AwacDevicePdm() = default;
uint32_t read_stat(void);
void write_ctrl(uint32_t addr, uint16_t value);
private:
uint16_t ctrl_regs[5] = {}; // 12-bit wide control registers
};
2023-02-11 13:51:03 +00:00
/** Offsets to MacIO sound codec registers. */
2020-03-15 16:24:40 +00:00
enum {
AWAC_SOUND_CTRL_REG = 0x00,
AWAC_CODEC_CTRL_REG = 0x10,
AWAC_CODEC_STATUS_REG = 0x20,
};
/** Apple source calls this kValidData but doesn't explain
what it actually means. It seems like it's used to check
if the sound codec is available.
*/
#define AWAC_AVAILABLE 0x40
/** Audio processor chip (TDA7433) emulation. */
class AudioProcessor : public I2CDevice {
public:
AudioProcessor() = default;
~AudioProcessor() = default;
2020-05-12 18:55:45 +00:00
void start_transaction() {
this->pos = 0;
};
2020-03-15 16:24:40 +00:00
bool send_subaddress(uint8_t sub_addr) {
if ((sub_addr & 0xF) > 6)
return false; // invalid subaddress -> no acknowledge
2020-03-15 16:24:40 +00:00
this->sub_addr = sub_addr & 0xF;
this->auto_inc = !!(sub_addr & 0x10);
LOG_F(9, "TDA7433 subaddress = 0x%X, auto increment = %d",
this->sub_addr, this->auto_inc);
2020-03-15 16:24:40 +00:00
this->pos++;
return true;
};
bool send_byte(uint8_t data) {
if (!this->pos) {
return send_subaddress(data);
} else if (this->sub_addr <= 6) {
LOG_F(9, "TDA7433 byte 0x%X received", data);
2020-05-12 18:55:45 +00:00
this->regs[this->sub_addr] = data;
2020-03-15 16:24:40 +00:00
if (this->auto_inc) {
this->sub_addr++;
}
return true;
} else {
return false; // invalid sub_addr -> no acknowledge
2020-03-15 16:24:40 +00:00
}
};
2020-05-12 18:55:45 +00:00
bool receive_byte(uint8_t* p_data) {
2020-03-15 16:24:40 +00:00
*p_data = this->regs[this->sub_addr];
LOG_F(9, "TDA7433 byte 0x%X sent", *p_data);
2020-03-15 16:24:40 +00:00
return true;
};
private:
uint8_t regs[7] = {}; // control registers, see TDA7433 datasheet
2020-03-15 16:24:40 +00:00
uint8_t sub_addr;
2020-05-12 18:55:45 +00:00
int pos;
int auto_inc;
2020-03-15 16:24:40 +00:00
};
2023-02-11 13:51:03 +00:00
/** Sound codec interface with the typical MacIO access. */
class MacioSndCodec : public AwacsBase {
2023-02-11 13:51:03 +00:00
public:
MacioSndCodec(std::string name) : AwacsBase(name) {};
2023-02-11 13:51:03 +00:00
virtual uint32_t snd_ctrl_read(uint32_t offset, int size) = 0;
virtual void snd_ctrl_write(uint32_t offset, uint32_t value, int size) = 0;
};
2023-11-03 09:28:30 +00:00
/** AWACs manufacturers and revisions. */
2023-02-11 13:51:03 +00:00
#define AWAC_MAKER_CRYSTAL 1
2023-11-03 09:28:30 +00:00
#define AWAC_REV_AWACS 2
2023-02-11 13:51:03 +00:00
#define AWAC_REV_SCREAMER 3
/** Screamer sound codec. */
class AwacsScreamer : public MacioSndCodec {
2020-03-15 16:24:40 +00:00
public:
2023-02-11 13:51:03 +00:00
AwacsScreamer(std::string name = "Screamer");
~AwacsScreamer() = default;
uint32_t snd_ctrl_read(uint32_t offset, int size);
void snd_ctrl_write(uint32_t offset, uint32_t value, int size);
2020-03-15 16:24:40 +00:00
2023-02-11 13:51:03 +00:00
int device_postinit();
2020-03-19 14:09:24 +00:00
2023-02-11 13:51:03 +00:00
static std::unique_ptr<HWComponent> create() {
return std::unique_ptr<AwacsScreamer>(new AwacsScreamer("Screamer"));
}
2020-03-19 14:09:24 +00:00
2020-03-15 16:24:40 +00:00
private:
uint32_t snd_ctrl_reg = 0;
uint16_t control_regs[8] = {}; // control registers, each 12-bits wide
2023-02-11 13:51:03 +00:00
uint8_t is_busy = 0;
std::unique_ptr<AudioProcessor> audio_proc;
2020-03-15 16:24:40 +00:00
};
#endif /* AWAC_H */