[alsa-devel] [PATCH] alsa: lx6464es - driver for the digigram lx6464es interface
Tim Blechmann
tim at klingt.org
Mon Mar 23 11:12:22 CET 2009
prototype of a driver for the digigram lx6464es 64 channel ethersound
interface.
Signed-off-by: Tim Blechmann <tim at klingt.org>
---
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.h
index 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/Kconfig
index 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_CODEC
diff --git a/sound/pci/Makefile b/sound/pci/Makefile
index 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.h
new file mode 100644
index 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/Makefile
new file mode 100644
index 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
+endif
diff --git a/sound/pci/lx6464es/PcxErr_e.h b/sound/pci/lx6464es/PcxErr_e.h
new file mode 100644
index 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.h
new file mode 100644
index 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).*/
+
+
+#endif
diff --git a/sound/pci/lx6464es/if_drv_mb.h b/sound/pci/lx6464es/if_drv_mb.h
new file mode 100644
index 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.*/
+
+#endif
diff --git a/sound/pci/lx6464es/lx6464es.c b/sound/pci/lx6464es/lx6464es.c
new file mode 100644
index 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 at klingt.org>
+ *
+ *
+ * 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.h
new file mode 100644
index 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 at klingt.org>
+ *
+ *
+ * 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.c
new file mode 100644
index 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 at klingt.org>
+ *
+ * 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.h
new file mode 100644
index 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 at klingt.org>
+ *
+ * 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
More information about the Alsa-devel
mailing list