149ab747fSPaolo Bonzini /* 249ab747fSPaolo Bonzini * QEMU Crystal CS4231 audio chip emulation 349ab747fSPaolo Bonzini * 449ab747fSPaolo Bonzini * Copyright (c) 2006 Fabrice Bellard 549ab747fSPaolo Bonzini * 649ab747fSPaolo Bonzini * Permission is hereby granted, free of charge, to any person obtaining a copy 749ab747fSPaolo Bonzini * of this software and associated documentation files (the "Software"), to deal 849ab747fSPaolo Bonzini * in the Software without restriction, including without limitation the rights 949ab747fSPaolo Bonzini * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 1049ab747fSPaolo Bonzini * copies of the Software, and to permit persons to whom the Software is 1149ab747fSPaolo Bonzini * furnished to do so, subject to the following conditions: 1249ab747fSPaolo Bonzini * 1349ab747fSPaolo Bonzini * The above copyright notice and this permission notice shall be included in 1449ab747fSPaolo Bonzini * all copies or substantial portions of the Software. 1549ab747fSPaolo Bonzini * 1649ab747fSPaolo Bonzini * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 1749ab747fSPaolo Bonzini * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 1849ab747fSPaolo Bonzini * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 1949ab747fSPaolo Bonzini * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 2049ab747fSPaolo Bonzini * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 2149ab747fSPaolo Bonzini * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN 2249ab747fSPaolo Bonzini * THE SOFTWARE. 2349ab747fSPaolo Bonzini */ 240b8fa32fSMarkus Armbruster 256086a565SPeter Maydell #include "qemu/osdep.h" 2649ab747fSPaolo Bonzini #include "hw/hw.h" 278a824e4dSEduardo Habkost #include "hw/audio/soundhw.h" 2849ab747fSPaolo Bonzini #include "audio/audio.h" 2964552b6bSMarkus Armbruster #include "hw/irq.h" 3049ab747fSPaolo Bonzini #include "hw/isa/isa.h" 3149ab747fSPaolo Bonzini #include "hw/qdev.h" 32*d6454270SMarkus Armbruster #include "migration/vmstate.h" 330b8fa32fSMarkus Armbruster #include "qemu/module.h" 3449ab747fSPaolo Bonzini #include "qemu/timer.h" 35c9073238SThomas Huth #include "qapi/error.h" 3649ab747fSPaolo Bonzini 3749ab747fSPaolo Bonzini /* 3849ab747fSPaolo Bonzini Missing features: 3949ab747fSPaolo Bonzini ADC 4049ab747fSPaolo Bonzini Loopback 4149ab747fSPaolo Bonzini Timer 4249ab747fSPaolo Bonzini ADPCM 4349ab747fSPaolo Bonzini More... 4449ab747fSPaolo Bonzini */ 4549ab747fSPaolo Bonzini 4649ab747fSPaolo Bonzini /* #define DEBUG */ 4749ab747fSPaolo Bonzini /* #define DEBUG_XLAW */ 4849ab747fSPaolo Bonzini 4949ab747fSPaolo Bonzini static struct { 5049ab747fSPaolo Bonzini int aci_counter; 5149ab747fSPaolo Bonzini } conf = {1}; 5249ab747fSPaolo Bonzini 5349ab747fSPaolo Bonzini #ifdef DEBUG 5449ab747fSPaolo Bonzini #define dolog(...) AUD_log ("cs4231a", __VA_ARGS__) 5549ab747fSPaolo Bonzini #else 5649ab747fSPaolo Bonzini #define dolog(...) 5749ab747fSPaolo Bonzini #endif 5849ab747fSPaolo Bonzini 5949ab747fSPaolo Bonzini #define lwarn(...) AUD_log ("cs4231a", "warning: " __VA_ARGS__) 6049ab747fSPaolo Bonzini #define lerr(...) AUD_log ("cs4231a", "error: " __VA_ARGS__) 6149ab747fSPaolo Bonzini 6249ab747fSPaolo Bonzini #define CS_REGS 16 6349ab747fSPaolo Bonzini #define CS_DREGS 32 6449ab747fSPaolo Bonzini 65a3dcca56SAndreas Färber #define TYPE_CS4231A "cs4231a" 66a3dcca56SAndreas Färber #define CS4231A(obj) OBJECT_CHECK (CSState, (obj), TYPE_CS4231A) 67a3dcca56SAndreas Färber 6849ab747fSPaolo Bonzini typedef struct CSState { 6949ab747fSPaolo Bonzini ISADevice dev; 7049ab747fSPaolo Bonzini QEMUSoundCard card; 7149ab747fSPaolo Bonzini MemoryRegion ioports; 7249ab747fSPaolo Bonzini qemu_irq pic; 7349ab747fSPaolo Bonzini uint32_t regs[CS_REGS]; 7449ab747fSPaolo Bonzini uint8_t dregs[CS_DREGS]; 7549ab747fSPaolo Bonzini uint32_t irq; 7649ab747fSPaolo Bonzini uint32_t dma; 7749ab747fSPaolo Bonzini uint32_t port; 782d011091SHervé Poussineau IsaDma *isa_dma; 7949ab747fSPaolo Bonzini int shift; 8049ab747fSPaolo Bonzini int dma_running; 8149ab747fSPaolo Bonzini int audio_free; 8249ab747fSPaolo Bonzini int transferred; 8349ab747fSPaolo Bonzini int aci_counter; 8449ab747fSPaolo Bonzini SWVoiceOut *voice; 8549ab747fSPaolo Bonzini int16_t *tab; 8649ab747fSPaolo Bonzini } CSState; 8749ab747fSPaolo Bonzini 8849ab747fSPaolo Bonzini #define MODE2 (1 << 6) 8949ab747fSPaolo Bonzini #define MCE (1 << 6) 9049ab747fSPaolo Bonzini #define PMCE (1 << 4) 9149ab747fSPaolo Bonzini #define CMCE (1 << 5) 9249ab747fSPaolo Bonzini #define TE (1 << 6) 9349ab747fSPaolo Bonzini #define PEN (1 << 0) 9449ab747fSPaolo Bonzini #define INT (1 << 0) 9549ab747fSPaolo Bonzini #define IEN (1 << 1) 9649ab747fSPaolo Bonzini #define PPIO (1 << 6) 9749ab747fSPaolo Bonzini #define PI (1 << 4) 9849ab747fSPaolo Bonzini #define CI (1 << 5) 9949ab747fSPaolo Bonzini #define TI (1 << 6) 10049ab747fSPaolo Bonzini 10149ab747fSPaolo Bonzini enum { 10249ab747fSPaolo Bonzini Index_Address, 10349ab747fSPaolo Bonzini Index_Data, 10449ab747fSPaolo Bonzini Status, 10549ab747fSPaolo Bonzini PIO_Data 10649ab747fSPaolo Bonzini }; 10749ab747fSPaolo Bonzini 10849ab747fSPaolo Bonzini enum { 10949ab747fSPaolo Bonzini Left_ADC_Input_Control, 11049ab747fSPaolo Bonzini Right_ADC_Input_Control, 11149ab747fSPaolo Bonzini Left_AUX1_Input_Control, 11249ab747fSPaolo Bonzini Right_AUX1_Input_Control, 11349ab747fSPaolo Bonzini Left_AUX2_Input_Control, 11449ab747fSPaolo Bonzini Right_AUX2_Input_Control, 11549ab747fSPaolo Bonzini Left_DAC_Output_Control, 11649ab747fSPaolo Bonzini Right_DAC_Output_Control, 11749ab747fSPaolo Bonzini FS_And_Playback_Data_Format, 11849ab747fSPaolo Bonzini Interface_Configuration, 11949ab747fSPaolo Bonzini Pin_Control, 12049ab747fSPaolo Bonzini Error_Status_And_Initialization, 12149ab747fSPaolo Bonzini MODE_And_ID, 12249ab747fSPaolo Bonzini Loopback_Control, 12349ab747fSPaolo Bonzini Playback_Upper_Base_Count, 12449ab747fSPaolo Bonzini Playback_Lower_Base_Count, 12549ab747fSPaolo Bonzini Alternate_Feature_Enable_I, 12649ab747fSPaolo Bonzini Alternate_Feature_Enable_II, 12749ab747fSPaolo Bonzini Left_Line_Input_Control, 12849ab747fSPaolo Bonzini Right_Line_Input_Control, 12949ab747fSPaolo Bonzini Timer_Low_Base, 13049ab747fSPaolo Bonzini Timer_High_Base, 13149ab747fSPaolo Bonzini RESERVED, 13249ab747fSPaolo Bonzini Alternate_Feature_Enable_III, 13349ab747fSPaolo Bonzini Alternate_Feature_Status, 13449ab747fSPaolo Bonzini Version_Chip_ID, 13549ab747fSPaolo Bonzini Mono_Input_And_Output_Control, 13649ab747fSPaolo Bonzini RESERVED_2, 13749ab747fSPaolo Bonzini Capture_Data_Format, 13849ab747fSPaolo Bonzini RESERVED_3, 13949ab747fSPaolo Bonzini Capture_Upper_Base_Count, 14049ab747fSPaolo Bonzini Capture_Lower_Base_Count 14149ab747fSPaolo Bonzini }; 14249ab747fSPaolo Bonzini 14349ab747fSPaolo Bonzini static int freqs[2][8] = { 14449ab747fSPaolo Bonzini { 8000, 16000, 27420, 32000, -1, -1, 48000, 9000 }, 14549ab747fSPaolo Bonzini { 5510, 11025, 18900, 22050, 37800, 44100, 33075, 6620 } 14649ab747fSPaolo Bonzini }; 14749ab747fSPaolo Bonzini 14849ab747fSPaolo Bonzini /* Tables courtesy http://hazelware.luggle.com/tutorials/mulawcompression.html */ 14949ab747fSPaolo Bonzini static int16_t MuLawDecompressTable[256] = 15049ab747fSPaolo Bonzini { 15149ab747fSPaolo Bonzini -32124,-31100,-30076,-29052,-28028,-27004,-25980,-24956, 15249ab747fSPaolo Bonzini -23932,-22908,-21884,-20860,-19836,-18812,-17788,-16764, 15349ab747fSPaolo Bonzini -15996,-15484,-14972,-14460,-13948,-13436,-12924,-12412, 15449ab747fSPaolo Bonzini -11900,-11388,-10876,-10364, -9852, -9340, -8828, -8316, 15549ab747fSPaolo Bonzini -7932, -7676, -7420, -7164, -6908, -6652, -6396, -6140, 15649ab747fSPaolo Bonzini -5884, -5628, -5372, -5116, -4860, -4604, -4348, -4092, 15749ab747fSPaolo Bonzini -3900, -3772, -3644, -3516, -3388, -3260, -3132, -3004, 15849ab747fSPaolo Bonzini -2876, -2748, -2620, -2492, -2364, -2236, -2108, -1980, 15949ab747fSPaolo Bonzini -1884, -1820, -1756, -1692, -1628, -1564, -1500, -1436, 16049ab747fSPaolo Bonzini -1372, -1308, -1244, -1180, -1116, -1052, -988, -924, 16149ab747fSPaolo Bonzini -876, -844, -812, -780, -748, -716, -684, -652, 16249ab747fSPaolo Bonzini -620, -588, -556, -524, -492, -460, -428, -396, 16349ab747fSPaolo Bonzini -372, -356, -340, -324, -308, -292, -276, -260, 16449ab747fSPaolo Bonzini -244, -228, -212, -196, -180, -164, -148, -132, 16549ab747fSPaolo Bonzini -120, -112, -104, -96, -88, -80, -72, -64, 16649ab747fSPaolo Bonzini -56, -48, -40, -32, -24, -16, -8, 0, 16749ab747fSPaolo Bonzini 32124, 31100, 30076, 29052, 28028, 27004, 25980, 24956, 16849ab747fSPaolo Bonzini 23932, 22908, 21884, 20860, 19836, 18812, 17788, 16764, 16949ab747fSPaolo Bonzini 15996, 15484, 14972, 14460, 13948, 13436, 12924, 12412, 17049ab747fSPaolo Bonzini 11900, 11388, 10876, 10364, 9852, 9340, 8828, 8316, 17149ab747fSPaolo Bonzini 7932, 7676, 7420, 7164, 6908, 6652, 6396, 6140, 17249ab747fSPaolo Bonzini 5884, 5628, 5372, 5116, 4860, 4604, 4348, 4092, 17349ab747fSPaolo Bonzini 3900, 3772, 3644, 3516, 3388, 3260, 3132, 3004, 17449ab747fSPaolo Bonzini 2876, 2748, 2620, 2492, 2364, 2236, 2108, 1980, 17549ab747fSPaolo Bonzini 1884, 1820, 1756, 1692, 1628, 1564, 1500, 1436, 17649ab747fSPaolo Bonzini 1372, 1308, 1244, 1180, 1116, 1052, 988, 924, 17749ab747fSPaolo Bonzini 876, 844, 812, 780, 748, 716, 684, 652, 17849ab747fSPaolo Bonzini 620, 588, 556, 524, 492, 460, 428, 396, 17949ab747fSPaolo Bonzini 372, 356, 340, 324, 308, 292, 276, 260, 18049ab747fSPaolo Bonzini 244, 228, 212, 196, 180, 164, 148, 132, 18149ab747fSPaolo Bonzini 120, 112, 104, 96, 88, 80, 72, 64, 18249ab747fSPaolo Bonzini 56, 48, 40, 32, 24, 16, 8, 0 18349ab747fSPaolo Bonzini }; 18449ab747fSPaolo Bonzini 18549ab747fSPaolo Bonzini static int16_t ALawDecompressTable[256] = 18649ab747fSPaolo Bonzini { 18749ab747fSPaolo Bonzini -5504, -5248, -6016, -5760, -4480, -4224, -4992, -4736, 18849ab747fSPaolo Bonzini -7552, -7296, -8064, -7808, -6528, -6272, -7040, -6784, 18949ab747fSPaolo Bonzini -2752, -2624, -3008, -2880, -2240, -2112, -2496, -2368, 19049ab747fSPaolo Bonzini -3776, -3648, -4032, -3904, -3264, -3136, -3520, -3392, 19149ab747fSPaolo Bonzini -22016,-20992,-24064,-23040,-17920,-16896,-19968,-18944, 19249ab747fSPaolo Bonzini -30208,-29184,-32256,-31232,-26112,-25088,-28160,-27136, 19349ab747fSPaolo Bonzini -11008,-10496,-12032,-11520,-8960, -8448, -9984, -9472, 19449ab747fSPaolo Bonzini -15104,-14592,-16128,-15616,-13056,-12544,-14080,-13568, 19549ab747fSPaolo Bonzini -344, -328, -376, -360, -280, -264, -312, -296, 19649ab747fSPaolo Bonzini -472, -456, -504, -488, -408, -392, -440, -424, 19749ab747fSPaolo Bonzini -88, -72, -120, -104, -24, -8, -56, -40, 19849ab747fSPaolo Bonzini -216, -200, -248, -232, -152, -136, -184, -168, 19949ab747fSPaolo Bonzini -1376, -1312, -1504, -1440, -1120, -1056, -1248, -1184, 20049ab747fSPaolo Bonzini -1888, -1824, -2016, -1952, -1632, -1568, -1760, -1696, 20149ab747fSPaolo Bonzini -688, -656, -752, -720, -560, -528, -624, -592, 20249ab747fSPaolo Bonzini -944, -912, -1008, -976, -816, -784, -880, -848, 20349ab747fSPaolo Bonzini 5504, 5248, 6016, 5760, 4480, 4224, 4992, 4736, 20449ab747fSPaolo Bonzini 7552, 7296, 8064, 7808, 6528, 6272, 7040, 6784, 20549ab747fSPaolo Bonzini 2752, 2624, 3008, 2880, 2240, 2112, 2496, 2368, 20649ab747fSPaolo Bonzini 3776, 3648, 4032, 3904, 3264, 3136, 3520, 3392, 20749ab747fSPaolo Bonzini 22016, 20992, 24064, 23040, 17920, 16896, 19968, 18944, 20849ab747fSPaolo Bonzini 30208, 29184, 32256, 31232, 26112, 25088, 28160, 27136, 20949ab747fSPaolo Bonzini 11008, 10496, 12032, 11520, 8960, 8448, 9984, 9472, 21049ab747fSPaolo Bonzini 15104, 14592, 16128, 15616, 13056, 12544, 14080, 13568, 21149ab747fSPaolo Bonzini 344, 328, 376, 360, 280, 264, 312, 296, 21249ab747fSPaolo Bonzini 472, 456, 504, 488, 408, 392, 440, 424, 21349ab747fSPaolo Bonzini 88, 72, 120, 104, 24, 8, 56, 40, 21449ab747fSPaolo Bonzini 216, 200, 248, 232, 152, 136, 184, 168, 21549ab747fSPaolo Bonzini 1376, 1312, 1504, 1440, 1120, 1056, 1248, 1184, 21649ab747fSPaolo Bonzini 1888, 1824, 2016, 1952, 1632, 1568, 1760, 1696, 21749ab747fSPaolo Bonzini 688, 656, 752, 720, 560, 528, 624, 592, 21849ab747fSPaolo Bonzini 944, 912, 1008, 976, 816, 784, 880, 848 21949ab747fSPaolo Bonzini }; 22049ab747fSPaolo Bonzini 221a3dcca56SAndreas Färber static void cs4231a_reset (DeviceState *dev) 22249ab747fSPaolo Bonzini { 223a3dcca56SAndreas Färber CSState *s = CS4231A (dev); 22449ab747fSPaolo Bonzini 22549ab747fSPaolo Bonzini s->regs[Index_Address] = 0x40; 22649ab747fSPaolo Bonzini s->regs[Index_Data] = 0x00; 22749ab747fSPaolo Bonzini s->regs[Status] = 0x00; 22849ab747fSPaolo Bonzini s->regs[PIO_Data] = 0x00; 22949ab747fSPaolo Bonzini 23049ab747fSPaolo Bonzini s->dregs[Left_ADC_Input_Control] = 0x00; 23149ab747fSPaolo Bonzini s->dregs[Right_ADC_Input_Control] = 0x00; 23249ab747fSPaolo Bonzini s->dregs[Left_AUX1_Input_Control] = 0x88; 23349ab747fSPaolo Bonzini s->dregs[Right_AUX1_Input_Control] = 0x88; 23449ab747fSPaolo Bonzini s->dregs[Left_AUX2_Input_Control] = 0x88; 23549ab747fSPaolo Bonzini s->dregs[Right_AUX2_Input_Control] = 0x88; 23649ab747fSPaolo Bonzini s->dregs[Left_DAC_Output_Control] = 0x80; 23749ab747fSPaolo Bonzini s->dregs[Right_DAC_Output_Control] = 0x80; 23849ab747fSPaolo Bonzini s->dregs[FS_And_Playback_Data_Format] = 0x00; 23949ab747fSPaolo Bonzini s->dregs[Interface_Configuration] = 0x08; 24049ab747fSPaolo Bonzini s->dregs[Pin_Control] = 0x00; 24149ab747fSPaolo Bonzini s->dregs[Error_Status_And_Initialization] = 0x00; 24249ab747fSPaolo Bonzini s->dregs[MODE_And_ID] = 0x8a; 24349ab747fSPaolo Bonzini s->dregs[Loopback_Control] = 0x00; 24449ab747fSPaolo Bonzini s->dregs[Playback_Upper_Base_Count] = 0x00; 24549ab747fSPaolo Bonzini s->dregs[Playback_Lower_Base_Count] = 0x00; 24649ab747fSPaolo Bonzini s->dregs[Alternate_Feature_Enable_I] = 0x00; 24749ab747fSPaolo Bonzini s->dregs[Alternate_Feature_Enable_II] = 0x00; 24849ab747fSPaolo Bonzini s->dregs[Left_Line_Input_Control] = 0x88; 24949ab747fSPaolo Bonzini s->dregs[Right_Line_Input_Control] = 0x88; 25049ab747fSPaolo Bonzini s->dregs[Timer_Low_Base] = 0x00; 25149ab747fSPaolo Bonzini s->dregs[Timer_High_Base] = 0x00; 25249ab747fSPaolo Bonzini s->dregs[RESERVED] = 0x00; 25349ab747fSPaolo Bonzini s->dregs[Alternate_Feature_Enable_III] = 0x00; 25449ab747fSPaolo Bonzini s->dregs[Alternate_Feature_Status] = 0x00; 25549ab747fSPaolo Bonzini s->dregs[Version_Chip_ID] = 0xa0; 25649ab747fSPaolo Bonzini s->dregs[Mono_Input_And_Output_Control] = 0xa0; 25749ab747fSPaolo Bonzini s->dregs[RESERVED_2] = 0x00; 25849ab747fSPaolo Bonzini s->dregs[Capture_Data_Format] = 0x00; 25949ab747fSPaolo Bonzini s->dregs[RESERVED_3] = 0x00; 26049ab747fSPaolo Bonzini s->dregs[Capture_Upper_Base_Count] = 0x00; 26149ab747fSPaolo Bonzini s->dregs[Capture_Lower_Base_Count] = 0x00; 26249ab747fSPaolo Bonzini } 26349ab747fSPaolo Bonzini 26449ab747fSPaolo Bonzini static void cs_audio_callback (void *opaque, int free) 26549ab747fSPaolo Bonzini { 26649ab747fSPaolo Bonzini CSState *s = opaque; 26749ab747fSPaolo Bonzini s->audio_free = free; 26849ab747fSPaolo Bonzini } 26949ab747fSPaolo Bonzini 27049ab747fSPaolo Bonzini static void cs_reset_voices (CSState *s, uint32_t val) 27149ab747fSPaolo Bonzini { 27249ab747fSPaolo Bonzini int xtal; 27349ab747fSPaolo Bonzini struct audsettings as; 2742d011091SHervé Poussineau IsaDmaClass *k = ISADMA_GET_CLASS(s->isa_dma); 27549ab747fSPaolo Bonzini 27649ab747fSPaolo Bonzini #ifdef DEBUG_XLAW 27749ab747fSPaolo Bonzini if (val == 0 || val == 32) 27849ab747fSPaolo Bonzini val = (1 << 4) | (1 << 5); 27949ab747fSPaolo Bonzini #endif 28049ab747fSPaolo Bonzini 28149ab747fSPaolo Bonzini xtal = val & 1; 28249ab747fSPaolo Bonzini as.freq = freqs[xtal][(val >> 1) & 7]; 28349ab747fSPaolo Bonzini 28449ab747fSPaolo Bonzini if (as.freq == -1) { 28549ab747fSPaolo Bonzini lerr ("unsupported frequency (val=%#x)\n", val); 28649ab747fSPaolo Bonzini goto error; 28749ab747fSPaolo Bonzini } 28849ab747fSPaolo Bonzini 28949ab747fSPaolo Bonzini as.nchannels = (val & (1 << 4)) ? 2 : 1; 29049ab747fSPaolo Bonzini as.endianness = 0; 29149ab747fSPaolo Bonzini s->tab = NULL; 29249ab747fSPaolo Bonzini 29349ab747fSPaolo Bonzini switch ((val >> 5) & ((s->dregs[MODE_And_ID] & MODE2) ? 7 : 3)) { 29449ab747fSPaolo Bonzini case 0: 29585bc5852SKővágó, Zoltán as.fmt = AUDIO_FORMAT_U8; 29649ab747fSPaolo Bonzini s->shift = as.nchannels == 2; 29749ab747fSPaolo Bonzini break; 29849ab747fSPaolo Bonzini 29949ab747fSPaolo Bonzini case 1: 30049ab747fSPaolo Bonzini s->tab = MuLawDecompressTable; 30149ab747fSPaolo Bonzini goto x_law; 30249ab747fSPaolo Bonzini case 3: 30349ab747fSPaolo Bonzini s->tab = ALawDecompressTable; 30449ab747fSPaolo Bonzini x_law: 30585bc5852SKővágó, Zoltán as.fmt = AUDIO_FORMAT_S16; 30649ab747fSPaolo Bonzini as.endianness = AUDIO_HOST_ENDIANNESS; 30749ab747fSPaolo Bonzini s->shift = as.nchannels == 2; 30849ab747fSPaolo Bonzini break; 30949ab747fSPaolo Bonzini 31049ab747fSPaolo Bonzini case 6: 31149ab747fSPaolo Bonzini as.endianness = 1; 312edd7541bSPaolo Bonzini /* fall through */ 31349ab747fSPaolo Bonzini case 2: 31485bc5852SKővágó, Zoltán as.fmt = AUDIO_FORMAT_S16; 31549ab747fSPaolo Bonzini s->shift = as.nchannels; 31649ab747fSPaolo Bonzini break; 31749ab747fSPaolo Bonzini 31849ab747fSPaolo Bonzini case 7: 31949ab747fSPaolo Bonzini case 4: 32049ab747fSPaolo Bonzini lerr ("attempt to use reserved format value (%#x)\n", val); 32149ab747fSPaolo Bonzini goto error; 32249ab747fSPaolo Bonzini 32349ab747fSPaolo Bonzini case 5: 32449ab747fSPaolo Bonzini lerr ("ADPCM 4 bit IMA compatible format is not supported\n"); 32549ab747fSPaolo Bonzini goto error; 32649ab747fSPaolo Bonzini } 32749ab747fSPaolo Bonzini 32849ab747fSPaolo Bonzini s->voice = AUD_open_out ( 32949ab747fSPaolo Bonzini &s->card, 33049ab747fSPaolo Bonzini s->voice, 33149ab747fSPaolo Bonzini "cs4231a", 33249ab747fSPaolo Bonzini s, 33349ab747fSPaolo Bonzini cs_audio_callback, 33449ab747fSPaolo Bonzini &as 33549ab747fSPaolo Bonzini ); 33649ab747fSPaolo Bonzini 33749ab747fSPaolo Bonzini if (s->dregs[Interface_Configuration] & PEN) { 33849ab747fSPaolo Bonzini if (!s->dma_running) { 3392d011091SHervé Poussineau k->hold_DREQ(s->isa_dma, s->dma); 34049ab747fSPaolo Bonzini AUD_set_active_out (s->voice, 1); 34149ab747fSPaolo Bonzini s->transferred = 0; 34249ab747fSPaolo Bonzini } 34349ab747fSPaolo Bonzini s->dma_running = 1; 34449ab747fSPaolo Bonzini } 34549ab747fSPaolo Bonzini else { 34649ab747fSPaolo Bonzini if (s->dma_running) { 3472d011091SHervé Poussineau k->release_DREQ(s->isa_dma, s->dma); 34849ab747fSPaolo Bonzini AUD_set_active_out (s->voice, 0); 34949ab747fSPaolo Bonzini } 35049ab747fSPaolo Bonzini s->dma_running = 0; 35149ab747fSPaolo Bonzini } 35249ab747fSPaolo Bonzini return; 35349ab747fSPaolo Bonzini 35449ab747fSPaolo Bonzini error: 35549ab747fSPaolo Bonzini if (s->dma_running) { 3562d011091SHervé Poussineau k->release_DREQ(s->isa_dma, s->dma); 35749ab747fSPaolo Bonzini AUD_set_active_out (s->voice, 0); 35849ab747fSPaolo Bonzini } 35949ab747fSPaolo Bonzini } 36049ab747fSPaolo Bonzini 36149ab747fSPaolo Bonzini static uint64_t cs_read (void *opaque, hwaddr addr, unsigned size) 36249ab747fSPaolo Bonzini { 36349ab747fSPaolo Bonzini CSState *s = opaque; 36449ab747fSPaolo Bonzini uint32_t saddr, iaddr, ret; 36549ab747fSPaolo Bonzini 36649ab747fSPaolo Bonzini saddr = addr; 36749ab747fSPaolo Bonzini iaddr = ~0U; 36849ab747fSPaolo Bonzini 36949ab747fSPaolo Bonzini switch (saddr) { 37049ab747fSPaolo Bonzini case Index_Address: 37149ab747fSPaolo Bonzini ret = s->regs[saddr] & ~0x80; 37249ab747fSPaolo Bonzini break; 37349ab747fSPaolo Bonzini 37449ab747fSPaolo Bonzini case Index_Data: 37549ab747fSPaolo Bonzini if (!(s->dregs[MODE_And_ID] & MODE2)) 37649ab747fSPaolo Bonzini iaddr = s->regs[Index_Address] & 0x0f; 37749ab747fSPaolo Bonzini else 37849ab747fSPaolo Bonzini iaddr = s->regs[Index_Address] & 0x1f; 37949ab747fSPaolo Bonzini 38049ab747fSPaolo Bonzini ret = s->dregs[iaddr]; 38149ab747fSPaolo Bonzini if (iaddr == Error_Status_And_Initialization) { 38249ab747fSPaolo Bonzini /* keep SEAL happy */ 38349ab747fSPaolo Bonzini if (s->aci_counter) { 38449ab747fSPaolo Bonzini ret |= 1 << 5; 38549ab747fSPaolo Bonzini s->aci_counter -= 1; 38649ab747fSPaolo Bonzini } 38749ab747fSPaolo Bonzini } 38849ab747fSPaolo Bonzini break; 38949ab747fSPaolo Bonzini 39049ab747fSPaolo Bonzini default: 39149ab747fSPaolo Bonzini ret = s->regs[saddr]; 39249ab747fSPaolo Bonzini break; 39349ab747fSPaolo Bonzini } 39449ab747fSPaolo Bonzini dolog ("read %d:%d -> %d\n", saddr, iaddr, ret); 39549ab747fSPaolo Bonzini return ret; 39649ab747fSPaolo Bonzini } 39749ab747fSPaolo Bonzini 39849ab747fSPaolo Bonzini static void cs_write (void *opaque, hwaddr addr, 39949ab747fSPaolo Bonzini uint64_t val64, unsigned size) 40049ab747fSPaolo Bonzini { 40149ab747fSPaolo Bonzini CSState *s = opaque; 40249ab747fSPaolo Bonzini uint32_t saddr, iaddr, val; 40349ab747fSPaolo Bonzini 40449ab747fSPaolo Bonzini saddr = addr; 40549ab747fSPaolo Bonzini val = val64; 40649ab747fSPaolo Bonzini 40749ab747fSPaolo Bonzini switch (saddr) { 40849ab747fSPaolo Bonzini case Index_Address: 40949ab747fSPaolo Bonzini if (!(s->regs[Index_Address] & MCE) && (val & MCE) 41049ab747fSPaolo Bonzini && (s->dregs[Interface_Configuration] & (3 << 3))) 41149ab747fSPaolo Bonzini s->aci_counter = conf.aci_counter; 41249ab747fSPaolo Bonzini 41349ab747fSPaolo Bonzini s->regs[Index_Address] = val & ~(1 << 7); 41449ab747fSPaolo Bonzini break; 41549ab747fSPaolo Bonzini 41649ab747fSPaolo Bonzini case Index_Data: 41749ab747fSPaolo Bonzini if (!(s->dregs[MODE_And_ID] & MODE2)) 41849ab747fSPaolo Bonzini iaddr = s->regs[Index_Address] & 0x0f; 41949ab747fSPaolo Bonzini else 42049ab747fSPaolo Bonzini iaddr = s->regs[Index_Address] & 0x1f; 42149ab747fSPaolo Bonzini 42249ab747fSPaolo Bonzini switch (iaddr) { 42349ab747fSPaolo Bonzini case RESERVED: 42449ab747fSPaolo Bonzini case RESERVED_2: 42549ab747fSPaolo Bonzini case RESERVED_3: 42649ab747fSPaolo Bonzini lwarn ("attempt to write %#x to reserved indirect register %d\n", 42749ab747fSPaolo Bonzini val, iaddr); 42849ab747fSPaolo Bonzini break; 42949ab747fSPaolo Bonzini 43049ab747fSPaolo Bonzini case FS_And_Playback_Data_Format: 43149ab747fSPaolo Bonzini if (s->regs[Index_Address] & MCE) { 43249ab747fSPaolo Bonzini cs_reset_voices (s, val); 43349ab747fSPaolo Bonzini } 43449ab747fSPaolo Bonzini else { 43549ab747fSPaolo Bonzini if (s->dregs[Alternate_Feature_Status] & PMCE) { 43649ab747fSPaolo Bonzini val = (val & ~0x0f) | (s->dregs[iaddr] & 0x0f); 43749ab747fSPaolo Bonzini cs_reset_voices (s, val); 43849ab747fSPaolo Bonzini } 43949ab747fSPaolo Bonzini else { 44049ab747fSPaolo Bonzini lwarn ("[P]MCE(%#x, %#x) is not set, val=%#x\n", 44149ab747fSPaolo Bonzini s->regs[Index_Address], 44249ab747fSPaolo Bonzini s->dregs[Alternate_Feature_Status], 44349ab747fSPaolo Bonzini val); 44449ab747fSPaolo Bonzini break; 44549ab747fSPaolo Bonzini } 44649ab747fSPaolo Bonzini } 44749ab747fSPaolo Bonzini s->dregs[iaddr] = val; 44849ab747fSPaolo Bonzini break; 44949ab747fSPaolo Bonzini 45049ab747fSPaolo Bonzini case Interface_Configuration: 45149ab747fSPaolo Bonzini val &= ~(1 << 5); /* D5 is reserved */ 45249ab747fSPaolo Bonzini s->dregs[iaddr] = val; 45349ab747fSPaolo Bonzini if (val & PPIO) { 45449ab747fSPaolo Bonzini lwarn ("PIO is not supported (%#x)\n", val); 45549ab747fSPaolo Bonzini break; 45649ab747fSPaolo Bonzini } 45749ab747fSPaolo Bonzini if (val & PEN) { 45849ab747fSPaolo Bonzini if (!s->dma_running) { 45949ab747fSPaolo Bonzini cs_reset_voices (s, s->dregs[FS_And_Playback_Data_Format]); 46049ab747fSPaolo Bonzini } 46149ab747fSPaolo Bonzini } 46249ab747fSPaolo Bonzini else { 46349ab747fSPaolo Bonzini if (s->dma_running) { 4642d011091SHervé Poussineau IsaDmaClass *k = ISADMA_GET_CLASS(s->isa_dma); 4652d011091SHervé Poussineau k->release_DREQ(s->isa_dma, s->dma); 46649ab747fSPaolo Bonzini AUD_set_active_out (s->voice, 0); 46749ab747fSPaolo Bonzini s->dma_running = 0; 46849ab747fSPaolo Bonzini } 46949ab747fSPaolo Bonzini } 47049ab747fSPaolo Bonzini break; 47149ab747fSPaolo Bonzini 47249ab747fSPaolo Bonzini case Error_Status_And_Initialization: 47349ab747fSPaolo Bonzini lwarn ("attempt to write to read only register %d\n", iaddr); 47449ab747fSPaolo Bonzini break; 47549ab747fSPaolo Bonzini 47649ab747fSPaolo Bonzini case MODE_And_ID: 47749ab747fSPaolo Bonzini dolog ("val=%#x\n", val); 47849ab747fSPaolo Bonzini if (val & MODE2) 47949ab747fSPaolo Bonzini s->dregs[iaddr] |= MODE2; 48049ab747fSPaolo Bonzini else 48149ab747fSPaolo Bonzini s->dregs[iaddr] &= ~MODE2; 48249ab747fSPaolo Bonzini break; 48349ab747fSPaolo Bonzini 48449ab747fSPaolo Bonzini case Alternate_Feature_Enable_I: 48549ab747fSPaolo Bonzini if (val & TE) 48649ab747fSPaolo Bonzini lerr ("timer is not yet supported\n"); 48749ab747fSPaolo Bonzini s->dregs[iaddr] = val; 48849ab747fSPaolo Bonzini break; 48949ab747fSPaolo Bonzini 49049ab747fSPaolo Bonzini case Alternate_Feature_Status: 49149ab747fSPaolo Bonzini if ((s->dregs[iaddr] & PI) && !(val & PI)) { 49249ab747fSPaolo Bonzini /* XXX: TI CI */ 49349ab747fSPaolo Bonzini qemu_irq_lower (s->pic); 49449ab747fSPaolo Bonzini s->regs[Status] &= ~INT; 49549ab747fSPaolo Bonzini } 49649ab747fSPaolo Bonzini s->dregs[iaddr] = val; 49749ab747fSPaolo Bonzini break; 49849ab747fSPaolo Bonzini 49949ab747fSPaolo Bonzini case Version_Chip_ID: 50049ab747fSPaolo Bonzini lwarn ("write to Version_Chip_ID register %#x\n", val); 50149ab747fSPaolo Bonzini s->dregs[iaddr] = val; 50249ab747fSPaolo Bonzini break; 50349ab747fSPaolo Bonzini 50449ab747fSPaolo Bonzini default: 50549ab747fSPaolo Bonzini s->dregs[iaddr] = val; 50649ab747fSPaolo Bonzini break; 50749ab747fSPaolo Bonzini } 50849ab747fSPaolo Bonzini dolog ("written value %#x to indirect register %d\n", val, iaddr); 50949ab747fSPaolo Bonzini break; 51049ab747fSPaolo Bonzini 51149ab747fSPaolo Bonzini case Status: 51249ab747fSPaolo Bonzini if (s->regs[Status] & INT) { 51349ab747fSPaolo Bonzini qemu_irq_lower (s->pic); 51449ab747fSPaolo Bonzini } 51549ab747fSPaolo Bonzini s->regs[Status] &= ~INT; 51649ab747fSPaolo Bonzini s->dregs[Alternate_Feature_Status] &= ~(PI | CI | TI); 51749ab747fSPaolo Bonzini break; 51849ab747fSPaolo Bonzini 51949ab747fSPaolo Bonzini case PIO_Data: 52049ab747fSPaolo Bonzini lwarn ("attempt to write value %#x to PIO register\n", val); 52149ab747fSPaolo Bonzini break; 52249ab747fSPaolo Bonzini } 52349ab747fSPaolo Bonzini } 52449ab747fSPaolo Bonzini 52549ab747fSPaolo Bonzini static int cs_write_audio (CSState *s, int nchan, int dma_pos, 52649ab747fSPaolo Bonzini int dma_len, int len) 52749ab747fSPaolo Bonzini { 52849ab747fSPaolo Bonzini int temp, net; 52949ab747fSPaolo Bonzini uint8_t tmpbuf[4096]; 5302d011091SHervé Poussineau IsaDmaClass *k = ISADMA_GET_CLASS(s->isa_dma); 53149ab747fSPaolo Bonzini 53249ab747fSPaolo Bonzini temp = len; 53349ab747fSPaolo Bonzini net = 0; 53449ab747fSPaolo Bonzini 53549ab747fSPaolo Bonzini while (temp) { 53649ab747fSPaolo Bonzini int left = dma_len - dma_pos; 53749ab747fSPaolo Bonzini int copied; 53849ab747fSPaolo Bonzini size_t to_copy; 53949ab747fSPaolo Bonzini 54049ab747fSPaolo Bonzini to_copy = audio_MIN (temp, left); 54149ab747fSPaolo Bonzini if (to_copy > sizeof (tmpbuf)) { 54249ab747fSPaolo Bonzini to_copy = sizeof (tmpbuf); 54349ab747fSPaolo Bonzini } 54449ab747fSPaolo Bonzini 5452d011091SHervé Poussineau copied = k->read_memory(s->isa_dma, nchan, tmpbuf, dma_pos, to_copy); 54649ab747fSPaolo Bonzini if (s->tab) { 54749ab747fSPaolo Bonzini int i; 54849ab747fSPaolo Bonzini int16_t linbuf[4096]; 54949ab747fSPaolo Bonzini 55049ab747fSPaolo Bonzini for (i = 0; i < copied; ++i) 55149ab747fSPaolo Bonzini linbuf[i] = s->tab[tmpbuf[i]]; 55249ab747fSPaolo Bonzini copied = AUD_write (s->voice, linbuf, copied << 1); 55349ab747fSPaolo Bonzini copied >>= 1; 55449ab747fSPaolo Bonzini } 55549ab747fSPaolo Bonzini else { 55649ab747fSPaolo Bonzini copied = AUD_write (s->voice, tmpbuf, copied); 55749ab747fSPaolo Bonzini } 55849ab747fSPaolo Bonzini 55949ab747fSPaolo Bonzini temp -= copied; 56049ab747fSPaolo Bonzini dma_pos = (dma_pos + copied) % dma_len; 56149ab747fSPaolo Bonzini net += copied; 56249ab747fSPaolo Bonzini 56349ab747fSPaolo Bonzini if (!copied) { 56449ab747fSPaolo Bonzini break; 56549ab747fSPaolo Bonzini } 56649ab747fSPaolo Bonzini } 56749ab747fSPaolo Bonzini 56849ab747fSPaolo Bonzini return net; 56949ab747fSPaolo Bonzini } 57049ab747fSPaolo Bonzini 57149ab747fSPaolo Bonzini static int cs_dma_read (void *opaque, int nchan, int dma_pos, int dma_len) 57249ab747fSPaolo Bonzini { 57349ab747fSPaolo Bonzini CSState *s = opaque; 57449ab747fSPaolo Bonzini int copy, written; 57549ab747fSPaolo Bonzini int till = -1; 57649ab747fSPaolo Bonzini 57749ab747fSPaolo Bonzini copy = s->voice ? (s->audio_free >> (s->tab != NULL)) : dma_len; 57849ab747fSPaolo Bonzini 57949ab747fSPaolo Bonzini if (s->dregs[Pin_Control] & IEN) { 58049ab747fSPaolo Bonzini till = (s->dregs[Playback_Lower_Base_Count] 58149ab747fSPaolo Bonzini | (s->dregs[Playback_Upper_Base_Count] << 8)) << s->shift; 58249ab747fSPaolo Bonzini till -= s->transferred; 58349ab747fSPaolo Bonzini copy = audio_MIN (till, copy); 58449ab747fSPaolo Bonzini } 58549ab747fSPaolo Bonzini 58649ab747fSPaolo Bonzini if ((copy <= 0) || (dma_len <= 0)) { 58749ab747fSPaolo Bonzini return dma_pos; 58849ab747fSPaolo Bonzini } 58949ab747fSPaolo Bonzini 59049ab747fSPaolo Bonzini written = cs_write_audio (s, nchan, dma_pos, dma_len, copy); 59149ab747fSPaolo Bonzini 59249ab747fSPaolo Bonzini dma_pos = (dma_pos + written) % dma_len; 59349ab747fSPaolo Bonzini s->audio_free -= (written << (s->tab != NULL)); 59449ab747fSPaolo Bonzini 59549ab747fSPaolo Bonzini if (written == till) { 59649ab747fSPaolo Bonzini s->regs[Status] |= INT; 59749ab747fSPaolo Bonzini s->dregs[Alternate_Feature_Status] |= PI; 59849ab747fSPaolo Bonzini s->transferred = 0; 59949ab747fSPaolo Bonzini qemu_irq_raise (s->pic); 60049ab747fSPaolo Bonzini } 60149ab747fSPaolo Bonzini else { 60249ab747fSPaolo Bonzini s->transferred += written; 60349ab747fSPaolo Bonzini } 60449ab747fSPaolo Bonzini 60549ab747fSPaolo Bonzini return dma_pos; 60649ab747fSPaolo Bonzini } 60749ab747fSPaolo Bonzini 60849ab747fSPaolo Bonzini static int cs4231a_pre_load (void *opaque) 60949ab747fSPaolo Bonzini { 61049ab747fSPaolo Bonzini CSState *s = opaque; 61149ab747fSPaolo Bonzini 61249ab747fSPaolo Bonzini if (s->dma_running) { 6132d011091SHervé Poussineau IsaDmaClass *k = ISADMA_GET_CLASS(s->isa_dma); 6142d011091SHervé Poussineau k->release_DREQ(s->isa_dma, s->dma); 61549ab747fSPaolo Bonzini AUD_set_active_out (s->voice, 0); 61649ab747fSPaolo Bonzini } 61749ab747fSPaolo Bonzini s->dma_running = 0; 61849ab747fSPaolo Bonzini return 0; 61949ab747fSPaolo Bonzini } 62049ab747fSPaolo Bonzini 62149ab747fSPaolo Bonzini static int cs4231a_post_load (void *opaque, int version_id) 62249ab747fSPaolo Bonzini { 62349ab747fSPaolo Bonzini CSState *s = opaque; 62449ab747fSPaolo Bonzini 62549ab747fSPaolo Bonzini if (s->dma_running && (s->dregs[Interface_Configuration] & PEN)) { 62649ab747fSPaolo Bonzini s->dma_running = 0; 62749ab747fSPaolo Bonzini cs_reset_voices (s, s->dregs[FS_And_Playback_Data_Format]); 62849ab747fSPaolo Bonzini } 62949ab747fSPaolo Bonzini return 0; 63049ab747fSPaolo Bonzini } 63149ab747fSPaolo Bonzini 63249ab747fSPaolo Bonzini static const VMStateDescription vmstate_cs4231a = { 63349ab747fSPaolo Bonzini .name = "cs4231a", 63449ab747fSPaolo Bonzini .version_id = 1, 63549ab747fSPaolo Bonzini .minimum_version_id = 1, 63649ab747fSPaolo Bonzini .pre_load = cs4231a_pre_load, 63749ab747fSPaolo Bonzini .post_load = cs4231a_post_load, 63849ab747fSPaolo Bonzini .fields = (VMStateField[]) { 63949ab747fSPaolo Bonzini VMSTATE_UINT32_ARRAY (regs, CSState, CS_REGS), 64049ab747fSPaolo Bonzini VMSTATE_BUFFER (dregs, CSState), 64149ab747fSPaolo Bonzini VMSTATE_INT32 (dma_running, CSState), 64249ab747fSPaolo Bonzini VMSTATE_INT32 (audio_free, CSState), 64349ab747fSPaolo Bonzini VMSTATE_INT32 (transferred, CSState), 64449ab747fSPaolo Bonzini VMSTATE_INT32 (aci_counter, CSState), 64549ab747fSPaolo Bonzini VMSTATE_END_OF_LIST () 64649ab747fSPaolo Bonzini } 64749ab747fSPaolo Bonzini }; 64849ab747fSPaolo Bonzini 64949ab747fSPaolo Bonzini static const MemoryRegionOps cs_ioport_ops = { 65049ab747fSPaolo Bonzini .read = cs_read, 65149ab747fSPaolo Bonzini .write = cs_write, 65249ab747fSPaolo Bonzini .impl = { 65349ab747fSPaolo Bonzini .min_access_size = 1, 65449ab747fSPaolo Bonzini .max_access_size = 1, 65549ab747fSPaolo Bonzini } 65649ab747fSPaolo Bonzini }; 65749ab747fSPaolo Bonzini 658db895a1eSAndreas Färber static void cs4231a_initfn (Object *obj) 65949ab747fSPaolo Bonzini { 660db895a1eSAndreas Färber CSState *s = CS4231A (obj); 66149ab747fSPaolo Bonzini 66264bde0f3SPaolo Bonzini memory_region_init_io (&s->ioports, OBJECT(s), &cs_ioport_ops, s, 66364bde0f3SPaolo Bonzini "cs4231a", 4); 664db895a1eSAndreas Färber } 665db895a1eSAndreas Färber 666db895a1eSAndreas Färber static void cs4231a_realizefn (DeviceState *dev, Error **errp) 667db895a1eSAndreas Färber { 668db895a1eSAndreas Färber ISADevice *d = ISA_DEVICE (dev); 669db895a1eSAndreas Färber CSState *s = CS4231A (dev); 6702d011091SHervé Poussineau IsaDmaClass *k; 671db895a1eSAndreas Färber 6722d011091SHervé Poussineau s->isa_dma = isa_get_dma(isa_bus_from_device(d), s->dma); 673c9073238SThomas Huth if (!s->isa_dma) { 674c9073238SThomas Huth error_setg(errp, "ISA controller does not support DMA"); 675c9073238SThomas Huth return; 676c9073238SThomas Huth } 677c9073238SThomas Huth 678c9073238SThomas Huth isa_init_irq(d, &s->pic, s->irq); 6792d011091SHervé Poussineau k = ISADMA_GET_CLASS(s->isa_dma); 6802d011091SHervé Poussineau k->register_channel(s->isa_dma, s->dma, cs_dma_read, s); 681db895a1eSAndreas Färber 682db895a1eSAndreas Färber isa_register_ioport (d, &s->ioports, s->port); 68349ab747fSPaolo Bonzini 68449ab747fSPaolo Bonzini AUD_register_card ("cs4231a", &s->card); 68549ab747fSPaolo Bonzini } 68649ab747fSPaolo Bonzini 68736cd6f6fSPaolo Bonzini static int cs4231a_init (ISABus *bus) 68849ab747fSPaolo Bonzini { 689a3dcca56SAndreas Färber isa_create_simple (bus, TYPE_CS4231A); 69049ab747fSPaolo Bonzini return 0; 69149ab747fSPaolo Bonzini } 69249ab747fSPaolo Bonzini 69349ab747fSPaolo Bonzini static Property cs4231a_properties[] = { 694c7bcc85dSPaolo Bonzini DEFINE_PROP_UINT32 ("iobase", CSState, port, 0x534), 69549ab747fSPaolo Bonzini DEFINE_PROP_UINT32 ("irq", CSState, irq, 9), 69649ab747fSPaolo Bonzini DEFINE_PROP_UINT32 ("dma", CSState, dma, 3), 69749ab747fSPaolo Bonzini DEFINE_PROP_END_OF_LIST (), 69849ab747fSPaolo Bonzini }; 69949ab747fSPaolo Bonzini 70049ab747fSPaolo Bonzini static void cs4231a_class_initfn (ObjectClass *klass, void *data) 70149ab747fSPaolo Bonzini { 70249ab747fSPaolo Bonzini DeviceClass *dc = DEVICE_CLASS (klass); 703db895a1eSAndreas Färber 704db895a1eSAndreas Färber dc->realize = cs4231a_realizefn; 705a3dcca56SAndreas Färber dc->reset = cs4231a_reset; 706125ee0edSMarcel Apfelbaum set_bit(DEVICE_CATEGORY_SOUND, dc->categories); 70749ab747fSPaolo Bonzini dc->desc = "Crystal Semiconductor CS4231A"; 70849ab747fSPaolo Bonzini dc->vmsd = &vmstate_cs4231a; 70949ab747fSPaolo Bonzini dc->props = cs4231a_properties; 71049ab747fSPaolo Bonzini } 71149ab747fSPaolo Bonzini 71249ab747fSPaolo Bonzini static const TypeInfo cs4231a_info = { 713a3dcca56SAndreas Färber .name = TYPE_CS4231A, 71449ab747fSPaolo Bonzini .parent = TYPE_ISA_DEVICE, 71549ab747fSPaolo Bonzini .instance_size = sizeof (CSState), 716db895a1eSAndreas Färber .instance_init = cs4231a_initfn, 71749ab747fSPaolo Bonzini .class_init = cs4231a_class_initfn, 71849ab747fSPaolo Bonzini }; 71949ab747fSPaolo Bonzini 72049ab747fSPaolo Bonzini static void cs4231a_register_types (void) 72149ab747fSPaolo Bonzini { 72249ab747fSPaolo Bonzini type_register_static (&cs4231a_info); 72336cd6f6fSPaolo Bonzini isa_register_soundhw("cs4231a", "CS4231A", cs4231a_init); 72449ab747fSPaolo Bonzini } 72549ab747fSPaolo Bonzini 72649ab747fSPaolo Bonzini type_init (cs4231a_register_types) 727