prototype of a driver for the digigram lx6464es 64 channel ethersoundinterface. Signed-off-by: Tim Blechmann <tim@xxxxxxxxxx>--- include/linux/pci_ids.h | 5 + sound/pci/Kconfig | 10 + sound/pci/Makefile | 1 + sound/pci/lx6464es/LXES_registers.h | 122 +++ sound/pci/lx6464es/Makefile | 13 + sound/pci/lx6464es/PcxErr_e.h | 680 ++++++++++++++++ sound/pci/lx6464es/ethersound.h | 55 ++ sound/pci/lx6464es/if_drv_mb.h | 287 +++++++ sound/pci/lx6464es/lx6464es.c | 1179 ++++++++++++++++++++++++++++ sound/pci/lx6464es/lx6464es.h | 119 +++ sound/pci/lx6464es/lx_core.c | 1453 +++++++++++++++++++++++++++++++++++ sound/pci/lx6464es/lx_core.h | 239 ++++++ 12 files changed, 4163 insertions(+), 0 deletions(-) create mode 100644 sound/pci/lx6464es/LXES_registers.h create mode 100644 sound/pci/lx6464es/Makefile create mode 100644 sound/pci/lx6464es/PcxErr_e.h create mode 100644 sound/pci/lx6464es/ethersound.h create mode 100644 sound/pci/lx6464es/if_drv_mb.h create mode 100644 sound/pci/lx6464es/lx6464es.c create mode 100644 sound/pci/lx6464es/lx6464es.h create mode 100644 sound/pci/lx6464es/lx_core.c create mode 100644 sound/pci/lx6464es/lx_core.h diff --git a/include/linux/pci_ids.h b/include/linux/pci_ids.hindex 02c18b9..0a8c36d 100644--- a/include/linux/pci_ids.h+++ b/include/linux/pci_ids.h@@ -976,6 +976,7 @@ #define PCI_DEVICE_ID_PLX_PCI200SYN 0x3196 #define PCI_DEVICE_ID_PLX_9030 0x9030 #define PCI_DEVICE_ID_PLX_9050 0x9050+#define PCI_DEVICE_ID_PLX_9056 0x9056 #define PCI_DEVICE_ID_PLX_9080 0x9080 #define PCI_DEVICE_ID_PLX_GTEK_SERIAL2 0xa001 @@ -1813,6 +1814,10 @@ #define PCI_SUBDEVICE_ID_HYPERCOPE_METRO 0x0107 #define PCI_SUBDEVICE_ID_HYPERCOPE_CHAMP2 0x0108 +#define PCI_VENDOR_ID_DIGIGRAM 0x1369+#define PCI_SUBDEVICE_ID_DIGIGRAM_LX6464ES_SERIAL_SUBSYSTEM 0xc001+#define PCI_SUBDEVICE_ID_DIGIGRAM_LX6464ES_CAE_SERIAL_SUBSYSTEM 0xc002+ #define PCI_VENDOR_ID_KAWASAKI 0x136b #define PCI_DEVICE_ID_MCHIP_KL5A72002 0xff01 diff --git a/sound/pci/Kconfig b/sound/pci/Kconfigindex ca25e61..8002ae4 100644--- a/sound/pci/Kconfig+++ b/sound/pci/Kconfig@@ -622,6 +622,16 @@ config SND_KORG1212 To compile this driver as a module, choose M here: the module will be called snd-korg1212. +config SND_LX6464ES+ tristate "Digigram LX6464ES"+ select SND_PCM+ help+ Say Y here to include support for Digigram LX6464ES boards.++ To compile this driver as a module, choose M here: the module+ will be called snd-lx6464es.++ config SND_MAESTRO3 tristate "ESS Allegro/Maestro3" select SND_AC97_CODECdiff --git a/sound/pci/Makefile b/sound/pci/Makefileindex 65b25d2..7d83e08 100644--- a/sound/pci/Makefile+++ b/sound/pci/Makefile@@ -62,6 +62,7 @@ obj-$(CONFIG_SND) += \ ca0106/ \ cs46xx/ \ cs5535audio/ \+ lx6464es/ \ echoaudio/ \ emu10k1/ \ hda/ \diff --git a/sound/pci/lx6464es/LXES_registers.h b/sound/pci/lx6464es/LXES_registers.hnew file mode 100644index 0000000..f665084--- /dev/null+++ b/sound/pci/lx6464es/LXES_registers.h@@ -0,0 +1,122 @@+/**+* Copyright (C) 2006 DIGIGRAM S.A.+*+* @file LXES_registers.h+* @brief offsets and values of the PCI mapped registers for LX6464ES+*/++#if !defined(AFX_PCIESDEFINE_H)+#define AFX_PCIESDEFINE_H++#define Reg_CSM 0x400+#define Reg_CSM_MR 0x00000002+#define Reg_CSM_MC 0x00000001+#define REG_CRM_NUMBER 12+#define Reg_CRM1 0x401+#define Reg_CRM2 0x402+#define Reg_CRM3 0x403+#define Reg_CRM4 0x404+#define Reg_CRM5 0x405+#define Reg_CRM6 0x406+#define Reg_CRM7 0x407+#define Reg_CRM8 0x408+#define Reg_CRM9 0x409+#define Reg_CRM10 0x40a+#define Reg_CRM11 0x40b+#define Reg_CRM12 0x40c++#define Reg_ICR 0x410+#define Reg_CVR 0x411+#define Reg_ISR 0x412+#define Reg_RXHTXH 0x413+#define Reg_RXMTXM 0x414+#define Reg_RHLTXL 0x415+#define Reg_RESETDSP 0x416++#define Reg_CSUF 0x420+// write+#define Reg_CSUF_TMS0 0x00000001+#define Reg_CSUF_TDI0 0x00000002+#define Reg_CSUF_V0 0x00000004+#define Reg_CSUF_TMS1 0x00000008+#define Reg_CSUF_TDI1 0x00000010+#define Reg_CSUF_V1 0x00000020+#define Reg_CSUF_TMS2 0x00000040+#define Reg_CSUF_TDI2 0x00000080+#define Reg_CSUF_V2 0x00000100+#define Reg_CSUF_TMS3 0x00000200+#define Reg_CSUF_TDI3 0x00000400+#define Reg_CSUF_V3 0x00000800+#define Reg_CSUF_TMS4 0x00001000+#define Reg_CSUF_TDI4 0x00002000+#define Reg_CSUF_V4 0x00004000+#define Reg_CSUF_ENABLE_JTAG 0x80000000+// read+#define Reg_CSUF_TDO0 0x00000001+#define Reg_CSUF_TDO1 0x00000002+#define Reg_CSUF_TDO2 0x00000004+#define Reg_CSUF_TDO3 0x00000008+#define Reg_CSUF_TDO4 0x00000010+#define Reg_CSUF_B 0x00000020+#define Reg_CSUF_VERIFY_PATTERN 0x0000FFC0 // bit31 = enable jtag + bits 15..6 forced to 1+#define Reg_CSUF_VERIFY_LAST_WRITE 0x7FFF0000+#define Reg_CSUF_VERIFY_LAST_WRITE_OFFSET 16+++#define Reg_CSES 0x430+#define Reg_CRESMSB 0x431+#define Reg_CRESLSB 0x432+#define Reg_ADMACESMSB 0x433+#define Reg_ADMACESLSB 0x434++#define Reg_CONFES 0x440++// ConfES : partie lue depuis le xilinx+//+#define CONFES_READ_PART_MASK 0x00070000+//+#define IOCR_OUTPUTS_OFFSET 0 /**< (rw) offset for the number of OUTs in the ConfES register. */+#define IOCR_INPUTS_OFFSET 8 /**< (rw) offset for the number of INs in the ConfES register. */+#define INTERPRETER_VERS_OFFSET 16 /**< (ro) offset for ES command interpreter version info. */+#define INTERPRETER_VERS_MASK 0x07 /**< bits 16/17/18 give the interpreter version, amongst the following values.*/+#define INTERPRETER_V2 0 /**< V1/V2 EtherSound, or early V3 prototypes .*/+#define INTERPRETER_ES100 1 /**< V3 EtherSound, ES100. */+//#define INTERPRETER_ESGIGA 2+++// ConfES : partie ecrite a partir des info dans la registry+//+#define CONFES_WRITE_PART_MASK 0x00F80000+//+#define FREQ_RATIO_OFFSET 19 /**< (rw) offset for frequency ratio in the ConfES register. */+#define MAC_PROG_SET_BIT 23 /**< (rw) MAC PROG bit. */+#define FREQ_RATIO_SINGLE_MODE 0x01 /**< value for single mode frequency ratio: sample rate = frequency rate. */+#define FREQ_RATIO_DUAL_MODE 0x02 /**< value for dual mode frequency ratio: sample rate = frequency rate * 2. */+#define FREQ_RATIO_QUAD_MODE 0x04 /**< value for quad mode frequency ratio: sample rate = frequency rate * 4. */++#define Reg_CONFESIOMR 0x441++// IOMR (32Bits)+//+#define IMR_OFFSET 16 /**< MSWord = IMR, LSWord = OMR*/+#define OMR_OFFSET 0++// offsets in OMR or IMR words+#define IOMR_PID_OFFSET 0 /**< offset for PID field in IOMR : unused for now. */+#define IOMR_PID_DEFAULT 0x0 /**< default PID value = 0. */+#define IOMR_PTY_OFFSET 6 /**< offset for Packet Type in IOMR : always audio 0x01. */+#define IOMR_PTY_AUDIO 0x1 /**< packet type AUDIO. */+#define IOMR_PTY_DEFAULT IOMR_PTY_AUDIO /**< default value for Packet Type. */+#define IOMR_BNP_OFFSET 8 /**< bundle number offset. */+#define IOMR_IOD_OFFSET 15 /**< offset for the bundle direction bit : significant if a bi-dir end of loop is configured in the network.*/+#define IOMR_IOD_UPSTREAM 0x1 /**< "upstream" value.*/+#define IOMR_IOD_DOWNSTREAM 0x0 /**< "downstream" value.*/+#define OMR_IOD_DEFAULT IOMR_IOD_UPSTREAM /**< default value for OMR:IOD.*/+#define IMR_IOD_DEFAULT IOMR_IOD_DOWNSTREAM /**< default value for IMR:IOD.*/+++#endif // !defined(AFX_PCIESDEFINE_H)++//*********************************************************************************+// $history:$+//diff --git a/sound/pci/lx6464es/Makefile b/sound/pci/lx6464es/Makefilenew file mode 100644index 0000000..a23257c--- /dev/null+++ b/sound/pci/lx6464es/Makefile@@ -0,0 +1,13 @@+ifeq ($(KERNELRELEASE),)+snd-lx6464es-objs := lx6464es.o lx_core.o+obj-m += snd-lx6464es.o+all:+ make -C /lib/modules/$(shell uname -r)/build M=$(PWD) modules++clean:+ make -C /lib/modules/$(shell uname -r)/build M=$(PWD) clean++else+snd-lx6464es-objs := lx6464es.o lx_core.o+obj-$(CONFIG_SND_LX6464ES) += snd-lx6464es.o+endifdiff --git a/sound/pci/lx6464es/PcxErr_e.h b/sound/pci/lx6464es/PcxErr_e.hnew file mode 100644index 0000000..dd48c46--- /dev/null+++ b/sound/pci/lx6464es/PcxErr_e.h@@ -0,0 +1,680 @@+// *************************************************************************+//+// COPYRIGHT 1996-2000 DIGIGRAM.+//+// DIGIGRAM+//+// **************************************************************************++#ifndef _PCXERROR_H_+//{+#define _PCXERROR_H_++// ********************************************************************+// Error code structure+// ********************+//+// An error code is 16 bits :+//+// Bit 15 : 1 = error, 0 = warning+//+// Bits 14-->11 : source of error/warning+// 0x1 = TOOLS source error/warning+// 0x2 = API source error/warning+// 0x3 = AUDIO source error/warning+// 0x4 = DRV source error/warning+// 0x5 = VPCX source error/warning+// 0x6 = DISPATCHER source error/warning+// 0x7 = reserved+// 0x8 = BOARD source error/warning+// 0x9-->0xe = reserved+// 0xf = free for USER usage+//+// Bits 8-->10 : Class code+//+// Bits 0-->7 : Code+//+// *********************************************************************++// Return value when OK+// ********************++#define SUCCESS 0+++// Bits masks+// **********++#define ERROR_MASK 0x8000++#define SOURCE_MASK 0x7800++#define E_SOURCE_BOARD 0x4000 //8 >> 1+#define E_SOURCE_DRV 0x2000 //4 >> 1+#define E_SOURCE_API 0x1000 //2 >> 1+// Error tools+#define E_SOURCE_TOOLS 0x0800 //1 >> 1+// Error pcxaudio+#define E_SOURCE_AUDIO 0x1800 //3 >> 1+// Error virtual pcx+#define E_SOURCE_VPCX 0x2800 //5 >> 1+// Error dispatcher+#define E_SOURCE_DISPATCHER 0x3000 //6 >> 1+// Error from CobraNet firmware+#define E_SOURCE_COBRANET 0x3800 //7 >> 1++#define E_SOURCE_USER 0x7800++#define CLASS_MASK 0x0700++#define CODE_MASK 0x00FF++// Bits values+// ***********++// Values for the error/warning bit+// --------------------------------+#define ERROR_VALUE 0x8000+#define WARNING_VALUE 0x0000++// Class values+// ------------+#define E_CLASS_GENERAL 0x0000+#define E_CLASS_INVALID_CMD 0x0100+#define E_CLASS_INVALID_STD_OBJECT 0x0200+#define E_CLASS_RSRC_IMPOSSIBLE 0x0300+#define E_CLASS_WRONG_CONTEXT 0x0400+#define E_CLASS_BAD_SPECIFIC_PARAMETER 0x0500+#define E_CLASS_REAL_TIME_ERROR 0x0600+#define E_CLASS_DIRECTSHOW 0x0700+#define E_CLASS_FREE 0x0700++// Complete API warning code for the general class+// -----------------------------------------------+#define WA_GN (WARNING_VALUE | E_SOURCE_API | E_CLASS_GENERAL)+#define WA_MORE_DATA (WA_GN | 0X01)+#define WA_NO_MORE_DATA (WA_GN | 0x02)+#define WA_ESTIMATED_VALUE (WA_GN | 0x03)+#define WA_GENERIC_WARNING (WA_GN | 0x04)++// Complete API error code for the general class+// ---------------------------------------------+#define EA_GN (ERROR_VALUE | E_SOURCE_API | E_CLASS_GENERAL)+#define EA_DRIVER_NOT_LOADED (EA_GN | 0x01)+#define EA_UNKNOWN_MEMORY_TYPE (EA_GN | 0x02)+#define EA_USE_XMS_MEMORY_IMPOSSIBLE (EA_GN | 0x03)+#define EA_CANT_TAKE_MUTEX (EA_GN | 0x04)+#define EA_VERSION_MISMATCH (EA_GN | 0x05)+#define EA_FILE_CORRUPTED (EA_GN | 0x06)+#define EA_DLL_NOT_LOADED (EA_GN | 0x07)+#define EA_UNKNOWN_BOARD (EA_GN | 0x08)+#define EA_CREATE_THREAD_FAILED (EA_GN | 0x09)+#define EA_FEATURE_NOT_AUTHORIZED (EA_GN | 0x0a)+#define EA_TOO_MUCH_DATA (EA_GN | 0x0b)+#define EA_COMMAND_NOT_AVAILABLE (EA_GN | 0x0c)++// Complete API warning code for the invalid command class+// -------------------------------------------------------+#define WA_IC (WARNING_VALUE | E_SOURCE_API | E_CLASS_INVALID_CMD)+#define WA_COMMAND_NOT_AVAILABLE (WA_IC | 0x01)++// Complete API error code for the invalid command class+// -----------------------------------------------------+#define EA_IC (ERROR_VALUE | E_SOURCE_API | E_CLASS_INVALID_CMD)++// Complete API warning code for the invalid standard object class+// -------------------------------------------------------------+#define WA_ISO (WARNING_VALUE | E_SOURCE_API | E_CLASS_INVALID_STD_OBJECT)+#define WA_INVALID_PARAMETER (WA_ISO | 0x01)++// Complete API error code for the invalid standard object class+// -------------------------------------------------------------+#define EA_ISO (ERROR_VALUE | E_SOURCE_API | E_CLASS_INVALID_STD_OBJECT)+#define EA_INVALID_BUFFER (EA_ISO | 0x01)+#define EA_INVALID_PCX_HANDLE (EA_ISO | 0x02)+#define EA_INVALID_PIPE (EA_ISO | 0x03)+#define EA_INVALID_PARAMETER (EA_ISO | 0x04)+#define EA_INVALID_DATA_LENGTH (EA_ISO | 0x05)+#define EA_INVALID_FUNCTION (EA_ISO | 0x06)++// Complete API error code for impossible resource allocation class+// ----------------------------------------------------------------+#define EA_RI (ERROR_VALUE | E_SOURCE_API | E_CLASS_RSRC_IMPOSSIBLE)+#define EA_ALLOCATE_ASYNC_IMPOSSIBLE (EA_RI | 0x01)+#define EA_ALLOCATE_CMD_BLK_IMPOSSIBLE (EA_RI | 0x02)+#define EA_ALLOCATE_MEMORY_IMPOSSIBLE (EA_RI | 0x03)++// Complete API error code for wrong call context class+// ----------------------------------------------------+#define EA_WCC (ERROR_VALUE | E_SOURCE_API | E_CLASS_WRONG_CONTEXT)+#define EA_BUFFER_REFUSED (EA_WCC | 0x01)+#define EA_WAIT_REQUEST_REFUSED (EA_WCC | 0x02)+#define EA_LOAD_DSP_REFUSED (EA_WCC | 0x03)+#define EA_SET_OEM_DLL_REFUSED (EA_WCC | 0x04)++// Complete API warning code for bad specific parameter class+// --------------------------------------------------------+#define WA_WSP (WARNING_VALUE | E_SOURCE_API | E_CLASS_BAD_SPECIFIC_PARAMETER)+#define WA_LEVEL_OUT_OF_RANGE (WA_WSP | 0x01)+#define WA_PARAMETER_ADJUSTED (WA_WSP | 0x02)++// Complete API error code for bad specific parameter class+// --------------------------------------------------------+#define EA_WSP (ERROR_VALUE | E_SOURCE_API | E_CLASS_BAD_SPECIFIC_PARAMETER)+#define EA_APPLICATION_NAME_SIZE (EA_WSP | 0x01)+#define EA_BAD_SYNC_SOURCE (EA_WSP | 0x02)+#define EA_OPEN_FILE_IMPOSSIBLE (EA_WSP | 0x03)+#define EA_READ_FILE_IMPOSSIBLE (EA_WSP | 0x04)+#define EA_FILE_TOO_LARGE (EA_WSP | 0x05)+#define EA_BAD_STREAM_NUMBER (EA_WSP | 0x06)+#define EA_BAD_AUDIO_NUMBER (EA_WSP | 0x07)+#define EA_BAD_EFFECT_ID (EA_WSP | 0x08)+#define EA_BAD_FREQUENCY_VALUE (EA_WSP | 0x09)+#define EA_MODE_NOT_AVAILABLE (EA_WSP | 0x0a)+#define EA_CONTEXT_NOT_AVAILABLE (EA_WSP | 0x0b)+#define EA_BAD_EFFECT_PARAMETER (EA_WSP | 0x0c)+#define EA_BAD_DSP_FEATURES (EA_WSP | 0x0d)+#define EA_BAD_PLAY_FORMATS (EA_WSP | 0x0e)+#define EA_BAD_RECORD_FORMATS (EA_WSP | 0x0f)+#define EA_BAD_PLAY_EFFECTS (EA_WSP | 0x10)+#define EA_BAD_RECORD_EFFECTS (EA_WSP | 0x11)+#define EA_BAD_SOUND_FORMAT (EA_WSP | 0x12)+#define EA_BAD_TIME_CODE_NUMBER (EA_WSP | 0x13)+#define EA_BAD_BAND_NUMBER (EA_WSP | 0x14)+#define EA_BAD_OEM_ID (EA_WSP | 0x15)+#define EA_INVALID_POINTER (EA_WSP | 0x16)+#define EA_BAD_MULTICHANNEL_FORMAT (EA_WSP | 0x17)++// VPCX ADD - Begin+// Complete API error code for direct show problems class+// ------------------------------------------------------+#define EA_DS (ERROR_VALUE | E_SOURCE_VPCX | E_CLASS_DIRECTSHOW)+#define EA_SRCFILTER_INSTANCIATION_FAILED (EA_DS | 0x01)+#define EA_DECODMIX_INSTANCIATION_FAILED (EA_DS | 0x02)+#define EA_RNDFILTER_INSTANCIATION_FAILED (EA_DS | 0x03)+#define EA_EFFECT_INSTANCIATION_FAILED (EA_DS | 0x04)+#define EA_DECFILTER_INSTANCIATION_FAILED (EA_DS | 0x05)+#define EA_CODFILTER_INSTANCIATION_FAILED (EA_DS | 0x06)+#define EA_CANNOT_GET_INTERFACE (EA_DS | 0x07)+#define EA_FILTER_GRAPH_ERROR (EA_DS | 0x08)+#define EA_NO_WAVE_DRIVER (EA_DS | 0x09)+#define EA_SRCFILTER_RUNTIME_ERROR (EA_DS | 0x0A)+#define EA_DECODMIX_RUNTIME_ERROR (EA_DS | 0x0B)+#define EA_RNDFILTER_RUNTIME_ERROR (EA_DS | 0x0C)+#define EA_FCFILTER_RUNTIME_ERROR (EA_DS | 0x0D)+#define EA_DEINTERLEAVEFILTER_INSTANCIATION_FAILED (EA_DS | 0x0E)+#define EA_INTERLEAVEFILTER_INSTANCIATION_FAILED (EA_DS | 0x0F)+#define EA_MPEGCODER_RUNTIME_ERROR (EA_DS | 0x10)+#define EA_PCCODERMPEG_RUNTIME_ERROR EA_MPEGCODER_RUNTIME_ERROR // for compatibility+#define EA_PCMCODER_RUNTIME_ERROR (EA_DS | 0x11)+#define EA_EFFECT_NOT_FOUND (EA_DS | 0x12)+#define EA_AACCODER_RUNTIME_ERROR (EA_DS | 0x13)+#define EA_WMACODER_RUNTIME_ERROR (EA_DS | 0x14)+// VPCX ADD - End+++// Complete API warning code for real time class+// ---------------------------------------------+#define WA_RT (WARNING_VALUE | E_SOURCE_API | E_CLASS_REAL_TIME_ERROR)+#define WA_CANNOT_CANCEL (WA_RT | 0x07)++// Complete DRV error code for the general class+// ---------------------------------------------+#define ED_GN (ERROR_VALUE | E_SOURCE_DRV | E_CLASS_GENERAL)+#define ED_CONCURRENCY (ED_GN | 0x01)+#define ED_DSP_CRASHED (ED_GN | 0x02)+#define ED_UNKNOWN_BOARD (ED_GN | 0x03)+#define ED_NOT_INSTALLED (ED_GN | 0x04)+#define ED_CANNOT_OPEN_SVC_MANAGER (ED_GN | 0x05)+#define ED_CANNOT_READ_REGISTRY (ED_GN | 0x06)+#define ED_DSP_VERSION_MISMATCH (ED_GN | 0x07)+#define ED_UNAVAILABLE_FEATURE (ED_GN | 0x08)+#define ED_CANCELLED (ED_GN | 0x09)+#define ED_NO_RESPONSE_AT_IRQA (ED_GN | 0x10)+#define ED_INVALID_ADDRESS (ED_GN | 0x11)+#define ED_DSP_CORRUPTED (ED_GN | 0x12)+#define ED_PENDING_OPERATION (ED_GN | 0x13)+#define ED_NET_ALLOCATE_MEMORY_IMPOSSIBLE (ED_GN | 0x14)+#define ED_NET_REGISTER_ERROR (ED_GN | 0x15)+#define ED_NET_THREAD_ERROR (ED_GN | 0x16)+#define ED_NET_OPEN_ERROR (ED_GN | 0x17)+#define ED_NET_CLOSE_ERROR (ED_GN | 0x18)+#define ED_NET_NO_MORE_PACKET (ED_GN | 0x19)+#define ED_NET_NO_MORE_BUFFER (ED_GN | 0x1A)+#define ED_NET_SEND_ERROR (ED_GN | 0x1B)+#define ED_NET_RECEIVE_ERROR (ED_GN | 0x1C)+#define ED_NET_WRONG_MSG_SIZE (ED_GN | 0x1D)+#define ED_NET_WAIT_ERROR (ED_GN | 0x1E)+#define ED_NET_EEPROM_ERROR (ED_GN | 0x1F)+#define ED_INVALID_RS232_COM_NUMBER (ED_GN | 0x20)+#define ED_INVALID_RS232_INIT (ED_GN | 0x21)+#define ED_FILE_ERROR (ED_GN | 0x22)+#define ED_INVALID_GPIO_CMD (ED_GN | 0x23)+#define ED_RS232_ALREADY_OPENED (ED_GN | 0x24)+#define ED_RS232_NOT_OPENED (ED_GN | 0x25)+#define ED_GPIO_ALREADY_OPENED (ED_GN | 0x26)+#define ED_GPIO_NOT_OPENED (ED_GN | 0x27)+#define ED_REGISTRY_ERROR (ED_GN | 0x28)// <- NCX+#define ED_INVALID_SERVICE (ED_GN | 0x29)// <- NCX++#define ED_READ_FILE_ALREADY_OPENED (ED_GN | 0x2a)// <- Decalage pour RCX (old 0x28)+#define ED_READ_FILE_INVALID_COMMAND (ED_GN | 0x2b)// ~+#define ED_READ_FILE_INVALID_PARAMETER (ED_GN | 0x2c)// ~+#define ED_READ_FILE_ALREADY_CLOSED (ED_GN | 0x2d)// ~+#define ED_READ_FILE_NO_INFORMATION (ED_GN | 0x2e)// ~+#define ED_READ_FILE_INVALID_HANDLE (ED_GN | 0x2f)// ~+#define ED_READ_FILE_END_OF_FILE (ED_GN | 0x30)// ~+#define ED_READ_FILE_ERROR (ED_GN | 0x31)// ~++#define ED_DSP_CRASHED_EXC_DSPSTACK_OVERFLOW (ED_GN | 0x32)// <- Decalage pour PCX (old 0x14)+#define ED_DSP_CRASHED_EXC_SYSSTACK_OVERFLOW (ED_GN | 0x33)// ~+#define ED_DSP_CRASHED_EXC_ILLEGAL (ED_GN | 0x34)// ~+#define ED_DSP_CRASHED_EXC_TIMER_REENTRY (ED_GN | 0x35)// ~+#define ED_DSP_CRASHED_EXC_FATAL_ERROR (ED_GN | 0x36)// ~++#define ED_FLASH_PCCARD_NOT_PRESENT (ED_GN | 0x37)++#define ED_NO_CURRENT_CLOCK (ED_GN | 0x38)+++// Complete DRV warning code for the general class+// ---------------------------------------------+#define WD_GN (WARNING_VALUE | E_SOURCE_DRV | E_CLASS_GENERAL)+#define WD_MORE_DATA (WD_GN | 0x01)+#define WD_NO_MORE_DATA (WD_GN | 0x02)+#define WD_DSP_VERSION_MISMATCH (WD_GN | 0x07)+#define WD_IBL_CLIPPED (WD_GN | 0x08) /**< on VX: when Internal Buffer Latency settings are not in accordance with granularity.*/+#define WD_FLASH_CARD_REINSERTED (WD_GN | 0x17)+++// Complete DRV error code for the invalid command class+// -----------------------------------------------------+#define ED_IC (ERROR_VALUE | E_SOURCE_DRV | E_CLASS_INVALID_CMD)+#define ED_INVALID_FAM (ED_IC | 0x01)+#define ED_INVALID_CMD_FAM1 (ED_IC | 0x02)+#define ED_INVALID_CMD_FAM2 (ED_IC | 0x03)+#define ED_INVALID_CMD_FAM3 (ED_IC | 0x04)+#define ED_INVALID_CMD_FAM4 (ED_IC | 0x05)+#define ED_INVALID_CMD_FAM5 (ED_IC | 0x06)+#define ED_INVALID_CMD_FAM6 (ED_IC | 0x07)+#define ED_INVALID_CMD_FAM7 (ED_IC | 0x08)+#define ED_INVALID_CMD_FAM8 (ED_IC | 0x09)+#define ED_INVALID_CMD_FAM9 (ED_IC | 0x0A)+#define ED_INVALID_CMD_FAM10 (ED_IC | 0x0B)+#define ED_INVALID_CMD_FAM11 (ED_IC | 0x0C)+#define ED_INVALID_CMD_FAM12 (ED_IC | 0x0D)+#define ED_INVALID_CMD_FAM13 (ED_IC | 0x0E)++// Complete DRV error code for the invalid standard object class+// -------------------------------------------------------------+#define ED_ISO (ERROR_VALUE | E_SOURCE_DRV | E_CLASS_INVALID_STD_OBJECT)+#define ED_INVALID_PCX_HANDLE (ED_ISO | 0x01)+#define ED_INVALID_PIPE (ED_ISO | 0x02)+#define ED_INVALID_STREAM (ED_ISO | 0x03)+#define ED_INVALID_PIPE_AUDIO (ED_ISO | 0x04)+#define ED_INVALID_BUFFER (ED_ISO | 0x05)+#define ED_INVALID_DSP_SOFTWARE (ED_ISO | 0x06)+#define ED_INVALID_BOARD (ED_ISO | 0x07)+#define ED_INVALID_DSP (ED_ISO | 0x08)+#define ED_INVALID_BOARD_AUDIO (ED_ISO | 0x09)+#define ED_INVALID_FREQUENCY (ED_ISO | 0x0a)++// Complete DRV warning code for impossible resource allocation class+// ------------------------------------------------------------------+#define WD_RI (WARNING_VALUE | E_SOURCE_DRV | E_CLASS_RSRC_IMPOSSIBLE)+#define WD_UNDER_ALLOCATE_BUFFER (WD_RI | 0x01)+#define WD_REPLY_TRUNCATED (WD_RI | 0x02)++// Complete DRV error code for impossible resource allocation class+// ----------------------------------------------------------------+#define ED_RI (ERROR_VALUE | E_SOURCE_DRV | E_CLASS_RSRC_IMPOSSIBLE)+#define ED_DECL_APP_IMPOSSIBLE (ED_RI | 0x01)+#define ED_SET_CLOCK_IMPOSSIBLE (ED_RI | 0x02)+#define ED_ALLOCATE_BUFFER_IMPOSSIBLE (ED_RI | 0x03)+#define ED_ALLOCATE_AUDIO_IMPOSSIBLE (ED_RI | 0x04)+#define ED_LOAD_DSP_IMPOSSIBLE (ED_RI | 0x05)+#define ED_REQUEST_BLOC_TOO_SHORT (ED_RI | 0x06)+#define ED_ALLOCATE_STREAM_IMPOSSIBLE (ED_RI | 0x07)+#define ED_REPLY_BLOC_TOO_SHORT (ED_RI | 0x08)+#define ED_ALLOCATE_MEMORY_IMPOSSIBLE (ED_RI | 0x09)+#define ED_NON_COBRANET_CLOCK_REFUSED (ED_RI | 0x0A)+#define ED_NON_ETHERSOUND_CLOCK_REFUSED (ED_RI | 0x0B)++// Complete DRV warning code for wrong call context class+// ----------------------------------------------------+#define WD_WCC (WARNING_VALUE | E_SOURCE_DRV | E_CLASS_WRONG_CONTEXT)+#define WD_PRIVATE_DATA (WD_WCC | 0x01)+#define WD_NO_HARDWARE_SUPPORT (WD_WCC | 0x02)+#define WD_COBRANET_CLOCK_SELECTED (WD_WCC | 0x03)+#define WD_AUDIO_MANAGED_BY_CONSOLE (WD_WCC | 0x04)++// Complete DRV error code for wrong call context class+// ----------------------------------------------------+#define ED_WCC (ERROR_VALUE | E_SOURCE_DRV | E_CLASS_WRONG_CONTEXT)+#define ED_USE_DSP_REFUSED (ED_WCC | 0x01)+#define ED_FREE_DSP_REFUSED (ED_WCC | 0x02)+#define ED_SUPPRESS_PIPE_REFUSED (ED_WCC | 0x03)+#define ED_SET_CLOCK_REFUSED (ED_WCC | 0x04)+#define ED_DIFFERED_CMD_REFUSED (ED_WCC | 0x05)+#define ED_SET_CLOCK_THESEPIPES_REFUSED (ED_WCC | 0x06)+#define ED_MANY_PIPES_REFUSED (ED_WCC | 0x07)+#define ED_MANY_AUDIOS_REFUSED (ED_WCC | 0x08)+#define ED_SET_PIPE_SOURCE_REFUSED (ED_WCC | 0x09)+#define ED_CONTEXT_REFUSED (ED_WCC | 0x0a)+#define ED_START_IN_PROGRESS (ED_WCC | 0x0b)+#define ED_START_ON_TIME_CODE_IN_PROGRESS (ED_WCC | 0x0c)+#define ED_START_ON_VSYNC_IN_PROGRESS (ED_WCC | 0x0d)+#define ED_AUDIO_MANAGED_BY_DHS_PANEL (ED_WCC | 0x0e)+#define ED_CLOCK_MANAGED_BY_DHS_PANEL (ED_WCC | 0x0f)+++// Complete DRV error code for bad specific parameter class+// ----------------------------------------------------------+#define ED_WSP (ERROR_VALUE | E_SOURCE_DRV | E_CLASS_BAD_SPECIFIC_PARAMETER)+#define ED_APPLICATION_NAME_SIZE (ED_WSP | 0x01)+#define ED_UNKNOWN_APPLICATION_ATTRIB (ED_WSP | 0x02)+#define ED_UNKNOWN_PIPE_ATTRIB (ED_WSP | 0x03)+#define ED_UNKNOWN_AUDIO_ATTRIB (ED_WSP | 0x04)+#define ED_INCOMPATIBLE_PIPE_AUDIO_ATTRIB (ED_WSP | 0x05)+#define ED_FORMAT_NOT_SUPPORTED (ED_WSP | 0x06)+#define ED_LEVEL_OUT_OF_RANGE (ED_WSP | 0x07)+#define ED_CLOCK_UER_BOARD (ED_WSP | 0x08)+#define ED_BAD_UER_FREQUENCY (ED_WSP | 0x09)+#define ED_BAD_UER_MODE (ED_WSP | 0x0A)+#define ED_BAD_CLOCK_FREQUENCY (ED_WSP | 0x0B)+#define ED_BAD_VALIDATION_MASK (ED_WSP | 0x0C)+#define ED_BAD_SOURCE (ED_WSP | 0x0D)+#define ED_CLOCK_WORD_CLOCK_BOARD (ED_WSP | 0x0E)+#define ED_CONTEXT_NOT_FOUND (ED_WSP | 0x0F)+#define ED_CMD_MUST_BE_DIFFERED (ED_WSP | 0x10)+#define ED_CLOCK_PROGRAMMABLE_CLOCK_BOARD (ED_WSP | 0x11)+#define ED_BAD_UER_EXTRA_PARAM (ED_WSP | 0x12)+#define ED_SRC_UNAVAILABLE (ED_WSP | 0x13)++// Complete DRV warning code for real time class+// ---------------------------------------------+#define WD_RT (WARNING_VALUE | E_SOURCE_DRV | E_CLASS_REAL_TIME_ERROR)+#define WD_UER_FREQUENCY_MODIFICATION (WD_RT | 0x01)+#define WD_WORD_CLOCK_FREQUENCY_MODIFICATION (WD_RT | 0x02)+#define WD_UER_CLOCK_LOST (WD_RT | 0x03)+#define WD_EXTERNAL_CLOCK_LOST WD_UER_CLOCK_LOST+#define WD_SRC_ASSIGNED (WD_RT | 0x04)+#define WD_CANNOT_CANCEL (WD_RT | 0x07)++// Complete DRV error code for real time class+// ---------------------------------------------+#define ED_RT (ERROR_VALUE | E_SOURCE_DRV | E_CLASS_REAL_TIME_ERROR)+#define ED_DSP_TIMED_OUT (ED_RT | 0x01)+#define ED_DSP_CHK_TIMED_OUT (ED_RT | 0x02)+#define ED_STREAM_OVERRUN (ED_RT | 0x03)+#define ED_DSP_BUSY (ED_RT | 0x04)+#define ED_DSP_SEMAPHORE_TIME_OUT (ED_RT | 0x05)+#define ED_BOARD_TIME_OUT (ED_RT | 0x06)+#define ED_XILINX_ERROR (ED_RT | 0x07)+#define ED_COBRANET_ITF_NOT_RESPONDING (ED_RT | 0x08)++// Complete BOARD error code for the general class+// -----------------------------------------------+#define EB_GN (ERROR_VALUE | E_SOURCE_BOARD | E_CLASS_GENERAL)++// Complete BOARD warning code for the invalid command class+// -------------------------------------------------------+#define WB_IC (WARNING_VALUE | E_SOURCE_BOARD | E_CLASS_INVALID_CMD)+#define WB_COMMAND_NOT_IMPLEMENTED_YET (WB_IC | 0x00)++// Complete BOARD error code for the invalid command class+// -------------------------------------------------------+#define EB_IC (ERROR_VALUE | E_SOURCE_BOARD | E_CLASS_INVALID_CMD)+#define EB_INVALID_COMMAND (EB_IC | 0x00)++// Complete BOARD error code for the invaid standard object class+// --------------------------------------------------------------+#define EB_ISO (ERROR_VALUE | E_SOURCE_BOARD | E_CLASS_INVALID_STD_OBJECT)+#define EB_INVALID_EFFECT (EB_ISO | 0x00)+#define EB_INVALID_PIPE (EB_ISO | 0x40)+#define EB_INVALID_STREAM (EB_ISO | 0x80)+#define EB_INVALID_AUDIO (EB_ISO | 0xC0)++// Complete BOARD warning code for impossible resource allocation class+// ------------------------------------------------------------------+#define WB_RI (WARNING_VALUE | E_SOURCE_BOARD | E_CLASS_RSRC_IMPOSSIBLE)+#define WB_LOW_DIFFERED_COMMAND (WB_RI | 0x06)+#define WB_COMMAND_WILL_NOT_BE_NOTIFIED (WB_RI | 0x07)+#define WB_WILL_NOT_BE_NOTIFIED WB_COMMAND_WILL_NOT_BE_NOTIFIED+#define WB_PREVIOUS_STATUS_DIALOG_IMPOSSIBLE (WB_RI | 0x1C)+#define WB_STATUS_TOO_BIG (WB_RI | 0x1F)++// Complete BOARD error code for impossible resource allocation class+// ------------------------------------------------------------------+#define EB_RI (ERROR_VALUE | E_SOURCE_BOARD | E_CLASS_RSRC_IMPOSSIBLE)+#define EB_ALLOCATE_ALL_STREAM_TRANSFERT_BUFFERS_IMPOSSIBLE (EB_RI | 0x01)+#define EB_ALLOCATE_PIPE_SAMPLE_BUFFER_IMPOSSIBLE (EB_RI | 0x02)++#define EB_ALLOCATE_MEM_STREAM_IMPOSSIBLE EB_ALLOCATE_ALL_STREAM_TRANSFERT_BUFFERS_IMPOSSIBLE+#define EB_ALLOCATE_MEM_PIPE_IMPOSSIBLE EB_ALLOCATE_PIPE_SAMPLE_BUFFER_IMPOSSIBLE++#define EB_ALLOCATE_DIFFERED_CMD_IMPOSSIBLE (EB_RI | 0x03)+#define EB_TOO_MANY_DIFFERED_CMD (EB_RI | 0x04)+#define EB_RBUFFERS_TABLE_OVERFLOW (EB_RI | 0x05)+#define EB_ALLOCATE_EFFECTS_IMPOSSIBLE (EB_RI | 0x08)+#define EB_ALLOCATE_EFFECT_POS_IMPOSSIBLE (EB_RI | 0x09)+#define EB_RBUFFER_NOT_AVAILABLE (EB_RI | 0x0A)+#define EB_ALLOCATE_CONTEXT_LIII_IMPOSSIBLE (EB_RI | 0x0B)+#define EB_STATUS_DIALOG_IMPOSSIBLE (EB_RI | 0x1D)+#define EB_CONTROL_CMD_IMPOSSIBLE (EB_RI | 0x1E)+#define EB_STATUS_SEND_IMPOSSIBLE (EB_RI | 0x1F)+#define EB_ALLOCATE_PIPE_IMPOSSIBLE (EB_RI | 0x40)+#define EB_ALLOCATE_STREAM_IMPOSSIBLE (EB_RI | 0x80)+#define EB_ALLOCATE_AUDIO_IMPOSSIBLE (EB_RI | 0xC0)+//Debut miXart specifique+#define EB_CLOCK_ALREADY_CAPTURED (EB_RI | 0xD0)+#define EB_NO_CURRENT_CLOCK (EB_RI | 0xD1)++//Fin miXart++// Complete BOARD error code for wrong call context class+// ------------------------------------------------------+#define EB_WCC (ERROR_VALUE | E_SOURCE_BOARD | E_CLASS_WRONG_CONTEXT)+#define EB_CMD_REFUSED (EB_WCC | 0x00)+#define EB_START_STREAM_REFUSED (EB_WCC | 0xFC)+#define EB_SPC_REFUSED (EB_WCC | 0xFD)+#define EB_CSN_REFUSED (EB_WCC | 0xFE)+#define EB_CSE_REFUSED (EB_WCC | 0xFF)++// Complete BOARD warning code for bad specific parameter class+// ------------------------------------------------------------+#define WB_WSP (WARNING_VALUE | E_SOURCE_BOARD | E_CLASS_BAD_SPECIFIC_PARAMETER)+#define WB_BAD_EFFECT_PARAMETER (WB_WSP | 0x42)++// Complete BOARD error code for bad specific parameter class+// ------------------------------------------------------------+#define EB_WSP (ERROR_VALUE | E_SOURCE_BOARD | E_CLASS_BAD_SPECIFIC_PARAMETER)+#define EB_BAD_DSP_MESSAGE_SIZE (EB_WSP | 0x01)++#define EB_BAD_CLOCK_FREQUENCY (EB_WSP | 0x30)++#define EB_BAD_PIPE_MESSAGE_SIZE (EB_WSP | 0x41)+#define EB_BAD_EFFECT_PARAMETER (EB_WSP | 0x42)+#define EB_BAD_STREAM_MESSAGE_SIZE (EB_WSP | 0x81)++#define EB_BAD_AUDIO_MESSAGE_SIZE (EB_WSP | 0xC1)+#define EB_BAD_DSP_MESSAGE_PARAMETER (EB_WSP | 0x20)+#define EB_BAD_PIPE_MESSAGE_PARAMETER (EB_WSP | 0x60)+#define EB_BAD_STREAM_MESSAGE_PARAMETER (EB_WSP | 0xA0)+#define EB_BAD_AUDIO_MESSAGE_PARAMETER (EB_WSP | 0xE0)++#define EB_BAD_DSPMESSAGE_SIZE EB_BAD_DSP_MESSAGE_SIZE+#define EB_BAD_PIPEMESSAGE_SIZE EB_BAD_PIPE_MESSAGE_SIZE++#define EB_BAD_STREAMMESSAGE_SIZE EB_BAD_STREAM_MESSAGE_SIZE+#define EB_BAD_AUDIOMESSAGE_SIZE EB_BAD_DSP_MESSAGE_PARAMETER+#define EB_BAD_DSPMESSAGE_PARAMETER EB_BAD_DSP_MESSAGE_PARAMETER+#define EB_BAD_PIPEMESSAGE_PARAMETER EB_BAD_PIPE_MESSAGE_PARAMETER+#define EB_BAD_STREAMMESSAGE_PARAMETER EB_BAD_STREAM_MESSAGE_PARAMETER+#define EB_BAD_AUDIOMESSAGE_PARAMETER EB_BAD_AUDIO_MESSAGE_PARAMETER++#define EB_CANNOT_READ_TIME_CODE (EB_WSP | 0xF0)+#define EB_CANNOT_READ_VSYNC (EB_WSP | 0xF1)+#define EB_CANNOT_WRITE_VSYNC (EB_WSP | 0xF2)++// Complete BOARD warning code for real time class+// ---------------------------------------------+#define WB_RT (WARNING_VALUE | E_SOURCE_BOARD | E_CLASS_REAL_TIME_ERROR)+#define WB_CANNOT_CANCEL (WB_RT | 0x0A)++// Complete BOARD error code for real time class+// ---------------------------------------------+#define EB_RT (ERROR_VALUE | E_SOURCE_BOARD | E_CLASS_REAL_TIME_ERROR)+#define EB_STREAM_UNDERRUN (EB_RT | 0x01)+#define EB_STREAM_FRAME_CRC_FAILED (EB_RT | 0x02)+#define EB_STREAM_INVALID_FRAME (EB_RT | 0x03)+#define EB_STREAM_INCOMPATIBLE_LAYER (EB_RT | 0x04)+#define EB_STREAM_RBUFFER_READ_ERROR (EB_RT | 0x05)+#define EB_STREAM_RBUFFER_TOO_SMALL (EB_RT | 0x06)+#define EB_STREAM_FORMAT_MISUSED (EB_RT | 0x07)+#define EB_STREAM_RBUFFER_WRITE_ERROR (EB_RT | 0x08)+#define EB_STREAM_LAYER3_FRAME_ERROR (EB_RT | 0x09)+#define EB_STREAM_MULTICHANNEL_ERROR (EB_RT | 0x0B)+#define EB_PIPE_UNDERRUN (EB_RT | 0x10)+#define EB_START_ON_TIME_CODE_FAILED (EB_RT | 0x11)++#define EB_STREAM_FAILED_CRC EB_STREAM_FRAME_CRC_FAILED+#define EB_STREAM_INVALID_LAYER EB_STREAM_INCOMPATIBLE_LAYER++// Complete BOARD error code for internal error class+// --------------------------------------------------+#define EB_IE (ERROR_VALUE | E_SOURCE_BOARD | E_CLASS_FREE)+#define EB_GENERIC_FATAL_ERROR (EB_IE | 0x00)+#define EB_CSNR_I1_FATAL_ERROR (EB_IE | 0x01)+#define EB_CSNR_I2_FATAL_ERROR (EB_IE | 0x02)+#define EB_CSNR_I3_FATAL_ERROR (EB_IE | 0x04)+#define EB_REENTRY_FATAL_ERROR (EB_IE | 0x08)+#define EB_CONN_RE_FATAL_ERROR (EB_IE | 0x10)+++//*************WARNING AND & ERROR for the new DISPATCHER*******************+//--------------------------------------------------------------------------+// Complete DISP warning code for the general class+#define WS_GN (WARNING_VALUE | E_SOURCE_DISPATCHER | E_CLASS_GENERAL)+#define WS_DISPATCHER (WS_GN | 0x00)+#define WS_NEW_MODE_ALREADY_DEFINE (WS_GN | 0x01)+#define WS_ALREADY_INITIALIZED (WS_GN | 0x02)+#define WS_INVALID_API_MODE_SELECTED_KEEP_OLD (WS_GN | 0x03)++// Complete DISP warning code for the invalid command class+#define WS_IC (WARNING_VALUE | E_SOURCE_DISPATCHER | E_CLASS_INVALID_CMD)++// Complete DISP warning code for the internal class+#define WS_IE (WARNING_VALUE | E_SOURCE_DISPATCHER | E_CLASS_FREE)++// Complete DISP warning code for the invalid standard object class+#define WS_ISO (WARNING_VALUE | E_SOURCE_DISPATCHER | E_CLASS_INVALID_STD_OBJECT)++// Complete DISP warning code for the bad specific parameter class+#define WS_WSP (WARNING_VALUE | E_SOURCE_DISPATCHER | E_CLASS_BAD_SPECIFIC_PARAMETER)++// Complete DISP warning code for the real time class+#define WS_RT (WARNING_VALUE | E_SOURCE_DISPATCHER | E_CLASS_REAL_TIME_ERROR)++// Complete DISP warning code for impossible resource allocation class+#define WS_RI (WARNING_VALUE | E_SOURCE_DISPATCHER | E_CLASS_RSRC_IMPOSSIBLE)++// Complete DISP warning code for wrong call context class+#define WS_WCC (WARNING_VALUE | E_SOURCE_DISPATCHER | E_CLASS_WRONG_CONTEXT)++// Complete DISP warning code for direct show problems class+#define WS_DS (WARNING_VALUE | E_SOURCE_DISPATCHER | E_CLASS_DIRECTSHOW)+++// Complete DISP error code for the general class+#define ES_GN (ERROR_VALUE | E_SOURCE_DISPATCHER | E_CLASS_GENERAL)+#define ES_CANNOT_FIND_PIPE_HANDLE (ES_GN | 0x00)+#define ES_CANNOT_FIND_BOARD_HANDLE (ES_GN | 0x01)+#define ES_CANNOT_FIND_LIST (ES_GN | 0x02)+#define ES_CANNOT_FIND_DATA (ES_GN | 0x03)+#define ES_CANNOT_ADD_ELEMENT_TO_LIST (ES_GN | 0x04)+#define ES_CANNOT_REMOVE_ELEMENT_FROM_LIST (ES_GN | 0x05)+#define ES_CANNOT_DISPATCH_FUNCTION (ES_GN | 0x06)+#define ES_FUNCTION_NOT_IMPLEMENTED (ES_GN | 0x07)+#define ES_ELEMENT_DOES_NOT_EXIST (ES_GN | 0x08)+#define ES_ELEMENT_DOES_NOT_EXIST_IN_LIST (ES_GN | 0x09)+#define ES_ELEMENT_NOT_FOUND_IN_LIST (ES_GN | 0x0A)+#define ES_LIST_DOES_NOT_EXIST (ES_GN | 0x0B)+#define ES_LIST_EMPTY (ES_GN | 0x0C)+#define ES_DATA_DOES_NOT_EXIST (ES_GN | 0x0D)+#define ES_DATA_NOT_FOUND_IN_LIST (ES_GN | 0x0E)+#define ES_FAILED (ES_GN | 0x0F)+#define ES_CANNOT_DETACH_DATA_FROM_LIST (ES_GN | 0x10)+#define ES_NOT_IMPLEMENTED (ES_GN | 0x11)+#define ES_BOARD_NOT_FOUND (ES_GN | 0x12)+#define ES_INVALID_MODE (ES_GN | 0x13)+#define ES_ALREADY_INITIALIZED (ES_GN | 0x14)+#define ES_NOT_INITIALIZED (ES_GN | 0x15)+#define ES_MAXIMUM_APPLI_REACH (ES_GN | 0x16)+#define ES_NEED_REGISTRATION (ES_GN | 0x17)+#define ES_DATA_ALREADY_IN_LIST (ES_GN | 0x18)+#define ES_NO_API_TO_LOAD (ES_GN | 0x19)+++// Complete DISP error code for the invalid command class+#define ES_IC (ERROR_VALUE | E_SOURCE_DISPATCHER | E_CLASS_INVALID_CMD)+#define ES_COMMAND_NOT_AVAILABLE (ES_IC | 0x00)++// Complete DISP error code for the internal class+#define ES_IE (ERROR_VALUE | E_SOURCE_DISPATCHER | E_CLASS_FREE)++// Complete DISP error code for the invalid standard object class+#define ES_ISO (ERROR_VALUE | E_SOURCE_DISPATCHER | E_CLASS_INVALID_STD_OBJECT)++// Complete DISP error code for the bad specific parameter class+#define ES_WSP (ERROR_VALUE | E_SOURCE_DISPATCHER | E_CLASS_BAD_SPECIFIC_PARAMETER)+#define ES_INVALID_PIPE (ES_WSP | 0x01)+#define ES_INVALID_PIPE_OUT (ES_WSP | 0x02)+#define ES_INVALID_PIPE_IN (ES_WSP | 0x03)+#define ES_INVALID_BOARD (ES_WSP | 0x04)+#define ES_INVALID_LIST (ES_WSP | 0x05)+#define ES_INVALID_DATA (ES_WSP | 0x06)+#define ES_INVALID_PARAMETER (ES_WSP | 0x07)+#define ES_INVALID_HANDLE (ES_WSP | 0x08)+#define ES_INVALID_LOGICAL_NUMBER (ES_WSP | 0x09)++// Complete DISP error code for the real time class+#define ES_RT (ERROR_VALUE | E_SOURCE_DISPATCHER | E_CLASS_REAL_TIME_ERROR)++// Complete DISP error code for impossible resource allocation class+#define ES_RI (ERROR_VALUE | E_SOURCE_DISPATCHER | E_CLASS_RSRC_IMPOSSIBLE)+#define ES_ALLOCATE_PIPE_HANDLE (ES_RI | 0x00)+#define ES_ALLOCATE_BOARD_HANDLE (ES_RI | 0x01)+#define ES_ALLOCATE_LIST (ES_RI | 0x02)+#define ES_ALLOCATE_ELEMENT (ES_RI | 0x03)+#define ES_ALLOCATE_MEMORY (ES_RI | 0x04)+#define ES_CANNOT_FREE_PIPE_HANDLE (ES_RI | 0x05)+#define ES_CANNOT_FREE_BOARD_HANDLE (ES_RI | 0x06)+#define ES_CANNOT_FREE_LIST (ES_RI | 0x07)+#define ES_CANNOT_FREE_ELEMENT (ES_RI | 0x08)+#define ES_CANNOT_FREE_BIG_BLOC (ES_RI | 0x09)+#define ES_LIST_OF_LIST_NOT_VALID (ES_RI | 0x0a)+#define ES_INVALID_POINTER (ES_RI | 0x0b)+#define ES_NO_MORE_ROOM_AVAILABLE (ES_RI | 0x0c)++// Complete DISP error code for wrong call context class+#define ES_WCC (ERROR_VALUE | E_SOURCE_DISPATCHER | E_CLASS_WRONG_CONTEXT)++// Complete DISP error code for direct show problems class+#define ES_DS (ERROR_VALUE | E_SOURCE_DISPATCHER | E_CLASS_DIRECTSHOW)+++//************* WARNINGS AND ERRORS coming form CobraNet firmware *******************+//--------------------------------------------------------------------------+// Complete warning codes for the general class+#define WC_GN (WARNING_VALUE | E_SOURCE_COBRANET | E_CLASS_GENERAL)+++// Complete error codes for the general class+#define EC_GN (ERROR_VALUE | E_SOURCE_COBRANET | E_CLASS_GENERAL)++//******************************FIN************************************++//}+#endif+diff --git a/sound/pci/lx6464es/ethersound.h b/sound/pci/lx6464es/ethersound.hnew file mode 100644index 0000000..0026f36--- /dev/null+++ b/sound/pci/lx6464es/ethersound.h@@ -0,0 +1,55 @@+//********************************************************************+//+// COPYRIGHT 2005 DIGIGRAM.+//+//********************************************************************+//+// ETHERSOUND.H+//+// Project : LXES+// Description : header file for Ethersound FPGA interface+//+// Last Mod $Date: 1/06/06 13:22 $ by $Author: Mtr $+//+//************************************************************************+#ifndef __ETHERSOUND_H_+#define __ETHERSOUND_H_++#define XES_BASEADDR 0x0007F800 // Ethersound fpga base address+//+#define XES_RW_GRANULARITY (XES_BASEADDR + 0x0000) // adresse du registre EtherSound contenant la granularité+#define XES_GRAN_MASK 0x00FFFF // masque des bits significatifs dans le registre de granularite++//+#define XES_RW_FREQUENCY (XES_BASEADDR + 0x0001) // adresse du registre EtherSound contenant la frequence des trames+#define XES_FREQ_FILTER 0x0000FFF0 // filtre pour notifier le changement de source / cadence.+#define XES_FREQ_MASK 0x0000FFFF // condition bits and counter filter mask.+#define XES_FREQ_IS_PM 0x00008000 // set if primary master+#define XES_FREQ_NOT_WCK 0x00004000 // (significant if primary master) clear if wordclock, set if local.+#define XES_FREQ_COUNT8_MASK 0x00001FFF // compteur 25MHz entre 8 ech.+#define XES_FREQ_COUNT8_44_MIN 0x00001288 // 25M / [ 44k - ( 44.1k + 48k ) / 2 ] * 8+#define XES_FREQ_COUNT8_44_MAX 0x000010F0 // 25M / [ ( 44.1k + 48k ) / 2 ] * 8+#define XES_FREQ_COUNT8_48_MAX 0x00000F08 // 25M / [ 48k + ( 44.1k + 48k ) / 2 ] * 8++//+#define XES_RO_PEAKMETER_IN0 (XES_BASEADDR + 0x0200)+#define XES_RO_PEAKMETER_OUT0 (XES_BASEADDR + 0x0210)+++// 16 bits pf a chaque fois+//+#define OFFSET_A 0 /**< channels 00 to 15.*/+#define OFFSET_B 1 /**< channels 16 to 31.*/+#define OFFSET_C 2 /**< channels 32 to 47.*/+#define OFFSET_D 3 /**< channels 48 to 63.*/+//+#define XES_W_MUTE_PLAY (XES_BASEADDR + 0x0020) // registre de mute (1) ou démute (0) des voies PLAY+#define XES_W_MUTE_RECORD (XES_BASEADDR + 0x0030) // registre de mute (1) ou démute (0) des voies RECORD+#define XES_W_ACTIVATE_PLAY (XES_BASEADDR + 0x0040) // registre d'activation (1) ou disable (0) des voies PLAY+#define XES_W_ACTIVATE_RECORD (XES_BASEADDR + 0x0050) // registre d'activation (1) ou disable (0) des voies REC++#define XES_W_RAM_FIFO_BASE (XES_BASEADDR + 0x0100) // registre de prog de l'addresse de base des fifo+#define XES_R_RAM_CUR_BANK (XES_BASEADDR + 0x0101) /**< current bank (bit 0).*/+++#endifdiff --git a/sound/pci/lx6464es/if_drv_mb.h b/sound/pci/lx6464es/if_drv_mb.hnew file mode 100644index 0000000..44bf687--- /dev/null+++ b/sound/pci/lx6464es/if_drv_mb.h@@ -0,0 +1,287 @@+//********************************************************************+//+// COPYRIGHT 2005 DIGIGRAM.+//+//********************************************************************+//+/** @file IF_DRV_MB.H+* @brief definitions shared between PC-side Driver and Embedded+*/+// Project : LXES+//+// Last Mod $Date: 16/11/06 17:46 $ by $Author: Mtr $+//+//*********************************************************************+++#ifndef __IF_DRV_MB_H_+#define __IF_DRV_MB_H_++///*********************************************************************+// compile/debug options+//*********************************************************************++#define DEBUG_MAIN 0 /** main debug switch.*/+#define DEBUG_UART 0 /** support for UART.*/+#define DEBUG_MASK64 0 // 64 bits mask class++//*********************************************************************+// general definitions+//*********************************************************************+#define SIZE_MAPPING 10 /* number of channels supported in m/c mode.*/+#define OPT_64_BITS_PCADDR 0 /* support for 64 bits PC address / DMA.*/++/** attention : si le max des I/O est < 32, il y a du code a revoir */++#define MAX_CARD_INPUTS 64+#define MAX_CARD_OUTPUTS 64+#define MAX_CANAUX_CARTE (MAX_CARD_INPUTS+MAX_CARD_OUTPUTS)++#define MAX_CARD_IN_PIPES 32+#define MAX_CARD_OUT_PIPES 32++//*********************************************************************+// DRV <---> uM commands definitions+//*********************************************************************++#define MB_REG_CSM 0+// DRV --> MB+#define MB_REG_CMD0 1+#define MB_REG_CMD1 2+#define MB_REG_CMD2 3+#define MB_REG_CMD3 4+#define MB_REG_CMD4 5+#define MB_REG_CMD5 6+#define MB_REG_CMD_MAX 7+// MB --> DRV+#define MB_REG_ERR 1+#define MB_REG_RSP0 2+#define MB_REG_RSP1 3+#define MB_REG_RSP2 4+#define MB_REG_RSP3 5+#define MB_REG_RSP4 6+#define MB_REG_RSP5 7+#define MB_REG_RSP6 8+#define MB_REG_RSP7 9+#define MB_REG_RSP8 10+#define MB_REG_RSP9 11++/*+ the capture bit position in the object_id field in driver commands+ depends upon the number of managed channels. For now, 64 IN + 64 OUT are+ supported. HOwever, the communication protocol forsees 1024 channels, hence+ bit 10 indicates a capture (input) object).+*/+#define ID_IS_CAPTURE ( 1L << 10 )+#define ID_OFFSET 13 /**< object ID is at the 13th bit in the 1st command word.*/+#define ID_CH_MASK 0x3F++#define OPCODE_OFFSET 24 /**< offset of the command opcode in the first command word.*/+++typedef enum {+ CMD_00_INFO_DEBUG = 0x00, // information de debug+ CMD_01_GET_SYS_CFG = 0x01, // lecture de la configuration système+ CMD_02_SET_GRANULARITY = 0x02, // programmation de la granularité+ CMD_03_SET_TIMER_IRQ = 0x03, // programmation de l'IRQ PCI Timer+ CMD_04_GET_EVENT = 0x04, // lecture des événements notifiés+ CMD_05_GET_PIPES = 0x05, // lecture de l'état des pipes+ //+ CMD_06_ALLOCATE_PIPE = 0x06, // réservation d'un pipe+ CMD_07_RELEASE_PIPE = 0x07, // libération d'un pipe+ CMD_08_ASK_BUFFERS = 0x08, // état des buffers d'un pipe+ CMD_09_STOP_PIPE = 0x09, // arret d'un pipe+ CMD_0A_GET_PIPE_SPL_COUNT = 0x0a, // lecture du compte d'échantillons pipe+ CMD_0B_TOGGLE_PIPE_STATE = 0x0b, // commutation START/PAUSE+ //+ CMD_0C_DEF_STREAM = 0x0c, // définition d'un stream+ CMD_0D_SET_MUTE = 0x0d, // mute des voies+ CMD_0E_GET_STREAM_SPL_COUNT = 0x0e, // lecture état et compte d'échantillons stream+ CMD_0F_UPDATE_BUFFER = 0x0f, // mise à jour d'un buffer PC+ CMD_10_GET_BUFFER = 0x10, // état d'un buffer PC+ CMD_11_CANCEL_BUFFER = 0x11, // annulation d'un buffer PC+ CMD_12_GET_PEAK = 0x12, // lecture des peak-metres audio+ CMD_13_SET_STREAM_STATE = 0x13,+ CMD_14_INVALID = 0x14,+} cmd_mb_opcodes;+++//*********** specific to CODECMD_INFO_DEBUG ***************+++//*********** specific to CODECMD_LEC_CONF_SYS ***************++#define FREQ_FIELD_OFFSET 15 // offset of the freq field in the response word++#define FREQ_48 0x00+#define FREQ_44 0x01+#define FREQ_96 0x02 // (drv must test the frequency mode)+#define FREQ_192 0x03 // (drv must test the frequency mode)+#define FREQ_UNKNOWN 0x0F // (drv must test the frequency mode)++//-- Version+#define VERSION_MAJOR_OFFSET 16+#define VERSION_MINOR_OFFSET 8+#define VERSION_BUILD_OFFSET 0+#define MICROBLAZE_VERSION ((PCX_VERSION_NUMBER << VERSION_MAJOR_OFFSET) \+ |(PCX_VERSION_RELEASE << VERSION_MINOR_OFFSET) \+ |(PCX_BUILD_NUMBER << VERSION_BUILD_OFFSET))+++//*********** specific to CODECMD_PROG_GRANULARITE ***************++// must be power of 2+//+#define MICROBLAZE_IBL_MIN 32 // 32 (0), 64 (1), 128 (2), 256 (3), 512 (4), 1024 (5)+#define MICROBLAZE_IBL_DEFAULT 128+#define MICROBLAZE_IBL_MAX 512+#define MASK_GRANULARITY (2*MICROBLAZE_IBL_MAX-1)+#define MICROBLAZE_IBL_CODED_DEFAULT 2 // coded value, for audio vhdl { 0, 1, 2, 3, 4, 5 }++//*********** specific to CODECMD_PROG_IRQ_TIMER ***************++#define MASK_TIMER_K 0x00FFFFFF++//*********** specific to CODECMD_LEC_EVE_NOTIFIES ***************++// -- mask definitions for SYS.STAT -----+//+#define MASK_SYS_STATUS_ERROR (1L << 31) // events that lead to a PCI irq if not yet pending+#define MASK_SYS_STATUS_URUN (1L << 30)+#define MASK_SYS_STATUS_ORUN (1L << 29)+#define MASK_SYS_STATUS_EOBO (1L << 28)+#define MASK_SYS_STATUS_EOBI (1L << 27)+#define MASK_SYS_STATUS_FREQ (1L << 26)+#define MASK_SYS_STATUS_ESA (1L << 25) // reserved, this is set by the XES+#define MASK_SYS_STATUS_TIMER (1L << 24)++#define MASK_SYS_ASYNC_EVENTS (MASK_SYS_STATUS_ERROR | MASK_SYS_STATUS_URUN | \+ MASK_SYS_STATUS_ORUN | MASK_SYS_STATUS_EOBO | \+ MASK_SYS_STATUS_EOBI | MASK_SYS_STATUS_FREQ | \+ MASK_SYS_STATUS_ESA)++#define MASK_SYS_PCI_EVENTS (MASK_SYS_ASYNC_EVENTS | MASK_SYS_STATUS_TIMER)++#define MASK_SYS_TIMER_COUNT 0x0000FFFF++//#define MASK_SYS_STATUS_CMD (1L << 23) // event that remains internal+#define MASK_SYS_STATUS_EOT_PLX (1L << 22) // event that remains internal : reserved fo end of plx dma+#define MASK_SYS_STATUS_XES (1L << 21) // event that remains internal : pending XES IRQ+#define MASK_SYS_STATUS_CMD_DONE (1L << 20) // alternate command management : notify driver instead of polling+++// behave like on PCX : the driver may not acknowledge+// the MASK_SYS_STATUS_TIMER IRQ source with command 0x04+// in this case, the uB resets it after sending the IRQ+//+#define OPT_AUTORESET_TIMER_IRQ 1++//*********** specific to CODECMD_ETAT_PIPES ***************++// this command is not mandatory infact+// default : do not compile+//+#define OPT_NO_CMD_GET_PIPES 1++//*********** specific to CODECMD_RESERV_PIPE ***************+//*********** specific to CODECMD_ETAT_BUFF_PIPE ***************++#define BUFF_FLAGS_OFFSET 24 /**< offset of the buffer flags in the response word.*/+#define MASK_DATA_SIZE 0x00FFFFFF /**<this must match the field size of datasize in the buffer_t structure.*/++/**+* Buffer Flags+*/+ typedef enum {+ BF_VALID = 0x80, /**< set if the buffer is valid, clear if free.*/+ BF_CURRENT = 0x40, /**< set if this is the current buffer (there is always a current buffer).*/+ BF_NOTIFY_EOB = 0x20, /**< set if this buffer must cause a PCI event when finished.*/+ BF_CIRCULAR = 0x10, /**< set if buffer[1] must be copied to buffer[0] by the end of this buffer.*/+ BF_64BITS_ADR = 0x08, /**< set if the hi part of the address is valid.*/+ BF_xx = 0x04, /**< futur extension.*/+ BF_EOB = 0x02, /**< set if finished, but not yet free.*/+ BF_PAUSE = 0x01, /**< pause stream at buffer end.*/+ BF_ZERO = 0x00, /**< no flags (init).*/+ } buffer_flags;++// flags set by driver.+#define MASK_BF_FLAGS (BF_NOTIFY_EOB|BF_CIRCULAR|BF_64BITS_ADR|BF_PAUSE|BF_xx)++//*********** specific to CODECMD_STOP_PIPE ***************+//*********** specific to CODECMD_LEC_CEP ***************++#define MASK_SPL_COUNT_HI 0x00FFFFFF // 4 MSBits are status bits+#define PSTATE_OFFSET 28 // 4 MSBits are status bits++/**+* pipe states+*/+typedef enum {+ PSTATE_IDLE = 0, /**< the pipe is not processed in the XES_IRQ (free or stopped, or paused). */+ PSTATE_RUN = 1, /**< sustained play/record state. */+ PSTATE_PURGE = 2, /**< the ES channels are now off, render pipes do not DMA, record pipe do a last DMA. */+ PSTATE_ACQUIRE = 3, /**< the ES channels are now on, render pipes do not yet increase their sample count, record pipes do not DMA. */+ PSTATE_CLOSING = 4, /**< the pipe is releasing, and may not yet receive an "alloc" command. */+} pipe_state_t;+++//*********** specific to CODECMD_START_PAUSE ***************++#define MASK_CMD_LISTE (1L << 12) // bit 12 indicates a list or mask of arguments is passed in the following comand words.++//*********** specific to CODECMD_DEF_STREAM ***************++#define MASK_STREAM_HAS_MAPPING ( 1L << 12)+#define MASK_STREAM_IS_ASIO ( 1L << 9)+#define STREAM_FMT_OFFSET 10 /**<the stream fmt bits start at the 10th bit in the command word.*/+// see DMA_conf_t structure aswell, must be the same values+#define STREAM_FMT_16b 0x02+#define STREAM_FMT_intel 0x01+#define STREAM_MAP_implicit 0x01+#define STREAM_MAP_explicit 0+++//*********** specific to CODECMD_LEC_ETAT_CES ***************++/**+* Stream Flags definitions+*/+ typedef enum {+ SF_ZERO = 0x00000000, /**< no flags (stream invalid).*/+ SF_VALID = 0x10000000, /**< the stream has a valid DMA_conf info (setstreamformat).*/+ SF_XRUN = 0x20000000, /**< the stream is un x-run state.*/+ SF_START = 0x40000000, /**< the DMA is running.*/+ SF_ASIO = 0x80000000, /**< ASIO.*/+ } stream_flags ;+++//*********** specific to CODECMD_MAJ_BUFFER_PC ***************+//#if DEBUG_MAIN == 1+//#define MAX_STREAM_BUFFER 3 /**< max amount of stream buffers.*/+//#else+#define MAX_STREAM_BUFFER 5 /**< max amount of stream buffers.*/+//#endif++//*********** specific to CODECMD_DEL_BUFF_PC ***************++#define MASK_BUFFER_ID 0xFF /**<the cancel command awaits a buffer ID, may be 0xFF for "current".*/++//*********** specific to CODECMD_LECTURE_PEAKMETRES ***************+++//*********** specific to CMD_13_SET_STREAM_STATE ***************+++/**+* stream states+*/+typedef enum {+ SSTATE_STOP = 0x00, /**< setting to stop resets the stream spl count.*/+ SSTATE_RUN = (0x01 << 0), /**< start DMA and spl count handling. */+ SSTATE_PAUSE = (0x01 << 1), /**< pause DMA and spl count handling. */+} stream_state_t;+++#define MASK_STREAM_STATE 0x0000000F /**<the state field in the 1st command word.*/++#endifdiff --git a/sound/pci/lx6464es/lx6464es.c b/sound/pci/lx6464es/lx6464es.cnew file mode 100644index 0000000..e1df693--- /dev/null+++ b/sound/pci/lx6464es/lx6464es.c@@ -0,0 +1,1179 @@+/* -*- linux-c -*- *+ *+ * ALSA driver for the digigram lx6464es interface+ *+ * Copyright (c) 2008, 2009 Tim Blechmann <tim@xxxxxxxxxx>+ *+ *+ * 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 2 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; see the file COPYING. If not, write to+ * the Free Software Foundation, Inc., 59 Temple Place - Suite 330,+ * Boston, MA 02111-1307, USA.+ *+ */++#include <linux/module.h>+#include <linux/init.h>+#include <linux/pci.h>+#include <linux/delay.h>++#include <sound/initval.h>+#include <sound/control.h>+#include <sound/info.h>++#include "lx6464es.h"++#include "LXES_registers.h"++MODULE_AUTHOR("Tim Blechmann");+MODULE_LICENSE("GPL");+MODULE_DESCRIPTION("digigram lx6464es");+MODULE_SUPPORTED_DEVICE("{digigram lx6464es{}}");+++static int index[SNDRV_CARDS] = SNDRV_DEFAULT_IDX;+static char *id[SNDRV_CARDS] = SNDRV_DEFAULT_STR;+static int enable[SNDRV_CARDS] = SNDRV_DEFAULT_ENABLE_PNP;++static const char card_name[] = "LX6464ES";+++/*+ * pci device stuff+ *+ * */+++#ifndef PCI_DEVICE_ID_PLX_9056 /* LATER: this should go to pci_ids.h */+#define PCI_DEVICE_ID_PLX_9056 0x9056+#endif++#define PCI_DEVICE_ID_PLX_LX6464ES PCI_DEVICE_ID_PLX_9056++#ifndef PCI_VENDOR_ID_DIGIGRAM+#define PCI_VENDOR_ID_DIGIGRAM 0x1369+#endif++#ifndef PCI_SUBDEVICE_ID_DIGIGRAM_LX6464ES_SERIAL_SUBSYSTEM+#define PCI_SUBDEVICE_ID_DIGIGRAM_LX6464ES_SERIAL_SUBSYSTEM 0xc001+#define PCI_SUBDEVICE_ID_DIGIGRAM_LX6464ES_CAE_SERIAL_SUBSYSTEM 0xc002+#endif++static struct pci_device_id snd_lx6464es_ids[] = {+ { PCI_DEVICE(PCI_VENDOR_ID_PLX, PCI_DEVICE_ID_PLX_LX6464ES),+ .subvendor = PCI_VENDOR_ID_DIGIGRAM,+ .subdevice = PCI_SUBDEVICE_ID_DIGIGRAM_LX6464ES_SERIAL_SUBSYSTEM+ }, /* LX6464ES */+ { PCI_DEVICE(PCI_VENDOR_ID_PLX, PCI_DEVICE_ID_PLX_LX6464ES),+ .subvendor = PCI_VENDOR_ID_DIGIGRAM,+ .subdevice = PCI_SUBDEVICE_ID_DIGIGRAM_LX6464ES_CAE_SERIAL_SUBSYSTEM+ }, /* LX6464ES-CAE */+ { 0, },+};++MODULE_DEVICE_TABLE(pci, snd_lx6464es_ids);++++/* PGO pour USERo dans le registre pci_0x06/loc_0xEC */+#define CHIPSC_RESET_XILINX (1L<<16)+++/* alsa callbacks */+static struct snd_pcm_hardware lx_caps = {+ .info = (SNDRV_PCM_INFO_MMAP |+ SNDRV_PCM_INFO_INTERLEAVED |+ SNDRV_PCM_INFO_MMAP_VALID |+ SNDRV_PCM_INFO_SYNC_START),+ .formats = (SNDRV_PCM_FMTBIT_S16_LE |+ SNDRV_PCM_FMTBIT_S16_BE |+ SNDRV_PCM_FMTBIT_S24_3LE |+ SNDRV_PCM_FMTBIT_S24_3BE),+ .rates = (SNDRV_PCM_RATE_CONTINUOUS |+ SNDRV_PCM_RATE_8000_192000),+ .rate_min = 8000,+ .rate_max = 192000,+ .channels_min = 2,+ .channels_max = 64,+ .buffer_bytes_max = 64*2*3*MICROBLAZE_IBL_MAX*MAX_STREAM_BUFFER,+ .period_bytes_min = (2*2*MICROBLAZE_IBL_MIN*2),+ .period_bytes_max = (4*64*MICROBLAZE_IBL_MAX*MAX_STREAM_BUFFER),+ .periods_min = 2,+ .periods_max = MAX_STREAM_BUFFER,+};++static int lx_set_granularity(struct lx6464es *chip, u32 gran);+++static int lx_hardware_open(struct lx6464es *chip,+ struct snd_pcm_substream *substream)+{+ int err = 0;+ struct snd_pcm_runtime *runtime = substream->runtime;+ int channels = runtime->channels;+ int is_capture = (substream->stream == SNDRV_PCM_STREAM_CAPTURE);++ snd_pcm_uframes_t period_size = runtime->period_size;++ snd_printd(LXP "allocating pipe for %d channels\n", channels);+ err = lx_pipe_allocate(chip, 0, is_capture, channels);+ if (err < 0) {+ snd_printk(KERN_ERR LXP "allocating pipe failed\n");+ return err;+ }++ err = lx_set_granularity(chip, period_size);+ if (err < 0) {+ snd_printk(KERN_ERR LXP "setting granularity to %ld failed\n",+ period_size);+ return err;+ }++ return 0;+}++static int lx_hardware_start(struct lx6464es *chip,+ struct snd_pcm_substream *substream)+{+ int err = 0;+ struct snd_pcm_runtime *runtime = substream->runtime;+ int is_capture = (substream->stream == SNDRV_PCM_STREAM_CAPTURE);++ snd_printd(LXP "setting stream format\n");+ err = lx_stream_set_format(chip, runtime, 0, is_capture);+ if (err < 0) {+ snd_printk(KERN_ERR LXP "setting stream format failed\n");+ return err;+ }++ snd_printd(LXP "starting pipe\n");+ err = lx_pipe_start(chip, 0, is_capture);+ if (err < 0) {+ snd_printk(KERN_ERR LXP "starting pipe failed\n");+ return err;+ }++ snd_printd(LXP "waiting for pipe to start\n");+ err = lx_pipe_wait_for_start(chip, 0, is_capture);+ if (err < 0) {+ snd_printk(KERN_ERR LXP "waiting for pipe failed\n");+ return err;+ }++ return err;+}+++static int lx_hardware_stop(struct lx6464es *chip,+ struct snd_pcm_substream *substream)+{+ int err = 0;+ int is_capture = (substream->stream == SNDRV_PCM_STREAM_CAPTURE);++ snd_printd(LXP "pausing pipe\n");+ err = lx_pipe_pause(chip, 0, is_capture);+ if (err < 0) {+ snd_printk(KERN_ERR LXP "pausing pipe failed\n");+ return err;+ }++ snd_printd(LXP "waiting for pipe to become idle\n");+ err = lx_pipe_wait_for_idle(chip, 0, is_capture);+ if (err < 0) {+ snd_printk(KERN_ERR LXP "waiting for pipe failed\n");+ return err;+ }++ snd_printd(LXP "stopping pipe\n");+ err = lx_pipe_stop(chip, 0, is_capture);+ if (err < 0) {+ snd_printk(LXP "stopping pipe failed\n");+ return err;+ }++ return err;+}+++static int lx_hardware_close(struct lx6464es *chip,+ struct snd_pcm_substream *substream)+{+ int err = 0;+ int is_capture = (substream->stream == SNDRV_PCM_STREAM_CAPTURE);++ snd_printd(LXP "releasing pipe\n");+ err = lx_pipe_release(chip, 0, is_capture);+ if (err < 0) {+ snd_printk(LXP "releasing pipe failed\n");+ return err;+ }++ return err;+}+++static int lx_pcm_open(struct snd_pcm_substream *substream)+{+ struct lx6464es *chip = snd_pcm_substream_chip(substream);+ struct snd_pcm_runtime *runtime = substream->runtime;+ int err = 0;+ int board_rate;++ snd_printdd("->lx_pcm_open\n");+ mutex_lock(&chip->setup_mutex);++ /* copy the struct snd_pcm_hardware struct */+ runtime->hw = lx_caps;++#if 0+ /* buffer-size should better be multiple of period-size */+ err = snd_pcm_hw_constraint_integer(runtime,+ SNDRV_PCM_HW_PARAM_PERIODS);+ if (err < 0) {+ snd_printk(KERN_WARNING LXP "could not constrain periods\n");+ goto exit;+ }+#endif++ /* the clock rate cannot be changed */+ board_rate = chip->board_sample_rate;+ err = snd_pcm_hw_constraint_minmax(runtime, SNDRV_PCM_HW_PARAM_RATE,+ board_rate, board_rate);++ if (err < 0) {+ snd_printk(KERN_WARNING LXP "could not constrain periods\n");+ goto exit;+ }++ /* constrain period size */+ err = snd_pcm_hw_constraint_minmax(runtime,+ SNDRV_PCM_HW_PARAM_PERIOD_SIZE,+ MICROBLAZE_IBL_MIN,+ MICROBLAZE_IBL_MAX);+ if (err < 0) {+ snd_printk(KERN_WARNING LXP+ "could not constrain period size\n");+ goto exit;+ }++ snd_pcm_hw_constraint_step(runtime, 0,+ SNDRV_PCM_HW_PARAM_BUFFER_SIZE, 32);++ snd_pcm_set_sync(substream);+ err = 0;++exit:+ runtime->private_data = chip;++ mutex_unlock(&chip->setup_mutex);+ snd_printdd("<-lx_pcm_open, %d\n", err);+ return err;+}++static int lx_pcm_close(struct snd_pcm_substream *substream)+{+ int err = 0;+ snd_printdd("->lx_pcm_close\n");+ return err;+}++static snd_pcm_uframes_t lx_pcm_stream_pointer(struct snd_pcm_substream+ *substream)+{+ struct lx6464es *chip = snd_pcm_substream_chip(substream);+ snd_pcm_uframes_t pos;+ unsigned long flags;+ int is_capture = (substream->stream == SNDRV_PCM_STREAM_CAPTURE);++ struct lx_stream *lx_stream = is_capture ? &chip->capture_stream :+ &chip->playback_stream;++ snd_printdd("->lx_pcm_stream_pointer\n");++ spin_lock_irqsave(&chip->lock, flags);+ pos = lx_stream->frame_pos * substream->runtime->period_size;+ spin_unlock_irqrestore(&chip->lock, flags);++ snd_printdd(LXP "stream_pointer at %ld\n", pos);+ return pos;+}++static int lx_pcm_prepare(struct snd_pcm_substream *substream)+{+ struct lx6464es *chip = snd_pcm_substream_chip(substream);+ int err = 0;+ const int is_capture = (substream->stream == SNDRV_PCM_STREAM_CAPTURE);++ snd_printdd("->lx_pcm_prepare\n");++ mutex_lock(&chip->setup_mutex);++ if (chip->hardware_running[is_capture]) {+ err = lx_hardware_stop(chip, substream);+ if (err < 0) {+ snd_printk(KERN_ERR LXP "failed to stop hardware. "+ "Error code %d\n", err);+ goto exit;+ }++ err = lx_hardware_close(chip, substream);+ if (err < 0) {+ snd_printk(KERN_ERR LXP "failed to close hardware. "+ "Error code %d\n", err);+ goto exit;+ }+ }++ snd_printd(LXP "opening hardware\n");+ err = lx_hardware_open(chip, substream);+ if (err < 0) {+ snd_printk(KERN_ERR LXP "failed to open hardware. "+ "Error code %d\n", err);+ goto exit;+ }++ err = lx_hardware_start(chip, substream);+ if (err < 0) {+ snd_printk(KERN_ERR LXP "failed to start hardware. "+ "Error code %d\n", err);+ goto exit;+ }++ chip->hardware_running[is_capture] = 1;++ if (chip->board_sample_rate != substream->runtime->rate) {+ if (!err)+ chip->board_sample_rate = substream->runtime->rate;+ }++exit:+ mutex_unlock(&chip->setup_mutex);+ return err;+}++static int lx_pcm_hw_params(struct snd_pcm_substream *substream,+ struct snd_pcm_hw_params *hw_params, int is_capture)+{+ struct lx6464es *chip = snd_pcm_substream_chip(substream);+ int err = 0;++ snd_printdd("->lx_pcm_hw_params\n");++ mutex_lock(&chip->setup_mutex);++ /* set dma buffer */+ err = snd_pcm_lib_malloc_pages(substream,+ params_buffer_bytes(hw_params));++ if (is_capture)+ chip->capture_stream.stream = substream;+ else+ chip->playback_stream.stream = substream;++ mutex_unlock(&chip->setup_mutex);+ return err;+}++static int lx_pcm_hw_params_playback(struct snd_pcm_substream *substream,+ struct snd_pcm_hw_params *hw_params)+{+ return lx_pcm_hw_params(substream, hw_params, 0);+}++static int lx_pcm_hw_params_capture(struct snd_pcm_substream *substream,+ struct snd_pcm_hw_params *hw_params)+{+ return lx_pcm_hw_params(substream, hw_params, 1);+}++static int lx_pcm_hw_free(struct snd_pcm_substream *substream)+{+ struct lx6464es *chip = snd_pcm_substream_chip(substream);+ int err = 0;+ int is_capture = (substream->stream == SNDRV_PCM_STREAM_CAPTURE);++ snd_printdd("->lx_pcm_hw_free\n");+ mutex_lock(&chip->setup_mutex);++ if (chip->hardware_running[is_capture]) {+ err = lx_hardware_stop(chip, substream);+ if (err < 0) {+ snd_printk(KERN_ERR LXP "failed to stop hardware. "+ "Error code %d\n", err);+ goto exit;+ }++ err = lx_hardware_close(chip, substream);+ if (err < 0) {+ snd_printk(KERN_ERR LXP "failed to close hardware. "+ "Error code %d\n", err);+ goto exit;+ }++ chip->hardware_running[is_capture] = 0;+ }++ err = snd_pcm_lib_free_pages(substream);++ if (is_capture)+ chip->capture_stream.stream = 0;+ else+ chip->playback_stream.stream = 0;++exit:+ mutex_unlock(&chip->setup_mutex);+ return err;+}++static void lx_trigger_start(struct lx6464es *chip, struct lx_stream *lx_stream)+{+ struct snd_pcm_substream *substream = lx_stream->stream;+ const int is_capture = lx_stream->is_capture;++ int err;++ const u32 channels = substream->runtime->channels;+ const u32 bytes_per_frame = channels * 3;+ const u32 period_size = substream->runtime->period_size;+ const u32 periods = substream->runtime->periods;+ const u32 period_bytes = period_size * bytes_per_frame;++ dma_addr_t buf = substream->dma_buffer.addr;+ int i;++ u32 needed, freed;+ u32 size_array[5];++ for (i = 0; i != periods; ++i) {+ u32 buf_lo = 0;+ u32 buf_hi = 0;+ u32 buffer_index = 0;++ err = lx_buffer_ask(chip, 0, is_capture, &needed, &freed,+ size_array);+ snd_printdd(LXP "starting: needed %d, freed %d\n",+ needed, freed);++ unpack_pointer(buf, &buf_lo, &buf_hi);+ err = lx_buffer_give(chip, 0, is_capture, period_bytes, buf_lo,+ buf_hi, &buffer_index);++ snd_printdd(LXP "starting: buffer index %x on %p (%d bytes)\n",+ buffer_index, (void*)buf, period_bytes);+ buf += period_bytes;+ }++ err = lx_buffer_ask(chip, 0, is_capture, &needed, &freed, size_array);+ snd_printdd(LXP "starting: needed %d, freed %d\n", needed, freed);++ snd_printd(LXP "starting: starting stream\n");+ err = lx_stream_start(chip, 0, is_capture);+ if (err < 0)+ snd_printk(KERN_ERR LXP "couldn't start stream\n");+ else+ lx_stream->status = LX_STREAM_STATUS_RUNNING;++ lx_stream->frame_pos = 0;+}++static void lx_trigger_stop(struct lx6464es *chip, struct lx_stream *lx_stream)+{+ const int is_capture = lx_stream->is_capture;+ int err;++ snd_printd(LXP "stopping: stopping stream\n");+ err = lx_stream_stop(chip, 0, is_capture);+ if (err < 0)+ snd_printk(KERN_ERR LXP "couldn't stop stream\n");+ else+ lx_stream->status = LX_STREAM_STATUS_FREE;++}++static void lx_trigger_tasklet_dispatch_stream(struct lx6464es *chip,+ struct lx_stream *lx_stream)+{+ switch (lx_stream->status) {+ case LX_STREAM_STATUS_SCHEDULE_RUN:+ lx_trigger_start(chip, lx_stream);+ break;++ case LX_STREAM_STATUS_SCHEDULE_STOP:+ lx_trigger_stop(chip, lx_stream);+ break;++ default:+ break;+ }+}++static void lx_trigger_tasklet(unsigned long data)+{+ struct lx6464es *chip = (struct lx6464es *)data;+ unsigned long flags;++ snd_printdd("->lx_trigger_tasklet\n");++ spin_lock_irqsave(&chip->lock, flags);+ lx_trigger_tasklet_dispatch_stream(chip, &chip->capture_stream);+ lx_trigger_tasklet_dispatch_stream(chip, &chip->playback_stream);+ spin_unlock_irqrestore(&chip->lock, flags);+}++static int lx_pcm_trigger_dispatch(struct lx6464es *chip,+ struct lx_stream *lx_stream, int cmd)+{+ int err = 0;++ switch (cmd) {+ case SNDRV_PCM_TRIGGER_START:+ lx_stream->status = LX_STREAM_STATUS_SCHEDULE_RUN;+ break;++ case SNDRV_PCM_TRIGGER_STOP:+ lx_stream->status = LX_STREAM_STATUS_SCHEDULE_STOP;+ break;++ default:+ err = -EINVAL;+ goto exit;+ }+ tasklet_schedule(&chip->trigger_tasklet);++exit:+ return err;+}+++static int lx_pcm_trigger(struct snd_pcm_substream *substream, int cmd)+{+ struct lx6464es *chip = snd_pcm_substream_chip(substream);+ const int is_capture = (substream->stream == SNDRV_PCM_STREAM_CAPTURE);+ struct lx_stream *stream = is_capture ? &chip->capture_stream :+ &chip->playback_stream;++ snd_printdd("->lx_pcm_trigger\n");++ return lx_pcm_trigger_dispatch(chip, stream, cmd);+}++static int snd_lx6464es_free(struct lx6464es *chip)+{+ snd_printdd("->snd_lx6464es_free\n");++ lx_irq_disable(chip);++ if (chip->irq >= 0)+ free_irq(chip->irq, chip);++ iounmap(chip->port_dsp_remapped);+ ioport_unmap(chip->port_plx_remapped);++ pci_release_regions(chip->pci);+ pci_disable_device(chip->pci);++ kfree(chip);++ return 0;+}++static int snd_lx6464es_dev_free(struct snd_device *device)+{+ return snd_lx6464es_free(device->device_data);+}++/* reset the dsp during initialization */+static int __devinit lx_init_xilinx_reset(struct lx6464es *chip)+{+ int i;+ u32 plx_reg = lx_plx_reg_read(chip, ePLX_CHIPSC);++ snd_printdd("->lx_init_xilinx_reset\n");++ /* activate reset of xilinx */+ plx_reg &= ~CHIPSC_RESET_XILINX;++ lx_plx_reg_write(chip, ePLX_CHIPSC, plx_reg);+ msleep(1);++ lx_plx_reg_write(chip, ePLX_MBOX3, 0);+ msleep(1);++ plx_reg |= CHIPSC_RESET_XILINX;+ lx_plx_reg_write(chip, ePLX_CHIPSC, plx_reg);++ /* deactivate reset of xilinx */+ for (i = 0; i != 100; ++i) {+ u32 reg_mbox3;+ msleep(10);+ reg_mbox3 = lx_plx_reg_read(chip, ePLX_MBOX3);+ if (reg_mbox3) {+ snd_printd(LXP "xilinx reset done\n");+ snd_printdd(LXP "xilinx took %d loops\n", i);+ break;+ }+ }++ /* todo: add some error handling? */++ /* clear mr */+ lx_dsp_reg_write(chip, eReg_CSM, 0);++ /* le xilinx ES peut ne pas etre encore pret, on attend. */+ msleep(600);++ return 0;+}++static int __devinit lx_init_xilinx_test(struct lx6464es *chip)+{+ u32 reg;++ snd_printdd("->lx_init_xilinx_test\n");++ /* TEST if we have access to Xilinx/MicroBlaze */+ lx_dsp_reg_write(chip, eReg_CSM, 0);++ reg = lx_dsp_reg_read(chip, eReg_CSM);++ if (reg) {+ snd_printk(KERN_ERR LXP "Problem: Reg_CSM %x.\n", reg);++ /* PCI9056_SPACE0_REMAP */+ lx_plx_reg_write(chip, ePLX_PCICR, 1);++ reg = lx_dsp_reg_read(chip, eReg_CSM);+ if (reg) {+ snd_printk(KERN_ERR LXP "Error: Reg_CSM %x.\n", reg);+ return -EAGAIN; /* seems to be appropriate */+ }+ }++ snd_printd(LXP "Xilinx/MicroBlaze access test successful\n");++ return 0;+}++/* initialize ethersound */+static int __devinit lx_init_ethersound_config(struct lx6464es *chip)+{+ int i;+ u32 orig_conf_es = lx_dsp_reg_read(chip, eReg_CONFES);++ u32 default_conf_es = (64 << IOCR_OUTPUTS_OFFSET) |+ (64 << IOCR_INPUTS_OFFSET) |+ (FREQ_RATIO_SINGLE_MODE << FREQ_RATIO_OFFSET);++ u32 conf_es = (orig_conf_es & CONFES_READ_PART_MASK)+ | (default_conf_es & CONFES_WRITE_PART_MASK);++ snd_printdd("->lx_init_ethersound\n");++ chip->freq_ratio = FREQ_RATIO_SINGLE_MODE;++ /*+ * write it to the card !+ * this actually kicks the ES xilinx, the first time since poweron.+ * the MAC address in the Reg_ADMACESMSB Reg_ADMACESLSB registers+ * is not ready before this is done, and the bit 2 in Reg_CSES is set.+ * */+ lx_dsp_reg_write(chip, eReg_CONFES, conf_es);++ for (i = 0; i != 1000; ++i) {+ if (lx_dsp_reg_read(chip, eReg_CSES) & 4) {+ snd_printd(LXP "ethersound initialized after %dms\n",+ i);+ goto ethersound_initialized;+ }+ msleep(1);+ }+ snd_printk(KERN_WARNING LXP+ "ethersound could not be initialized after %dms\n", i);+ return -ETIMEDOUT;++ ethersound_initialized:+ snd_printd(LXP "ethersound initialized\n");+ return 0;+}++static int __devinit lx_init_get_version_features(struct lx6464es *chip)+{+ u32 dsp_version;++ int err;++ snd_printdd("->lx_init_get_version_features\n");++ err = lx_dsp_get_version(chip, &dsp_version);++ if (err == 0) {+ u32 freq;++ snd_printk(LXP "DSP version: V%02d.%02d #%d\n",+ (dsp_version>>16) & 0xff, (dsp_version>>8) & 0xff,+ dsp_version & 0xff);++ /* later: what firmware version do we expect? */++ /* retrieve Play/Rec features */+ /* done here because we may have to handle alternate+ * DSP files. */+ /* later */++ /* init the EtherSound sample rate */+ err = lx_dsp_get_clock_frequency(chip, &freq);+ if (err == 0)+ chip->board_sample_rate = freq;+ snd_printd(LXP "actual clock frequency %d\n", freq);+ } else {+ snd_printk(KERN_ERR LXP "DSP corrupted \n");+ err = -EAGAIN;+ }++ return err;+}++static int lx_set_granularity(struct lx6464es *chip, u32 gran)+{+ int err = 0;+ u32 snapped_gran = MICROBLAZE_IBL_MIN;++ snd_printdd("->lx_set_granularity\n");++ /* blocksize is a power of 2 */+ while ((snapped_gran < gran) &&+ (snapped_gran < MICROBLAZE_IBL_MAX)) {+ snapped_gran *= 2;+ }++ if (snapped_gran == chip->pcm_granularity)+ return 0;++ err = lx_dsp_set_granularity(chip, snapped_gran);+ if (err < 0) {+ snd_printk(KERN_WARNING LXP "could not set granularity\n");+ err = -EAGAIN;+ }++ if (snapped_gran != gran)+ snd_printk(LXP "snapped blocksize to %d\n", snapped_gran);++ snd_printd(LXP "set blocksize on board %d\n", snapped_gran);+ chip->pcm_granularity = snapped_gran;++ return err;+}++/* initialize and test the xilinx dsp chip */+static int __devinit lx_init_dsp(struct lx6464es *chip)+{+ int err;+ u8 mac_address[6];+ int i;++ snd_printdd("->lx_init_dsp\n");++ snd_printd(LXP "initialize board\n");+ err = lx_init_xilinx_reset(chip);+ if (err)+ return err;++ snd_printd(LXP "testing board\n");+ err = lx_init_xilinx_test(chip);+ if (err)+ return err;++ snd_printd(LXP "initialize ethersound configuration\n");+ err = lx_init_ethersound_config(chip);+ if (err)+ return err;++ lx_irq_enable(chip);++ /** \todo the mac address should be ready by not, but it isn't,+ * so we wait for it */+ for (i = 0; i != 1000; ++i) {+ err = lx_dsp_get_mac(chip, mac_address);+ if (err)+ return err;+ if (mac_address[0] || mac_address[1] || mac_address[2] ||+ mac_address[3] || mac_address[4] || mac_address[5])+ goto mac_ready;+ msleep(1);+ }+ return -ETIMEDOUT;++mac_ready:+ snd_printd(LXP "mac address ready read after: %dms\n", i);+ snd_printk(LXP "mac address: %02X.%02X.%02X.%02X.%02X.%02X\n",+ mac_address[0], mac_address[1], mac_address[2],+ mac_address[3], mac_address[4], mac_address[5]);++ err = lx_init_get_version_features(chip);+ if (err)+ return err;++ lx_set_granularity(chip, MICROBLAZE_IBL_DEFAULT);++ chip->playback_mute = 0;++ return err;+}++static struct snd_pcm_ops lx_ops_playback = {+ .open = lx_pcm_open,+ .close = lx_pcm_close,+ .ioctl = snd_pcm_lib_ioctl,+ .prepare = lx_pcm_prepare,+ .hw_params = lx_pcm_hw_params_playback,+ .hw_free = lx_pcm_hw_free,+ .trigger = lx_pcm_trigger,+ .pointer = lx_pcm_stream_pointer,+};++static struct snd_pcm_ops lx_ops_capture = {+ .open = lx_pcm_open,+ .close = lx_pcm_close,+ .ioctl = snd_pcm_lib_ioctl,+ .prepare = lx_pcm_prepare,+ .hw_params = lx_pcm_hw_params_capture,+ .hw_free = lx_pcm_hw_free,+ .trigger = lx_pcm_trigger,+ .pointer = lx_pcm_stream_pointer,+};++static int __devinit lx_pcm_create(struct lx6464es *chip)+{+ int err;+ struct snd_pcm *pcm;++ u32 size = 64 * /* channels */+ 3 * /* 24 bit samples */+ MAX_STREAM_BUFFER * /* periods */+ MICROBLAZE_IBL_MAX * /* frames per period */+ 2; /* duplex */++ size = PAGE_ALIGN(size);++ /* hardcoded device name & channel count */+ err = snd_pcm_new(chip->card, (char *)card_name, 0,+ 1, 1, &pcm);++ pcm->private_data = chip;++ snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_PLAYBACK, &lx_ops_playback);+ snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_CAPTURE, &lx_ops_capture);++ pcm->info_flags = 0;+ strcpy(pcm->name, card_name);++ err = snd_pcm_lib_preallocate_pages_for_all(pcm, SNDRV_DMA_TYPE_DEV,+ snd_dma_pci_data(chip->pci),+ size, size);+ if (err < 0)+ return err;++ chip->pcm = pcm;+ chip->capture_stream.is_capture = 1;++ return 0;+}++static int lx_control_playback_info(struct snd_kcontrol *kcontrol,+ struct snd_ctl_elem_info *uinfo)+{+ uinfo->type = SNDRV_CTL_ELEM_TYPE_BOOLEAN;+ uinfo->count = 1;+ uinfo->value.integer.min = 0;+ uinfo->value.integer.max = 1;+ return 0;+}++static int lx_control_playback_get(struct snd_kcontrol *kcontrol,+ struct snd_ctl_elem_value *ucontrol)+{+ struct lx6464es *chip = snd_kcontrol_chip(kcontrol);+ ucontrol->value.integer.value[0] = chip->playback_mute;+ return 0;+}++static int lx_control_playback_put(struct snd_kcontrol *kcontrol,+ struct snd_ctl_elem_value *ucontrol)+{+ struct lx6464es *chip = snd_kcontrol_chip(kcontrol);+ int changed = 0;+ int current_value = chip->playback_mute;++ if (current_value != ucontrol->value.integer.value[0]) {+ lx_level_unmute(chip, 0, !current_value);+ chip->playback_mute = !current_value;+ changed = 1;+ }+ return changed;+}++static struct snd_kcontrol_new lx_control_playback_switch __devinitdata = {+ .iface = SNDRV_CTL_ELEM_IFACE_MIXER,+ .name = "PCM Playback Switch",+ .index = 0,+ .access = SNDRV_CTL_ELEM_ACCESS_READWRITE,+ .private_value = 0,+ .info = lx_control_playback_info,+ .get = lx_control_playback_get,+ .put = lx_control_playback_put+};++++static void lx_proc_levels_read(struct snd_info_entry *entry,+ struct snd_info_buffer *buffer)+{+ u32 levels[64];+ int err;+ int i, j;+ struct lx6464es *chip = entry->private_data;++ snd_iprintf(buffer, "capture levels:\n");+ err = lx_level_peaks(chip, 1, 64, levels);+ if (err < 0)+ return;++ for (i = 0; i != 8; ++i) {+ for (j = 0; j != 8; ++j)+ snd_iprintf(buffer, "%08x ", levels[i*8+j]);+ snd_iprintf(buffer, "\n");+ }++ snd_iprintf(buffer, "\nplayback levels:\n");++ err = lx_level_peaks(chip, 0, 64, levels);+ if (err < 0)+ return;++ for (i = 0; i != 8; ++i) {+ for (j = 0; j != 8; ++j)+ snd_iprintf(buffer, "%08x ", levels[i*8+j]);+ snd_iprintf(buffer, "\n");+ }++ snd_iprintf(buffer, "\n");+}++static int lx_proc_create(struct snd_card *card, struct lx6464es *chip)+{+ struct snd_info_entry *entry;+ int err = snd_card_proc_new(card, "levels", &entry);+ if (err < 0)+ return err;++ snd_info_set_text_ops(entry, chip, lx_proc_levels_read);+ return 0;+}+++static int __devinit snd_lx6464es_create(struct snd_card *card,+ struct pci_dev *pci,+ struct lx6464es **rchip)+{+ struct lx6464es *chip;+ int err;++ static struct snd_device_ops ops = {+ .dev_free = snd_lx6464es_dev_free,+ };++ snd_printdd("->snd_lx6464es_create\n");++ *rchip = NULL;++ /* enable PCI device */+ err = pci_enable_device(pci);+ if (err < 0)+ return err;++ pci_set_master(pci);++ /* check if we can restrict PCI DMA transfers to 32 bits */+ err = pci_set_dma_mask(pci, DMA_32BIT_MASK);+ if (err < 0) {+ snd_printk(KERN_ERR "architecture does not support "+ "32bit PCI busmaster DMA\n");+ pci_disable_device(pci);+ return -ENXIO;+ }++ chip = kzalloc(sizeof(*chip), GFP_KERNEL);+ if (chip == NULL) {+ err = -ENOMEM;+ goto alloc_failed;+ }++ chip->card = card;+ chip->pci = pci;+ chip->irq = -1;++ /* initialize synchronization structs */+ spin_lock_init(&chip->lock);+ spin_lock_init(&chip->msg_lock);+ mutex_init(&chip->setup_mutex);+ tasklet_init(&chip->trigger_tasklet, lx_trigger_tasklet,+ (unsigned long)chip);+ tasklet_init(&chip->tasklet_capture, lx_tasklet_capture,+ (unsigned long)chip);+ tasklet_init(&chip->tasklet_playback, lx_tasklet_playback,+ (unsigned long)chip);++ /* request resources */+ err = pci_request_regions(pci, card_name);+ if (err < 0)+ goto request_regions_failed;++ chip->port_mem = pci_resource_start(pci, 0);++ /* plx port */+ chip->port_plx = pci_resource_start(pci, 1);+ chip->port_plx_remapped = ioport_map(chip->port_plx,+ pci_resource_len(pci, 1));++ /* dsp port */+ chip->port_dsp = pci_resource_start(pci, 2);+ chip->port_dsp_resource = request_mem_region(chip->port_dsp,+ pci_resource_len(pci, 2),+ card_name);+ chip->port_dsp_remapped = ioremap_nocache(chip->port_dsp,+ pci_resource_len(pci, 2));++ err = request_irq(pci->irq, lx_interrupt, IRQF_SHARED,+ card_name, chip);+ if (err) {+ snd_printk(KERN_ERR LXP "unable to grab IRQ %d\n", pci->irq);+ goto request_irq_failed;+ }+ chip->irq = pci->irq;++ err = snd_device_new(card, SNDRV_DEV_LOWLEVEL, chip, &ops);+ if (err < 0)+ return err;++ err = lx_init_dsp(chip);+ if (err < 0) {+ snd_printk(KERN_ERR LXP "error during DSP initialization\n");+ return err;+ }++ err = lx_pcm_create(chip);+ if (err < 0)+ return err;++ err = lx_proc_create(card, chip);+ if (err < 0)+ return err;++ err = snd_ctl_add(card, snd_ctl_new1(&lx_control_playback_switch,+ chip));+ if (err < 0)+ return err;++ snd_card_set_dev(card, &pci->dev);++ *rchip = chip;+ return 0;++request_irq_failed:+ pci_release_regions(pci);++request_regions_failed:+ kfree(chip);++alloc_failed:+ pci_disable_device(pci);++ return err;+}++static int __devinit snd_lx6464es_probe(struct pci_dev *pci,+ const struct pci_device_id *pci_id)+{+ static int dev;+ struct snd_card *card;+ struct lx6464es *chip;+ int err;++ snd_printdd("->snd_lx6464es_probe\n");++ if (dev >= SNDRV_CARDS)+ return -ENODEV;+ if (!enable[dev]) {+ dev++;+ return -ENOENT;+ }++ card = snd_card_new(index[dev], id[dev], THIS_MODULE, 0);+ if (card == NULL)+ return -ENOMEM;++ err = snd_lx6464es_create(card, pci, &chip);+ if (err < 0) {+ snd_printk(KERN_ERR LXP "error during snd_lx6464es_create\n");+ goto out_free;+ }++ strcpy(card->driver, "lx6464es");+ strcpy(card->shortname, "Digigram LX6464ES");+ sprintf(card->longname, "%s at 0x%lx, 0x%lx, 0x%lx, irq %i",+ card->shortname, chip->port_mem, chip->port_plx,+ chip->port_dsp, chip->irq);++ err = snd_card_register(card);+ if (err < 0)+ goto out_free;++ snd_printdd(LXP "initialization successful\n");+ pci_set_drvdata(pci, card);+ dev++;+ return 0;++out_free:+ snd_card_free(card);+ return err;++}++static void __devexit snd_lx6464es_remove(struct pci_dev *pci)+{+ snd_card_free(pci_get_drvdata(pci));+ pci_set_drvdata(pci, NULL);+}+++static struct pci_driver driver = {+ .name = "Digigram LX6464ES",+ .id_table = snd_lx6464es_ids,+ .probe = snd_lx6464es_probe,+ .remove = __devexit_p(snd_lx6464es_remove),+};+++/* module initialization */+static int __init mod_init(void)+{+ return pci_register_driver(&driver);+}++static void __exit mod_exit(void)+{+ pci_unregister_driver(&driver);+}++module_init(mod_init);+module_exit(mod_exit);diff --git a/sound/pci/lx6464es/lx6464es.h b/sound/pci/lx6464es/lx6464es.hnew file mode 100644index 0000000..474c5a7--- /dev/null+++ b/sound/pci/lx6464es/lx6464es.h@@ -0,0 +1,119 @@+/* -*- linux-c -*- *+ *+ * ALSA driver for the digigram lx6464es interface+ *+ * Copyright (c) 2009 Tim Blechmann <tim@xxxxxxxxxx>+ *+ *+ * 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 2 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; see the file COPYING. If not, write to+ * the Free Software Foundation, Inc., 59 Temple Place - Suite 330,+ * Boston, MA 02111-1307, USA.+ *+ */++#ifndef LX6464ES_H+#define LX6464ES_H++#include <linux/spinlock.h>+#include <asm/atomic.h>++#include <sound/core.h>+#include <sound/pcm.h>++#include "lx_core.h"++#define LXP "LX6464ES: "++enum {+ ES_cmd_free = 0, /* no command executing */+ ES_cmd_processing = 1, /* execution of a read/write command */+ ES_read_pending = 2, /* a asynchron read command is pending */+ ES_read_finishing = 3, /* a read command has finished waiting (set by+ * Interrupt or CancelIrp) */+};++enum lx_stream_status {+ LX_STREAM_STATUS_FREE,+/* LX_STREAM_STATUS_OPEN, */+ LX_STREAM_STATUS_SCHEDULE_RUN,+/* LX_STREAM_STATUS_STARTED, */+ LX_STREAM_STATUS_RUNNING,+ LX_STREAM_STATUS_SCHEDULE_STOP,+/* LX_STREAM_STATUS_STOPPED, */+/* LX_STREAM_STATUS_PAUSED */+};+++struct lx_stream {+ struct snd_pcm_substream *stream;+ snd_pcm_uframes_t frame_pos;+ enum lx_stream_status status; /* free, open, running, draining+ * pause */+ int is_capture:1;+};+++struct lx6464es {+ struct snd_card *card;+ struct pci_dev *pci;+ int irq;++ spinlock_t lock; /* interrupt spinlock */+ struct mutex setup_mutex; /* mutex used in hw_params, open+ * and close */++ struct tasklet_struct trigger_tasklet; /* trigger tasklet */+ struct tasklet_struct tasklet_capture;+ struct tasklet_struct tasklet_playback;++ /* ports */+ unsigned long port_mem; /* memory port (32-bit,+ * non-prefetchable,+ * size=512) */+ unsigned long port_plx; /* io port (size=256) */+ void __iomem *port_plx_remapped; /* remapped plx port */+ unsigned long port_dsp; /* memory port (32-bit,+ * non-prefetchable,+ * size=8K) */+ void __iomem *port_dsp_remapped; /* remapped dsp port */+ struct resource *port_dsp_resource; /* its resource */++ /* messaging */+ spinlock_t msg_lock; /* message spinlock */+ atomic_t send_message_locked;+ struct lx_rmh rmh;++ /* configuration */+ uint freq_ratio : 2;+ uint playback_mute : 1;+ uint hardware_running[2];+ u32 board_sample_rate; /* sample rate read from+ * board */+ u32 sample_rate; /* our sample rate */+ u16 pcm_granularity; /* board blocksize */++ /* dma */+ struct snd_dma_buffer capture_dma_buf;+ struct snd_dma_buffer playback_dma_buf;++ /* pcm */+ struct snd_pcm *pcm;++ /* streams */+ struct lx_stream capture_stream;+ struct lx_stream playback_stream;+};+++#endif /* LX6464ES_H */diff --git a/sound/pci/lx6464es/lx_core.c b/sound/pci/lx6464es/lx_core.cnew file mode 100644index 0000000..0a60b18--- /dev/null+++ b/sound/pci/lx6464es/lx_core.c@@ -0,0 +1,1453 @@+/* -*- linux-c -*- *+ *+ * ALSA driver for the digigram lx6464es interface+ * low-level interface+ *+ * Copyright (c) 2009 Tim Blechmann <tim@xxxxxxxxxx>+ *+ * 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 2 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; see the file COPYING. If not, write to+ * the Free Software Foundation, Inc., 59 Temple Place - Suite 330,+ * Boston, MA 02111-1307, USA.+ *+ */++/* #define RMH_DEBUG 1 */++#include <linux/module.h>+#include <linux/pci.h>+#include <linux/delay.h>++#include "lx6464es.h"+#include "lx_core.h"++#include "ethersound.h"++/* low-level register access */++static const unsigned long dsp_port_offsets[] = {+ 0,+ 0x400,+ 0x401,+ 0x402,+ 0x403,+ 0x404,+ 0x405,+ 0x406,+ 0x407,+ 0x408,+ 0x409,+ 0x40a,+ 0x40b,+ 0x40c,++ 0x410,+ 0x411,+ 0x412,+ 0x413,+ 0x414,+ 0x415,+ 0x416,++ 0x420,+ 0x430,+ 0x431,+ 0x432,+ 0x433,+ 0x434,+ 0x440+};++static void __iomem *lx_dsp_register(struct lx6464es *chip, int port)+{+ void __iomem *base_address = chip->port_dsp_remapped;+ return base_address + dsp_port_offsets[port]*4;+}++unsigned long lx_dsp_reg_read(struct lx6464es *chip, int port)+{+ void __iomem *address = lx_dsp_register(chip, port);+ return ioread32(address);+}++void lx_dsp_reg_readbuf(struct lx6464es *chip, int port, u32 *data, u32 len)+{+ void __iomem *address = lx_dsp_register(chip, port);+ memcpy_fromio(data, address, len*sizeof(u32));+}+++void lx_dsp_reg_write(struct lx6464es *chip, int port, unsigned data)+{+ void __iomem *address = lx_dsp_register(chip, port);+ iowrite32(data, address);+}++void lx_dsp_reg_writebuf(struct lx6464es *chip, int port, const u32 *data,+ u32 len)+{+ void __iomem *address = lx_dsp_register(chip, port);+ memcpy_toio(address, data, len*sizeof(u32));+}+++static const unsigned long plx_port_offsets[] = {+ 0x04,+ 0x40,+ 0x44,+ 0x48,+ 0x4c,+ 0x50,+ 0x54,+ 0x58,+ 0x5c,+ 0x64,+ 0x68,+ 0x6C+};++static void __iomem *lx_plx_register(struct lx6464es *chip, int port)+{+ void __iomem *base_address = chip->port_plx_remapped;+ return base_address + plx_port_offsets[port];+}++unsigned long lx_plx_reg_read(struct lx6464es *chip, int port)+{+ void __iomem *address = lx_plx_register(chip, port);+ return ioread32(address);+}++void lx_plx_reg_write(struct lx6464es *chip, int port, u32 data)+{+ void __iomem *address = lx_plx_register(chip, port);+ iowrite32(data, address);+}++u32 lx_plx_mbox_read(struct lx6464es *chip, int mbox_nr)+{+ int index;++ switch (mbox_nr) {+ case 1:+ index = ePLX_MBOX1; break;+ case 2:+ index = ePLX_MBOX2; break;+ case 3:+ index = ePLX_MBOX3; break;+ case 4:+ index = ePLX_MBOX4; break;+ case 5:+ index = ePLX_MBOX5; break;+ case 6:+ index = ePLX_MBOX6; break;+ case 7:+ index = ePLX_MBOX7; break;+ case 0: /* reserved for HF flags */+ snd_BUG();+ default:+ return 0xdeadbeef;+ }++ return lx_plx_reg_read(chip, index);+}++int lx_plx_mbox_write(struct lx6464es *chip, int mbox_nr, u32 value)+{+ int index = -1;++ switch (mbox_nr) {+ case 1:+ index = ePLX_MBOX1; break;+ case 3:+ index = ePLX_MBOX3; break;+ case 4:+ index = ePLX_MBOX4; break;+ case 5:+ index = ePLX_MBOX5; break;+ case 6:+ index = ePLX_MBOX6; break;+ case 7:+ index = ePLX_MBOX7; break;+ case 0: /* reserved for HF flags */+ case 2: /* reserved for Pipe States+ * the DSP keeps an image of it */+ snd_BUG();+ return -EBADRQC;+ }++ lx_plx_reg_write(chip, index, value);+ return 0;+}+++/* rmh */++#ifdef CONFIG_SND_DEBUG+#define CMD_NAME(a) a+#else+#define CMD_NAME(a) NULL+#endif++#define OPCODE_OFFSET 24 /**< offset of the command opcode in the first command word.*/++/* specific to CODECMD_MAJ_BUFFER_PC */+#define MAX_STREAM_BUFFER 5 /**< max amount of stream buffers.*/++#define Reg_CSM_MR 0x00000002+#define Reg_CSM_MC 0x00000001++struct dsp_cmd_info {+ u32 dcCodeOp; /**< Op Code of the command (usually 1st 24-bits word).*/+ u16 dcCmdLength; /**< Command length in words of 24 bits.*/+ u16 dcStatusType; /**< Status type: 0 for fixed length, 1 for random.*/+ u16 dcStatusLength; /**< Status length (if fixed).*/+ char *dcOpName;+};++/**+* @brief Initialization and control data for the Microblaze interface+*+* - OpCode: the opcode field of the command set at the proper offset+* - CmdLength the number of command words+* - StatusType offset in the status registers : 0 means that the return value may be different from SUCCESS, and must be read+* - StatusLength the number of status words (in addition to the return value)+*/++static struct dsp_cmd_info dsp_commands[] =+{+ // ----------------------------------------------------------------------------------------+ // OpCode | CmdLength | StatusType | StatusLength+ // ----------------------------------------------------------------------------------------+ { (CMD_00_INFO_DEBUG << OPCODE_OFFSET) ,1 /*custom*/ ,1 ,0 /**/ , CMD_NAME("INFO_DEBUG") },+ { (CMD_01_GET_SYS_CFG << OPCODE_OFFSET) ,1 /**/ ,1 ,2 /**/ , CMD_NAME("GET_SYS_CFG") },+ { (CMD_02_SET_GRANULARITY << OPCODE_OFFSET) ,1 /**/ ,1 ,0 /**/ , CMD_NAME("SET_GRANULARITY") },+ { (CMD_03_SET_TIMER_IRQ << OPCODE_OFFSET) ,1 /**/ ,1 ,0 /**/ , CMD_NAME("SET_TIMER_IRQ") },+ { (CMD_04_GET_EVENT << OPCODE_OFFSET) ,1 /**/ ,1 ,0 /*up to 10*/ , CMD_NAME("GET_EVENT") },+ { (CMD_05_GET_PIPES << OPCODE_OFFSET) ,1 /**/ ,1 ,2 /*up to 4*/ , CMD_NAME("GET_PIPES") },+ { (CMD_06_ALLOCATE_PIPE << OPCODE_OFFSET) ,1 /**/ ,0 ,0 /**/ , CMD_NAME("ALLOCATE_PIPE") },+ { (CMD_07_RELEASE_PIPE << OPCODE_OFFSET) ,1 /**/ ,0 ,0 /**/ , CMD_NAME("RELEASE_PIPE") },+ { (CMD_08_ASK_BUFFERS << OPCODE_OFFSET) ,1 /**/ ,1 ,MAX_STREAM_BUFFER , CMD_NAME("ASK_BUFFERS") },+ { (CMD_09_STOP_PIPE << OPCODE_OFFSET) ,1 /**/ ,0 ,0 /*up to 2*/ , CMD_NAME("STOP_PIPE") },+ { (CMD_0A_GET_PIPE_SPL_COUNT << OPCODE_OFFSET) ,1 /**/ ,1 ,1 /*up to 2*/ , CMD_NAME("GET_PIPE_SPL_COUNT") },+ { (CMD_0B_TOGGLE_PIPE_STATE << OPCODE_OFFSET) ,1 /*up to 5*/ ,1 ,0 /**/ , CMD_NAME("TOGGLE_PIPE_STATE") },+ { (CMD_0C_DEF_STREAM << OPCODE_OFFSET) ,1 /*up to 4*/ ,1 ,0 /**/ , CMD_NAME("DEF_STREAM") },+ { (CMD_0D_SET_MUTE << OPCODE_OFFSET) ,3 /**/ ,1 ,0 /**/ , CMD_NAME("SET_MUTE") },+ { (CMD_0E_GET_STREAM_SPL_COUNT << OPCODE_OFFSET) ,1/**/ ,1 ,2 /**/ , CMD_NAME("GET_STREAM_SPL_COUNT") },+ { (CMD_0F_UPDATE_BUFFER << OPCODE_OFFSET) ,3 /*up to 4*/ ,0 ,1 /**/ , CMD_NAME("UPDATE_BUFFER") },+ { (CMD_10_GET_BUFFER << OPCODE_OFFSET) ,1 /**/ ,1 ,4 /**/ , CMD_NAME("GET_BUFFER") },+ { (CMD_11_CANCEL_BUFFER << OPCODE_OFFSET) ,1 /**/ ,1 ,1 /*up to 4*/ , CMD_NAME("CANCEL_BUFFER") },+ { (CMD_12_GET_PEAK << OPCODE_OFFSET) ,1 /**/ ,1 ,1 /**/ , CMD_NAME("GET_PEAK") },+ { (CMD_13_SET_STREAM_STATE << OPCODE_OFFSET) ,1 /**/ ,1 ,0 /**/ , CMD_NAME("SET_STREAM_STATE") },+};++static void lx_message_init(struct lx_rmh *rmh, cmd_mb_opcodes cmd)+{+ snd_BUG_ON(cmd >= CMD_14_INVALID);++ rmh->cmd[0] = dsp_commands[cmd].dcCodeOp;+ rmh->cmd_len = dsp_commands[cmd].dcCmdLength;+ rmh->stat_len = dsp_commands[cmd].dcStatusLength;+ rmh->dsp_stat = dsp_commands[cmd].dcStatusType;+ rmh->cmd_idx = cmd;+ memset(&rmh->cmd[1], 0, (REG_CRM_NUMBER - 1) * sizeof(u32));++#ifdef CONFIG_SND_DEBUG+ memset(rmh->stat, 0, REG_CRM_NUMBER * sizeof(u32));+#endif+#ifdef RMH_DEBUG+ rmh->cmd_idx = cmd;+#endif+}++#ifdef RMH_DEBUG+#define LXRMH "lx6464es rmh: "+static void lx_message_dump(struct lx_rmh *rmh)+{+ u8 idx = rmh->cmd_idx;+ int i;++ snd_printk(LXRMH "command %s\n", dsp_commands[idx].dcOpName);++ for (i = 0; i != rmh->cmd_len; ++i)+ snd_printk(LXRMH "\tcmd[%d] %08x\n", i, rmh->cmd[i]);++ for (i = 0; i != rmh->stat_len; ++i)+ snd_printk(LXRMH "\tstat[%d]: %08x\n", i, rmh->stat[i]);+ snd_printk("\n");+}+#else+static inline void lx_message_dump(struct lx_rmh *rmh)+{}+#endif++++/* sleep 500 - 100 = 400 times 100us -> the timeout is >= 40 ms */+#define XILINX_TIMEOUT_MS 40+#define XILINX_POLL_NO_SLEEP 100+#define XILINX_POLL_ITERATIONS 150++static int lx_message_send(struct lx6464es *chip, struct lx_rmh *rmh)+{+ u32 reg = ED_DSP_TIMED_OUT;+ int dwloop;+ int answer_received;++ if (lx_dsp_reg_read(chip, eReg_CSM) & (Reg_CSM_MC | Reg_CSM_MR)) {+ snd_printk(KERN_ERR LXP "PIOSendMessage eReg_CSM %x\n", reg);+ return -EBUSY;+ }++ /* write command */+ lx_dsp_reg_writebuf(chip, eReg_CRM1, rmh->cmd, rmh->cmd_len);++ snd_BUG_ON(atomic_read(&chip->send_message_locked) != 0);+ atomic_set(&chip->send_message_locked, 1);++ /* MicoBlaze gogogo */+ lx_dsp_reg_write(chip, eReg_CSM, Reg_CSM_MC);++ /* wait for interrupt to answer */+ for (dwloop = 0; dwloop != XILINX_TIMEOUT_MS; ++dwloop) {+ answer_received = atomic_read(&chip->send_message_locked);+ if (answer_received == 0)+ break;+ msleep(1);+ }++ if (answer_received == 0) {+ /* in Debug mode verify Reg_CSM_MR */+ snd_BUG_ON(!(lx_dsp_reg_read(chip, eReg_CSM) & Reg_CSM_MR));++ /* command finished, read status */+ if (rmh->dsp_stat == 0)+ reg = lx_dsp_reg_read(chip, eReg_CRM1);+ else+ reg = SUCCESS;+ } else {+ int i;+ snd_printk(KERN_WARNING LXP "TIMEOUT lx_message_send! "+ "Interrupts disabled?\n");++ /* attente bit Reg_CSM_MR */+ for (i = 0; i != XILINX_POLL_ITERATIONS; i++) {+ if ((lx_dsp_reg_read(chip, eReg_CSM) & Reg_CSM_MR)) {+ if (rmh->dsp_stat == 0)+ reg = lx_dsp_reg_read(chip, eReg_CRM1);+ else+ reg = SUCCESS;+ goto polling_successful;+ }++ if (i > XILINX_POLL_NO_SLEEP)+ msleep(1);+ }+ snd_printk(KERN_WARNING LXP "TIMEOUT lx_message_send! "+ "polling failed\n");++polling_successful:+ atomic_set(&chip->send_message_locked, 0);+ }++ if ((reg & ERROR_VALUE) == 0) {+ /* read response */+ if (rmh->stat_len) {+ snd_BUG_ON(rmh->stat_len >= (REG_CRM_NUMBER-1));++ lx_dsp_reg_readbuf(chip, eReg_CRM2, rmh->stat,+ rmh->stat_len);+ }+ } else+ snd_printk(KERN_WARNING LXP "lx_message_send: error_value %x\n",+ reg);++ /* clear Reg_CSM_MR */+ lx_dsp_reg_write(chip, eReg_CSM, 0);++ switch (reg) {+ case ED_DSP_TIMED_OUT:+ snd_printk(KERN_WARNING LXP "lx_message_send: dsp timeout\n");+ return -ETIMEDOUT;++ case ED_DSP_CRASHED:+ snd_printk(KERN_WARNING LXP "lx_message_send: dsp crashed\n");+ return -EAGAIN;+ }++ lx_message_dump(rmh);+ return 0;+}++static int lx_message_send_atomic(struct lx6464es *chip, struct lx_rmh *rmh)+{+ u32 reg = ED_DSP_TIMED_OUT;+ int dwloop;++ if (lx_dsp_reg_read(chip, eReg_CSM) & (Reg_CSM_MC | Reg_CSM_MR)) {+ snd_printk(KERN_ERR LXP "PIOSendMessage eReg_CSM %x\n", reg);+ return -EBUSY;+ }++ /* write command */+ lx_dsp_reg_writebuf(chip, eReg_CRM1, rmh->cmd, rmh->cmd_len);++ /* MicoBlaze gogogo */+ lx_dsp_reg_write(chip, eReg_CSM, Reg_CSM_MC);++ /* wait for interrupt to answer */+ for (dwloop = 0; dwloop != XILINX_TIMEOUT_MS * 1000; ++dwloop) {+ if (lx_dsp_reg_read(chip, eReg_CSM) & Reg_CSM_MR) {+ if (rmh->dsp_stat == 0)+ reg = lx_dsp_reg_read(chip, eReg_CRM1);+ else+ reg = SUCCESS;+ goto polling_successful;+ } else+ udelay(1);+ }+ snd_printk(KERN_WARNING LXP "TIMEOUT lx_message_send_atomic! "+ "polling failed\n");++polling_successful:+ if ((reg & ERROR_VALUE) == 0) {+ /* read response */+ if (rmh->stat_len) {+ snd_BUG_ON(rmh->stat_len >= (REG_CRM_NUMBER-1));+ lx_dsp_reg_readbuf(chip, eReg_CRM2, rmh->stat,+ rmh->stat_len);+ }+ } else+ snd_printk(LXP "rmh error: %08x\n", reg);++ /* clear Reg_CSM_MR */+ lx_dsp_reg_write(chip, eReg_CSM, 0);++ switch (reg) {+ case ED_DSP_TIMED_OUT:+ snd_printk(KERN_WARNING LXP "lx_message_send: dsp timeout\n");+ return -ETIMEDOUT;++ case ED_DSP_CRASHED:+ snd_printk(KERN_WARNING LXP "lx_message_send: dsp crashed\n");+ return -EAGAIN;+ }++ lx_message_dump(rmh);++ return reg;+}+++/* low-level dsp access */+int __devinit lx_dsp_get_version(struct lx6464es *chip, u32 *rdsp_version)+{+ u16 ret;+ unsigned long flags;++ spin_lock_irqsave(&chip->msg_lock, flags);++ lx_message_init(&chip->rmh, CMD_01_GET_SYS_CFG);+ ret = lx_message_send_atomic(chip, &chip->rmh);++ *rdsp_version = chip->rmh.stat[1];+ spin_unlock_irqrestore(&chip->msg_lock, flags);+ return ret;+}++int lx_dsp_get_clock_frequency(struct lx6464es *chip, u32 *rfreq)+{+ u16 ret = SUCCESS;+ unsigned long flags;+ u32 freq_raw = 0;+ u32 freq = 0;+ u32 frequency = 0;++ spin_lock_irqsave(&chip->msg_lock, flags);++ lx_message_init(&chip->rmh, CMD_01_GET_SYS_CFG);+ ret = lx_message_send_atomic(chip, &chip->rmh);++ if (ret == SUCCESS) {+ freq_raw = chip->rmh.stat[0] >> FREQ_FIELD_OFFSET;+ freq = freq_raw & XES_FREQ_COUNT8_MASK;++ if ((freq < XES_FREQ_COUNT8_48_MAX) ||+ (freq > XES_FREQ_COUNT8_44_MIN))+ frequency = 0; /* unknown */+ else if (freq >= XES_FREQ_COUNT8_44_MAX)+ frequency = 44100;+ else+ frequency = 48000;+ }++ spin_unlock_irqrestore(&chip->msg_lock, flags);++ *rfreq = frequency * chip->freq_ratio;++ return ret;+}++int lx_dsp_get_mac(struct lx6464es *chip, u8 *mac_address)+{+ u32 macmsb, maclsb;++ macmsb = lx_dsp_reg_read(chip, eReg_ADMACESMSB) & 0x00FFFFFF;+ maclsb = lx_dsp_reg_read(chip, eReg_ADMACESLSB) & 0x00FFFFFF;++ /* todo: endianess handling */+ mac_address[5] = ((u8 *)(&maclsb))[0];+ mac_address[4] = ((u8 *)(&maclsb))[1];+ mac_address[3] = ((u8 *)(&maclsb))[2];+ mac_address[2] = ((u8 *)(&macmsb))[0];+ mac_address[1] = ((u8 *)(&macmsb))[1];+ mac_address[0] = ((u8 *)(&macmsb))[2];++ return 0;+}+++int lx_dsp_set_granularity(struct lx6464es *chip, u32 gran)+{+ unsigned long flags;+ int ret;++ spin_lock_irqsave(&chip->msg_lock, flags);++ lx_message_init(&chip->rmh, CMD_02_SET_GRANULARITY);+ chip->rmh.cmd[0] |= gran;++ ret = lx_message_send_atomic(chip, &chip->rmh);+ spin_unlock_irqrestore(&chip->msg_lock, flags);+ return ret;+}++int lx_dsp_read_async_events(struct lx6464es *chip, u32 *data)+{+ unsigned long flags;+ int ret;++ spin_lock_irqsave(&chip->msg_lock, flags);++ lx_message_init(&chip->rmh, CMD_04_GET_EVENT);+ chip->rmh.stat_len = 9; /* we don't necessarily need the full length */++ ret = lx_message_send_atomic(chip, &chip->rmh);++ if (!ret)+ memcpy(data, chip->rmh.stat, chip->rmh.stat_len * sizeof(u32));++ spin_unlock_irqrestore(&chip->msg_lock, flags);+ return ret;+}++#define CSES_TIMEOUT 100 /* microseconds */+#define CSES_CE 0x0001+#define CSES_BROADCAST 0x0002+#define CSES_UPDATE_LDSV 0x0004++int lx_dsp_es_check_pipeline(struct lx6464es *chip)+{+ int i;++ for (i = 0; i != CSES_TIMEOUT; ++i) {+ /*+ * le bit CSES_UPDATE_LDSV est à 1 dés que le macprog est pret.+ * il re-passe à 0 lorsque le premier read a été fait.+ * pour l'instant on retire le test car ce bit passe a 1 environ 200 à 400 ms+ * aprés que le registre confES à été écrit (kick du xilinx ES).+ *+ * On ne teste que le bit CE.+ * */++ u32 cses = lx_dsp_reg_read(chip, eReg_CSES);++ if ((cses & CSES_CE) == 0)+ return 0;++ udelay(1);+ }++ return -ETIMEDOUT;+}+++#define PIPE_INFO_TO_CMD(capture, pipe) \+ ((u32)((u32)(pipe) | ((capture) ? ID_IS_CAPTURE : 0L)) << ID_OFFSET)++++/* low-level pipe handling */+int lx_pipe_allocate(struct lx6464es *chip, u32 pipe, int is_capture,+ int channels)+{+ int err;+ unsigned long flags;++ u32 pipe_cmd = PIPE_INFO_TO_CMD(is_capture, pipe);++ spin_lock_irqsave(&chip->msg_lock, flags);+ lx_message_init(&chip->rmh, CMD_06_ALLOCATE_PIPE);++ chip->rmh.cmd[0] |= pipe_cmd;+ chip->rmh.cmd[0] |= channels;++ err = lx_message_send_atomic(chip, &chip->rmh);+ spin_unlock_irqrestore(&chip->msg_lock, flags);++ if (err != SUCCESS)+ snd_printk(KERN_ERR "lx6464es: could not allocate pipe\n");++ return err;+}++int lx_pipe_release(struct lx6464es *chip, u32 pipe, int is_capture)+{+ int err;+ unsigned long flags;++ u32 pipe_cmd = PIPE_INFO_TO_CMD(is_capture, pipe);++ spin_lock_irqsave(&chip->msg_lock, flags);+ lx_message_init(&chip->rmh, CMD_07_RELEASE_PIPE);++ chip->rmh.cmd[0] |= pipe_cmd;++ err = lx_message_send_atomic(chip, &chip->rmh);+ spin_unlock_irqrestore(&chip->msg_lock, flags);++ return err;+}++int lx_buffer_ask(struct lx6464es *chip, u32 pipe, int is_capture,+ u32 *r_needed, u32 *r_freed, u32 *size_array)+{+ int err;+ unsigned long flags;++ u32 pipe_cmd = PIPE_INFO_TO_CMD(is_capture, pipe);++#ifdef CONFIG_SND_DEBUG+ if (size_array)+ memset(size_array, 0, sizeof(u32)*MAX_STREAM_BUFFER);+#endif++ *r_needed = 0;+ *r_freed = 0;++ spin_lock_irqsave(&chip->msg_lock, flags);+ lx_message_init(&chip->rmh, CMD_08_ASK_BUFFERS);++ chip->rmh.cmd[0] |= pipe_cmd;++ err = lx_message_send_atomic(chip, &chip->rmh);++ if (!err) {+ int i;+ for (i = 0; i < MAX_STREAM_BUFFER; ++i) {+ u32 stat = chip->rmh.stat[i];+ if (stat & (BF_EOB << BUFF_FLAGS_OFFSET)) {+ /* finished */+ *r_freed += 1;+ if (size_array)+ size_array[i] = stat & MASK_DATA_SIZE;+ } else if ((stat & (BF_VALID << BUFF_FLAGS_OFFSET))+ == 0)+ /* free */+ *r_needed += 1;+ }++#if 0+ snd_printdd(LXP "CMD_08_ASK_BUFFERS: needed %d, freed %d\n",+ *r_needed, *r_freed);+ for (i = 0; i < MAX_STREAM_BUFFER; ++i) {+ for (i = 0; i != chip->rmh.stat_len; ++i)+ snd_printdd(" stat[%d]: %x, %x\n", i,+ chip->rmh.stat[i],+ chip->rmh.stat[i] & MASK_DATA_SIZE);+ }+#endif+ }++ spin_unlock_irqrestore(&chip->msg_lock, flags);+ return err;+}+++int lx_pipe_stop(struct lx6464es *chip, u32 pipe, int is_capture)+{+ int err;+ unsigned long flags;++ u32 pipe_cmd = PIPE_INFO_TO_CMD(is_capture, pipe);++ spin_lock_irqsave(&chip->msg_lock, flags);+ lx_message_init(&chip->rmh, CMD_09_STOP_PIPE);++ chip->rmh.cmd[0] |= pipe_cmd;++ err = lx_message_send_atomic(chip, &chip->rmh);++ spin_unlock_irqrestore(&chip->msg_lock, flags);+ return err;+}++static int lx_pipe_toggle_state(struct lx6464es *chip, u32 pipe, int is_capture)+{+ int err;+ unsigned long flags;++ u32 pipe_cmd = PIPE_INFO_TO_CMD(is_capture, pipe);++ spin_lock_irqsave(&chip->msg_lock, flags);+ lx_message_init(&chip->rmh, CMD_0B_TOGGLE_PIPE_STATE);++ chip->rmh.cmd[0] |= pipe_cmd;++ err = lx_message_send_atomic(chip, &chip->rmh);++ spin_unlock_irqrestore(&chip->msg_lock, flags);+ return err;+}+++int lx_pipe_start(struct lx6464es *chip, u32 pipe, int is_capture)+{+ int err;++ err = lx_pipe_wait_for_idle(chip, pipe, is_capture);+ if (err < 0)+ return err;++ err = lx_pipe_toggle_state(chip, pipe, is_capture);++ return err;+}++int lx_pipe_pause(struct lx6464es *chip, u32 pipe, int is_capture)+{+ int err = 0;++ err = lx_pipe_wait_for_start(chip, pipe, is_capture);+ if (err < 0)+ return err;++ err = lx_pipe_toggle_state(chip, pipe, is_capture);++ return err;+}+++int lx_pipe_sample_count(struct lx6464es *chip, u32 pipe, int is_capture,+ u64 *rsample_count)+{+ int err;+ unsigned long flags;++ u32 pipe_cmd = PIPE_INFO_TO_CMD(is_capture, pipe);++ spin_lock_irqsave(&chip->msg_lock, flags);+ lx_message_init(&chip->rmh, CMD_0A_GET_PIPE_SPL_COUNT);++ chip->rmh.cmd[0] |= pipe_cmd;+ chip->rmh.stat_len = 2; /* need all words here! */++ err = lx_message_send_atomic(chip, &chip->rmh); /* don't sleep! */++ if (err != SUCCESS)+ snd_printk(KERN_ERR+ "lx6464es: could not query pipe's sample count\n");+ else {+ *rsample_count = ((u64)(chip->rmh.stat[0] & MASK_SPL_COUNT_HI)+ << 24) /* hi part */+ + chip->rmh.stat[1]; /* lo part */+ }++ spin_unlock_irqrestore(&chip->msg_lock, flags);+ return err;+}++int lx_pipe_state(struct lx6464es *chip, u32 pipe, int is_capture, u16 *rstate)+{+ int err;+ unsigned long flags;++ u32 pipe_cmd = PIPE_INFO_TO_CMD(is_capture, pipe);++ spin_lock_irqsave(&chip->msg_lock, flags);+ lx_message_init(&chip->rmh, CMD_0A_GET_PIPE_SPL_COUNT);++ chip->rmh.cmd[0] |= pipe_cmd;++ err = lx_message_send_atomic(chip, &chip->rmh);++ if (err != SUCCESS)+ snd_printk(KERN_ERR "lx6464es: could not query pipe's state\n");+ else+ *rstate = (chip->rmh.stat[0] >> PSTATE_OFFSET) & 0x0F;++ spin_unlock_irqrestore(&chip->msg_lock, flags);+ return err;+}++static int lx_pipe_wait_for_state(struct lx6464es *chip, u32 pipe,+ int is_capture, u16 state)+{+ int i;++ /* max 2*PCMOnlyGranularity = 2*1024 at 44100 = < 50 ms:+ * timeout 50 ms */+ for (i = 0; i != 50; ++i) {+ u16 current_state;+ int err = lx_pipe_state(chip, pipe, is_capture, ¤t_state);++ if (err < 0)+ return err;++ if (current_state == state)+ return 0;++ mdelay(1);+ }++ return -ETIMEDOUT;+}++int lx_pipe_wait_for_start(struct lx6464es *chip, u32 pipe, int is_capture)+{+ return lx_pipe_wait_for_state(chip, pipe, is_capture, PSTATE_RUN);+}++int lx_pipe_wait_for_idle(struct lx6464es *chip, u32 pipe, int is_capture)+{+ return lx_pipe_wait_for_state(chip, pipe, is_capture, PSTATE_IDLE);+}++/* low-level stream handling */+int lx_stream_set_state(struct lx6464es *chip, u32 pipe,+ int is_capture, stream_state_t state)+{+ int err;+ unsigned long flags;++ u32 pipe_cmd = PIPE_INFO_TO_CMD(is_capture, pipe);++ spin_lock_irqsave(&chip->msg_lock, flags);+ lx_message_init(&chip->rmh, CMD_13_SET_STREAM_STATE);++ chip->rmh.cmd[0] |= pipe_cmd;+ chip->rmh.cmd[0] |= state;++ err = lx_message_send_atomic(chip, &chip->rmh);+ spin_unlock_irqrestore(&chip->msg_lock, flags);++ return err;+}++int lx_stream_set_format(struct lx6464es *chip, struct snd_pcm_runtime *runtime,+ u32 pipe, int is_capture)+{+ int err;+ unsigned long flags;++ u32 pipe_cmd = PIPE_INFO_TO_CMD(is_capture, pipe);++ u32 channels = runtime->channels;++ if (runtime->channels != channels)+ snd_printk(KERN_ERR LXP "channel count mismatch: %d vs %d",+ runtime->channels, channels);++ spin_lock_irqsave(&chip->msg_lock, flags);+ lx_message_init(&chip->rmh, CMD_0C_DEF_STREAM);++ chip->rmh.cmd[0] |= pipe_cmd;++ if (runtime->sample_bits == 16)+ /* 16 bit format */+ chip->rmh.cmd[0] |= (STREAM_FMT_16b << STREAM_FMT_OFFSET);++ if (snd_pcm_format_little_endian(runtime->format))+ /* little endian/intel format */+ chip->rmh.cmd[0] |= (STREAM_FMT_intel << STREAM_FMT_OFFSET);++ chip->rmh.cmd[0] |= channels-1;++ err = lx_message_send_atomic(chip, &chip->rmh);+ spin_unlock_irqrestore(&chip->msg_lock, flags);++ return err;+}++int lx_stream_state(struct lx6464es *chip, u32 pipe, int is_capture,+ int *rstate)+{+ int err;+ unsigned long flags;++ u32 pipe_cmd = PIPE_INFO_TO_CMD(is_capture, pipe);++ spin_lock_irqsave(&chip->msg_lock, flags);+ lx_message_init(&chip->rmh, CMD_0E_GET_STREAM_SPL_COUNT);++ chip->rmh.cmd[0] |= pipe_cmd;++ err = lx_message_send_atomic(chip, &chip->rmh);++ *rstate = (chip->rmh.stat[0] & SF_START) ? START_STATE : PAUSE_STATE;++ spin_unlock_irqrestore(&chip->msg_lock, flags);+ return err;+}++int lx_stream_sample_position(struct lx6464es *chip, u32 pipe, int is_capture,+ u64 *r_bytepos)+{+ int err;+ unsigned long flags;++ u32 pipe_cmd = PIPE_INFO_TO_CMD(is_capture, pipe);++ spin_lock_irqsave(&chip->msg_lock, flags);+ lx_message_init(&chip->rmh, CMD_0E_GET_STREAM_SPL_COUNT);++ chip->rmh.cmd[0] |= pipe_cmd;++ err = lx_message_send_atomic(chip, &chip->rmh);++ *r_bytepos = ((u64) (chip->rmh.stat[0] & MASK_SPL_COUNT_HI)+ << 32) /* hi part */+ + chip->rmh.stat[1]; /* lo part */++ spin_unlock_irqrestore(&chip->msg_lock, flags);+ return err;+}++/* low-level buffer handling */+int lx_buffer_give(struct lx6464es *chip, u32 pipe, int is_capture,+ u32 buffer_size, u32 buf_address_lo, u32 buf_address_hi,+ u32 *r_buffer_index)+{+ int err;+ unsigned long flags;++ u32 pipe_cmd = PIPE_INFO_TO_CMD(is_capture, pipe);++ spin_lock_irqsave(&chip->msg_lock, flags);+ lx_message_init(&chip->rmh, CMD_0F_UPDATE_BUFFER);++ chip->rmh.cmd[0] |= pipe_cmd;+ chip->rmh.cmd[0] |= BF_NOTIFY_EOB; /* request interrupt notification */++ /* todo: pause request, circular buffer */++ chip->rmh.cmd[1] = buffer_size & MASK_DATA_SIZE;+ chip->rmh.cmd[2] = buf_address_lo;++ if (buf_address_hi) {+ chip->rmh.cmd_len = 4;+ chip->rmh.cmd[3] = buf_address_hi;+ chip->rmh.cmd[0] |= BF_64BITS_ADR;+ }++ err = lx_message_send_atomic(chip, &chip->rmh);++ if (err == SUCCESS) {+ *r_buffer_index = chip->rmh.stat[0];+ goto done;+ }++ if (err == EB_RBUFFERS_TABLE_OVERFLOW)+ snd_printk(LXP "lx_buffer_give EB_RBUFFERS_TABLE_OVERFLOW\n");++ if (err == EB_INVALID_STREAM)+ snd_printk(LXP "lx_buffer_give EB_INVALID_STREAM\n");++ if (err == EB_CMD_REFUSED)+ snd_printk(LXP "lx_buffer_give EB_CMD_REFUSED\n");++ done:+ spin_unlock_irqrestore(&chip->msg_lock, flags);+ return err;+}++int lx_buffer_free(struct lx6464es *chip, u32 pipe, int is_capture,+ u32 *r_buffer_size)+{+ int err;+ unsigned long flags;++ u32 pipe_cmd = PIPE_INFO_TO_CMD(is_capture, pipe);++ spin_lock_irqsave(&chip->msg_lock, flags);+ lx_message_init(&chip->rmh, CMD_11_CANCEL_BUFFER);++ chip->rmh.cmd[0] |= pipe_cmd;+ chip->rmh.cmd[0] |= MASK_BUFFER_ID; /* ask for the current buffer: the+ * microblaze will seek for it */++ err = lx_message_send_atomic(chip, &chip->rmh);++ if (err == SUCCESS)+ *r_buffer_size = chip->rmh.stat[0] & MASK_DATA_SIZE;++ spin_unlock_irqrestore(&chip->msg_lock, flags);+ return err;+}++int lx_buffer_cancel(struct lx6464es *chip, u32 pipe, int is_capture,+ u32 buffer_index)+{+ int err;+ unsigned long flags;++ u32 pipe_cmd = PIPE_INFO_TO_CMD(is_capture, pipe);++ spin_lock_irqsave(&chip->msg_lock, flags);+ lx_message_init(&chip->rmh, CMD_11_CANCEL_BUFFER);++ chip->rmh.cmd[0] |= pipe_cmd;+ chip->rmh.cmd[0] |= buffer_index;++ err = lx_message_send_atomic(chip, &chip->rmh);++ spin_unlock_irqrestore(&chip->msg_lock, flags);+ return err;+}+++/* low-level gain/peak handling+ *+ * \todo: can we unmute capture/playback channels independently?+ *+ * */+int lx_level_unmute(struct lx6464es *chip, int is_capture, int unmute)+{+ int err;+ unsigned long flags;++ /* bit set to 1: channel muted */+ u64 mute_mask = unmute ? 0 : 0xFFFFFFFFFFFFFFFFLLU;++ spin_lock_irqsave(&chip->msg_lock, flags);+ lx_message_init(&chip->rmh, CMD_0D_SET_MUTE);++ chip->rmh.cmd[0] |= PIPE_INFO_TO_CMD(is_capture, 0);++ chip->rmh.cmd[1] = (u32)(mute_mask >> (u64)32); /* hi part */+ chip->rmh.cmd[2] = (u32)(mute_mask & (u64)0xFFFFFFFF); /* lo part */++ snd_printk("mute %x %x %x\n", chip->rmh.cmd[0], chip->rmh.cmd[1],+ chip->rmh.cmd[2]);++ err = lx_message_send_atomic(chip, &chip->rmh);++ spin_unlock_irqrestore(&chip->msg_lock, flags);+ return err;+}++static u32 peak_map[] = {+ 0x00000109, /* -90.308dB */+ 0x0000083B, /* -72.247dB */+ 0x000020C4, /* -60.205dB */+ 0x00008273, /* -48.030dB */+ 0x00020756, /* -36.005dB */+ 0x00040C37, /* -30.001dB */+ 0x00081385, /* -24.002dB */+ 0x00101D3F, /* -18.000dB */+ 0x0016C310, /* -15.000dB */+ 0x002026F2, /* -12.001dB */+ 0x002D6A86, /* -9.000dB */+ 0x004026E6, /* -6.004dB */+ 0x005A9DF6, /* -3.000dB */+ 0x0065AC8B, /* -2.000dB */+ 0x00721481, /* -1.000dB */+ 0x007FFFFF, /* FS */+};++int lx_level_peaks(struct lx6464es *chip, int is_capture, int channels,+ u32 *r_levels)+{+ int err = 0;+ unsigned long flags;+ int i;+ spin_lock_irqsave(&chip->msg_lock, flags);++ for (i = 0; i < channels; i += 4) {+ u32 s0, s1, s2, s3;++ lx_message_init(&chip->rmh, CMD_12_GET_PEAK);+ chip->rmh.cmd[0] |= PIPE_INFO_TO_CMD(is_capture, i);++ err = lx_message_send_atomic(chip, &chip->rmh);++ if (err == 0) {+ s0 = peak_map[chip->rmh.stat[0] & 0x0F];+ s1 = peak_map[(chip->rmh.stat[0] >> 4) & 0xf];+ s2 = peak_map[(chip->rmh.stat[0] >> 8) & 0xf];+ s3 = peak_map[(chip->rmh.stat[0] >> 12) & 0xf];+ } else+ s0 = s1 = s2 = s3 = 0;++ r_levels[0] = s0;+ r_levels[1] = s1;+ r_levels[2] = s2;+ r_levels[3] = s3;++ r_levels += 4;+ }++ spin_unlock_irqrestore(&chip->msg_lock, flags);+ return err;+}++/* later: the card also supports hardware peak metering */+++/* interrupt handling */+#define PCX_IRQ_NONE 0+#define IRQCS_ACTIVE_PCIDB 0x00002000L /* Bit nø 13 */+#define IRQCS_ENABLE_PCIIRQ 0x00000100L /* Bit nø 08 */+#define IRQCS_ENABLE_PCIDB 0x00000200L /* Bit nø 09 */++/* -- mask definitions for SYS.STAT ----- */+#define MASK_SYS_STATUS_ERROR (1L << 31) /* events that lead to a PCI irq if not yet pending */+#define MASK_SYS_STATUS_URUN (1L << 30)+#define MASK_SYS_STATUS_ORUN (1L << 29)+#define MASK_SYS_STATUS_EOBO (1L << 28)+#define MASK_SYS_STATUS_EOBI (1L << 27)+#define MASK_SYS_STATUS_FREQ (1L << 26)+#define MASK_SYS_STATUS_ESA (1L << 25) /* reserved, this is set by the XES */+#define MASK_SYS_STATUS_TIMER (1L << 24)++#define MASK_SYS_ASYNC_EVENTS (MASK_SYS_STATUS_ERROR | MASK_SYS_STATUS_URUN | \+ MASK_SYS_STATUS_ORUN | MASK_SYS_STATUS_EOBO | \+ MASK_SYS_STATUS_EOBI | MASK_SYS_STATUS_FREQ | \+ MASK_SYS_STATUS_ESA)++#define MASK_SYS_PCI_EVENTS (MASK_SYS_ASYNC_EVENTS | MASK_SYS_STATUS_TIMER)++#define MASK_SYS_TIMER_COUNT 0x0000FFFF++/* #define MASK_SYS_STATUS_CMD (1L << 23) /\* event that remains internal *\/ */+#define MASK_SYS_STATUS_EOT_PLX (1L << 22) /* event that remains internal : reserved fo end of plx dma */+#define MASK_SYS_STATUS_XES (1L << 21) /* event that remains internal : pending XES IRQ */+#define MASK_SYS_STATUS_CMD_DONE (1L << 20) /* alternate command management : notify driver instead of polling */+++static u32 lx_interrupt_test_ack(struct lx6464es *chip)+{+ u32 irqcs = lx_plx_reg_read(chip, ePLX_IRQCS);++ /* Test if PCI Doorbell interrupt is active */+ if (irqcs & IRQCS_ACTIVE_PCIDB) {+ u32 temp;+ irqcs = PCX_IRQ_NONE;++ while ((temp = lx_plx_reg_read(chip, ePLX_L2PCIDB))) {+ /* RAZ interrupt */+ irqcs |= temp;+ lx_plx_reg_write(chip, ePLX_L2PCIDB, temp);+ }++ return irqcs;+ }+ return PCX_IRQ_NONE;+}++static int lx_interrupt_ack(struct lx6464es *chip, u32 *r_irqsrc,+ int *r_async_pending, int *r_async_escmd)+{+ u32 irq_async;+ u32 irqsrc = lx_interrupt_test_ack(chip);++ if (irqsrc == PCX_IRQ_NONE)+ return 0;++ *r_irqsrc = irqsrc;++ irq_async = irqsrc & MASK_SYS_ASYNC_EVENTS; /* + EtherSound response+ * (set by xilinx) + EOB */++ if (irq_async & MASK_SYS_STATUS_ESA) {+ irq_async &= ~MASK_SYS_STATUS_ESA;+ *r_async_escmd = 1;+ }++ if (irqsrc & MASK_SYS_STATUS_CMD_DONE)+ /* xilinx command notification */+ atomic_set(&chip->send_message_locked, 0);++ if (irq_async) {+ /* snd_printd("interrupt: async event pending\n"); */+ *r_async_pending = 1;+ }++ return 1;+}++static int lx_interrupt_handle_async_events(struct lx6464es *chip, u32 irqsrc,+ int *r_freq_changed,+ u64 *r_notified_in_pipe_mask,+ u64 *r_notified_out_pipe_mask)+{+ int err;+ u32 stat[9]; /* answer from CMD_04_GET_EVENT */++ /* On peut optimiser pour ne pas lire les evenements vides+ * les mots de réponse sont dans l'ordre suivant :+ * Stat[0] mot de status général+ * Stat[1] fin de buffer OUT pF+ * Stat[2] fin de buffer OUT pf+ * Stat[3] fin de buffer IN pF+ * Stat[4] fin de buffer IN pf+ * Stat[5] underrun poid fort+ * Stat[6] underrun poid faible+ * Stat[7] overrun poid fort+ * Stat[8] overrun poid faible+ * */++ u64 orun_mask;+ u64 urun_mask;+#if 0+ int has_underrun = (irqsrc & MASK_SYS_STATUS_URUN) ? 1 : 0;+ int has_overrun = (irqsrc & MASK_SYS_STATUS_ORUN) ? 1 : 0;+#endif+ int eb_pending_out = (irqsrc & MASK_SYS_STATUS_EOBO) ? 1 : 0;+ int eb_pending_in = (irqsrc & MASK_SYS_STATUS_EOBI) ? 1 : 0;++ *r_freq_changed = (irqsrc & MASK_SYS_STATUS_FREQ) ? 1 : 0;++ err = lx_dsp_read_async_events(chip, stat);+ if (err < 0)+ return err;++ if (eb_pending_in) {+ *r_notified_in_pipe_mask = ((u64)stat[3] << 32)+ + stat[4];+ snd_printdd(LXP "interrupt: EOBI pending %llx\n",+ *r_notified_in_pipe_mask);+ }+ if (eb_pending_out) {+ *r_notified_out_pipe_mask = ((u64)stat[1] << 32)+ + stat[2];+ snd_printdd(LXP "interrupt: EOBO pending %llx\n",+ *r_notified_out_pipe_mask);+ }++ orun_mask = ((u64)stat[7] << 32) + stat[8];+ urun_mask = ((u64)stat[5] << 32) + stat[6];++ /* todo: handle xrun notification */++ return err;+}++static int lx_interrupt_request_new_buffer(struct lx6464es *chip,+ struct lx_stream *lx_stream)+{+ struct snd_pcm_substream *substream = lx_stream->stream;+ int is_capture = lx_stream->is_capture;+ int err;+ unsigned long flags;++ const u32 channels = substream->runtime->channels;+ const u32 bytes_per_frame = channels * 3;+ const u32 period_size = substream->runtime->period_size;+ const u32 period_bytes = period_size * bytes_per_frame;+ const u32 pos = lx_stream->frame_pos;+ const u32 next_pos = ((pos+1) == substream->runtime->periods) ?+ 0 : pos + 1;++ dma_addr_t buf = substream->dma_buffer.addr + pos * period_bytes;+ u32 buf_hi = 0;+ u32 buf_lo = 0;+ u32 buffer_index = 0;++ u32 needed, freed;+ u32 size_array[MAX_STREAM_BUFFER];++ snd_printdd("->lx_interrupt_request_new_buffer\n");++ spin_lock_irqsave(&chip->lock, flags);++ err = lx_buffer_ask(chip, 0, is_capture, &needed, &freed, size_array);+ snd_printdd(LXP "interrupt: needed %d, freed %d\n", needed, freed);++ unpack_pointer(buf, &buf_lo, &buf_hi);+ err = lx_buffer_give(chip, 0, is_capture, period_bytes, buf_lo, buf_hi,+ &buffer_index);+ snd_printdd(LXP "interrupt: gave buffer index %x on %p (%d bytes)\n",+ buffer_index, (void*)buf, period_bytes);++ lx_stream->frame_pos = next_pos;+ spin_unlock_irqrestore(&chip->lock, flags);++ return err;+}++void lx_tasklet_playback(unsigned long data)+{+ struct lx6464es *chip = (struct lx6464es *)data;+ struct lx_stream *lx_stream = &chip->playback_stream;+ int err;++ snd_printdd("->lx_tasklet_playback\n");++ err = lx_interrupt_request_new_buffer(chip, lx_stream);+ if (err < 0)+ snd_printk(KERN_ERR LXP+ "cannot request new buffer for playback\n");++ snd_pcm_period_elapsed(lx_stream->stream);+}++void lx_tasklet_capture(unsigned long data)+{+ struct lx6464es *chip = (struct lx6464es *)data;+ struct lx_stream *lx_stream = &chip->capture_stream;+ int err;++ snd_printdd("->lx_tasklet_capture\n");+ err = lx_interrupt_request_new_buffer(chip, lx_stream);+ if (err < 0)+ snd_printk(KERN_ERR LXP+ "cannot request new buffer for capture\n");++ snd_pcm_period_elapsed(lx_stream->stream);+}++++static int lx_interrupt_handle_audio_transfer(struct lx6464es *chip,+ u64 notified_in_pipe_mask,+ u64 notified_out_pipe_mask)+{+ int err = 0;++ if (notified_in_pipe_mask) {+ snd_printdd(LXP "requesting audio transfer for capture\n");+ tasklet_hi_schedule(&chip->tasklet_capture);+ }++ if (notified_out_pipe_mask) {+ snd_printdd(LXP "requesting audio transfer for playback\n");+ tasklet_hi_schedule(&chip->tasklet_playback);+ }++ return err;+}+++irqreturn_t lx_interrupt(int irq, void *dev_id)+{+ struct lx6464es *chip = dev_id;+ int async_pending, async_escmd;+ u32 irqsrc;++ spin_lock(&chip->lock);++ snd_printdd("**************************************************\n");++ if (!lx_interrupt_ack(chip, &irqsrc, &async_pending, &async_escmd)) {+ spin_unlock(&chip->lock);+ snd_printdd("IRQ_NONE\n");+ return IRQ_NONE; /* this device did not cause the interrupt */+ }++ if (irqsrc & MASK_SYS_STATUS_CMD_DONE)+ goto exit;++#if 0+ if (irqsrc & MASK_SYS_STATUS_EOBI)+ snd_printdd(LXP "interrupt: EOBI\n");++ if (irqsrc & MASK_SYS_STATUS_EOBO)+ snd_printdd(LXP "interrupt: EOBO\n");++ if (irqsrc & MASK_SYS_STATUS_URUN)+ snd_printdd(LXP "interrupt: URUN\n");++ if (irqsrc & MASK_SYS_STATUS_ORUN)+ snd_printdd(LXP "interrupt: ORUN\n");+#endif++ if (async_pending) {+ u64 notified_in_pipe_mask = 0;+ u64 notified_out_pipe_mask = 0;+ int freq_changed;+ int err;++ /* handle async events */+ err = lx_interrupt_handle_async_events(chip, irqsrc,+ &freq_changed,+ ¬ified_in_pipe_mask,+ ¬ified_out_pipe_mask);+ if (err)+ snd_printk(KERN_ERR LXP+ "error handling async events\n");++ err = lx_interrupt_handle_audio_transfer(chip,+ notified_in_pipe_mask,+ notified_out_pipe_mask+ );+ if (err)+ snd_printk(KERN_ERR LXP+ "error during audio transfer\n");+ }++ if (async_escmd) {+#if 0+ /* backdoor for ethersound commands+ *+ * for now, we do not need this+ *+ * */++ snd_printdd("lx6464es: interrupt requests escmd handling\n");+#endif+ }++exit:+ spin_unlock(&chip->lock);+ return IRQ_HANDLED; /* this device caused the interrupt */+}+++static void lx_irq_set(struct lx6464es *chip, int enable)+{+ u32 reg = lx_plx_reg_read(chip, ePLX_IRQCS);++ /* enable/disable interrupts+ *+ * Set the Doorbell and PCI interrupt enable bits+ *+ * */+ if (enable)+ reg |= (IRQCS_ENABLE_PCIIRQ | IRQCS_ENABLE_PCIDB);+ else+ reg &= ~(IRQCS_ENABLE_PCIIRQ | IRQCS_ENABLE_PCIDB);+ lx_plx_reg_write(chip, ePLX_IRQCS, reg);+}++void lx_irq_enable(struct lx6464es *chip)+{+ snd_printdd("->lx_irq_enable\n");+ lx_irq_set(chip, 1);+}++void lx_irq_disable(struct lx6464es *chip)+{+ snd_printdd("->lx_irq_disable\n");+ lx_irq_set(chip, 0);+}diff --git a/sound/pci/lx6464es/lx_core.h b/sound/pci/lx6464es/lx_core.hnew file mode 100644index 0000000..901d443--- /dev/null+++ b/sound/pci/lx6464es/lx_core.h@@ -0,0 +1,239 @@+/* -*- linux-c -*- *+ *+ * ALSA driver for the digigram lx6464es interface+ * low-level interface+ *+ * Copyright (c) 2009 Tim Blechmann <tim@xxxxxxxxxx>+ *+ * 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 2 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; see the file COPYING. If not, write to+ * the Free Software Foundation, Inc., 59 Temple Place - Suite 330,+ * Boston, MA 02111-1307, USA.+ *+ */++#ifndef LX_CORE_H+#define LX_CORE_H++#include <linux/interrupt.h>++#include "PcxErr_e.h"+#include "if_drv_mb.h"++#define REG_CRM_NUMBER 12++struct lx6464es;++/* low-level register access */++/* dsp register access */+enum {+ eReg_BASE,+ eReg_CSM,+ eReg_CRM1,+ eReg_CRM2,+ eReg_CRM3,+ eReg_CRM4,+ eReg_CRM5,+ eReg_CRM6,+ eReg_CRM7,+ eReg_CRM8,+ eReg_CRM9,+ eReg_CRM10,+ eReg_CRM11,+ eReg_CRM12,++ eReg_ICR,+ eReg_CVR,+ eReg_ISR,+ eReg_RXHTXH,+ eReg_RXMTXM,+ eReg_RHLTXL,+ eReg_RESETDSP,++ eReg_CSUF,+ eReg_CSES,+ eReg_CRESMSB,+ eReg_CRESLSB,+ eReg_ADMACESMSB,+ eReg_ADMACESLSB,+ eReg_CONFES,++ eMaxPortLx+};++unsigned long lx_dsp_reg_read(struct lx6464es *chip, int port);+void lx_dsp_reg_readbuf(struct lx6464es *chip, int port, u32 *data, u32 len);+void lx_dsp_reg_write(struct lx6464es *chip, int port, unsigned data);+void lx_dsp_reg_writebuf(struct lx6464es *chip, int port, const u32 *data,+ u32 len);++/* plx register access */+enum {+ ePLX_PCICR,++ ePLX_MBOX0,+ ePLX_MBOX1,+ ePLX_MBOX2,+ ePLX_MBOX3,+ ePLX_MBOX4,+ ePLX_MBOX5,+ ePLX_MBOX6,+ ePLX_MBOX7,++ ePLX_L2PCIDB,+ ePLX_IRQCS,+ ePLX_CHIPSC,++ eMaxPort+};++unsigned long lx_plx_reg_read(struct lx6464es *chip, int port);+void lx_plx_reg_write(struct lx6464es *chip, int port, u32 data);++/* rhm */+struct lx_rmh {+ u16 cmd_len; /* length of the command to send (WORDs) */+ u16 stat_len; /* length of the status received (WORDs) */+ u16 dsp_stat; /* status type, RMP_SSIZE_XXX */+ u16 cmd_idx; /* index of the command */+ u32 cmd[REG_CRM_NUMBER];+ u32 stat[REG_CRM_NUMBER];+};+++/* low-level dsp access */+int __devinit lx_dsp_get_version(struct lx6464es *chip, u32 *rdsp_version);+int lx_dsp_get_clock_frequency(struct lx6464es *chip, u32 *rfreq);+int lx_dsp_set_granularity(struct lx6464es *chip, u32 gran);+int lx_dsp_read_async_events(struct lx6464es *chip, u32 *data);+int lx_dsp_get_mac(struct lx6464es *chip, u8 *mac_address);+++/* low-level pipe handling */+int lx_pipe_allocate(struct lx6464es *chip, u32 pipe, int is_capture,+ int channels);+int lx_pipe_release(struct lx6464es *chip, u32 pipe, int is_capture);+int lx_pipe_sample_count(struct lx6464es *chip, u32 pipe, int is_capture,+ u64 *rsample_count);+int lx_pipe_state(struct lx6464es *chip, u32 pipe, int is_capture, u16 *rstate);+int lx_pipe_stop(struct lx6464es *chip, u32 pipe, int is_capture);+int lx_pipe_start(struct lx6464es *chip, u32 pipe, int is_capture);+int lx_pipe_pause(struct lx6464es *chip, u32 pipe, int is_capture);++int lx_pipe_wait_for_start(struct lx6464es *chip, u32 pipe, int is_capture);+int lx_pipe_wait_for_idle(struct lx6464es *chip, u32 pipe, int is_capture);++/* low-level stream handling */+int lx_stream_set_format(struct lx6464es *chip, struct snd_pcm_runtime *runtime,+ u32 pipe, int is_capture);+int lx_stream_state(struct lx6464es *chip, u32 pipe, int is_capture,+ int *rstate);+int lx_stream_sample_position(struct lx6464es *chip, u32 pipe, int is_capture,+ u64 *r_bytepos);++int lx_stream_set_state(struct lx6464es *chip, u32 pipe,+ int is_capture, stream_state_t state);++static inline int lx_stream_start(struct lx6464es *chip, u32 pipe,+ int is_capture)+{+ snd_printdd("->lx_stream_start\n");+ return lx_stream_set_state(chip, pipe, is_capture, SSTATE_RUN);+}++static inline int lx_stream_pause(struct lx6464es *chip, u32 pipe,+ int is_capture)+{+ snd_printdd("->lx_stream_pause\n");+ return lx_stream_set_state(chip, pipe, is_capture, SSTATE_PAUSE);+}++static inline int lx_stream_stop(struct lx6464es *chip, u32 pipe,+ int is_capture)+{+ snd_printdd("->lx_stream_stop\n");+ return lx_stream_set_state(chip, pipe, is_capture, SSTATE_STOP);+}++/* low-level buffer handling */+int lx_buffer_ask(struct lx6464es *chip, u32 pipe, int is_capture,+ u32 *r_needed, u32 *r_freed, u32 *size_array);+int lx_buffer_give(struct lx6464es *chip, u32 pipe, int is_capture,+ u32 buffer_size, u32 buf_address_lo, u32 buf_address_hi,+ u32 *r_buffer_index);+int lx_buffer_free(struct lx6464es *chip, u32 pipe, int is_capture,+ u32 *r_buffer_size);+int lx_buffer_cancel(struct lx6464es *chip, u32 pipe, int is_capture,+ u32 buffer_index);++/* low-level gain/peak handling */+int lx_level_unmute(struct lx6464es *chip, int is_capture, int unmute);+int lx_level_peaks(struct lx6464es *chip, int is_capture, int channels,+ u32 *r_levels);+++/* interrupt handling */+irqreturn_t lx_interrupt(int irq, void *dev_id);+void lx_irq_enable(struct lx6464es *chip);+void lx_irq_disable(struct lx6464es *chip);++void lx_tasklet_capture(unsigned long data);+void lx_tasklet_playback(unsigned long data);+++/* Stream Format Header Defines (for LIN and IEEE754) */+#define HEADER_FMT_BASE HEADER_FMT_BASE_LIN+#define HEADER_FMT_BASE_LIN 0xFED00000+#define HEADER_FMT_BASE_FLOAT 0xFAD00000+#define HEADER_FMT_MONO 0x00000080 /**< bit 23 in header_lo. WARNING : old bit 22 is ignored in float format*/+#define HEADER_FMT_INTEL 0x00008000+#define HEADER_FMT_16BITS 0x00002000+#define HEADER_FMT_24BITS 0x00004000+#define HEADER_FMT_UPTO11 0x00000200 /**< frequency is less or equ. to 11k.*/+#define HEADER_FMT_UPTO32 0x00000100 /**< frequency is over 11k and less then 32k.*/+++#define BIT_FMP_HEADER 23+#define BIT_FMP_SD 22+#define BIT_FMP_MULTICHANNEL 19++#define START_STATE 1+#define PAUSE_STATE 0++++++/* from PcxAll_e.h */+/* Start/Pause condition for pipes (PCXStartPipe, PCXPausePipe) */+#define START_PAUSE_IMMEDIATE 0+#define START_PAUSE_ON_SYNCHRO 1+#define START_PAUSE_ON_TIME_CODE 2+++/* Pipe / Stream state */+#define START_STATE 1+#define PAUSE_STATE 0++static inline void unpack_pointer(dma_addr_t ptr, u32 *r_low, u32 *r_high)+{+ *r_low = (u32)(ptr & 0xffffffff);+#if BITS_PER_LONG == 32+ *r_high = 0;+#else+ *r_high = (u32)((u64)ptr>>32);+#endif+}++#endif /* LX_CORE_H */-- 1.6.2.1_______________________________________________Alsa-devel mailing listAlsa-devel@xxxxxxxxxxxxxxxxxxxx://mailman.alsa-project.org/mailman/listinfo/alsa-devel