[PATCH] alsa: lx6464es - driver for the digigram lx6464es interface

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



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

[Index of Archives]     [ALSA User]     [Linux Audio Users]     [Pulse Audio]     [Kernel Archive]     [Asterisk PBX]     [Photo Sharing]     [Linux Sound]     [Video 4 Linux]     [Gimp]     [Yosemite News]

  Powered by Linux