diff options
Diffstat (limited to 'packages/linux')
56 files changed, 66509 insertions, 47 deletions
diff --git a/packages/linux/ep93xx-kernel/dynamic-phys-offset-2.6.20-rc7.diff b/packages/linux/ep93xx-kernel/dynamic-phys-offset-2.6.20-rc7.diff new file mode 100644 index 0000000000..1f59be644e --- /dev/null +++ b/packages/linux/ep93xx-kernel/dynamic-phys-offset-2.6.20-rc7.diff @@ -0,0 +1,199 @@ +On the Cirrus Logic ep93xx, system RAM isn't one nice physically +contiguous region as it is on most SoCs, but it is spread out over +between one and four memory banks. + +What's worse, RAM doesn't necessarily start at any fixed physical +memory location. The start of RAM (PHYS_OFFSET) is not only board- +specific, but on some boards also depends on jumper settings (whether +async or sync boot mode is selected.) + +The attached patch adds the RUNTIME_PHYS_OFFSET config option, which, +if selected, turns PHYS_OFFSET into a variable which is determined and +set by __create_page_tables by looking at the current pc. This allows +booting a single kernel image on all the different flavors of ep93xx +boards, reducing user confusion and hopefully pleasing our kautobuild +admin :-) If the option isn't selected, there's zero impact. + +Signed-off-by: Lennert Buytenhek <buytenh@wantstofly.org> +=================================================================== +20070202 modified to apply cleanly to linux-2.6.20-rc7 - NZG +=================================================================== +diff -Naur linux-2.6.20-rc7/arch/arm/Kconfig linux-2.6.20-rc7-e1.0/arch/arm/Kconfig +--- linux-2.6.20-rc7/arch/arm/Kconfig 2007-02-02 10:26:21.000000000 -0600 ++++ linux-2.6.20-rc7-e1.0/arch/arm/Kconfig 2007-02-02 16:23:05.000000000 -0600 +@@ -113,6 +113,9 @@ + help + The base address of exception vectors. + ++config RUNTIME_PHYS_OFFSET ++ bool ++ + source "init/Kconfig" + + menu "System Type" +@@ -187,6 +190,7 @@ + bool "EP93xx-based" + select ARM_AMBA + select ARM_VIC ++ select RUNTIME_PHYS_OFFSET + help + This enables support for the Cirrus EP93xx series of CPUs. + +diff -Naur linux-2.6.20-rc7/arch/arm/boot/compressed/Makefile linux-2.6.20-rc7-e1.0/arch/arm/boot/compressed/Makefile +--- linux-2.6.20-rc7/arch/arm/boot/compressed/Makefile 2006-11-29 15:57:37.000000000 -0600 ++++ linux-2.6.20-rc7-e1.0/arch/arm/boot/compressed/Makefile 2007-02-02 16:23:05.000000000 -0600 +@@ -78,13 +78,10 @@ + EXTRA_CFLAGS := -fpic + EXTRA_AFLAGS := + +-# Supply ZRELADDR, INITRD_PHYS and PARAMS_PHYS to the decompressor via +-# linker symbols. We only define initrd_phys and params_phys if the +-# machine class defined the corresponding makefile variable. ++# Supply ZRELADDR and PARAMS_PHYS to the decompressor via linker ++# symbols. We only define params_phys if the machine class defined ++# the corresponding makefile variable. + LDFLAGS_vmlinux := --defsym zreladdr=$(ZRELADDR) +-ifneq ($(INITRD_PHYS),) +-LDFLAGS_vmlinux += --defsym initrd_phys=$(INITRD_PHYS) +-endif + ifneq ($(PARAMS_PHYS),) + LDFLAGS_vmlinux += --defsym params_phys=$(PARAMS_PHYS) + endif +diff -Naur linux-2.6.20-rc7/arch/arm/boot/compressed/head.S linux-2.6.20-rc7-e1.0/arch/arm/boot/compressed/head.S +--- linux-2.6.20-rc7/arch/arm/boot/compressed/head.S 2006-11-29 15:57:37.000000000 -0600 ++++ linux-2.6.20-rc7-e1.0/arch/arm/boot/compressed/head.S 2007-02-02 16:23:05.000000000 -0600 +@@ -156,6 +156,11 @@ + .text + adr r0, LC0 + ldmia r0, {r1, r2, r3, r4, r5, r6, ip, sp} ++#ifdef CONFIG_RUNTIME_PHYS_OFFSET ++ and r10, pc, #0xf0000000 @ fix up zreladdr ++ add r4, r4, r10 ++#endif ++ + subs r0, r0, r1 @ calculate the delta offset + + @ if delta is zero, we are +diff -Naur linux-2.6.20-rc7/arch/arm/kernel/head.S linux-2.6.20-rc7-e1.0/arch/arm/kernel/head.S +--- linux-2.6.20-rc7/arch/arm/kernel/head.S 2007-02-02 10:26:21.000000000 -0600 ++++ linux-2.6.20-rc7-e1.0/arch/arm/kernel/head.S 2007-02-02 16:36:21.000000000 -0600 +@@ -43,8 +43,8 @@ + .globl swapper_pg_dir + .equ swapper_pg_dir, KERNEL_RAM_VADDR - 0x4000 + +- .macro pgtbl, rd +- ldr \rd, =(KERNEL_RAM_PADDR - 0x4000) ++ .macro pgtbl, rd, phys_offset ++ add \rd, \phys_offset, #(TEXT_OFFSET - 0x4000) + .endm + + #ifdef CONFIG_XIP_KERNEL +@@ -206,10 +206,22 @@ + * Returns: + * r0, r3, r6, r7 corrupted + * r4 = physical page table address ++ * r5 = PHYS_OFFSET + */ + .type __create_page_tables, %function + __create_page_tables: +- pgtbl r4 @ page table address ++#ifdef CONFIG_RUNTIME_PHYS_OFFSET ++ adr r5, stext ++ sub r5, r5, #TEXT_OFFSET @ r5 = phys_offset ++ ++ ldr r4, =(phys_offset - PAGE_OFFSET) ++ add r4, r4, r5 ++ str r5, [r4] @ save phys_offset ++#else ++ mov r5, #PHYS_OFFSET @ r5 = phys_offset ++#endif ++ ++ pgtbl r4, r5 @ r4 = page table address + + /* + * Clear the 16K level 1 swapper page table +@@ -255,8 +267,7 @@ + * Then map first 1MB of ram in case it contains our boot params. + */ + add r0, r4, #PAGE_OFFSET >> 18 +- orr r6, r7, #(PHYS_OFFSET & 0xff000000) +- orr r6, r6, #(PHYS_OFFSET & 0x00e00000) ++ orr r6, r7, r5 + str r6, [r0] + + #ifdef CONFIG_XIP_KERNEL +diff -Naur linux-2.6.20-rc7/arch/arm/kernel/setup.c linux-2.6.20-rc7-e1.0/arch/arm/kernel/setup.c +--- linux-2.6.20-rc7/arch/arm/kernel/setup.c 2007-02-02 10:26:21.000000000 -0600 ++++ linux-2.6.20-rc7-e1.0/arch/arm/kernel/setup.c 2007-02-02 16:23:05.000000000 -0600 +@@ -59,6 +59,16 @@ + extern int root_mountflags; + extern void _stext, _text, _etext, __data_start, _edata, _end; + ++#ifdef CONFIG_RUNTIME_PHYS_OFFSET ++/* ++ * The assignment is here solely to prevent this variable from ending ++ * up in bss. As the early startup code writes to it, we don't want it ++ * to be zeroed again later. ++ */ ++unsigned long phys_offset = 0xdeadbeef; ++EXPORT_SYMBOL(phys_offset); ++#endif ++ + unsigned int processor_id; + unsigned int __machine_arch_type; + EXPORT_SYMBOL(__machine_arch_type); +@@ -749,7 +759,7 @@ + { tag_size(tag_core), ATAG_CORE }, + { 1, PAGE_SIZE, 0xff }, + { tag_size(tag_mem32), ATAG_MEM }, +- { MEM_SIZE, PHYS_OFFSET }, ++ { MEM_SIZE, 0 }, + { 0, ATAG_NONE } + }; + +@@ -770,6 +780,8 @@ + struct machine_desc *mdesc; + char *from = default_command_line; + ++ init_tags.mem.start = PHYS_OFFSET; ++ + setup_processor(); + mdesc = setup_machine(machine_arch_type); + machine_name = mdesc->name; +diff -Naur linux-2.6.20-rc7/arch/arm/mach-ep93xx/Makefile.boot linux-2.6.20-rc7-e1.0/arch/arm/mach-ep93xx/Makefile.boot +--- linux-2.6.20-rc7/arch/arm/mach-ep93xx/Makefile.boot 2006-11-29 15:57:37.000000000 -0600 ++++ linux-2.6.20-rc7-e1.0/arch/arm/mach-ep93xx/Makefile.boot 2007-02-02 16:23:05.000000000 -0600 +@@ -1,2 +1 @@ + zreladdr-y := 0x00008000 +-params_phys-y := 0x00000100 +diff -Naur linux-2.6.20-rc7/include/asm-arm/arch-ep93xx/memory.h linux-2.6.20-rc7-e1.0/include/asm-arm/arch-ep93xx/memory.h +--- linux-2.6.20-rc7/include/asm-arm/arch-ep93xx/memory.h 2006-11-29 15:57:37.000000000 -0600 ++++ linux-2.6.20-rc7-e1.0/include/asm-arm/arch-ep93xx/memory.h 2007-02-02 16:23:05.000000000 -0600 +@@ -5,7 +5,9 @@ + #ifndef __ASM_ARCH_MEMORY_H + #define __ASM_ARCH_MEMORY_H + ++#ifndef CONFIG_RUNTIME_PHYS_OFFSET + #define PHYS_OFFSET UL(0x00000000) ++#endif + + #define __bus_to_virt(x) __phys_to_virt(x) + #define __virt_to_bus(x) __virt_to_phys(x) +diff -Naur linux-2.6.20-rc7/include/asm-arm/memory.h linux-2.6.20-rc7-e1.0/include/asm-arm/memory.h +--- linux-2.6.20-rc7/include/asm-arm/memory.h 2007-02-02 10:26:27.000000000 -0600 ++++ linux-2.6.20-rc7-e1.0/include/asm-arm/memory.h 2007-02-02 16:23:05.000000000 -0600 +@@ -73,6 +73,14 @@ + */ + #define IOREMAP_MAX_ORDER 24 + ++/* ++ * PHYS_OFFSET determined at run time? ++ */ ++#if defined(CONFIG_RUNTIME_PHYS_OFFSET) && !defined(__ASSEMBLY__) ++extern unsigned long phys_offset; ++#define PHYS_OFFSET (phys_offset) ++#endif ++ + #else /* CONFIG_MMU */ + + /* diff --git a/packages/linux/ep93xx-kernel_2.6.19+2.6.20-rc7.bb b/packages/linux/ep93xx-kernel_2.6.19+2.6.20-rc7.bb new file mode 100644 index 0000000000..81c359f7fa --- /dev/null +++ b/packages/linux/ep93xx-kernel_2.6.19+2.6.20-rc7.bb @@ -0,0 +1,56 @@ +DESCRIPTION = "Linux Kernel for Cirrus Logic ep39xx compatible machines" +SECTION = "kernel" +LICENSE = "GPL" +PR = "r1" + +COMPATIBLE_MACHINE = "ep93xx" + +SRC_URI = "${KERNELORG_MIRROR}/pub/linux/kernel/v2.6/linux-2.6.19.tar.bz2 \ + ${KERNELORG_MIRROR}/pub/linux/kernel/v2.6/testing/patch-2.6.20-rc7.bz2;patch=1 \ + file://dynamic-phys-offset-2.6.20-rc7.diff;patch=1 \ + file://defconfig \ + " + +S = "${WORKDIR}/linux-2.6.19" + +inherit kernel + +KERNEL_IMAGETYPE = "zImage" + + + +do_configure() { + rm -f ${S}/.config + + if [ ! -e ${WORKDIR}/defconfig ]; then + die "No default configuration for ${MACHINE} available." + fi + + + if [ "${TARGET_OS}" == "linux-gnueabi" -o "${TARGET_OS}" == "linux-uclibcgnueabi" ]; then + echo "CONFIG_AEABI=y" >> ${S}/.config + echo "CONFIG_OABI_COMPAT=y" >> ${S}/.config + else + echo "# CONFIG_AEABI is not set" >> ${S}/.config + echo "# CONFIG_OABI_COMPAT is not set" >> ${S}/.config + fi + + sed -e '/CONFIG_AEABI/d' \ + -e '/CONFIG_OABI_COMPAT=/d' \ + '${WORKDIR}/defconfig' >>'${S}/.config' + + yes '' | oe_runmake oldconfig + + +} + +do_deploy() { + install -d ${DEPLOY_DIR_IMAGE} + install -m 0644 arch/${ARCH}/boot/${KERNEL_IMAGETYPE} ${DEPLOY_DIR_IMAGE}/${KERNEL_IMAGETYPE}-${PV}-${MACHINE}-${DATETIME} +} + +do_deploy[dirs] = "${S}" + +addtask deploy before do_build after do_compile + + diff --git a/packages/linux/ixp4xx-kernel/2.6.19/defconfig b/packages/linux/ixp4xx-kernel/2.6.19/defconfig index 4d88930912..00750f843a 100644 --- a/packages/linux/ixp4xx-kernel/2.6.19/defconfig +++ b/packages/linux/ixp4xx-kernel/2.6.19/defconfig @@ -244,7 +244,7 @@ CONFIG_NET=y # Networking options # # CONFIG_NETDEBUG is not set -CONFIG_PACKET=m +CONFIG_PACKET=y CONFIG_PACKET_MMAP=y CONFIG_UNIX=y CONFIG_XFRM=y diff --git a/packages/linux/ixp4xx-kernel/2.6.20/defconfig b/packages/linux/ixp4xx-kernel/2.6.20/defconfig index 255d952cb9..a7cdacc960 100644 --- a/packages/linux/ixp4xx-kernel/2.6.20/defconfig +++ b/packages/linux/ixp4xx-kernel/2.6.20/defconfig @@ -251,7 +251,7 @@ CONFIG_NET=y # Networking options # # CONFIG_NETDEBUG is not set -CONFIG_PACKET=m +CONFIG_PACKET=y CONFIG_PACKET_MMAP=y CONFIG_UNIX=y CONFIG_XFRM=y diff --git a/packages/linux/ixp4xx-kernel_2.6.19.bb b/packages/linux/ixp4xx-kernel_2.6.19.bb index 3be13fd971..6841c93d00 100644 --- a/packages/linux/ixp4xx-kernel_2.6.19.bb +++ b/packages/linux/ixp4xx-kernel_2.6.19.bb @@ -6,7 +6,7 @@ # http://trac.nslu2-linux.org/kernel/ # # The revision that is pulled from SVN is specified below -IXP4XX_KERNEL_SVN_REV = "681" +IXP4XX_KERNEL_SVN_REV = "682" # # The directory containing the patches to be applied is # specified below @@ -23,3 +23,16 @@ require ixp4xx-kernel-svnpatch.inc SRC_URI += "file://defconfig" SRC_URI += "file://series" + +# Remove the specific cmdline hacking patches if we are not building for nslu2. +addtask remove_cmdline_hacks before do_patch after do_unpack +do_remove_cmdline_hacks() { + +# if [ "${MACHINE}" != "nslu2" ] ; then + sed -e '/88-nas100d-dflt-cmdline.patch/d' \ + -e '/88-nslu2-dflt-cmdline.patch/d' \ + '${WORKDIR}/series' > '${WORKDIR}/series.new' + mv '${WORKDIR}/series.new' '${WORKDIR}/series' +# fi + +} diff --git a/packages/linux/ixp4xx-kernel_2.6.19+2.6.20-rc6.bb b/packages/linux/ixp4xx-kernel_2.6.20.bb index 5ab6d5c6e7..0cc6b387f7 100644 --- a/packages/linux/ixp4xx-kernel_2.6.19+2.6.20-rc6.bb +++ b/packages/linux/ixp4xx-kernel_2.6.20.bb @@ -6,7 +6,7 @@ # http://trac.nslu2-linux.org/kernel/ # # The revision that is pulled from SVN is specified below -IXP4XX_KERNEL_SVN_REV = "681" +IXP4XX_KERNEL_SVN_REV = "682" # # The directory containing the patches to be applied is # specified below @@ -23,3 +23,16 @@ require ixp4xx-kernel-svnpatch.inc SRC_URI += "file://defconfig" SRC_URI += "file://series" + +# Remove the specific cmdline hacking patches if we are not building for nslu2. +addtask remove_cmdline_hacks before do_patch after do_unpack +do_remove_cmdline_hacks() { + + if [ "${MACHINE}" != "nslu2" ] ; then + sed -e '/88-nas100d-dflt-cmdline.patch/d' \ + -e '/88-nslu2-dflt-cmdline.patch/d' \ + '${WORKDIR}/series' > '${WORKDIR}/series.new' + mv '${WORKDIR}/series.new' '${WORKDIR}/series' + fi + +} diff --git a/packages/linux/linux-hackndev-2.6/palmz72/defconfig b/packages/linux/linux-hackndev-2.6/palmz72/defconfig index db628a2424..c31d272ac6 100644 --- a/packages/linux/linux-hackndev-2.6/palmz72/defconfig +++ b/packages/linux/linux-hackndev-2.6/palmz72/defconfig @@ -1,7 +1,7 @@ # # Automatically generated make config: don't edit # Linux kernel version: 2.6.17-hnd0 -# Thu Jan 18 12:36:38 2007 +# Sat Feb 3 23:56:17 2007 # CONFIG_ARM=y CONFIG_MMU=y @@ -143,23 +143,24 @@ CONFIG_ARCH_PXA=y # CONFIG_MACH_XSCALE_PALMLD is not set # CONFIG_MACH_XSCALE_PALMTREO650 is not set # CONFIG_MACH_XSCALE_TREO680 is not set +# CONFIG_MACH_T700WX is not set # CONFIG_MACH_TUNGE2 is not set # CONFIG_MACH_T3XSCALE is not set # CONFIG_MACH_XSCALE_PALMTT5 is not set # CONFIG_MACH_XSCALE_PALMTX is not set CONFIG_MACH_PALMZ72=y -CONFIG_PALMZ72_AC97=y CONFIG_PALMZ72_PM=y +CONFIG_PALMZ72_BATTERY=m CONFIG_GPIOED=m CONFIG_GPIOEDNG=m # CONFIG_MACH_OMAP_PALMTC is not set +# CONFIG_MACH_ZIRE31 is not set # CONFIG_PXA_SHARPSL is not set CONFIG_PXA27x=y CONFIG_PXA_RTC_EPOCH=1970 # CONFIG_SA1100_H3100 is not set # CONFIG_SA1100_H3600 is not set # CONFIG_SA1100_H3800 is not set -# CONFIG_GPIO_KEYS is not set # # Linux As Bootloader @@ -253,7 +254,6 @@ CONFIG_FPE_NWFPE=y CONFIG_BINFMT_ELF=y # CONFIG_BINFMT_AOUT is not set # CONFIG_BINFMT_MISC is not set -# CONFIG_ARTHUR is not set # # Power management options @@ -335,7 +335,7 @@ CONFIG_TCP_CONG_BIC=y # # CONFIG_NET_PKTGEN is not set # CONFIG_HAMRADIO is not set -CONFIG_IRDA=y +CONFIG_IRDA=m # # IrDA protocols @@ -359,7 +359,7 @@ CONFIG_IRDA=y # # SIR device drivers # -CONFIG_IRTTY_SIR=y +CONFIG_IRTTY_SIR=m # # Dongle support @@ -378,24 +378,23 @@ CONFIG_IRTTY_SIR=y # # FIR device drivers # -CONFIG_PXA_FICP=y -CONFIG_BT=y +CONFIG_PXA_FICP=m +CONFIG_BT=m # CONFIG_BT_L2CAP is not set # CONFIG_BT_SCO is not set # # Bluetooth device drivers # -CONFIG_BT_HCIUART=y +CONFIG_BT_HCIUART=m # CONFIG_BT_HCIUART_H4 is not set # CONFIG_BT_HCIUART_BCSP is not set -# CONFIG_BT_HCIBCM203X is not set # CONFIG_BT_HCIVHCI is not set -CONFIG_IEEE80211=y +CONFIG_IEEE80211=m # CONFIG_IEEE80211_DEBUG is not set -# CONFIG_IEEE80211_CRYPT_WEP is not set -# CONFIG_IEEE80211_CRYPT_CCMP is not set -CONFIG_IEEE80211_SOFTMAC=y +CONFIG_IEEE80211_CRYPT_WEP=m +CONFIG_IEEE80211_CRYPT_CCMP=m +CONFIG_IEEE80211_SOFTMAC=m # CONFIG_IEEE80211_SOFTMAC_DEBUG is not set CONFIG_WIRELESS_EXT=y @@ -540,6 +539,10 @@ CONFIG_PPP=y # CONFIG_INPUT=y +# +# Userland interfaces +# +# CONFIG_INPUT_MOUSEDEV is not set # CONFIG_INPUT_JOYDEV is not set CONFIG_INPUT_TSDEV=y CONFIG_INPUT_TSDEV_SCREEN_X=320 @@ -559,10 +562,20 @@ CONFIG_INPUT_KEYBOARD=y # CONFIG_KEYBOARD_STOWAWAY is not set CONFIG_KEYBOARD_PXA27x=y # CONFIG_KEYBOARD_PALMIR is not set -CONFIG_KEYBOARD_PALMWK=y +CONFIG_KEYBOARD_PALMWK=m # CONFIG_INPUT_MOUSE is not set # CONFIG_INPUT_JOYSTICK is not set -# CONFIG_INPUT_TOUCHSCREEN is not set +CONFIG_INPUT_TOUCHSCREEN=y +# CONFIG_TOUCHSCREEN_ADS7846 is not set +# CONFIG_TOUCHSCREEN_GUNZE is not set +# CONFIG_TOUCHSCREEN_ELO is not set +# CONFIG_TOUCHSCREEN_MTOUCH is not set +# CONFIG_TOUCHSCREEN_MK712 is not set +CONFIG_TOUCHSCREEN_WM97XX=y +# CONFIG_TOUCHSCREEN_WM9705 is not set +CONFIG_TOUCHSCREEN_WM9712=y +# CONFIG_TOUCHSCREEN_WM9713 is not set +# CONFIG_TOUCHSCREEN_WM97XX_PXA is not set # CONFIG_INPUT_MISC is not set # @@ -1068,7 +1081,31 @@ CONFIG_FRAME_POINTER=y # # Cryptographic options # -# CONFIG_CRYPTO is not set +CONFIG_CRYPTO=y +# CONFIG_CRYPTO_HMAC is not set +# CONFIG_CRYPTO_NULL is not set +# CONFIG_CRYPTO_MD4 is not set +# CONFIG_CRYPTO_MD5 is not set +# CONFIG_CRYPTO_SHA1 is not set +# CONFIG_CRYPTO_SHA256 is not set +# CONFIG_CRYPTO_SHA512 is not set +# CONFIG_CRYPTO_WP512 is not set +# CONFIG_CRYPTO_TGR192 is not set +# CONFIG_CRYPTO_DES is not set +# CONFIG_CRYPTO_BLOWFISH is not set +# CONFIG_CRYPTO_TWOFISH is not set +# CONFIG_CRYPTO_SERPENT is not set +CONFIG_CRYPTO_AES=m +# CONFIG_CRYPTO_CAST5 is not set +# CONFIG_CRYPTO_CAST6 is not set +# CONFIG_CRYPTO_TEA is not set +CONFIG_CRYPTO_ARC4=m +# CONFIG_CRYPTO_KHAZAD is not set +# CONFIG_CRYPTO_ANUBIS is not set +# CONFIG_CRYPTO_DEFLATE is not set +# CONFIG_CRYPTO_MICHAEL_MIC is not set +# CONFIG_CRYPTO_CRC32C is not set +# CONFIG_CRYPTO_TEST is not set # # Hardware crypto devices diff --git a/packages/linux/linux-hackndev-2.6_svn.bb b/packages/linux/linux-hackndev-2.6_svn.bb index 7ef5ad581e..427923af2c 100644 --- a/packages/linux/linux-hackndev-2.6_svn.bb +++ b/packages/linux/linux-hackndev-2.6_svn.bb @@ -2,7 +2,7 @@ DESCRIPTION = "Hack&Dev's Linux kernel for Palm devices." HOMEPAGE = "http://www.hackndev.com/" SECTION = "kernel" LICENSE = "GPL" -PR = "r1" +PR = "r2" COMPATIBLE_MACHINE = "(palmtx|palmld|palmz72)" diff --git a/packages/linux/linux-rp-2.6.18/defconfig-spitz b/packages/linux/linux-rp-2.6.18/defconfig-spitz index f425956323..bda9cfbd53 100644 --- a/packages/linux/linux-rp-2.6.18/defconfig-spitz +++ b/packages/linux/linux-rp-2.6.18/defconfig-spitz @@ -1563,3 +1563,4 @@ CONFIG_CRC32=y CONFIG_LIBCRC32C=m CONFIG_ZLIB_INFLATE=y CONFIG_ZLIB_DEFLATE=y +CONFIG_SHARPSL_RC=y diff --git a/packages/linux/linux-rp-2.6.20/.mtn2git_empty b/packages/linux/linux-rp-2.6.20/.mtn2git_empty new file mode 100644 index 0000000000..e69de29bb2 --- /dev/null +++ b/packages/linux/linux-rp-2.6.20/.mtn2git_empty diff --git a/packages/linux/linux-rp-2.6.20/add-oz-release-string.patch b/packages/linux/linux-rp-2.6.20/add-oz-release-string.patch new file mode 100644 index 0000000000..a80a1d5528 --- /dev/null +++ b/packages/linux/linux-rp-2.6.20/add-oz-release-string.patch @@ -0,0 +1,28 @@ +--- + Makefile | 1 + + init/version.c | 2 +- + 2 files changed, 2 insertions(+), 1 deletion(-) + +Index: git/init/version.c +=================================================================== +--- git.orig/init/version.c 2006-10-31 16:09:47.000000000 +0000 ++++ git/init/version.c 2006-10-31 16:24:54.000000000 +0000 +@@ -35,5 +35,5 @@ struct uts_namespace init_uts_ns = { + EXPORT_SYMBOL_GPL(init_uts_ns); + + const char linux_banner[] = +- "Linux version " UTS_RELEASE " (" LINUX_COMPILE_BY "@" ++ "Linux version " UTS_RELEASE OPENZAURUS_RELEASE " (" LINUX_COMPILE_BY "@" + LINUX_COMPILE_HOST ") (" LINUX_COMPILER ") " UTS_VERSION "\n"; +Index: git/Makefile +=================================================================== +--- git.orig/Makefile 2006-10-31 16:08:28.000000000 +0000 ++++ git/Makefile 2006-10-31 16:27:02.000000000 +0000 +@@ -905,6 +905,7 @@ endef + define filechk_version.h + (echo \#define LINUX_VERSION_CODE $(shell \ + expr $(VERSION) \* 65536 + $(PATCHLEVEL) \* 256 + $(SUBLEVEL)); \ ++ echo \#define OPENZAURUS_RELEASE \"$(OPENZAURUS_RELEASE)\"; \ + echo '#define KERNEL_VERSION(a,b,c) (((a) << 16) + ((b) << 8) + (c))';) + endef + diff --git a/packages/linux/linux-rp-2.6.20/connectplus-remove-ide-HACK.patch b/packages/linux/linux-rp-2.6.20/connectplus-remove-ide-HACK.patch new file mode 100644 index 0000000000..4414b21191 --- /dev/null +++ b/packages/linux/linux-rp-2.6.20/connectplus-remove-ide-HACK.patch @@ -0,0 +1,12 @@ +Index: linux-2.6.13/drivers/ide/legacy/ide-cs.c +=================================================================== +--- linux-2.6.13.orig/drivers/ide/legacy/ide-cs.c 2005-09-01 22:43:46.000000000 +0100 ++++ linux-2.6.13/drivers/ide/legacy/ide-cs.c 2005-09-01 22:45:46.000000000 +0100 +@@ -488,7 +488,6 @@ + PCMCIA_DEVICE_PROD_ID123("KODAK Picture Card ", "KODAK ", "V100K", 0x94a0d8f3, 0xe4fc3ea0, 0xe5e7eed4), + PCMCIA_DEVICE_PROD_ID1("STI Flash", 0xe4a13209), + PCMCIA_DEVICE_PROD_ID12("STI", "Flash 5.0", 0xbf2df18d, 0x8cb57a0e), +- PCMCIA_MFC_DEVICE_PROD_ID12(1, "SanDisk", "ConnectPlus", 0x7a954bd9, 0x74be00c6), + PCMCIA_DEVICE_NULL, + }; + MODULE_DEVICE_TABLE(pcmcia, ide_ids); diff --git a/packages/linux/linux-rp-2.6.20/defconfig-akita b/packages/linux/linux-rp-2.6.20/defconfig-akita new file mode 100644 index 0000000000..249466ac87 --- /dev/null +++ b/packages/linux/linux-rp-2.6.20/defconfig-akita @@ -0,0 +1,1572 @@ +# +# Automatically generated make config: don't edit +# Linux kernel version: 2.6.16 +# Thu Mar 23 22:11:12 2006 +# +CONFIG_ARM=y +CONFIG_MMU=y +CONFIG_RWSEM_GENERIC_SPINLOCK=y +CONFIG_GENERIC_CALIBRATE_DELAY=y +CONFIG_ARCH_MTD_XIP=y + +# +# Code maturity level options +# +CONFIG_EXPERIMENTAL=y +CONFIG_BROKEN_ON_SMP=y +CONFIG_LOCK_KERNEL=y +CONFIG_INIT_ENV_ARG_LIMIT=32 + +# +# General setup +# +CONFIG_LOCALVERSION="" +CONFIG_LOCALVERSION_AUTO=y +CONFIG_SWAP=y +CONFIG_SYSVIPC=y +# CONFIG_POSIX_MQUEUE is not set +CONFIG_BSD_PROCESS_ACCT=y +# CONFIG_BSD_PROCESS_ACCT_V3 is not set +CONFIG_SYSCTL=y +# CONFIG_AUDIT is not set +# CONFIG_IKCONFIG is not set +CONFIG_INITRAMFS_SOURCE="" +CONFIG_UID16=y +CONFIG_CC_OPTIMIZE_FOR_SIZE=y +CONFIG_EMBEDDED=y +CONFIG_KALLSYMS=y +# CONFIG_KALLSYMS_ALL is not set +# CONFIG_KALLSYMS_EXTRA_PASS is not set +CONFIG_HOTPLUG=y +CONFIG_PRINTK=y +CONFIG_BUG=y +CONFIG_ELF_CORE=y +CONFIG_BASE_FULL=y +CONFIG_FUTEX=y +CONFIG_EPOLL=y +CONFIG_SHMEM=y +CONFIG_CC_ALIGN_FUNCTIONS=0 +CONFIG_CC_ALIGN_LABELS=0 +CONFIG_CC_ALIGN_LOOPS=0 +CONFIG_CC_ALIGN_JUMPS=0 +CONFIG_SLAB=y +# CONFIG_TINY_SHMEM is not set +CONFIG_BASE_SMALL=0 +# CONFIG_SLOB is not set + +# +# Loadable module support +# +CONFIG_MODULES=y +CONFIG_MODULE_UNLOAD=y +CONFIG_MODULE_FORCE_UNLOAD=y +CONFIG_OBSOLETE_MODPARM=y +# CONFIG_MODVERSIONS is not set +# CONFIG_MODULE_SRCVERSION_ALL is not set +CONFIG_KMOD=y + +# +# Block layer +# + +# +# IO Schedulers +# +CONFIG_IOSCHED_NOOP=y +CONFIG_IOSCHED_AS=y +CONFIG_IOSCHED_DEADLINE=m +CONFIG_IOSCHED_CFQ=m +CONFIG_DEFAULT_AS=y +# CONFIG_DEFAULT_DEADLINE is not set +# CONFIG_DEFAULT_CFQ is not set +# CONFIG_DEFAULT_NOOP is not set +CONFIG_DEFAULT_IOSCHED="anticipatory" + +# +# System Type +# +# CONFIG_ARCH_CLPS7500 is not set +# CONFIG_ARCH_CLPS711X is not set +# CONFIG_ARCH_CO285 is not set +# CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_FOOTBRIDGE is not set +# CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_IOP3XX is not set +# CONFIG_ARCH_IXP4XX is not set +# CONFIG_ARCH_IXP2000 is not set +# CONFIG_ARCH_L7200 is not set +CONFIG_ARCH_PXA=y +# CONFIG_ARCH_RPC is not set +# CONFIG_ARCH_SA1100 is not set +# CONFIG_ARCH_S3C2410 is not set +# CONFIG_ARCH_SHARK is not set +# CONFIG_ARCH_LH7A40X is not set +# CONFIG_ARCH_OMAP is not set +# CONFIG_ARCH_VERSATILE is not set +# CONFIG_ARCH_REALVIEW is not set +# CONFIG_ARCH_IMX is not set +# CONFIG_ARCH_H720X is not set +# CONFIG_ARCH_AAEC2000 is not set +# CONFIG_ARCH_AT91RM9200 is not set + +# +# Intel PXA2xx Implementations +# +# CONFIG_ARCH_LUBBOCK is not set +# CONFIG_MACH_MAINSTONE is not set +# CONFIG_ARCH_PXA_IDP is not set +CONFIG_PXA_SHARPSL=y +# CONFIG_MACH_HX2750 is not set +# CONFIG_PXA_SHARPSL_25x is not set +CONFIG_PXA_SHARPSL_27x=y +CONFIG_MACH_AKITA=y +CONFIG_MACH_SPITZ=y +CONFIG_MACH_BORZOI=y +CONFIG_PXA27x=y +# CONFIG_PXA_KEYS is not set +CONFIG_IWMMXT=y +CONFIG_PXA_SHARP_Cxx00=y +CONFIG_PXA_SSP=y + +# +# Processor Type +# +CONFIG_CPU_32=y +CONFIG_CPU_XSCALE=y +CONFIG_CPU_32v5=y +CONFIG_CPU_ABRT_EV5T=y +CONFIG_CPU_CACHE_VIVT=y +CONFIG_CPU_TLB_V4WBI=y + +# +# Processor Features +# +CONFIG_ARM_THUMB=y +CONFIG_XSCALE_PMU=y +CONFIG_KEXEC=y +CONFIG_SHARP_PARAM=y +CONFIG_SHARPSL_PM=y +CONFIG_SHARP_SCOOP=y + +# +# Bus support +# + +# +# PCCARD (PCMCIA/CardBus) support +# +CONFIG_PCCARD=y +# CONFIG_PCMCIA_DEBUG is not set +CONFIG_PCMCIA=y +CONFIG_PCMCIA_LOAD_CIS=y +CONFIG_PCMCIA_IOCTL=y + +# +# PC-card bridges +# +CONFIG_PCMCIA_PXA2XX=y + +# +# Kernel Features +# +CONFIG_PREEMPT=y +CONFIG_NO_IDLE_HZ=y +# CONFIG_AEABI is not set +# CONFIG_ARCH_DISCONTIGMEM_ENABLE is not set +CONFIG_SELECT_MEMORY_MODEL=y +CONFIG_FLATMEM_MANUAL=y +# CONFIG_DISCONTIGMEM_MANUAL is not set +# CONFIG_SPARSEMEM_MANUAL is not set +CONFIG_FLATMEM=y +CONFIG_FLAT_NODE_MEM_MAP=y +# CONFIG_SPARSEMEM_STATIC is not set +CONFIG_SPLIT_PTLOCK_CPUS=4096 +CONFIG_ALIGNMENT_TRAP=y + +# +# Boot options +# +CONFIG_ZBOOT_ROM_TEXT=0x0 +CONFIG_ZBOOT_ROM_BSS=0x0 +# CONFIG_XIP_KERNEL is not set + +# +# CPU Frequency scaling +# +# CONFIG_CPU_FREQ is not set + +# +# Floating point emulation +# + +# +# At least one emulation must be selected +# +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_NWFPE_XP is not set +# CONFIG_FPE_FASTFPE is not set + +# +# Userspace binary formats +# +CONFIG_BINFMT_ELF=y +CONFIG_BINFMT_AOUT=m +CONFIG_BINFMT_MISC=m +# CONFIG_ARTHUR is not set + +# +# Power management options +# +CONFIG_PM=y +# CONFIG_PM_LEGACY is not set +# CONFIG_PM_DEBUG is not set +CONFIG_APM=y + +# +# Networking +# +CONFIG_NET=y + +# +# Networking options +# +# CONFIG_NETDEBUG is not set +CONFIG_PACKET=y +CONFIG_PACKET_MMAP=y +CONFIG_UNIX=y +CONFIG_XFRM=y +# CONFIG_XFRM_USER is not set +# CONFIG_NET_KEY is not set +CONFIG_INET=y +# CONFIG_IP_MULTICAST is not set +# CONFIG_IP_ADVANCED_ROUTER is not set +CONFIG_IP_FIB_HASH=y +# CONFIG_IP_PNP is not set +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set +CONFIG_SYN_COOKIES=y +# CONFIG_INET_AH is not set +# CONFIG_INET_ESP is not set +# CONFIG_INET_IPCOMP is not set +# CONFIG_INET_TUNNEL is not set +CONFIG_INET_DIAG=m +CONFIG_INET_TCP_DIAG=m +# CONFIG_TCP_CONG_ADVANCED is not set +CONFIG_TCP_CONG_BIC=y + +# +# IP: Virtual Server Configuration +# +# CONFIG_IP_VS is not set +CONFIG_IPV6=m +# CONFIG_IPV6_PRIVACY is not set +CONFIG_INET6_AH=m +CONFIG_INET6_ESP=m +CONFIG_INET6_IPCOMP=m +CONFIG_INET6_TUNNEL=m +CONFIG_IPV6_TUNNEL=m +CONFIG_NETFILTER=y +# CONFIG_NETFILTER_DEBUG is not set + +# +# Core Netfilter Configuration +# +# CONFIG_NETFILTER_NETLINK is not set +# CONFIG_NETFILTER_XTABLES is not set + +# +# IP: Netfilter Configuration +# +CONFIG_IP_NF_CONNTRACK=m +# CONFIG_IP_NF_CT_ACCT is not set +# CONFIG_IP_NF_CONNTRACK_MARK is not set +# CONFIG_IP_NF_CONNTRACK_EVENTS is not set +CONFIG_IP_NF_CT_PROTO_SCTP=m +CONFIG_IP_NF_FTP=m +CONFIG_IP_NF_IRC=m +# CONFIG_IP_NF_NETBIOS_NS is not set +CONFIG_IP_NF_TFTP=m +CONFIG_IP_NF_AMANDA=m +# CONFIG_IP_NF_PPTP is not set +CONFIG_IP_NF_QUEUE=m + +# +# IPv6: Netfilter Configuration (EXPERIMENTAL) +# +# CONFIG_IP6_NF_QUEUE is not set + +# +# DCCP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_DCCP is not set + +# +# SCTP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_SCTP is not set + +# +# TIPC Configuration (EXPERIMENTAL) +# +# CONFIG_TIPC is not set +# CONFIG_ATM is not set +# CONFIG_BRIDGE is not set +# CONFIG_VLAN_8021Q is not set +# CONFIG_DECNET is not set +# CONFIG_LLC2 is not set +# CONFIG_IPX is not set +# CONFIG_ATALK is not set +# CONFIG_X25 is not set +# CONFIG_LAPB is not set +# CONFIG_NET_DIVERT is not set +# CONFIG_ECONET is not set +# CONFIG_WAN_ROUTER is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network testing +# +# CONFIG_NET_PKTGEN is not set +# CONFIG_HAMRADIO is not set +CONFIG_IRDA=m + +# +# IrDA protocols +# +CONFIG_IRLAN=m +CONFIG_IRNET=m +CONFIG_IRCOMM=m +# CONFIG_IRDA_ULTRA is not set + +# +# IrDA options +# +# CONFIG_IRDA_CACHE_LAST_LSAP is not set +# CONFIG_IRDA_FAST_RR is not set +# CONFIG_IRDA_DEBUG is not set + +# +# Infrared-port device drivers +# + +# +# SIR device drivers +# +# CONFIG_IRTTY_SIR is not set + +# +# Dongle support +# + +# +# Old SIR device drivers +# +# CONFIG_IRPORT_SIR is not set + +# +# Old Serial dongle support +# + +# +# FIR device drivers +# +# CONFIG_USB_IRDA is not set +# CONFIG_SIGMATEL_FIR is not set +CONFIG_PXA_FICP=m +CONFIG_BT=m +CONFIG_BT_L2CAP=m +CONFIG_BT_SCO=m +CONFIG_BT_RFCOMM=m +CONFIG_BT_RFCOMM_TTY=y +CONFIG_BT_BNEP=m +CONFIG_BT_BNEP_MC_FILTER=y +CONFIG_BT_BNEP_PROTO_FILTER=y +CONFIG_BT_HIDP=m + +# +# Bluetooth device drivers +# +CONFIG_BT_HCIUSB=m +# CONFIG_BT_HCIUSB_SCO is not set +CONFIG_BT_HCIUART=m +CONFIG_BT_HCIUART_H4=y +CONFIG_BT_HCIUART_BCSP=y +CONFIG_BT_HCIBCM203X=m +CONFIG_BT_HCIBPA10X=m +CONFIG_BT_HCIBFUSB=m +CONFIG_BT_HCIDTL1=m +CONFIG_BT_HCIBT3C=m +CONFIG_BT_HCIBLUECARD=m +CONFIG_BT_HCIBTUART=m +CONFIG_BT_HCIVHCI=m +CONFIG_IEEE80211=m +# CONFIG_IEEE80211_DEBUG is not set +CONFIG_IEEE80211_CRYPT_WEP=m +CONFIG_IEEE80211_CRYPT_CCMP=m +CONFIG_IEEE80211_CRYPT_TKIP=m + +# +# Device Drivers +# + +# +# Generic Driver Options +# +CONFIG_STANDALONE=y +CONFIG_PREVENT_FIRMWARE_BUILD=y +CONFIG_FW_LOADER=y +# CONFIG_DEBUG_DRIVER is not set + +# +# Connector - unified userspace <-> kernelspace linker +# +# CONFIG_CONNECTOR is not set + +# +# Memory Technology Devices (MTD) +# +CONFIG_MTD=y +# CONFIG_MTD_DEBUG is not set +# CONFIG_MTD_CONCAT is not set +CONFIG_MTD_PARTITIONS=y +# CONFIG_MTD_REDBOOT_PARTS is not set +CONFIG_MTD_CMDLINE_PARTS=y +# CONFIG_MTD_AFS_PARTS is not set + +# +# User Modules And Translation Layers +# +CONFIG_MTD_CHAR=y +CONFIG_MTD_BLOCK=y +# CONFIG_FTL is not set +# CONFIG_NFTL is not set +# CONFIG_INFTL is not set +# CONFIG_RFD_FTL is not set + +# +# RAM/ROM/Flash chip drivers +# +# CONFIG_MTD_CFI is not set +# CONFIG_MTD_JEDECPROBE is not set +CONFIG_MTD_MAP_BANK_WIDTH_1=y +CONFIG_MTD_MAP_BANK_WIDTH_2=y +CONFIG_MTD_MAP_BANK_WIDTH_4=y +# CONFIG_MTD_MAP_BANK_WIDTH_8 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_16 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_32 is not set +CONFIG_MTD_CFI_I1=y +CONFIG_MTD_CFI_I2=y +# CONFIG_MTD_CFI_I4 is not set +# CONFIG_MTD_CFI_I8 is not set +# CONFIG_MTD_RAM is not set +CONFIG_MTD_ROM=y +# CONFIG_MTD_ABSENT is not set +# CONFIG_MTD_OBSOLETE_CHIPS is not set + +# +# Mapping drivers for chip access +# +CONFIG_MTD_COMPLEX_MAPPINGS=y +CONFIG_MTD_SHARP_SL=y +# CONFIG_MTD_PLATRAM is not set + +# +# Self-contained MTD device drivers +# +# CONFIG_MTD_SLRAM is not set +# CONFIG_MTD_PHRAM is not set +# CONFIG_MTD_MTDRAM is not set +# CONFIG_MTD_BLKMTD is not set +# CONFIG_MTD_BLOCK2MTD is not set + +# +# Disk-On-Chip Device Drivers +# +# CONFIG_MTD_DOC2000 is not set +# CONFIG_MTD_DOC2001 is not set +# CONFIG_MTD_DOC2001PLUS is not set + +# +# NAND Flash Device Drivers +# +CONFIG_MTD_NAND=y +CONFIG_MTD_NAND_VERIFY_WRITE=y +# CONFIG_MTD_NAND_H1900 is not set +CONFIG_MTD_NAND_IDS=y +# CONFIG_MTD_NAND_DISKONCHIP is not set +CONFIG_MTD_NAND_SHARPSL=y +# CONFIG_MTD_NAND_NANDSIM is not set + +# +# OneNAND Flash Device Drivers +# +# CONFIG_MTD_ONENAND is not set + +# +# Parallel port support +# +# CONFIG_PARPORT is not set + +# +# Plug and Play support +# + +# +# Block devices +# +# CONFIG_BLK_DEV_COW_COMMON is not set +CONFIG_BLK_DEV_LOOP=y +# CONFIG_BLK_DEV_CRYPTOLOOP is not set +# CONFIG_BLK_DEV_NBD is not set +# CONFIG_BLK_DEV_UB is not set +# CONFIG_BLK_DEV_RAM is not set +CONFIG_BLK_DEV_RAM_COUNT=16 +# CONFIG_CDROM_PKTCDVD is not set +# CONFIG_ATA_OVER_ETH is not set + +# +# ATA/ATAPI/MFM/RLL support +# +CONFIG_IDE=y +CONFIG_BLK_DEV_IDE=y + +# +# Please see Documentation/ide.txt for help/info on IDE drives +# +# CONFIG_BLK_DEV_IDE_SATA is not set +CONFIG_BLK_DEV_IDEDISK=y +# CONFIG_IDEDISK_MULTI_MODE is not set +CONFIG_BLK_DEV_IDECS=y +# CONFIG_BLK_DEV_IDECD is not set +# CONFIG_BLK_DEV_IDETAPE is not set +# CONFIG_BLK_DEV_IDEFLOPPY is not set +# CONFIG_BLK_DEV_IDESCSI is not set +# CONFIG_IDE_TASK_IOCTL is not set + +# +# IDE chipset support/bugfixes +# +CONFIG_IDE_GENERIC=y +# CONFIG_IDE_ARM is not set +# CONFIG_BLK_DEV_IDEDMA is not set +# CONFIG_IDEDMA_AUTO is not set +# CONFIG_BLK_DEV_HD is not set + +# +# SCSI device support +# +# CONFIG_RAID_ATTRS is not set +CONFIG_SCSI=m +CONFIG_SCSI_PROC_FS=y + +# +# SCSI support type (disk, tape, CD-ROM) +# +CONFIG_BLK_DEV_SD=m +CONFIG_CHR_DEV_ST=m +CONFIG_CHR_DEV_OSST=m +CONFIG_BLK_DEV_SR=m +# CONFIG_BLK_DEV_SR_VENDOR is not set +CONFIG_CHR_DEV_SG=m +# CONFIG_CHR_DEV_SCH is not set + +# +# Some SCSI devices (e.g. CD jukebox) support multiple LUNs +# +CONFIG_SCSI_MULTI_LUN=y +# CONFIG_SCSI_CONSTANTS is not set +# CONFIG_SCSI_LOGGING is not set + +# +# SCSI Transport Attributes +# +# CONFIG_SCSI_SPI_ATTRS is not set +# CONFIG_SCSI_FC_ATTRS is not set +# CONFIG_SCSI_ISCSI_ATTRS is not set +# CONFIG_SCSI_SAS_ATTRS is not set + +# +# SCSI low-level drivers +# +# CONFIG_ISCSI_TCP is not set +# CONFIG_SCSI_SATA is not set +# CONFIG_SCSI_DEBUG is not set + +# +# PCMCIA SCSI adapter support +# +# CONFIG_PCMCIA_AHA152X is not set +# CONFIG_PCMCIA_FDOMAIN is not set +# CONFIG_PCMCIA_NINJA_SCSI is not set +# CONFIG_PCMCIA_QLOGIC is not set +# CONFIG_PCMCIA_SYM53C500 is not set + +# +# Multi-device support (RAID and LVM) +# +CONFIG_MD=y +CONFIG_BLK_DEV_DM=m +CONFIG_DM_CRYPT=m +CONFIG_DM_SNAPSHOT=m +CONFIG_DM_MIRROR=m +CONFIG_DM_ZERO=m +CONFIG_DM_MULTIPATH=m +CONFIG_DM_MULTIPATH_EMC=m + +# +# Fusion MPT device support +# +# CONFIG_FUSION is not set + +# +# IEEE 1394 (FireWire) support +# + +# +# I2O device support +# + +# +# Network device support +# +CONFIG_NETDEVICES=y +# CONFIG_DUMMY is not set +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +CONFIG_TUN=m + +# +# PHY device support +# +# CONFIG_PHYLIB is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +CONFIG_MII=m +# CONFIG_SMC91X is not set +# CONFIG_DM9000 is not set + +# +# Ethernet (1000 Mbit) +# + +# +# Ethernet (10000 Mbit) +# + +# +# Token Ring devices +# + +# +# Wireless LAN (non-hamradio) +# +CONFIG_NET_RADIO=y + +# +# Obsolete Wireless cards support (pre-802.11) +# +# CONFIG_STRIP is not set +# CONFIG_PCMCIA_WAVELAN is not set +# CONFIG_PCMCIA_NETWAVE is not set + +# +# Wireless 802.11 Frequency Hopping cards support +# +# CONFIG_PCMCIA_RAYCS is not set + +# +# Wireless 802.11b ISA/PCI cards support +# +CONFIG_HERMES=m +# CONFIG_ATMEL is not set + +# +# Wireless 802.11b Pcmcia/Cardbus cards support +# +CONFIG_PCMCIA_HERMES=m +CONFIG_PCMCIA_SPECTRUM=m +# CONFIG_AIRO_CS is not set +# CONFIG_PCMCIA_WL3501 is not set +CONFIG_HOSTAP=m +CONFIG_HOSTAP_FIRMWARE=y +# CONFIG_HOSTAP_FIRMWARE_NVRAM is not set +CONFIG_HOSTAP_CS=m +CONFIG_NET_WIRELESS=y + +# +# PCMCIA network device support +# +CONFIG_NET_PCMCIA=y +# CONFIG_PCMCIA_3C589 is not set +# CONFIG_PCMCIA_3C574 is not set +# CONFIG_PCMCIA_FMVJ18X is not set +CONFIG_PCMCIA_PCNET=m +# CONFIG_PCMCIA_NMCLAN is not set +# CONFIG_PCMCIA_SMC91C92 is not set +# CONFIG_PCMCIA_XIRC2PS is not set +# CONFIG_PCMCIA_AXNET is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set +CONFIG_PPP=m +# CONFIG_PPP_MULTILINK is not set +# CONFIG_PPP_FILTER is not set +CONFIG_PPP_ASYNC=m +# CONFIG_PPP_SYNC_TTY is not set +CONFIG_PPP_DEFLATE=m +CONFIG_PPP_BSDCOMP=m +# CONFIG_PPP_MPPE is not set +# CONFIG_PPPOE is not set +# CONFIG_SLIP is not set +# CONFIG_SHAPER is not set +# CONFIG_NETCONSOLE is not set +# CONFIG_NETPOLL is not set +# CONFIG_NET_POLL_CONTROLLER is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Input device support +# +CONFIG_INPUT=y + +# +# Userland interfaces +# +CONFIG_INPUT_MOUSEDEV=m +# CONFIG_INPUT_MOUSEDEV_PSAUX is not set +CONFIG_INPUT_MOUSEDEV_SCREEN_X=640 +CONFIG_INPUT_MOUSEDEV_SCREEN_Y=480 +# CONFIG_INPUT_JOYDEV is not set +# CONFIG_INPUT_TSDEV is not set +CONFIG_INPUT_EVDEV=y +# CONFIG_INPUT_EVBUG is not set +CONFIG_INPUT_POWER=y + +# +# Input Device Drivers +# +CONFIG_INPUT_KEYBOARD=y +# CONFIG_KEYBOARD_ATKBD is not set +# CONFIG_KEYBOARD_SUNKBD is not set +# CONFIG_KEYBOARD_LKKBD is not set +# CONFIG_KEYBOARD_XTKBD is not set +# CONFIG_KEYBOARD_NEWTON is not set +# CONFIG_KEYBOARD_CORGI is not set +CONFIG_KEYBOARD_SPITZ=y +# CONFIG_INPUT_MOUSE is not set +# CONFIG_INPUT_JOYSTICK is not set +CONFIG_INPUT_TOUCHSCREEN=y +CONFIG_TOUCHSCREEN_CORGI=y +# CONFIG_TOUCHSCREEN_GUNZE is not set +# CONFIG_TOUCHSCREEN_ELO is not set +# CONFIG_TOUCHSCREEN_MTOUCH is not set +# CONFIG_TOUCHSCREEN_MK712 is not set +CONFIG_INPUT_MISC=y +CONFIG_INPUT_UINPUT=m + +# +# Hardware I/O ports +# +# CONFIG_SERIO is not set +# CONFIG_GAMEPORT is not set + +# +# Character devices +# +CONFIG_VT=y +CONFIG_VT_CONSOLE=y +CONFIG_HW_CONSOLE=y +# CONFIG_SERIAL_NONSTANDARD is not set + +# +# Serial drivers +# +CONFIG_SERIAL_8250=m +CONFIG_SERIAL_8250_CS=m +CONFIG_SERIAL_8250_NR_UARTS=4 +CONFIG_SERIAL_8250_RUNTIME_UARTS=4 +# CONFIG_SERIAL_8250_EXTENDED is not set + +# +# Non-8250 serial port support +# +CONFIG_SERIAL_PXA=y +CONFIG_SERIAL_PXA_CONSOLE=y +CONFIG_SERIAL_CORE=y +CONFIG_SERIAL_CORE_CONSOLE=y +CONFIG_UNIX98_PTYS=y +# CONFIG_LEGACY_PTYS is not set + +# +# IPMI +# +# CONFIG_IPMI_HANDLER is not set + +# +# Watchdog Cards +# +# CONFIG_WATCHDOG is not set +# CONFIG_NVRAM is not set +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set + +# +# Ftape, the floppy tape device driver +# + +# +# PCMCIA character devices +# +# CONFIG_SYNCLINK_CS is not set +# CONFIG_CARDMAN_4000 is not set +# CONFIG_CARDMAN_4040 is not set +# CONFIG_RAW_DRIVER is not set + +# +# TPM devices +# +# CONFIG_TCG_TPM is not set +# CONFIG_TELCLOCK is not set + +# +# I2C support +# +CONFIG_I2C=y +# CONFIG_I2C_CHARDEV is not set + +# +# I2C Algorithms +# +# CONFIG_I2C_ALGOBIT is not set +# CONFIG_I2C_ALGOPCF is not set +# CONFIG_I2C_ALGOPCA is not set + +# +# I2C Hardware Bus support +# +CONFIG_I2C_PXA=y +# CONFIG_I2C_PXA_SLAVE is not set +# CONFIG_I2C_PARPORT_LIGHT is not set +# CONFIG_I2C_STUB is not set +# CONFIG_I2C_PCA_ISA is not set + +# +# Miscellaneous I2C Chip support +# +# CONFIG_SENSORS_DS1337 is not set +# CONFIG_SENSORS_DS1374 is not set +# CONFIG_SENSORS_EEPROM is not set +# CONFIG_SENSORS_PCF8574 is not set +# CONFIG_SENSORS_PCA9539 is not set +# CONFIG_SENSORS_PCF8591 is not set +# CONFIG_SENSORS_MAX6875 is not set +# CONFIG_I2C_DEBUG_CORE is not set +# CONFIG_I2C_DEBUG_ALGO is not set +# CONFIG_I2C_DEBUG_BUS is not set +# CONFIG_I2C_DEBUG_CHIP is not set + +# +# SPI support +# +# CONFIG_SPI is not set +# CONFIG_SPI_MASTER is not set + +# +# Dallas's 1-wire bus +# +# CONFIG_W1 is not set + +# +# Hardware Monitoring support +# +# CONFIG_HWMON is not set +# CONFIG_HWMON_VID is not set + +# +# Misc devices +# + +# +# Multimedia Capabilities Port drivers +# + +# +# Multi-Function Devices +# + +# +# LED devices +# +CONFIG_NEW_LEDS=y +CONFIG_LEDS_CLASS=y +CONFIG_LEDS_TRIGGERS=y +CONFIG_LEDS_SPITZ=y +# CONFIG_LEDS_TOSA is not set +CONFIG_LEDS_TRIGGER_TIMER=y +CONFIG_LEDS_TRIGGER_IDE_DISK=y + +# +# Multimedia devices +# +# CONFIG_VIDEO_DEV is not set + +# +# Digital Video Broadcasting Devices +# +# CONFIG_DVB is not set + +# +# Graphics support +# +CONFIG_FB=y +CONFIG_FB_CFB_FILLRECT=y +CONFIG_FB_CFB_COPYAREA=y +CONFIG_FB_CFB_IMAGEBLIT=y +# CONFIG_FB_MACMODES is not set +# CONFIG_FB_MODE_HELPERS is not set +# CONFIG_FB_TILEBLITTING is not set +# CONFIG_FB_S1D13XXX is not set +CONFIG_FB_PXA=y +# CONFIG_FB_PXA_PARAMETERS is not set +# CONFIG_FB_W100 is not set +# CONFIG_FB_VIRTUAL is not set + +# +# Console display driver support +# +# CONFIG_VGA_CONSOLE is not set +CONFIG_DUMMY_CONSOLE=y +CONFIG_FRAMEBUFFER_CONSOLE=y +CONFIG_FRAMEBUFFER_CONSOLE_ROTATION=y +CONFIG_FONTS=y +# CONFIG_FONT_8x8 is not set +CONFIG_FONT_8x16=y +# CONFIG_FONT_6x11 is not set +# CONFIG_FONT_7x14 is not set +# CONFIG_FONT_PEARL_8x8 is not set +# CONFIG_FONT_ACORN_8x8 is not set +# CONFIG_FONT_MINI_4x6 is not set +# CONFIG_FONT_SUN8x16 is not set +# CONFIG_FONT_SUN12x22 is not set +# CONFIG_FONT_10x18 is not set + +# +# Logo configuration +# +# CONFIG_LOGO is not set +CONFIG_BACKLIGHT_LCD_SUPPORT=y +CONFIG_BACKLIGHT_CLASS_DEVICE=y +CONFIG_BACKLIGHT_DEVICE=y +CONFIG_LCD_CLASS_DEVICE=y +CONFIG_LCD_DEVICE=y +CONFIG_BACKLIGHT_CORGI=y +# CONFIG_BACKLIGHT_HP680 is not set + +# +# Sound +# +CONFIG_SOUND=m + +# +# Advanced Linux Sound Architecture +# +CONFIG_SND=m +CONFIG_SND_TIMER=m +CONFIG_SND_PCM=m +CONFIG_SND_SEQUENCER=m +# CONFIG_SND_SEQ_DUMMY is not set +CONFIG_SND_OSSEMUL=y +CONFIG_SND_MIXER_OSS=m +CONFIG_SND_PCM_OSS=m +# CONFIG_SND_SEQUENCER_OSS is not set +# CONFIG_SND_DYNAMIC_MINORS is not set +CONFIG_SND_SUPPORT_OLD_API=y +CONFIG_SND_VERBOSE_PRINTK=y +CONFIG_SND_DEBUG=y +# CONFIG_SND_DEBUG_DETECT is not set + +# +# Generic devices +# +CONFIG_SND_AC97_CODEC=m +CONFIG_SND_AC97_BUS=m +# CONFIG_SND_DUMMY is not set +# CONFIG_SND_VIRMIDI is not set +# CONFIG_SND_MTPAV is not set +# CONFIG_SND_SERIAL_U16550 is not set +# CONFIG_SND_MPU401 is not set + +# +# ALSA ARM devices +# +CONFIG_SND_PXA2XX_PCM=m +CONFIG_SND_PXA2XX_AC97=m + +# +# USB devices +# +CONFIG_SND_USB_AUDIO=m + +# +# PCMCIA devices +# + +# +# SoC audio support +# +CONFIG_SND_SOC=m + +# +# Soc Platforms +# + +# +# SoC Audio for the Intel PXA2xx +# +CONFIG_SND_PXA2xx_SOC=m +CONFIG_SND_PXA2xx_SOC_I2S=m +# CONFIG_SND_PXA2xx_SOC_MAINSTONE is not set +# CONFIG_SND_PXA2xx_SOC_MAINSTONE_WM8753 is not set +# CONFIG_SND_PXA2xx_SOC_MAINSTONE_WM9713 is not set +# CONFIG_SND_PXA2xx_SOC_MAINSTONE_WM9712 is not set +# CONFIG_SND_PXA2xx_SOC_CORGI is not set +CONFIG_SND_PXA2xx_SOC_SPITZ=m +# CONFIG_SND_PXA2xx_SOC_TOSA is not set + +# +# Soc Codecs +# +# CONFIG_SND_SOC_AC97_CODEC is not set +# CONFIG_SND_SOC_WM8731 is not set +CONFIG_SND_SOC_WM8750=m +# CONFIG_SND_SOC_WM8753 is not set +# CONFIG_SND_SOC_WM8772 is not set +# CONFIG_SND_SOC_WM8971 is not set +# CONFIG_SND_SOC_WM9713 is not set +# CONFIG_SND_SOC_WM9712 is not set +# CONFIG_SND_SOC_UDA1380 is not set +# CONFIG_SND_SOC_AK4535 is not set + +# +# Open Sound System +# +# CONFIG_SOUND_PRIME is not set + +# +# USB support +# +CONFIG_USB_ARCH_HAS_HCD=y +CONFIG_USB_ARCH_HAS_OHCI=y +CONFIG_USB=m +# CONFIG_USB_DEBUG is not set + +# +# Miscellaneous USB options +# +CONFIG_USB_DEVICEFS=y +# CONFIG_USB_BANDWIDTH is not set +# CONFIG_USB_DYNAMIC_MINORS is not set +# CONFIG_USB_SUSPEND is not set +# CONFIG_USB_OTG is not set + +# +# USB Host Controller Drivers +# +# CONFIG_USB_ISP116X_HCD is not set +CONFIG_USB_OHCI_HCD=m +# CONFIG_USB_OHCI_BIG_ENDIAN is not set +CONFIG_USB_OHCI_LITTLE_ENDIAN=y +CONFIG_USB_SL811_HCD=m +CONFIG_USB_SL811_CS=m + +# +# USB Device Class drivers +# +# CONFIG_OBSOLETE_OSS_USB_DRIVER is not set +CONFIG_USB_ACM=m +CONFIG_USB_PRINTER=m + +# +# NOTE: USB_STORAGE enables SCSI, and 'SCSI disk support' +# + +# +# may also be needed; see USB_STORAGE Help for more information +# +CONFIG_USB_STORAGE=m +# CONFIG_USB_STORAGE_DEBUG is not set +# CONFIG_USB_STORAGE_DATAFAB is not set +# CONFIG_USB_STORAGE_FREECOM is not set +# CONFIG_USB_STORAGE_ISD200 is not set +# CONFIG_USB_STORAGE_DPCM is not set +# CONFIG_USB_STORAGE_USBAT is not set +# CONFIG_USB_STORAGE_SDDR09 is not set +# CONFIG_USB_STORAGE_SDDR55 is not set +# CONFIG_USB_STORAGE_JUMPSHOT is not set +# CONFIG_USB_STORAGE_ALAUDA is not set +# CONFIG_USB_LIBUSUAL is not set + +# +# USB Input Devices +# +CONFIG_USB_HID=m +CONFIG_USB_HIDINPUT=y +# CONFIG_USB_HIDINPUT_POWERBOOK is not set +# CONFIG_HID_FF is not set +# CONFIG_USB_HIDDEV is not set + +# +# USB HID Boot Protocol drivers +# +CONFIG_USB_KBD=m +CONFIG_USB_MOUSE=m +CONFIG_USB_AIPTEK=m +CONFIG_USB_WACOM=m +# CONFIG_USB_ACECAD is not set +CONFIG_USB_KBTAB=m +CONFIG_USB_POWERMATE=m +CONFIG_USB_MTOUCH=m +# CONFIG_USB_ITMTOUCH is not set +CONFIG_USB_EGALAX=m +# CONFIG_USB_YEALINK is not set +CONFIG_USB_XPAD=m +CONFIG_USB_ATI_REMOTE=m +# CONFIG_USB_ATI_REMOTE2 is not set +# CONFIG_USB_KEYSPAN_REMOTE is not set +# CONFIG_USB_APPLETOUCH is not set + +# +# USB Imaging devices +# +CONFIG_USB_MDC800=m +CONFIG_USB_MICROTEK=m + +# +# USB Multimedia devices +# +CONFIG_USB_DABUSB=m + +# +# Video4Linux support is needed for USB Multimedia device support +# + +# +# USB Network Adapters +# +CONFIG_USB_CATC=m +CONFIG_USB_KAWETH=m +CONFIG_USB_PEGASUS=m +CONFIG_USB_RTL8150=m +CONFIG_USB_USBNET=m +CONFIG_USB_NET_AX8817X=m +CONFIG_USB_NET_CDCETHER=m +# CONFIG_USB_NET_GL620A is not set +CONFIG_USB_NET_NET1080=m +# CONFIG_USB_NET_PLUSB is not set +# CONFIG_USB_NET_RNDIS_HOST is not set +# CONFIG_USB_NET_CDC_SUBSET is not set +CONFIG_USB_NET_ZAURUS=m +# CONFIG_USB_ZD1201 is not set +CONFIG_USB_MON=y + +# +# USB port drivers +# + +# +# USB Serial Converter support +# +CONFIG_USB_SERIAL=m +CONFIG_USB_SERIAL_GENERIC=y +# CONFIG_USB_SERIAL_AIRPRIME is not set +# CONFIG_USB_SERIAL_ANYDATA is not set +CONFIG_USB_SERIAL_BELKIN=m +# CONFIG_USB_SERIAL_WHITEHEAT is not set +CONFIG_USB_SERIAL_DIGI_ACCELEPORT=m +# CONFIG_USB_SERIAL_CP2101 is not set +CONFIG_USB_SERIAL_CYPRESS_M8=m +CONFIG_USB_SERIAL_EMPEG=m +CONFIG_USB_SERIAL_FTDI_SIO=m +CONFIG_USB_SERIAL_VISOR=m +CONFIG_USB_SERIAL_IPAQ=m +CONFIG_USB_SERIAL_IR=m +CONFIG_USB_SERIAL_EDGEPORT=m +CONFIG_USB_SERIAL_EDGEPORT_TI=m +CONFIG_USB_SERIAL_GARMIN=m +CONFIG_USB_SERIAL_IPW=m +CONFIG_USB_SERIAL_KEYSPAN_PDA=m +CONFIG_USB_SERIAL_KEYSPAN=m +# CONFIG_USB_SERIAL_KEYSPAN_MPR is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XA is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XB is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA18X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19QW is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19QI is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49WLC is not set +CONFIG_USB_SERIAL_KLSI=m +CONFIG_USB_SERIAL_KOBIL_SCT=m +CONFIG_USB_SERIAL_MCT_U232=m +CONFIG_USB_SERIAL_PL2303=m +# CONFIG_USB_SERIAL_HP4X is not set +CONFIG_USB_SERIAL_SAFE=m +# CONFIG_USB_SERIAL_SAFE_PADDED is not set +CONFIG_USB_SERIAL_TI=m +CONFIG_USB_SERIAL_CYBERJACK=m +CONFIG_USB_SERIAL_XIRCOM=m +# CONFIG_USB_SERIAL_OPTION is not set +CONFIG_USB_SERIAL_OMNINET=m +CONFIG_USB_EZUSB=y + +# +# USB Miscellaneous drivers +# +CONFIG_USB_EMI62=m +CONFIG_USB_EMI26=m +CONFIG_USB_AUERSWALD=m +CONFIG_USB_RIO500=m +CONFIG_USB_LEGOTOWER=m +CONFIG_USB_LCD=m +CONFIG_USB_LED=m +CONFIG_USB_CYTHERM=m +CONFIG_USB_PHIDGETKIT=m +CONFIG_USB_PHIDGETSERVO=m +CONFIG_USB_IDMOUSE=m +# CONFIG_USB_LD is not set +# CONFIG_USB_TEST is not set + +# +# USB DSL modem support +# + +# +# USB Gadget Support +# +CONFIG_USB_GADGET=m +# CONFIG_USB_GADGET_DEBUG_FILES is not set +CONFIG_USB_GADGET_SELECTED=y +# CONFIG_USB_GADGET_NET2280 is not set +# CONFIG_USB_GADGET_PXA2XX is not set +CONFIG_USB_GADGET_PXA27X=y +CONFIG_USB_PXA27X=m +# CONFIG_USB_GADGET_GOKU is not set +# CONFIG_USB_GADGET_LH7A40X is not set +# CONFIG_USB_GADGET_OMAP is not set +# CONFIG_USB_GADGET_DUMMY_HCD is not set +# CONFIG_USB_GADGET_DUALSPEED is not set +CONFIG_USB_ZERO=m +CONFIG_USB_ETH=m +CONFIG_USB_ETH_RNDIS=y +CONFIG_USB_GADGETFS=m +CONFIG_USB_FILE_STORAGE=m +# CONFIG_USB_FILE_STORAGE_TEST is not set +CONFIG_USB_G_SERIAL=m + +# +# MMC/SD Card support +# +CONFIG_MMC=y +# CONFIG_MMC_DEBUG is not set +CONFIG_MMC_BLOCK=y +CONFIG_MMC_PXA=y + +# +# Real Time Clock +# +CONFIG_RTC_CLASS=y +CONFIG_RTC_HCTOSYS=y +CONFIG_RTC_HCTOSYS_DEVICE="rtc0" + +# +# RTC interfaces +# +CONFIG_RTC_INTF_SYSFS=y +CONFIG_RTC_INTF_PROC=y +CONFIG_RTC_INTF_DEV=y + +# +# RTC drivers +# +# CONFIG_RTC_DRV_X1205 is not set +# CONFIG_RTC_DRV_DS1672 is not set +# CONFIG_RTC_DRV_PCF8563 is not set +# CONFIG_RTC_DRV_RS5C372 is not set +CONFIG_RTC_DRV_SA1100=y +# CONFIG_RTC_DRV_TEST is not set + +# +# File systems +# +CONFIG_EXT2_FS=y +# CONFIG_EXT2_FS_XATTR is not set +# CONFIG_EXT2_FS_XIP is not set +CONFIG_EXT3_FS=m +# CONFIG_EXT3_FS_XATTR is not set +CONFIG_JBD=y +# CONFIG_JBD_DEBUG is not set +# CONFIG_REISERFS_FS is not set +# CONFIG_JFS_FS is not set +# CONFIG_FS_POSIX_ACL is not set +# CONFIG_XFS_FS is not set +# CONFIG_OCFS2_FS is not set +# CONFIG_MINIX_FS is not set +# CONFIG_ROMFS_FS is not set +CONFIG_INOTIFY=y +# CONFIG_QUOTA is not set +CONFIG_DNOTIFY=y +# CONFIG_AUTOFS_FS is not set +# CONFIG_AUTOFS4_FS is not set +# CONFIG_FUSE_FS is not set + +# +# CD-ROM/DVD Filesystems +# +# CONFIG_ISO9660_FS is not set +# CONFIG_UDF_FS is not set + +# +# DOS/FAT/NT Filesystems +# +CONFIG_FAT_FS=y +# CONFIG_MSDOS_FS is not set +CONFIG_VFAT_FS=y +CONFIG_FAT_DEFAULT_CODEPAGE=437 +CONFIG_FAT_DEFAULT_IOCHARSET="iso8859-1" +# CONFIG_NTFS_FS is not set + +# +# Pseudo filesystems +# +CONFIG_PROC_FS=y +CONFIG_SYSFS=y +CONFIG_TMPFS=y +# CONFIG_HUGETLB_PAGE is not set +CONFIG_RAMFS=y +# CONFIG_RELAYFS_FS is not set +# CONFIG_CONFIGFS_FS is not set + +# +# Miscellaneous filesystems +# +# CONFIG_ADFS_FS is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_HFSPLUS_FS is not set +# CONFIG_BEFS_FS is not set +# CONFIG_BFS_FS is not set +# CONFIG_EFS_FS is not set +# CONFIG_JFFS_FS is not set +CONFIG_JFFS2_FS=y +CONFIG_JFFS2_FS_DEBUG=0 +CONFIG_JFFS2_FS_WRITEBUFFER=y +CONFIG_JFFS2_SUMMARY=y +CONFIG_JFFS2_COMPRESSION_OPTIONS=y +CONFIG_JFFS2_ZLIB=y +CONFIG_JFFS2_RTIME=y +CONFIG_JFFS2_RUBIN=y +# CONFIG_JFFS2_CMODE_NONE is not set +CONFIG_JFFS2_CMODE_PRIORITY=y +# CONFIG_JFFS2_CMODE_SIZE is not set +CONFIG_CRAMFS=m +CONFIG_SQUASHFS=m +# CONFIG_SQUASHFS_EMBEDDED is not set +CONFIG_SQUASHFS_FRAGMENT_CACHE_SIZE=3 +# CONFIG_SQUASHFS_VMALLOC is not set +# CONFIG_VXFS_FS is not set +# CONFIG_HPFS_FS is not set +# CONFIG_QNX4FS_FS is not set +# CONFIG_SYSV_FS is not set +# CONFIG_UFS_FS is not set + +# +# Network File Systems +# +CONFIG_NFS_FS=m +CONFIG_NFS_V3=y +# CONFIG_NFS_V3_ACL is not set +CONFIG_NFS_V4=y +# CONFIG_NFS_DIRECTIO is not set +# CONFIG_NFSD is not set +CONFIG_LOCKD=m +CONFIG_LOCKD_V4=y +CONFIG_NFS_COMMON=y +CONFIG_SUNRPC=m +CONFIG_SUNRPC_GSS=m +CONFIG_RPCSEC_GSS_KRB5=m +# CONFIG_RPCSEC_GSS_SPKM3 is not set +CONFIG_SMB_FS=m +CONFIG_SMB_NLS_DEFAULT=y +CONFIG_SMB_NLS_REMOTE="cp437" +CONFIG_CIFS=m +# CONFIG_CIFS_STATS is not set +# CONFIG_CIFS_XATTR is not set +# CONFIG_CIFS_EXPERIMENTAL is not set +# CONFIG_NCP_FS is not set +# CONFIG_CODA_FS is not set +# CONFIG_AFS_FS is not set +# CONFIG_9P_FS is not set + +# +# Partition Types +# +CONFIG_PARTITION_ADVANCED=y +# CONFIG_ACORN_PARTITION is not set +# CONFIG_OSF_PARTITION is not set +# CONFIG_AMIGA_PARTITION is not set +# CONFIG_ATARI_PARTITION is not set +# CONFIG_MAC_PARTITION is not set +CONFIG_MSDOS_PARTITION=y +# CONFIG_BSD_DISKLABEL is not set +# CONFIG_MINIX_SUBPARTITION is not set +# CONFIG_SOLARIS_X86_PARTITION is not set +# CONFIG_UNIXWARE_DISKLABEL is not set +# CONFIG_LDM_PARTITION is not set +# CONFIG_SGI_PARTITION is not set +# CONFIG_ULTRIX_PARTITION is not set +# CONFIG_SUN_PARTITION is not set +# CONFIG_KARMA_PARTITION is not set +# CONFIG_EFI_PARTITION is not set + +# +# Native Language Support +# +CONFIG_NLS=y +CONFIG_NLS_DEFAULT="cp437" +CONFIG_NLS_CODEPAGE_437=y +CONFIG_NLS_CODEPAGE_737=m +CONFIG_NLS_CODEPAGE_775=m +CONFIG_NLS_CODEPAGE_850=m +CONFIG_NLS_CODEPAGE_852=m +CONFIG_NLS_CODEPAGE_855=m +CONFIG_NLS_CODEPAGE_857=m +CONFIG_NLS_CODEPAGE_860=m +CONFIG_NLS_CODEPAGE_861=m +CONFIG_NLS_CODEPAGE_862=m +CONFIG_NLS_CODEPAGE_863=m +CONFIG_NLS_CODEPAGE_864=m +CONFIG_NLS_CODEPAGE_865=m +CONFIG_NLS_CODEPAGE_866=m +CONFIG_NLS_CODEPAGE_869=m +CONFIG_NLS_CODEPAGE_936=m +CONFIG_NLS_CODEPAGE_950=m +CONFIG_NLS_CODEPAGE_932=m +CONFIG_NLS_CODEPAGE_949=m +CONFIG_NLS_CODEPAGE_874=m +CONFIG_NLS_ISO8859_8=m +CONFIG_NLS_CODEPAGE_1250=m +CONFIG_NLS_CODEPAGE_1251=m +CONFIG_NLS_ASCII=m +CONFIG_NLS_ISO8859_1=y +CONFIG_NLS_ISO8859_2=m +CONFIG_NLS_ISO8859_3=m +CONFIG_NLS_ISO8859_4=m +CONFIG_NLS_ISO8859_5=m +CONFIG_NLS_ISO8859_6=m +CONFIG_NLS_ISO8859_7=m +CONFIG_NLS_ISO8859_9=m +CONFIG_NLS_ISO8859_13=m +CONFIG_NLS_ISO8859_14=m +CONFIG_NLS_ISO8859_15=m +CONFIG_NLS_KOI8_R=m +CONFIG_NLS_KOI8_U=m +CONFIG_NLS_UTF8=y + +# +# Profiling support +# +CONFIG_PROFILING=y +CONFIG_OPROFILE=m + +# +# Kernel hacking +# +# CONFIG_PRINTK_TIME is not set +CONFIG_MAGIC_SYSRQ=y +CONFIG_DEBUG_KERNEL=y +CONFIG_LOG_BUF_SHIFT=14 +# CONFIG_DETECT_SOFTLOCKUP is not set +# CONFIG_SCHEDSTATS is not set +# CONFIG_DEBUG_SLAB is not set +# CONFIG_DEBUG_PREEMPT is not set +# CONFIG_DEBUG_MUTEXES is not set +# CONFIG_DEBUG_SPINLOCK is not set +# CONFIG_DEBUG_SPINLOCK_SLEEP is not set +# CONFIG_DEBUG_KOBJECT is not set +CONFIG_DEBUG_BUGVERBOSE=y +# CONFIG_DEBUG_INFO is not set +# CONFIG_DEBUG_FS is not set +# CONFIG_DEBUG_VM is not set +CONFIG_FRAME_POINTER=y +# CONFIG_FORCED_INLINING is not set +# CONFIG_RCU_TORTURE_TEST is not set +# CONFIG_DEBUG_USER is not set +# CONFIG_DEBUG_WAITQ is not set +CONFIG_DEBUG_ERRORS=y +# CONFIG_DEBUG_LL is not set + +# +# Security options +# +# CONFIG_KEYS is not set +# CONFIG_SECURITY is not set + +# +# Cryptographic options +# +CONFIG_CRYPTO=y +CONFIG_CRYPTO_HMAC=y +CONFIG_CRYPTO_NULL=m +CONFIG_CRYPTO_MD4=m +CONFIG_CRYPTO_MD5=m +CONFIG_CRYPTO_SHA1=m +CONFIG_CRYPTO_SHA256=m +CONFIG_CRYPTO_SHA512=m +CONFIG_CRYPTO_WP512=m +# CONFIG_CRYPTO_TGR192 is not set +CONFIG_CRYPTO_DES=m +CONFIG_CRYPTO_BLOWFISH=m +CONFIG_CRYPTO_TWOFISH=m +CONFIG_CRYPTO_SERPENT=m +CONFIG_CRYPTO_AES=m +CONFIG_CRYPTO_CAST5=m +CONFIG_CRYPTO_CAST6=m +CONFIG_CRYPTO_TEA=m +CONFIG_CRYPTO_ARC4=m +CONFIG_CRYPTO_KHAZAD=m +CONFIG_CRYPTO_ANUBIS=m +CONFIG_CRYPTO_DEFLATE=m +CONFIG_CRYPTO_MICHAEL_MIC=m +CONFIG_CRYPTO_CRC32C=m +CONFIG_CRYPTO_TEST=m + +# +# Hardware crypto devices +# + +# +# Library routines +# +CONFIG_CRC_CCITT=y +# CONFIG_CRC16 is not set +CONFIG_CRC32=y +CONFIG_LIBCRC32C=m +CONFIG_ZLIB_INFLATE=y +CONFIG_ZLIB_DEFLATE=y diff --git a/packages/linux/linux-rp-2.6.20/defconfig-c7x0 b/packages/linux/linux-rp-2.6.20/defconfig-c7x0 new file mode 100644 index 0000000000..1d702f60c0 --- /dev/null +++ b/packages/linux/linux-rp-2.6.20/defconfig-c7x0 @@ -0,0 +1,1608 @@ +# +# Automatically generated make config: don't edit +# Linux kernel version: 2.6.16 +# Mon May 22 09:00:01 2006 +# +CONFIG_ARM=y +CONFIG_MMU=y +CONFIG_RWSEM_GENERIC_SPINLOCK=y +CONFIG_GENERIC_CALIBRATE_DELAY=y +CONFIG_ARCH_MTD_XIP=y + +# +# Code maturity level options +# +CONFIG_EXPERIMENTAL=y +CONFIG_BROKEN_ON_SMP=y +CONFIG_LOCK_KERNEL=y +CONFIG_INIT_ENV_ARG_LIMIT=32 + +# +# General setup +# +CONFIG_LOCALVERSION="" +CONFIG_LOCALVERSION_AUTO=y +CONFIG_SWAP=y +CONFIG_SYSVIPC=y +# CONFIG_POSIX_MQUEUE is not set +CONFIG_BSD_PROCESS_ACCT=y +# CONFIG_BSD_PROCESS_ACCT_V3 is not set +CONFIG_SYSCTL=y +# CONFIG_AUDIT is not set +# CONFIG_IKCONFIG is not set +CONFIG_INITRAMFS_SOURCE="" +CONFIG_UID16=y +CONFIG_CC_OPTIMIZE_FOR_SIZE=y +CONFIG_EMBEDDED=y +CONFIG_KALLSYMS=y +# CONFIG_KALLSYMS_ALL is not set +# CONFIG_KALLSYMS_EXTRA_PASS is not set +CONFIG_HOTPLUG=y +CONFIG_PRINTK=y +CONFIG_BUG=y +CONFIG_ELF_CORE=y +CONFIG_BASE_FULL=y +CONFIG_FUTEX=y +CONFIG_EPOLL=y +CONFIG_SHMEM=y +CONFIG_CC_ALIGN_FUNCTIONS=0 +CONFIG_CC_ALIGN_LABELS=0 +CONFIG_CC_ALIGN_LOOPS=0 +CONFIG_CC_ALIGN_JUMPS=0 +CONFIG_SLAB=y +# CONFIG_TINY_SHMEM is not set +CONFIG_BASE_SMALL=0 +# CONFIG_SLOB is not set + +# +# Loadable module support +# +CONFIG_MODULES=y +CONFIG_MODULE_UNLOAD=y +CONFIG_MODULE_FORCE_UNLOAD=y +CONFIG_OBSOLETE_MODPARM=y +# CONFIG_MODVERSIONS is not set +# CONFIG_MODULE_SRCVERSION_ALL is not set +CONFIG_KMOD=y + +# +# Block layer +# + +# +# IO Schedulers +# +CONFIG_IOSCHED_NOOP=y +CONFIG_IOSCHED_AS=y +CONFIG_IOSCHED_DEADLINE=m +CONFIG_IOSCHED_CFQ=m +CONFIG_DEFAULT_AS=y +# CONFIG_DEFAULT_DEADLINE is not set +# CONFIG_DEFAULT_CFQ is not set +# CONFIG_DEFAULT_NOOP is not set +CONFIG_DEFAULT_IOSCHED="anticipatory" + +# +# System Type +# +# CONFIG_ARCH_CLPS7500 is not set +# CONFIG_ARCH_CLPS711X is not set +# CONFIG_ARCH_CO285 is not set +# CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_FOOTBRIDGE is not set +# CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_IOP3XX is not set +# CONFIG_ARCH_IXP4XX is not set +# CONFIG_ARCH_IXP2000 is not set +# CONFIG_ARCH_L7200 is not set +CONFIG_ARCH_PXA=y +# CONFIG_ARCH_RPC is not set +# CONFIG_ARCH_SA1100 is not set +# CONFIG_ARCH_S3C2410 is not set +# CONFIG_ARCH_SHARK is not set +# CONFIG_ARCH_LH7A40X is not set +# CONFIG_ARCH_OMAP is not set +# CONFIG_ARCH_VERSATILE is not set +# CONFIG_ARCH_REALVIEW is not set +# CONFIG_ARCH_IMX is not set +# CONFIG_ARCH_H720X is not set +# CONFIG_ARCH_AAEC2000 is not set +# CONFIG_ARCH_AT91RM9200 is not set + +# +# Intel PXA2xx Implementations +# +# CONFIG_ARCH_LUBBOCK is not set +# CONFIG_MACH_MAINSTONE is not set +# CONFIG_ARCH_PXA_IDP is not set +CONFIG_PXA_SHARPSL=y +# CONFIG_MACH_HX2750 is not set +CONFIG_PXA_SHARPSL_25x=y +# CONFIG_PXA_SHARPSL_27x is not set +# CONFIG_MACH_POODLE is not set +CONFIG_MACH_CORGI=y +CONFIG_MACH_SHEPHERD=y +CONFIG_MACH_HUSKY=y +# CONFIG_MACH_TOSA is not set +CONFIG_PXA25x=y +# CONFIG_PXA_KEYS is not set +CONFIG_PXA_SHARP_C7xx=y +CONFIG_PXA_SSP=y + +# +# Processor Type +# +CONFIG_CPU_32=y +CONFIG_CPU_XSCALE=y +CONFIG_CPU_32v5=y +CONFIG_CPU_ABRT_EV5T=y +CONFIG_CPU_CACHE_VIVT=y +CONFIG_CPU_TLB_V4WBI=y + +# +# Processor Features +# +CONFIG_ARM_THUMB=y +CONFIG_XSCALE_PMU=y +CONFIG_KEXEC=y +CONFIG_SHARP_PARAM=y +CONFIG_SHARPSL_PM=y +CONFIG_SHARP_SCOOP=y + +# +# Bus support +# + +# +# PCCARD (PCMCIA/CardBus) support +# +CONFIG_PCCARD=y +# CONFIG_PCMCIA_DEBUG is not set +CONFIG_PCMCIA=y +CONFIG_PCMCIA_LOAD_CIS=y +CONFIG_PCMCIA_IOCTL=y + +# +# PC-card bridges +# +CONFIG_PCMCIA_PXA2XX=y + +# +# Kernel Features +# +CONFIG_PREEMPT=y +CONFIG_NO_IDLE_HZ=y +# CONFIG_AEABI is not set +# CONFIG_ARCH_DISCONTIGMEM_ENABLE is not set +CONFIG_SELECT_MEMORY_MODEL=y +CONFIG_FLATMEM_MANUAL=y +# CONFIG_DISCONTIGMEM_MANUAL is not set +# CONFIG_SPARSEMEM_MANUAL is not set +CONFIG_FLATMEM=y +CONFIG_FLAT_NODE_MEM_MAP=y +# CONFIG_SPARSEMEM_STATIC is not set +CONFIG_SPLIT_PTLOCK_CPUS=4096 +CONFIG_ALIGNMENT_TRAP=y + +# +# Boot options +# +CONFIG_ZBOOT_ROM_TEXT=0x0 +CONFIG_ZBOOT_ROM_BSS=0x0 +# CONFIG_XIP_KERNEL is not set + +# +# CPU Frequency scaling +# +CONFIG_CPU_FREQ=y +CONFIG_CPU_FREQ_TABLE=y +CONFIG_CPU_FREQ_DEBUG=y +CONFIG_CPU_FREQ_STAT=y +# CONFIG_CPU_FREQ_STAT_DETAILS is not set +CONFIG_CPU_FREQ_DEFAULT_GOV_PERFORMANCE=y +# CONFIG_CPU_FREQ_DEFAULT_GOV_USERSPACE is not set +CONFIG_CPU_FREQ_GOV_PERFORMANCE=y +CONFIG_CPU_FREQ_GOV_POWERSAVE=y +CONFIG_CPU_FREQ_GOV_USERSPACE=y +CONFIG_CPU_FREQ_GOV_ONDEMAND=y +CONFIG_CPU_FREQ_GOV_CONSERVATIVE=y +CONFIG_CPU_FREQ_PXA25x=y + +# +# Floating point emulation +# + +# +# At least one emulation must be selected +# +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_NWFPE_XP is not set +# CONFIG_FPE_FASTFPE is not set + +# +# Userspace binary formats +# +CONFIG_BINFMT_ELF=y +CONFIG_BINFMT_AOUT=m +CONFIG_BINFMT_MISC=m +# CONFIG_ARTHUR is not set + +# +# Power management options +# +CONFIG_PM=y +# CONFIG_PM_LEGACY is not set +# CONFIG_PM_DEBUG is not set +CONFIG_APM=y + +# +# Networking +# +CONFIG_NET=y + +# +# Networking options +# +# CONFIG_NETDEBUG is not set +CONFIG_PACKET=y +CONFIG_PACKET_MMAP=y +CONFIG_UNIX=y +CONFIG_XFRM=y +CONFIG_XFRM_USER=m +# CONFIG_NET_KEY is not set +CONFIG_INET=y +# CONFIG_IP_MULTICAST is not set +# CONFIG_IP_ADVANCED_ROUTER is not set +CONFIG_IP_FIB_HASH=y +# CONFIG_IP_PNP is not set +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set +CONFIG_SYN_COOKIES=y +# CONFIG_INET_AH is not set +# CONFIG_INET_ESP is not set +# CONFIG_INET_IPCOMP is not set +# CONFIG_INET_TUNNEL is not set +CONFIG_INET_DIAG=m +CONFIG_INET_TCP_DIAG=m +# CONFIG_TCP_CONG_ADVANCED is not set +CONFIG_TCP_CONG_BIC=y + +# +# IP: Virtual Server Configuration +# +# CONFIG_IP_VS is not set +CONFIG_IPV6=m +# CONFIG_IPV6_PRIVACY is not set +CONFIG_INET6_AH=m +CONFIG_INET6_ESP=m +CONFIG_INET6_IPCOMP=m +CONFIG_INET6_TUNNEL=m +CONFIG_IPV6_TUNNEL=m +CONFIG_NETFILTER=y +# CONFIG_NETFILTER_DEBUG is not set + +# +# Core Netfilter Configuration +# +# CONFIG_NETFILTER_NETLINK is not set +# CONFIG_NETFILTER_XTABLES is not set + +# +# IP: Netfilter Configuration +# +CONFIG_IP_NF_CONNTRACK=m +# CONFIG_IP_NF_CT_ACCT is not set +# CONFIG_IP_NF_CONNTRACK_MARK is not set +# CONFIG_IP_NF_CONNTRACK_EVENTS is not set +CONFIG_IP_NF_CT_PROTO_SCTP=m +CONFIG_IP_NF_FTP=m +CONFIG_IP_NF_IRC=m +# CONFIG_IP_NF_NETBIOS_NS is not set +CONFIG_IP_NF_TFTP=m +CONFIG_IP_NF_AMANDA=m +# CONFIG_IP_NF_PPTP is not set +CONFIG_IP_NF_QUEUE=m + +# +# IPv6: Netfilter Configuration (EXPERIMENTAL) +# +# CONFIG_IP6_NF_QUEUE is not set + +# +# DCCP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_DCCP is not set + +# +# SCTP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_SCTP is not set + +# +# TIPC Configuration (EXPERIMENTAL) +# +# CONFIG_TIPC is not set +# CONFIG_ATM is not set +# CONFIG_BRIDGE is not set +# CONFIG_VLAN_8021Q is not set +# CONFIG_DECNET is not set +# CONFIG_LLC2 is not set +# CONFIG_IPX is not set +# CONFIG_ATALK is not set +# CONFIG_X25 is not set +# CONFIG_LAPB is not set +# CONFIG_NET_DIVERT is not set +# CONFIG_ECONET is not set +# CONFIG_WAN_ROUTER is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network testing +# +# CONFIG_NET_PKTGEN is not set +# CONFIG_HAMRADIO is not set +CONFIG_IRDA=m + +# +# IrDA protocols +# +CONFIG_IRLAN=m +CONFIG_IRNET=m +CONFIG_IRCOMM=m +# CONFIG_IRDA_ULTRA is not set + +# +# IrDA options +# +# CONFIG_IRDA_CACHE_LAST_LSAP is not set +# CONFIG_IRDA_FAST_RR is not set +# CONFIG_IRDA_DEBUG is not set + +# +# Infrared-port device drivers +# + +# +# SIR device drivers +# +# CONFIG_IRTTY_SIR is not set + +# +# Dongle support +# + +# +# Old SIR device drivers +# +# CONFIG_IRPORT_SIR is not set + +# +# Old Serial dongle support +# + +# +# FIR device drivers +# +# CONFIG_USB_IRDA is not set +# CONFIG_SIGMATEL_FIR is not set +CONFIG_PXA_FICP=m +CONFIG_BT=m +CONFIG_BT_L2CAP=m +CONFIG_BT_SCO=m +CONFIG_BT_RFCOMM=m +CONFIG_BT_RFCOMM_TTY=y +CONFIG_BT_BNEP=m +CONFIG_BT_BNEP_MC_FILTER=y +CONFIG_BT_BNEP_PROTO_FILTER=y +CONFIG_BT_HIDP=m + +# +# Bluetooth device drivers +# +CONFIG_BT_HCIUSB=m +# CONFIG_BT_HCIUSB_SCO is not set +CONFIG_BT_HCIUART=m +CONFIG_BT_HCIUART_H4=y +CONFIG_BT_HCIUART_BCSP=y +CONFIG_BT_HCIBCM203X=m +CONFIG_BT_HCIBPA10X=m +CONFIG_BT_HCIBFUSB=m +CONFIG_BT_HCIDTL1=m +CONFIG_BT_HCIBT3C=m +CONFIG_BT_HCIBLUECARD=m +CONFIG_BT_HCIBTUART=m +CONFIG_BT_HCIVHCI=m +CONFIG_IEEE80211=m +# CONFIG_IEEE80211_DEBUG is not set +CONFIG_IEEE80211_CRYPT_WEP=m +CONFIG_IEEE80211_CRYPT_CCMP=m +CONFIG_IEEE80211_CRYPT_TKIP=m + +# +# Device Drivers +# + +# +# Generic Driver Options +# +CONFIG_STANDALONE=y +CONFIG_PREVENT_FIRMWARE_BUILD=y +CONFIG_FW_LOADER=y +# CONFIG_DEBUG_DRIVER is not set + +# +# Connector - unified userspace <-> kernelspace linker +# +# CONFIG_CONNECTOR is not set + +# +# Memory Technology Devices (MTD) +# +CONFIG_MTD=y +# CONFIG_MTD_DEBUG is not set +# CONFIG_MTD_CONCAT is not set +CONFIG_MTD_PARTITIONS=y +# CONFIG_MTD_REDBOOT_PARTS is not set +CONFIG_MTD_CMDLINE_PARTS=y +# CONFIG_MTD_AFS_PARTS is not set + +# +# User Modules And Translation Layers +# +CONFIG_MTD_CHAR=y +CONFIG_MTD_BLOCK=y +# CONFIG_FTL is not set +# CONFIG_NFTL is not set +# CONFIG_INFTL is not set +# CONFIG_RFD_FTL is not set + +# +# RAM/ROM/Flash chip drivers +# +# CONFIG_MTD_CFI is not set +# CONFIG_MTD_JEDECPROBE is not set +CONFIG_MTD_MAP_BANK_WIDTH_1=y +CONFIG_MTD_MAP_BANK_WIDTH_2=y +CONFIG_MTD_MAP_BANK_WIDTH_4=y +# CONFIG_MTD_MAP_BANK_WIDTH_8 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_16 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_32 is not set +CONFIG_MTD_CFI_I1=y +CONFIG_MTD_CFI_I2=y +# CONFIG_MTD_CFI_I4 is not set +# CONFIG_MTD_CFI_I8 is not set +# CONFIG_MTD_RAM is not set +CONFIG_MTD_ROM=y +# CONFIG_MTD_ABSENT is not set +# CONFIG_MTD_OBSOLETE_CHIPS is not set + +# +# Mapping drivers for chip access +# +CONFIG_MTD_COMPLEX_MAPPINGS=y +CONFIG_MTD_SHARP_SL=y +# CONFIG_MTD_PLATRAM is not set + +# +# Self-contained MTD device drivers +# +# CONFIG_MTD_SLRAM is not set +# CONFIG_MTD_PHRAM is not set +# CONFIG_MTD_MTDRAM is not set +# CONFIG_MTD_BLKMTD is not set +# CONFIG_MTD_BLOCK2MTD is not set + +# +# Disk-On-Chip Device Drivers +# +# CONFIG_MTD_DOC2000 is not set +# CONFIG_MTD_DOC2001 is not set +# CONFIG_MTD_DOC2001PLUS is not set + +# +# NAND Flash Device Drivers +# +CONFIG_MTD_NAND=y +CONFIG_MTD_NAND_VERIFY_WRITE=y +# CONFIG_MTD_NAND_H1900 is not set +CONFIG_MTD_NAND_IDS=y +# CONFIG_MTD_NAND_DISKONCHIP is not set +CONFIG_MTD_NAND_SHARPSL=y +# CONFIG_MTD_NAND_NANDSIM is not set + +# +# OneNAND Flash Device Drivers +# +# CONFIG_MTD_ONENAND is not set + +# +# Parallel port support +# +# CONFIG_PARPORT is not set + +# +# Plug and Play support +# + +# +# Block devices +# +# CONFIG_BLK_DEV_COW_COMMON is not set +CONFIG_BLK_DEV_LOOP=y +# CONFIG_BLK_DEV_CRYPTOLOOP is not set +# CONFIG_BLK_DEV_NBD is not set +# CONFIG_BLK_DEV_UB is not set +# CONFIG_BLK_DEV_RAM is not set +CONFIG_BLK_DEV_RAM_COUNT=16 +# CONFIG_CDROM_PKTCDVD is not set +# CONFIG_ATA_OVER_ETH is not set + +# +# ATA/ATAPI/MFM/RLL support +# +CONFIG_IDE=y +CONFIG_BLK_DEV_IDE=y + +# +# Please see Documentation/ide.txt for help/info on IDE drives +# +# CONFIG_BLK_DEV_IDE_SATA is not set +CONFIG_BLK_DEV_IDEDISK=y +# CONFIG_IDEDISK_MULTI_MODE is not set +CONFIG_BLK_DEV_IDECS=y +# CONFIG_BLK_DEV_IDECD is not set +# CONFIG_BLK_DEV_IDETAPE is not set +# CONFIG_BLK_DEV_IDEFLOPPY is not set +# CONFIG_BLK_DEV_IDESCSI is not set +# CONFIG_IDE_TASK_IOCTL is not set + +# +# IDE chipset support/bugfixes +# +CONFIG_IDE_GENERIC=y +# CONFIG_IDE_ARM is not set +# CONFIG_BLK_DEV_IDEDMA is not set +# CONFIG_IDEDMA_AUTO is not set +# CONFIG_BLK_DEV_HD is not set + +# +# SCSI device support +# +# CONFIG_RAID_ATTRS is not set +CONFIG_SCSI=m +CONFIG_SCSI_PROC_FS=y + +# +# SCSI support type (disk, tape, CD-ROM) +# +CONFIG_BLK_DEV_SD=m +CONFIG_CHR_DEV_ST=m +CONFIG_CHR_DEV_OSST=m +CONFIG_BLK_DEV_SR=m +# CONFIG_BLK_DEV_SR_VENDOR is not set +CONFIG_CHR_DEV_SG=m +# CONFIG_CHR_DEV_SCH is not set + +# +# Some SCSI devices (e.g. CD jukebox) support multiple LUNs +# +CONFIG_SCSI_MULTI_LUN=y +# CONFIG_SCSI_CONSTANTS is not set +# CONFIG_SCSI_LOGGING is not set + +# +# SCSI Transport Attributes +# +# CONFIG_SCSI_SPI_ATTRS is not set +# CONFIG_SCSI_FC_ATTRS is not set +# CONFIG_SCSI_ISCSI_ATTRS is not set +# CONFIG_SCSI_SAS_ATTRS is not set + +# +# SCSI low-level drivers +# +# CONFIG_ISCSI_TCP is not set +# CONFIG_SCSI_SATA is not set +# CONFIG_SCSI_DEBUG is not set + +# +# PCMCIA SCSI adapter support +# +# CONFIG_PCMCIA_AHA152X is not set +# CONFIG_PCMCIA_FDOMAIN is not set +# CONFIG_PCMCIA_NINJA_SCSI is not set +# CONFIG_PCMCIA_QLOGIC is not set +# CONFIG_PCMCIA_SYM53C500 is not set + +# +# Multi-device support (RAID and LVM) +# +CONFIG_MD=y +CONFIG_BLK_DEV_DM=m +CONFIG_DM_CRYPT=m +CONFIG_DM_SNAPSHOT=m +CONFIG_DM_MIRROR=m +CONFIG_DM_ZERO=m +CONFIG_DM_MULTIPATH=m +CONFIG_DM_MULTIPATH_EMC=m + +# +# Fusion MPT device support +# +# CONFIG_FUSION is not set + +# +# IEEE 1394 (FireWire) support +# + +# +# I2O device support +# + +# +# Network device support +# +CONFIG_NETDEVICES=y +# CONFIG_DUMMY is not set +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +CONFIG_TUN=m + +# +# PHY device support +# +# CONFIG_PHYLIB is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +CONFIG_MII=m +# CONFIG_SMC91X is not set +# CONFIG_DM9000 is not set + +# +# Ethernet (1000 Mbit) +# + +# +# Ethernet (10000 Mbit) +# + +# +# Token Ring devices +# + +# +# Wireless LAN (non-hamradio) +# +CONFIG_NET_RADIO=y + +# +# Obsolete Wireless cards support (pre-802.11) +# +# CONFIG_STRIP is not set +# CONFIG_PCMCIA_WAVELAN is not set +# CONFIG_PCMCIA_NETWAVE is not set + +# +# Wireless 802.11 Frequency Hopping cards support +# +# CONFIG_PCMCIA_RAYCS is not set + +# +# Wireless 802.11b ISA/PCI cards support +# +CONFIG_HERMES=m +# CONFIG_ATMEL is not set + +# +# Wireless 802.11b Pcmcia/Cardbus cards support +# +CONFIG_PCMCIA_HERMES=m +CONFIG_PCMCIA_SPECTRUM=m +# CONFIG_AIRO_CS is not set +# CONFIG_PCMCIA_WL3501 is not set +CONFIG_HOSTAP=m +CONFIG_HOSTAP_FIRMWARE=y +# CONFIG_HOSTAP_FIRMWARE_NVRAM is not set +CONFIG_HOSTAP_CS=m +CONFIG_NET_WIRELESS=y + +# +# PCMCIA network device support +# +CONFIG_NET_PCMCIA=y +# CONFIG_PCMCIA_3C589 is not set +# CONFIG_PCMCIA_3C574 is not set +# CONFIG_PCMCIA_FMVJ18X is not set +CONFIG_PCMCIA_PCNET=m +# CONFIG_PCMCIA_NMCLAN is not set +# CONFIG_PCMCIA_SMC91C92 is not set +# CONFIG_PCMCIA_XIRC2PS is not set +# CONFIG_PCMCIA_AXNET is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set +CONFIG_PPP=m +# CONFIG_PPP_MULTILINK is not set +# CONFIG_PPP_FILTER is not set +CONFIG_PPP_ASYNC=m +# CONFIG_PPP_SYNC_TTY is not set +CONFIG_PPP_DEFLATE=m +CONFIG_PPP_BSDCOMP=m +# CONFIG_PPP_MPPE is not set +# CONFIG_PPPOE is not set +# CONFIG_SLIP is not set +# CONFIG_SHAPER is not set +# CONFIG_NETCONSOLE is not set +# CONFIG_NETPOLL is not set +# CONFIG_NET_POLL_CONTROLLER is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Input device support +# +CONFIG_INPUT=y + +# +# Userland interfaces +# +CONFIG_INPUT_MOUSEDEV=m +# CONFIG_INPUT_MOUSEDEV_PSAUX is not set +CONFIG_INPUT_MOUSEDEV_SCREEN_X=640 +CONFIG_INPUT_MOUSEDEV_SCREEN_Y=480 +# CONFIG_INPUT_JOYDEV is not set +# CONFIG_INPUT_TSDEV is not set +CONFIG_INPUT_EVDEV=y +# CONFIG_INPUT_EVBUG is not set +CONFIG_INPUT_POWER=y + +# +# Input Device Drivers +# +CONFIG_INPUT_KEYBOARD=y +# CONFIG_KEYBOARD_ATKBD is not set +# CONFIG_KEYBOARD_SUNKBD is not set +# CONFIG_KEYBOARD_LKKBD is not set +# CONFIG_KEYBOARD_XTKBD is not set +# CONFIG_KEYBOARD_NEWTON is not set +CONFIG_KEYBOARD_CORGI=y +# CONFIG_KEYBOARD_SPITZ is not set +# CONFIG_INPUT_MOUSE is not set +# CONFIG_INPUT_JOYSTICK is not set +CONFIG_INPUT_TOUCHSCREEN=y +CONFIG_TOUCHSCREEN_CORGI=y +# CONFIG_TOUCHSCREEN_GUNZE is not set +# CONFIG_TOUCHSCREEN_ELO is not set +# CONFIG_TOUCHSCREEN_MTOUCH is not set +# CONFIG_TOUCHSCREEN_MK712 is not set +CONFIG_INPUT_MISC=y +CONFIG_INPUT_UINPUT=m + +# +# Hardware I/O ports +# +# CONFIG_SERIO is not set +# CONFIG_GAMEPORT is not set + +# +# Character devices +# +CONFIG_VT=y +CONFIG_VT_CONSOLE=y +CONFIG_HW_CONSOLE=y +# CONFIG_SERIAL_NONSTANDARD is not set + +# +# Serial drivers +# +CONFIG_SERIAL_8250=m +CONFIG_SERIAL_8250_CS=m +CONFIG_SERIAL_8250_NR_UARTS=4 +CONFIG_SERIAL_8250_RUNTIME_UARTS=4 +# CONFIG_SERIAL_8250_EXTENDED is not set + +# +# Non-8250 serial port support +# +CONFIG_SERIAL_PXA=y +CONFIG_SERIAL_PXA_CONSOLE=y +CONFIG_SERIAL_CORE=y +CONFIG_SERIAL_CORE_CONSOLE=y +CONFIG_UNIX98_PTYS=y +# CONFIG_LEGACY_PTYS is not set + +# +# IPMI +# +# CONFIG_IPMI_HANDLER is not set + +# +# Watchdog Cards +# +# CONFIG_WATCHDOG is not set +# CONFIG_NVRAM is not set +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set + +# +# Ftape, the floppy tape device driver +# + +# +# PCMCIA character devices +# +# CONFIG_SYNCLINK_CS is not set +# CONFIG_CARDMAN_4000 is not set +# CONFIG_CARDMAN_4040 is not set +# CONFIG_RAW_DRIVER is not set + +# +# TPM devices +# +# CONFIG_TCG_TPM is not set +# CONFIG_TELCLOCK is not set + +# +# I2C support +# +CONFIG_I2C=y +# CONFIG_I2C_CHARDEV is not set + +# +# I2C Algorithms +# +CONFIG_I2C_ALGOBIT=y +# CONFIG_I2C_ALGOPCF is not set +# CONFIG_I2C_ALGOPCA is not set + +# +# I2C Hardware Bus support +# +CONFIG_I2C_PXA=y +# CONFIG_I2C_PXA_SLAVE is not set +# CONFIG_I2C_PARPORT_LIGHT is not set +# CONFIG_I2C_STUB is not set +# CONFIG_I2C_PCA_ISA is not set + +# +# Miscellaneous I2C Chip support +# +# CONFIG_SENSORS_DS1337 is not set +# CONFIG_SENSORS_DS1374 is not set +# CONFIG_SENSORS_EEPROM is not set +# CONFIG_SENSORS_PCF8574 is not set +# CONFIG_SENSORS_PCA9539 is not set +# CONFIG_SENSORS_PCF8591 is not set +# CONFIG_SENSORS_MAX6875 is not set +# CONFIG_I2C_DEBUG_CORE is not set +# CONFIG_I2C_DEBUG_ALGO is not set +# CONFIG_I2C_DEBUG_BUS is not set +# CONFIG_I2C_DEBUG_CHIP is not set + +# +# SPI support +# +# CONFIG_SPI is not set +# CONFIG_SPI_MASTER is not set + +# +# Dallas's 1-wire bus +# +# CONFIG_W1 is not set + +# +# Hardware Monitoring support +# +# CONFIG_HWMON is not set +# CONFIG_HWMON_VID is not set + +# +# Misc devices +# + +# +# Multimedia Capabilities Port drivers +# + +# +# Multi-Function Devices +# + +# +# LED devices +# +CONFIG_NEW_LEDS=y +CONFIG_LEDS_CLASS=y +CONFIG_LEDS_TRIGGERS=y +CONFIG_LEDS_CORGI=y +# CONFIG_LEDS_TOSA is not set +CONFIG_LEDS_TRIGGER_TIMER=y +CONFIG_LEDS_TRIGGER_IDE_DISK=y + +# +# Multimedia devices +# +CONFIG_VIDEO_DEV=m + +# +# Video For Linux +# + +# +# Video Adapters +# +# CONFIG_VIDEO_ADV_DEBUG is not set +# CONFIG_VIDEO_CPIA is not set +# CONFIG_VIDEO_SAA5246A is not set +# CONFIG_VIDEO_SAA5249 is not set +# CONFIG_TUNER_3036 is not set +# CONFIG_VIDEO_EM28XX is not set +# CONFIG_VIDEO_OVCAMCHIP is not set +# CONFIG_VIDEO_AUDIO_DECODER is not set +# CONFIG_VIDEO_DECODER is not set + +# +# Radio Adapters +# +# CONFIG_RADIO_MAESTRO is not set + +# +# Digital Video Broadcasting Devices +# +# CONFIG_DVB is not set + +# +# Graphics support +# +CONFIG_FB=y +CONFIG_FB_CFB_FILLRECT=y +CONFIG_FB_CFB_COPYAREA=y +CONFIG_FB_CFB_IMAGEBLIT=y +# CONFIG_FB_MACMODES is not set +# CONFIG_FB_MODE_HELPERS is not set +# CONFIG_FB_TILEBLITTING is not set +# CONFIG_FB_S1D13XXX is not set +# CONFIG_FB_PXA is not set +CONFIG_FB_W100=y +# CONFIG_FB_VIRTUAL is not set + +# +# Console display driver support +# +# CONFIG_VGA_CONSOLE is not set +CONFIG_DUMMY_CONSOLE=y +CONFIG_FRAMEBUFFER_CONSOLE=y +# CONFIG_FRAMEBUFFER_CONSOLE_ROTATION is not set +CONFIG_FONTS=y +# CONFIG_FONT_8x8 is not set +CONFIG_FONT_8x16=y +# CONFIG_FONT_6x11 is not set +# CONFIG_FONT_7x14 is not set +# CONFIG_FONT_PEARL_8x8 is not set +# CONFIG_FONT_ACORN_8x8 is not set +# CONFIG_FONT_MINI_4x6 is not set +# CONFIG_FONT_SUN8x16 is not set +# CONFIG_FONT_SUN12x22 is not set +# CONFIG_FONT_10x18 is not set + +# +# Logo configuration +# +CONFIG_LOGO=y +# CONFIG_LOGO_LINUX_MONO is not set +# CONFIG_LOGO_LINUX_VGA16 is not set +CONFIG_LOGO_LINUX_CLUT224=y +CONFIG_BACKLIGHT_LCD_SUPPORT=y +CONFIG_BACKLIGHT_CLASS_DEVICE=y +CONFIG_BACKLIGHT_DEVICE=y +# CONFIG_LCD_CLASS_DEVICE is not set +CONFIG_BACKLIGHT_CORGI=y +# CONFIG_BACKLIGHT_HP680 is not set + +# +# Sound +# +CONFIG_SOUND=m + +# +# Advanced Linux Sound Architecture +# +CONFIG_SND=m +CONFIG_SND_TIMER=m +CONFIG_SND_PCM=m +CONFIG_SND_SEQUENCER=m +# CONFIG_SND_SEQ_DUMMY is not set +CONFIG_SND_OSSEMUL=y +CONFIG_SND_MIXER_OSS=m +CONFIG_SND_PCM_OSS=m +# CONFIG_SND_SEQUENCER_OSS is not set +# CONFIG_SND_DYNAMIC_MINORS is not set +CONFIG_SND_SUPPORT_OLD_API=y +CONFIG_SND_VERBOSE_PRINTK=y +CONFIG_SND_DEBUG=y +# CONFIG_SND_DEBUG_DETECT is not set + +# +# Generic devices +# +CONFIG_SND_AC97_CODEC=m +CONFIG_SND_AC97_BUS=m +# CONFIG_SND_DUMMY is not set +# CONFIG_SND_VIRMIDI is not set +# CONFIG_SND_MTPAV is not set +# CONFIG_SND_SERIAL_U16550 is not set +# CONFIG_SND_MPU401 is not set + +# +# ALSA ARM devices +# +CONFIG_SND_PXA2XX_PCM=m +CONFIG_SND_PXA2XX_AC97=m + +# +# USB devices +# +CONFIG_SND_USB_AUDIO=m + +# +# PCMCIA devices +# + +# +# SoC audio support +# +CONFIG_SND_SOC=m + +# +# Soc Platforms +# + +# +# SoC Audio for the Intel PXA2xx +# +CONFIG_SND_PXA2xx_SOC=m +CONFIG_SND_PXA2xx_SOC_I2S=m +# CONFIG_SND_PXA2xx_SOC_MAINSTONE is not set +# CONFIG_SND_PXA2xx_SOC_MAINSTONE_WM8753 is not set +# CONFIG_SND_PXA2xx_SOC_MAINSTONE_WM9713 is not set +# CONFIG_SND_PXA2xx_SOC_MAINSTONE_WM9712 is not set +CONFIG_SND_PXA2xx_SOC_CORGI=m +# CONFIG_SND_PXA2xx_SOC_SPITZ is not set +# CONFIG_SND_PXA2xx_SOC_TOSA is not set + +# +# Soc Codecs +# +# CONFIG_SND_SOC_AC97_CODEC is not set +CONFIG_SND_SOC_WM8731=m +# CONFIG_SND_SOC_WM8750 is not set +# CONFIG_SND_SOC_WM8753 is not set +# CONFIG_SND_SOC_WM8772 is not set +# CONFIG_SND_SOC_WM8971 is not set +# CONFIG_SND_SOC_WM9713 is not set +# CONFIG_SND_SOC_WM9712 is not set +# CONFIG_SND_SOC_UDA1380 is not set +# CONFIG_SND_SOC_AK4535 is not set + +# +# Open Sound System +# +# CONFIG_SOUND_PRIME is not set + +# +# USB support +# +CONFIG_USB_ARCH_HAS_HCD=y +# CONFIG_USB_ARCH_HAS_OHCI is not set +CONFIG_USB=m +# CONFIG_USB_DEBUG is not set + +# +# Miscellaneous USB options +# +CONFIG_USB_DEVICEFS=y +# CONFIG_USB_BANDWIDTH is not set +# CONFIG_USB_DYNAMIC_MINORS is not set +# CONFIG_USB_SUSPEND is not set +# CONFIG_USB_OTG is not set + +# +# USB Host Controller Drivers +# +# CONFIG_USB_ISP116X_HCD is not set +CONFIG_USB_SL811_HCD=m +CONFIG_USB_SL811_CS=m + +# +# USB Device Class drivers +# +# CONFIG_OBSOLETE_OSS_USB_DRIVER is not set +CONFIG_USB_ACM=m +CONFIG_USB_PRINTER=m + +# +# NOTE: USB_STORAGE enables SCSI, and 'SCSI disk support' +# + +# +# may also be needed; see USB_STORAGE Help for more information +# +CONFIG_USB_STORAGE=m +# CONFIG_USB_STORAGE_DEBUG is not set +# CONFIG_USB_STORAGE_DATAFAB is not set +# CONFIG_USB_STORAGE_FREECOM is not set +# CONFIG_USB_STORAGE_ISD200 is not set +# CONFIG_USB_STORAGE_DPCM is not set +# CONFIG_USB_STORAGE_USBAT is not set +# CONFIG_USB_STORAGE_SDDR09 is not set +# CONFIG_USB_STORAGE_SDDR55 is not set +# CONFIG_USB_STORAGE_JUMPSHOT is not set +# CONFIG_USB_STORAGE_ALAUDA is not set +# CONFIG_USB_LIBUSUAL is not set + +# +# USB Input Devices +# +CONFIG_USB_HID=m +CONFIG_USB_HIDINPUT=y +# CONFIG_USB_HIDINPUT_POWERBOOK is not set +# CONFIG_HID_FF is not set +# CONFIG_USB_HIDDEV is not set + +# +# USB HID Boot Protocol drivers +# +CONFIG_USB_KBD=m +CONFIG_USB_MOUSE=m +CONFIG_USB_AIPTEK=m +CONFIG_USB_WACOM=m +# CONFIG_USB_ACECAD is not set +CONFIG_USB_KBTAB=m +CONFIG_USB_POWERMATE=m +CONFIG_USB_MTOUCH=m +# CONFIG_USB_ITMTOUCH is not set +CONFIG_USB_EGALAX=m +# CONFIG_USB_YEALINK is not set +CONFIG_USB_XPAD=m +CONFIG_USB_ATI_REMOTE=m +# CONFIG_USB_ATI_REMOTE2 is not set +# CONFIG_USB_KEYSPAN_REMOTE is not set +# CONFIG_USB_APPLETOUCH is not set + +# +# USB Imaging devices +# +CONFIG_USB_MDC800=m +CONFIG_USB_MICROTEK=m + +# +# USB Multimedia devices +# +CONFIG_USB_DABUSB=m +CONFIG_USB_VICAM=m +CONFIG_USB_DSBR=m +# CONFIG_USB_ET61X251 is not set +CONFIG_USB_IBMCAM=m +CONFIG_USB_KONICAWC=m +CONFIG_USB_OV511=m +CONFIG_USB_SE401=m +CONFIG_USB_SN9C102=m +CONFIG_USB_STV680=m +# CONFIG_USB_PWC is not set + +# +# USB Network Adapters +# +CONFIG_USB_CATC=m +CONFIG_USB_KAWETH=m +CONFIG_USB_PEGASUS=m +CONFIG_USB_RTL8150=m +CONFIG_USB_USBNET=m +CONFIG_USB_NET_AX8817X=m +CONFIG_USB_NET_CDCETHER=m +# CONFIG_USB_NET_GL620A is not set +CONFIG_USB_NET_NET1080=m +# CONFIG_USB_NET_PLUSB is not set +# CONFIG_USB_NET_RNDIS_HOST is not set +# CONFIG_USB_NET_CDC_SUBSET is not set +CONFIG_USB_NET_ZAURUS=m +# CONFIG_USB_ZD1201 is not set +CONFIG_USB_MON=y + +# +# USB port drivers +# + +# +# USB Serial Converter support +# +CONFIG_USB_SERIAL=m +CONFIG_USB_SERIAL_GENERIC=y +# CONFIG_USB_SERIAL_AIRPRIME is not set +# CONFIG_USB_SERIAL_ANYDATA is not set +CONFIG_USB_SERIAL_BELKIN=m +# CONFIG_USB_SERIAL_WHITEHEAT is not set +CONFIG_USB_SERIAL_DIGI_ACCELEPORT=m +# CONFIG_USB_SERIAL_CP2101 is not set +CONFIG_USB_SERIAL_CYPRESS_M8=m +CONFIG_USB_SERIAL_EMPEG=m +CONFIG_USB_SERIAL_FTDI_SIO=m +CONFIG_USB_SERIAL_VISOR=m +CONFIG_USB_SERIAL_IPAQ=m +CONFIG_USB_SERIAL_IR=m +CONFIG_USB_SERIAL_EDGEPORT=m +CONFIG_USB_SERIAL_EDGEPORT_TI=m +CONFIG_USB_SERIAL_GARMIN=m +CONFIG_USB_SERIAL_IPW=m +CONFIG_USB_SERIAL_KEYSPAN_PDA=m +CONFIG_USB_SERIAL_KEYSPAN=m +# CONFIG_USB_SERIAL_KEYSPAN_MPR is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XA is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XB is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA18X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19QW is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19QI is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49WLC is not set +CONFIG_USB_SERIAL_KLSI=m +CONFIG_USB_SERIAL_KOBIL_SCT=m +CONFIG_USB_SERIAL_MCT_U232=m +CONFIG_USB_SERIAL_PL2303=m +# CONFIG_USB_SERIAL_HP4X is not set +CONFIG_USB_SERIAL_SAFE=m +# CONFIG_USB_SERIAL_SAFE_PADDED is not set +CONFIG_USB_SERIAL_TI=m +CONFIG_USB_SERIAL_CYBERJACK=m +CONFIG_USB_SERIAL_XIRCOM=m +CONFIG_USB_SERIAL_OMNINET=m +CONFIG_USB_EZUSB=y + +# +# USB Miscellaneous drivers +# +CONFIG_USB_EMI62=m +CONFIG_USB_EMI26=m +CONFIG_USB_AUERSWALD=m +CONFIG_USB_RIO500=m +CONFIG_USB_LEGOTOWER=m +CONFIG_USB_LCD=m +CONFIG_USB_LED=m +CONFIG_USB_CYTHERM=m +CONFIG_USB_PHIDGETKIT=m +CONFIG_USB_PHIDGETSERVO=m +CONFIG_USB_IDMOUSE=m +# CONFIG_USB_LD is not set +# CONFIG_USB_TEST is not set + +# +# USB DSL modem support +# + +# +# USB Gadget Support +# +CONFIG_USB_GADGET=y +# CONFIG_USB_GADGET_DEBUG_FILES is not set +CONFIG_USB_GADGET_SELECTED=y +# CONFIG_USB_GADGET_NET2280 is not set +CONFIG_USB_GADGET_PXA2XX=y +CONFIG_USB_PXA2XX=y +# CONFIG_USB_PXA2XX_SMALL is not set +# CONFIG_USB_GADGET_PXA27X is not set +# CONFIG_USB_GADGET_GOKU is not set +# CONFIG_USB_GADGET_LH7A40X is not set +# CONFIG_USB_GADGET_OMAP is not set +# CONFIG_USB_GADGET_DUMMY_HCD is not set +# CONFIG_USB_GADGET_DUALSPEED is not set +CONFIG_USB_ZERO=m +CONFIG_USB_ETH=m +CONFIG_USB_ETH_RNDIS=y +CONFIG_USB_GADGETFS=m +CONFIG_USB_FILE_STORAGE=m +# CONFIG_USB_FILE_STORAGE_TEST is not set +CONFIG_USB_G_SERIAL=m + +# +# MMC/SD Card support +# +CONFIG_MMC=y +# CONFIG_MMC_DEBUG is not set +CONFIG_MMC_BLOCK=y +CONFIG_MMC_PXA=y + +# +# Real Time Clock +# +CONFIG_RTC_CLASS=y +CONFIG_RTC_HCTOSYS=y +CONFIG_RTC_HCTOSYS_DEVICE="rtc0" + +# +# RTC interfaces +# +CONFIG_RTC_INTF_SYSFS=y +CONFIG_RTC_INTF_PROC=y +CONFIG_RTC_INTF_DEV=y + +# +# RTC drivers +# +# CONFIG_RTC_DRV_X1205 is not set +# CONFIG_RTC_DRV_DS1672 is not set +# CONFIG_RTC_DRV_PCF8563 is not set +# CONFIG_RTC_DRV_RS5C372 is not set +CONFIG_RTC_DRV_SA1100=y +# CONFIG_RTC_DRV_TEST is not set + +# +# File systems +# +CONFIG_EXT2_FS=y +# CONFIG_EXT2_FS_XATTR is not set +# CONFIG_EXT2_FS_XIP is not set +# CONFIG_EXT3_FS is not set +# CONFIG_REISERFS_FS is not set +# CONFIG_JFS_FS is not set +# CONFIG_FS_POSIX_ACL is not set +# CONFIG_XFS_FS is not set +# CONFIG_OCFS2_FS is not set +# CONFIG_MINIX_FS is not set +# CONFIG_ROMFS_FS is not set +CONFIG_INOTIFY=y +# CONFIG_QUOTA is not set +CONFIG_DNOTIFY=y +# CONFIG_AUTOFS_FS is not set +# CONFIG_AUTOFS4_FS is not set +# CONFIG_FUSE_FS is not set + +# +# CD-ROM/DVD Filesystems +# +# CONFIG_ISO9660_FS is not set +# CONFIG_UDF_FS is not set + +# +# DOS/FAT/NT Filesystems +# +CONFIG_FAT_FS=y +# CONFIG_MSDOS_FS is not set +CONFIG_VFAT_FS=y +CONFIG_FAT_DEFAULT_CODEPAGE=437 +CONFIG_FAT_DEFAULT_IOCHARSET="iso8859-1" +# CONFIG_NTFS_FS is not set + +# +# Pseudo filesystems +# +CONFIG_PROC_FS=y +CONFIG_SYSFS=y +CONFIG_TMPFS=y +# CONFIG_HUGETLB_PAGE is not set +CONFIG_RAMFS=y +# CONFIG_RELAYFS_FS is not set +# CONFIG_CONFIGFS_FS is not set + +# +# Miscellaneous filesystems +# +# CONFIG_ADFS_FS is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_HFSPLUS_FS is not set +# CONFIG_BEFS_FS is not set +# CONFIG_BFS_FS is not set +# CONFIG_EFS_FS is not set +# CONFIG_JFFS_FS is not set +CONFIG_JFFS2_FS=y +CONFIG_JFFS2_FS_DEBUG=0 +CONFIG_JFFS2_FS_WRITEBUFFER=y +CONFIG_JFFS2_SUMMARY=y +CONFIG_JFFS2_COMPRESSION_OPTIONS=y +CONFIG_JFFS2_ZLIB=y +CONFIG_JFFS2_RTIME=y +CONFIG_JFFS2_RUBIN=y +# CONFIG_JFFS2_CMODE_NONE is not set +CONFIG_JFFS2_CMODE_PRIORITY=y +# CONFIG_JFFS2_CMODE_SIZE is not set +CONFIG_CRAMFS=m +CONFIG_SQUASHFS=m +# CONFIG_SQUASHFS_EMBEDDED is not set +CONFIG_SQUASHFS_FRAGMENT_CACHE_SIZE=3 +# CONFIG_SQUASHFS_VMALLOC is not set +# CONFIG_VXFS_FS is not set +# CONFIG_HPFS_FS is not set +# CONFIG_QNX4FS_FS is not set +# CONFIG_SYSV_FS is not set +# CONFIG_UFS_FS is not set + +# +# Network File Systems +# +CONFIG_NFS_FS=m +CONFIG_NFS_V3=y +# CONFIG_NFS_V3_ACL is not set +CONFIG_NFS_V4=y +# CONFIG_NFS_DIRECTIO is not set +# CONFIG_NFSD is not set +CONFIG_LOCKD=m +CONFIG_LOCKD_V4=y +CONFIG_NFS_COMMON=y +CONFIG_SUNRPC=m +CONFIG_SUNRPC_GSS=m +CONFIG_RPCSEC_GSS_KRB5=m +# CONFIG_RPCSEC_GSS_SPKM3 is not set +CONFIG_SMB_FS=m +CONFIG_SMB_NLS_DEFAULT=y +CONFIG_SMB_NLS_REMOTE="cp437" +CONFIG_CIFS=m +# CONFIG_CIFS_STATS is not set +# CONFIG_CIFS_XATTR is not set +# CONFIG_CIFS_EXPERIMENTAL is not set +# CONFIG_NCP_FS is not set +# CONFIG_CODA_FS is not set +# CONFIG_AFS_FS is not set +# CONFIG_9P_FS is not set + +# +# Partition Types +# +CONFIG_PARTITION_ADVANCED=y +# CONFIG_ACORN_PARTITION is not set +# CONFIG_OSF_PARTITION is not set +# CONFIG_AMIGA_PARTITION is not set +# CONFIG_ATARI_PARTITION is not set +# CONFIG_MAC_PARTITION is not set +CONFIG_MSDOS_PARTITION=y +# CONFIG_BSD_DISKLABEL is not set +# CONFIG_MINIX_SUBPARTITION is not set +# CONFIG_SOLARIS_X86_PARTITION is not set +# CONFIG_UNIXWARE_DISKLABEL is not set +# CONFIG_LDM_PARTITION is not set +# CONFIG_SGI_PARTITION is not set +# CONFIG_ULTRIX_PARTITION is not set +# CONFIG_SUN_PARTITION is not set +# CONFIG_KARMA_PARTITION is not set +# CONFIG_EFI_PARTITION is not set + +# +# Native Language Support +# +CONFIG_NLS=y +CONFIG_NLS_DEFAULT="cp437" +CONFIG_NLS_CODEPAGE_437=y +CONFIG_NLS_CODEPAGE_737=m +CONFIG_NLS_CODEPAGE_775=m +CONFIG_NLS_CODEPAGE_850=m +CONFIG_NLS_CODEPAGE_852=m +CONFIG_NLS_CODEPAGE_855=m +CONFIG_NLS_CODEPAGE_857=m +CONFIG_NLS_CODEPAGE_860=m +CONFIG_NLS_CODEPAGE_861=m +CONFIG_NLS_CODEPAGE_862=m +CONFIG_NLS_CODEPAGE_863=m +CONFIG_NLS_CODEPAGE_864=m +CONFIG_NLS_CODEPAGE_865=m +CONFIG_NLS_CODEPAGE_866=m +CONFIG_NLS_CODEPAGE_869=m +CONFIG_NLS_CODEPAGE_936=m +CONFIG_NLS_CODEPAGE_950=m +CONFIG_NLS_CODEPAGE_932=m +CONFIG_NLS_CODEPAGE_949=m +CONFIG_NLS_CODEPAGE_874=m +CONFIG_NLS_ISO8859_8=m +CONFIG_NLS_CODEPAGE_1250=m +CONFIG_NLS_CODEPAGE_1251=m +CONFIG_NLS_ASCII=m +CONFIG_NLS_ISO8859_1=y +CONFIG_NLS_ISO8859_2=m +CONFIG_NLS_ISO8859_3=m +CONFIG_NLS_ISO8859_4=m +CONFIG_NLS_ISO8859_5=m +CONFIG_NLS_ISO8859_6=m +CONFIG_NLS_ISO8859_7=m +CONFIG_NLS_ISO8859_9=m +CONFIG_NLS_ISO8859_13=m +CONFIG_NLS_ISO8859_14=m +CONFIG_NLS_ISO8859_15=m +CONFIG_NLS_KOI8_R=m +CONFIG_NLS_KOI8_U=m +CONFIG_NLS_UTF8=y + +# +# Profiling support +# +CONFIG_PROFILING=y +CONFIG_OPROFILE=m + +# +# Kernel hacking +# +# CONFIG_PRINTK_TIME is not set +CONFIG_MAGIC_SYSRQ=y +CONFIG_DEBUG_KERNEL=y +CONFIG_LOG_BUF_SHIFT=14 +CONFIG_DETECT_SOFTLOCKUP=y +# CONFIG_SCHEDSTATS is not set +# CONFIG_DEBUG_SLAB is not set +# CONFIG_DEBUG_PREEMPT is not set +# CONFIG_DEBUG_MUTEXES is not set +# CONFIG_DEBUG_SPINLOCK is not set +# CONFIG_DEBUG_SPINLOCK_SLEEP is not set +# CONFIG_DEBUG_KOBJECT is not set +CONFIG_DEBUG_BUGVERBOSE=y +# CONFIG_DEBUG_INFO is not set +# CONFIG_DEBUG_FS is not set +# CONFIG_DEBUG_VM is not set +CONFIG_FRAME_POINTER=y +# CONFIG_FORCED_INLINING is not set +# CONFIG_RCU_TORTURE_TEST is not set +# CONFIG_DEBUG_USER is not set +# CONFIG_DEBUG_WAITQ is not set +CONFIG_DEBUG_ERRORS=y +# CONFIG_DEBUG_LL is not set + +# +# Security options +# +# CONFIG_KEYS is not set +# CONFIG_SECURITY is not set + +# +# Cryptographic options +# +CONFIG_CRYPTO=y +CONFIG_CRYPTO_HMAC=y +CONFIG_CRYPTO_NULL=m +CONFIG_CRYPTO_MD4=m +CONFIG_CRYPTO_MD5=m +CONFIG_CRYPTO_SHA1=m +CONFIG_CRYPTO_SHA256=m +CONFIG_CRYPTO_SHA512=m +CONFIG_CRYPTO_WP512=m +# CONFIG_CRYPTO_TGR192 is not set +CONFIG_CRYPTO_DES=m +CONFIG_CRYPTO_BLOWFISH=m +CONFIG_CRYPTO_TWOFISH=m +CONFIG_CRYPTO_SERPENT=m +CONFIG_CRYPTO_AES=m +CONFIG_CRYPTO_CAST5=m +CONFIG_CRYPTO_CAST6=m +CONFIG_CRYPTO_TEA=m +CONFIG_CRYPTO_ARC4=m +CONFIG_CRYPTO_KHAZAD=m +CONFIG_CRYPTO_ANUBIS=m +CONFIG_CRYPTO_DEFLATE=m +CONFIG_CRYPTO_MICHAEL_MIC=m +CONFIG_CRYPTO_CRC32C=m +CONFIG_CRYPTO_TEST=m + +# +# Hardware crypto devices +# + +# +# Library routines +# +CONFIG_CRC_CCITT=y +# CONFIG_CRC16 is not set +CONFIG_CRC32=y +CONFIG_LIBCRC32C=m +CONFIG_ZLIB_INFLATE=y +CONFIG_ZLIB_DEFLATE=y diff --git a/packages/linux/linux-rp-2.6.20/defconfig-collie b/packages/linux/linux-rp-2.6.20/defconfig-collie new file mode 100644 index 0000000000..f488d42d93 --- /dev/null +++ b/packages/linux/linux-rp-2.6.20/defconfig-collie @@ -0,0 +1,1641 @@ +# +# Automatically generated make config: don't edit +# Linux kernel version: 2.6.17 +# Tue Jul 4 16:35:35 2006 +# +CONFIG_ARM=y +CONFIG_MMU=y +CONFIG_RWSEM_GENERIC_SPINLOCK=y +CONFIG_GENERIC_HWEIGHT=y +CONFIG_GENERIC_CALIBRATE_DELAY=y +CONFIG_ARCH_MTD_XIP=y +CONFIG_VECTORS_BASE=0xffff0000 + +# +# Code maturity level options +# +CONFIG_EXPERIMENTAL=y +CONFIG_BROKEN_ON_SMP=y +CONFIG_LOCK_KERNEL=y +CONFIG_INIT_ENV_ARG_LIMIT=32 + +# +# General setup +# +CONFIG_LOCALVERSION="" +CONFIG_LOCALVERSION_AUTO=y +CONFIG_SWAP=y +CONFIG_SYSVIPC=y +# CONFIG_POSIX_MQUEUE is not set +CONFIG_BSD_PROCESS_ACCT=y +# CONFIG_BSD_PROCESS_ACCT_V3 is not set +CONFIG_SYSCTL=y +# CONFIG_AUDIT is not set +# CONFIG_IKCONFIG is not set +# CONFIG_RELAY is not set +CONFIG_INITRAMFS_SOURCE="" +CONFIG_UID16=y +CONFIG_CC_OPTIMIZE_FOR_SIZE=y +CONFIG_EMBEDDED=y +CONFIG_KALLSYMS=y +# CONFIG_KALLSYMS_ALL is not set +# CONFIG_KALLSYMS_EXTRA_PASS is not set +CONFIG_HOTPLUG=y +CONFIG_PRINTK=y +CONFIG_BUG=y +CONFIG_ELF_CORE=y +CONFIG_BASE_FULL=y +CONFIG_FUTEX=y +CONFIG_EPOLL=y +CONFIG_SHMEM=y +CONFIG_SLAB=y +# CONFIG_TINY_SHMEM is not set +CONFIG_BASE_SMALL=0 +# CONFIG_SLOB is not set + +# +# Loadable module support +# +CONFIG_MODULES=y +CONFIG_MODULE_UNLOAD=y +CONFIG_MODULE_FORCE_UNLOAD=y +# CONFIG_MODVERSIONS is not set +# CONFIG_MODULE_SRCVERSION_ALL is not set +CONFIG_KMOD=y + +# +# Block layer +# +# CONFIG_BLK_DEV_IO_TRACE is not set + +# +# IO Schedulers +# +CONFIG_IOSCHED_NOOP=y +CONFIG_IOSCHED_AS=y +CONFIG_IOSCHED_DEADLINE=m +CONFIG_IOSCHED_CFQ=m +CONFIG_DEFAULT_AS=y +# CONFIG_DEFAULT_DEADLINE is not set +# CONFIG_DEFAULT_CFQ is not set +# CONFIG_DEFAULT_NOOP is not set +CONFIG_DEFAULT_IOSCHED="anticipatory" + +# +# System Type +# +# CONFIG_ARCH_CLPS7500 is not set +# CONFIG_ARCH_CLPS711X is not set +# CONFIG_ARCH_CO285 is not set +# CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_EP93XX is not set +# CONFIG_ARCH_FOOTBRIDGE is not set +# CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_IOP3XX is not set +# CONFIG_ARCH_IXP4XX is not set +# CONFIG_ARCH_IXP2000 is not set +# CONFIG_ARCH_IXP23XX is not set +# CONFIG_ARCH_L7200 is not set +# CONFIG_ARCH_PXA is not set +# CONFIG_ARCH_RPC is not set +CONFIG_ARCH_SA1100=y +# CONFIG_ARCH_S3C2410 is not set +# CONFIG_ARCH_SHARK is not set +# CONFIG_ARCH_LH7A40X is not set +# CONFIG_ARCH_OMAP is not set +# CONFIG_ARCH_VERSATILE is not set +# CONFIG_ARCH_REALVIEW is not set +# CONFIG_ARCH_IMX is not set +# CONFIG_ARCH_H720X is not set +# CONFIG_ARCH_AAEC2000 is not set +# CONFIG_ARCH_AT91RM9200 is not set + +# +# SA11x0 Implementations +# +# CONFIG_SA1100_ASSABET is not set +# CONFIG_SA1100_CERF is not set +CONFIG_SA1100_COLLIE=y +# CONFIG_SA1100_H3100 is not set +# CONFIG_SA1100_H3600 is not set +# CONFIG_SA1100_H3800 is not set +# CONFIG_SA1100_BADGE4 is not set +# CONFIG_SA1100_JORNADA720 is not set +# CONFIG_SA1100_HACKKIT is not set +# CONFIG_SA1100_LART is not set +# CONFIG_SA1100_PLEB is not set +# CONFIG_SA1100_SHANNON is not set +# CONFIG_SA1100_SIMPAD is not set +# CONFIG_SA1100_SSP is not set + +# +# Processor Type +# +CONFIG_CPU_32=y +CONFIG_CPU_SA1100=y +CONFIG_CPU_32v4=y +CONFIG_CPU_ABRT_EV4=y +CONFIG_CPU_CACHE_V4WB=y +CONFIG_CPU_CACHE_VIVT=y +CONFIG_CPU_TLB_V4WB=y + +# +# Processor Features +# +CONFIG_KEXEC=y +CONFIG_SHARP_LOCOMO=y +CONFIG_SHARP_PARAM=y +CONFIG_SHARP_SCOOP=y + +# +# Bus support +# +CONFIG_ISA=y + +# +# PCCARD (PCMCIA/CardBus) support +# +CONFIG_PCCARD=y +# CONFIG_PCMCIA_DEBUG is not set +CONFIG_PCMCIA=y +CONFIG_PCMCIA_LOAD_CIS=y +CONFIG_PCMCIA_IOCTL=y + +# +# PC-card bridges +# +# CONFIG_I82365 is not set +# CONFIG_TCIC is not set +CONFIG_PCMCIA_SA1100=y + +# +# Kernel Features +# +CONFIG_PREEMPT=y +CONFIG_NO_IDLE_HZ=y +CONFIG_HZ=100 +# CONFIG_AEABI is not set +CONFIG_ARCH_DISCONTIGMEM_ENABLE=y +CONFIG_NODES_SHIFT=2 +CONFIG_SELECT_MEMORY_MODEL=y +# CONFIG_FLATMEM_MANUAL is not set +CONFIG_DISCONTIGMEM_MANUAL=y +# CONFIG_SPARSEMEM_MANUAL is not set +CONFIG_DISCONTIGMEM=y +CONFIG_FLAT_NODE_MEM_MAP=y +CONFIG_NEED_MULTIPLE_NODES=y +# CONFIG_SPARSEMEM_STATIC is not set +CONFIG_SPLIT_PTLOCK_CPUS=4096 +# CONFIG_LEDS is not set +CONFIG_ALIGNMENT_TRAP=y + +# +# Boot options +# +CONFIG_ZBOOT_ROM_TEXT=0x0 +CONFIG_ZBOOT_ROM_BSS=0x0 +CONFIG_CMDLINE="console=ttyS0,115200n8 console=tty1 noinitrd root=/dev/mtdblock2 rootfstype=jffs2 mem=32M fbcon=rotate:1 dyntick=enable quiet" +# CONFIG_XIP_KERNEL is not set + +# +# CPU Frequency scaling +# +# CONFIG_CPU_FREQ is not set + +# +# Floating point emulation +# + +# +# At least one emulation must be selected +# +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_NWFPE_XP is not set +# CONFIG_FPE_FASTFPE is not set + +# +# Userspace binary formats +# +CONFIG_BINFMT_ELF=y +CONFIG_BINFMT_AOUT=m +CONFIG_BINFMT_MISC=m +# CONFIG_ARTHUR is not set + +# +# Power management options +# +CONFIG_PM=y +# CONFIG_PM_LEGACY is not set +# CONFIG_PM_DEBUG is not set +CONFIG_APM=y + +# +# Networking +# +CONFIG_NET=y + +# +# Networking options +# +# CONFIG_NETDEBUG is not set +CONFIG_PACKET=y +CONFIG_PACKET_MMAP=y +CONFIG_UNIX=y +CONFIG_XFRM=y +CONFIG_XFRM_USER=m +# CONFIG_NET_KEY is not set +CONFIG_INET=y +# CONFIG_IP_MULTICAST is not set +# CONFIG_IP_ADVANCED_ROUTER is not set +CONFIG_IP_FIB_HASH=y +# CONFIG_IP_PNP is not set +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set +CONFIG_SYN_COOKIES=y +# CONFIG_INET_AH is not set +# CONFIG_INET_ESP is not set +# CONFIG_INET_IPCOMP is not set +# CONFIG_INET_XFRM_TUNNEL is not set +# CONFIG_INET_TUNNEL is not set +CONFIG_INET_DIAG=m +CONFIG_INET_TCP_DIAG=m +# CONFIG_TCP_CONG_ADVANCED is not set +CONFIG_TCP_CONG_BIC=y + +# +# IP: Virtual Server Configuration +# +# CONFIG_IP_VS is not set +CONFIG_IPV6=m +# CONFIG_IPV6_PRIVACY is not set +# CONFIG_IPV6_ROUTER_PREF is not set +CONFIG_INET6_AH=m +CONFIG_INET6_ESP=m +CONFIG_INET6_IPCOMP=m +CONFIG_INET6_XFRM_TUNNEL=m +CONFIG_INET6_TUNNEL=m +CONFIG_IPV6_TUNNEL=m +CONFIG_NETFILTER=y +# CONFIG_NETFILTER_DEBUG is not set + +# +# Core Netfilter Configuration +# +# CONFIG_NETFILTER_NETLINK is not set +# CONFIG_NETFILTER_XTABLES is not set + +# +# IP: Netfilter Configuration +# +CONFIG_IP_NF_CONNTRACK=m +# CONFIG_IP_NF_CT_ACCT is not set +# CONFIG_IP_NF_CONNTRACK_MARK is not set +# CONFIG_IP_NF_CONNTRACK_EVENTS is not set +CONFIG_IP_NF_CT_PROTO_SCTP=m +CONFIG_IP_NF_FTP=m +CONFIG_IP_NF_IRC=m +# CONFIG_IP_NF_NETBIOS_NS is not set +CONFIG_IP_NF_TFTP=m +CONFIG_IP_NF_AMANDA=m +# CONFIG_IP_NF_PPTP is not set +# CONFIG_IP_NF_H323 is not set +CONFIG_IP_NF_QUEUE=m + +# +# IPv6: Netfilter Configuration (EXPERIMENTAL) +# +# CONFIG_IP6_NF_QUEUE is not set + +# +# DCCP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_DCCP is not set + +# +# SCTP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_SCTP is not set + +# +# TIPC Configuration (EXPERIMENTAL) +# +# CONFIG_TIPC is not set +# CONFIG_ATM is not set +# CONFIG_BRIDGE is not set +# CONFIG_VLAN_8021Q is not set +# CONFIG_DECNET is not set +# CONFIG_LLC2 is not set +# CONFIG_IPX is not set +# CONFIG_ATALK is not set +# CONFIG_X25 is not set +# CONFIG_LAPB is not set +# CONFIG_NET_DIVERT is not set +# CONFIG_ECONET is not set +# CONFIG_WAN_ROUTER is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network testing +# +# CONFIG_NET_PKTGEN is not set +# CONFIG_HAMRADIO is not set +CONFIG_IRDA=m + +# +# IrDA protocols +# +CONFIG_IRLAN=m +CONFIG_IRNET=m +CONFIG_IRCOMM=m +# CONFIG_IRDA_ULTRA is not set + +# +# IrDA options +# +# CONFIG_IRDA_CACHE_LAST_LSAP is not set +# CONFIG_IRDA_FAST_RR is not set +# CONFIG_IRDA_DEBUG is not set + +# +# Infrared-port device drivers +# + +# +# SIR device drivers +# +CONFIG_IRTTY_SIR=m + +# +# Dongle support +# +# CONFIG_DONGLE is not set + +# +# Old SIR device drivers +# +# CONFIG_IRPORT_SIR is not set + +# +# Old Serial dongle support +# + +# +# FIR device drivers +# +CONFIG_USB_IRDA=m +# CONFIG_SIGMATEL_FIR is not set +CONFIG_SA1100_FIR=m +CONFIG_BT=m +CONFIG_BT_L2CAP=m +CONFIG_BT_SCO=m +CONFIG_BT_RFCOMM=m +CONFIG_BT_RFCOMM_TTY=y +CONFIG_BT_BNEP=m +CONFIG_BT_BNEP_MC_FILTER=y +CONFIG_BT_BNEP_PROTO_FILTER=y +CONFIG_BT_HIDP=m + +# +# Bluetooth device drivers +# +# CONFIG_BT_HCIUSB is not set +CONFIG_BT_HCIUART=m +CONFIG_BT_HCIUART_H4=y +CONFIG_BT_HCIUART_BCSP=y +# CONFIG_BT_HCIBCM203X is not set +# CONFIG_BT_HCIBPA10X is not set +# CONFIG_BT_HCIBFUSB is not set +CONFIG_BT_HCIDTL1=m +CONFIG_BT_HCIBT3C=m +CONFIG_BT_HCIBLUECARD=m +CONFIG_BT_HCIBTUART=m +CONFIG_BT_HCIVHCI=m +CONFIG_IEEE80211=m +# CONFIG_IEEE80211_DEBUG is not set +CONFIG_IEEE80211_CRYPT_WEP=m +CONFIG_IEEE80211_CRYPT_CCMP=m +CONFIG_IEEE80211_CRYPT_TKIP=m +# CONFIG_IEEE80211_SOFTMAC is not set +CONFIG_WIRELESS_EXT=y + +# +# Device Drivers +# + +# +# Generic Driver Options +# +CONFIG_STANDALONE=y +CONFIG_PREVENT_FIRMWARE_BUILD=y +CONFIG_FW_LOADER=y +# CONFIG_DEBUG_DRIVER is not set + +# +# Connector - unified userspace <-> kernelspace linker +# +# CONFIG_CONNECTOR is not set + +# +# Memory Technology Devices (MTD) +# +CONFIG_MTD=y +# CONFIG_MTD_DEBUG is not set +# CONFIG_MTD_CONCAT is not set +CONFIG_MTD_PARTITIONS=y +# CONFIG_MTD_REDBOOT_PARTS is not set +# CONFIG_MTD_CMDLINE_PARTS is not set +# CONFIG_MTD_AFS_PARTS is not set + +# +# User Modules And Translation Layers +# +CONFIG_MTD_CHAR=y +CONFIG_MTD_BLOCK=y +# CONFIG_FTL is not set +# CONFIG_NFTL is not set +# CONFIG_INFTL is not set +# CONFIG_RFD_FTL is not set + +# +# RAM/ROM/Flash chip drivers +# +# CONFIG_MTD_CFI is not set +# CONFIG_MTD_JEDECPROBE is not set +CONFIG_MTD_MAP_BANK_WIDTH_1=y +CONFIG_MTD_MAP_BANK_WIDTH_2=y +CONFIG_MTD_MAP_BANK_WIDTH_4=y +# CONFIG_MTD_MAP_BANK_WIDTH_8 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_16 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_32 is not set +CONFIG_MTD_CFI_I1=y +CONFIG_MTD_CFI_I2=y +# CONFIG_MTD_CFI_I4 is not set +# CONFIG_MTD_CFI_I8 is not set +# CONFIG_MTD_RAM is not set +CONFIG_MTD_ROM=y +# CONFIG_MTD_ABSENT is not set +CONFIG_MTD_OBSOLETE_CHIPS=y +CONFIG_MTD_SHARP=y + +# +# Mapping drivers for chip access +# +# CONFIG_MTD_COMPLEX_MAPPINGS is not set +CONFIG_MTD_SA1100=y +# CONFIG_MTD_PLATRAM is not set + +# +# Self-contained MTD device drivers +# +# CONFIG_MTD_SLRAM is not set +# CONFIG_MTD_PHRAM is not set +# CONFIG_MTD_MTDRAM is not set +# CONFIG_MTD_BLOCK2MTD is not set + +# +# Disk-On-Chip Device Drivers +# +# CONFIG_MTD_DOC2000 is not set +# CONFIG_MTD_DOC2001 is not set +# CONFIG_MTD_DOC2001PLUS is not set + +# +# NAND Flash Device Drivers +# +# CONFIG_MTD_NAND is not set + +# +# OneNAND Flash Device Drivers +# +# CONFIG_MTD_ONENAND is not set + +# +# Parallel port support +# +# CONFIG_PARPORT is not set + +# +# Plug and Play support +# +# CONFIG_PNP is not set + +# +# Block devices +# +# CONFIG_BLK_DEV_COW_COMMON is not set +CONFIG_BLK_DEV_LOOP=m +# CONFIG_BLK_DEV_CRYPTOLOOP is not set +# CONFIG_BLK_DEV_NBD is not set +# CONFIG_BLK_DEV_UB is not set +# CONFIG_BLK_DEV_RAM is not set +# CONFIG_BLK_DEV_INITRD is not set +# CONFIG_CDROM_PKTCDVD is not set +# CONFIG_ATA_OVER_ETH is not set + +# +# ATA/ATAPI/MFM/RLL support +# +CONFIG_IDE=y +CONFIG_BLK_DEV_IDE=m + +# +# Please see Documentation/ide.txt for help/info on IDE drives +# +# CONFIG_BLK_DEV_IDE_SATA is not set +CONFIG_BLK_DEV_IDEDISK=m +# CONFIG_IDEDISK_MULTI_MODE is not set +CONFIG_BLK_DEV_IDECS=m +# CONFIG_BLK_DEV_IDECD is not set +# CONFIG_BLK_DEV_IDETAPE is not set +# CONFIG_BLK_DEV_IDEFLOPPY is not set +# CONFIG_BLK_DEV_IDESCSI is not set +# CONFIG_IDE_TASK_IOCTL is not set + +# +# IDE chipset support/bugfixes +# +# CONFIG_IDE_GENERIC is not set +# CONFIG_IDE_ARM is not set +# CONFIG_IDE_CHIPSETS is not set +# CONFIG_BLK_DEV_IDEDMA is not set +# CONFIG_IDEDMA_AUTO is not set +# CONFIG_BLK_DEV_HD is not set + +# +# SCSI device support +# +# CONFIG_RAID_ATTRS is not set +CONFIG_SCSI=m +CONFIG_SCSI_PROC_FS=y + +# +# SCSI support type (disk, tape, CD-ROM) +# +CONFIG_BLK_DEV_SD=m +CONFIG_CHR_DEV_ST=m +CONFIG_CHR_DEV_OSST=m +CONFIG_BLK_DEV_SR=m +# CONFIG_BLK_DEV_SR_VENDOR is not set +CONFIG_CHR_DEV_SG=m +# CONFIG_CHR_DEV_SCH is not set + +# +# Some SCSI devices (e.g. CD jukebox) support multiple LUNs +# +CONFIG_SCSI_MULTI_LUN=y +# CONFIG_SCSI_CONSTANTS is not set +# CONFIG_SCSI_LOGGING is not set + +# +# SCSI Transport Attributes +# +# CONFIG_SCSI_SPI_ATTRS is not set +# CONFIG_SCSI_FC_ATTRS is not set +# CONFIG_SCSI_ISCSI_ATTRS is not set +# CONFIG_SCSI_SAS_ATTRS is not set + +# +# SCSI low-level drivers +# +# CONFIG_ISCSI_TCP is not set +# CONFIG_SCSI_AHA152X is not set +# CONFIG_SCSI_AIC7XXX_OLD is not set +# CONFIG_SCSI_IN2000 is not set +# CONFIG_SCSI_SATA is not set +# CONFIG_SCSI_DTC3280 is not set +# CONFIG_SCSI_FUTURE_DOMAIN is not set +# CONFIG_SCSI_GENERIC_NCR5380 is not set +# CONFIG_SCSI_GENERIC_NCR5380_MMIO is not set +# CONFIG_SCSI_NCR53C406A is not set +# CONFIG_SCSI_PAS16 is not set +# CONFIG_SCSI_PSI240I is not set +# CONFIG_SCSI_QLOGIC_FAS is not set +# CONFIG_SCSI_SYM53C416 is not set +# CONFIG_SCSI_T128 is not set +# CONFIG_SCSI_DEBUG is not set + +# +# PCMCIA SCSI adapter support +# +# CONFIG_PCMCIA_AHA152X is not set +# CONFIG_PCMCIA_FDOMAIN is not set +# CONFIG_PCMCIA_NINJA_SCSI is not set +# CONFIG_PCMCIA_QLOGIC is not set +# CONFIG_PCMCIA_SYM53C500 is not set + +# +# Multi-device support (RAID and LVM) +# +CONFIG_MD=y +CONFIG_BLK_DEV_DM=m +CONFIG_DM_CRYPT=m +CONFIG_DM_SNAPSHOT=m +CONFIG_DM_MIRROR=m +CONFIG_DM_ZERO=m +CONFIG_DM_MULTIPATH=m +CONFIG_DM_MULTIPATH_EMC=m + +# +# Fusion MPT device support +# +# CONFIG_FUSION is not set + +# +# IEEE 1394 (FireWire) support +# + +# +# I2O device support +# + +# +# Network device support +# +CONFIG_NETDEVICES=y +# CONFIG_DUMMY is not set +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +CONFIG_TUN=m + +# +# ARCnet devices +# +# CONFIG_ARCNET is not set + +# +# PHY device support +# +# CONFIG_PHYLIB is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +CONFIG_MII=m +# CONFIG_NET_VENDOR_3COM is not set +# CONFIG_NET_VENDOR_SMC is not set +# CONFIG_SMC91X is not set +# CONFIG_DM9000 is not set +# CONFIG_NET_VENDOR_RACAL is not set +# CONFIG_AT1700 is not set +# CONFIG_DEPCA is not set +# CONFIG_HP100 is not set +# CONFIG_NET_ISA is not set +# CONFIG_NET_PCI is not set + +# +# Ethernet (1000 Mbit) +# + +# +# Ethernet (10000 Mbit) +# + +# +# Token Ring devices +# +# CONFIG_TR is not set + +# +# Wireless LAN (non-hamradio) +# +CONFIG_NET_RADIO=y +# CONFIG_NET_WIRELESS_RTNETLINK is not set + +# +# Obsolete Wireless cards support (pre-802.11) +# +# CONFIG_STRIP is not set +# CONFIG_ARLAN is not set +# CONFIG_WAVELAN is not set +# CONFIG_PCMCIA_WAVELAN is not set +# CONFIG_PCMCIA_NETWAVE is not set + +# +# Wireless 802.11 Frequency Hopping cards support +# +# CONFIG_PCMCIA_RAYCS is not set + +# +# Wireless 802.11b ISA/PCI cards support +# +CONFIG_HERMES=m +# CONFIG_ATMEL is not set + +# +# Wireless 802.11b Pcmcia/Cardbus cards support +# +CONFIG_PCMCIA_HERMES=m +CONFIG_PCMCIA_SPECTRUM=m +# CONFIG_AIRO_CS is not set +# CONFIG_PCMCIA_WL3501 is not set +CONFIG_HOSTAP=m +CONFIG_HOSTAP_FIRMWARE=y +# CONFIG_HOSTAP_FIRMWARE_NVRAM is not set +CONFIG_HOSTAP_CS=m +CONFIG_NET_WIRELESS=y + +# +# PCMCIA network device support +# +CONFIG_NET_PCMCIA=y +# CONFIG_PCMCIA_3C589 is not set +# CONFIG_PCMCIA_3C574 is not set +# CONFIG_PCMCIA_FMVJ18X is not set +CONFIG_PCMCIA_PCNET=m +# CONFIG_PCMCIA_NMCLAN is not set +# CONFIG_PCMCIA_SMC91C92 is not set +# CONFIG_PCMCIA_XIRC2PS is not set +# CONFIG_PCMCIA_AXNET is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set +CONFIG_PPP=m +# CONFIG_PPP_MULTILINK is not set +# CONFIG_PPP_FILTER is not set +CONFIG_PPP_ASYNC=m +# CONFIG_PPP_SYNC_TTY is not set +CONFIG_PPP_DEFLATE=m +CONFIG_PPP_BSDCOMP=m +# CONFIG_PPP_MPPE is not set +# CONFIG_PPPOE is not set +# CONFIG_SLIP is not set +# CONFIG_SHAPER is not set +# CONFIG_NETCONSOLE is not set +# CONFIG_NETPOLL is not set +# CONFIG_NET_POLL_CONTROLLER is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Input device support +# +CONFIG_INPUT=y + +# +# Userland interfaces +# +CONFIG_INPUT_MOUSEDEV=m +# CONFIG_INPUT_MOUSEDEV_PSAUX is not set +CONFIG_INPUT_MOUSEDEV_SCREEN_X=480 +CONFIG_INPUT_MOUSEDEV_SCREEN_Y=640 +# CONFIG_INPUT_JOYDEV is not set +# CONFIG_INPUT_TSDEV is not set +CONFIG_INPUT_EVDEV=y +# CONFIG_INPUT_EVBUG is not set +CONFIG_INPUT_POWER=y + +# +# Input Device Drivers +# +CONFIG_INPUT_KEYBOARD=y +# CONFIG_KEYBOARD_ATKBD is not set +# CONFIG_KEYBOARD_SUNKBD is not set +# CONFIG_KEYBOARD_LKKBD is not set +CONFIG_KEYBOARD_LOCOMO=y +# CONFIG_KEYBOARD_XTKBD is not set +# CONFIG_KEYBOARD_NEWTON is not set +# CONFIG_INPUT_MOUSE is not set +# CONFIG_INPUT_JOYSTICK is not set +# CONFIG_INPUT_TOUCHSCREEN is not set +# CONFIG_INPUT_MISC is not set + +# +# Hardware I/O ports +# +# CONFIG_SERIO is not set +# CONFIG_GAMEPORT is not set + +# +# Character devices +# +CONFIG_VT=y +CONFIG_VT_CONSOLE=y +CONFIG_HW_CONSOLE=y +# CONFIG_SERIAL_NONSTANDARD is not set + +# +# Serial drivers +# +CONFIG_SERIAL_8250=m +CONFIG_SERIAL_8250_CS=m +CONFIG_SERIAL_8250_NR_UARTS=4 +CONFIG_SERIAL_8250_RUNTIME_UARTS=4 +# CONFIG_SERIAL_8250_EXTENDED is not set + +# +# Non-8250 serial port support +# +CONFIG_SERIAL_SA1100=y +CONFIG_SERIAL_SA1100_CONSOLE=y +CONFIG_SERIAL_CORE=y +CONFIG_SERIAL_CORE_CONSOLE=y +CONFIG_UNIX98_PTYS=y +# CONFIG_LEGACY_PTYS is not set + +# +# IPMI +# +# CONFIG_IPMI_HANDLER is not set + +# +# Watchdog Cards +# +# CONFIG_WATCHDOG is not set +# CONFIG_NVRAM is not set +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set + +# +# Ftape, the floppy tape device driver +# + +# +# PCMCIA character devices +# +# CONFIG_SYNCLINK_CS is not set +# CONFIG_CARDMAN_4000 is not set +# CONFIG_CARDMAN_4040 is not set +# CONFIG_RAW_DRIVER is not set + +# +# TPM devices +# +# CONFIG_TCG_TPM is not set +# CONFIG_TELCLOCK is not set + +# +# I2C support +# +CONFIG_I2C=m +# CONFIG_I2C_CHARDEV is not set + +# +# I2C Algorithms +# +# CONFIG_I2C_ALGOBIT is not set +# CONFIG_I2C_ALGOPCF is not set +# CONFIG_I2C_ALGOPCA is not set + +# +# I2C Hardware Bus support +# +# CONFIG_I2C_ELEKTOR is not set +# CONFIG_I2C_PARPORT_LIGHT is not set +# CONFIG_I2C_STUB is not set +# CONFIG_I2C_PCA_ISA is not set + +# +# Miscellaneous I2C Chip support +# +# CONFIG_SENSORS_DS1337 is not set +# CONFIG_SENSORS_DS1374 is not set +# CONFIG_SENSORS_EEPROM is not set +# CONFIG_SENSORS_PCF8574 is not set +# CONFIG_SENSORS_PCA9539 is not set +# CONFIG_SENSORS_PCF8591 is not set +# CONFIG_SENSORS_MAX6875 is not set +# CONFIG_I2C_DEBUG_CORE is not set +# CONFIG_I2C_DEBUG_ALGO is not set +# CONFIG_I2C_DEBUG_BUS is not set +# CONFIG_I2C_DEBUG_CHIP is not set + +# +# SPI support +# +# CONFIG_SPI is not set +# CONFIG_SPI_MASTER is not set + +# +# Dallas's 1-wire bus +# +# CONFIG_W1 is not set + +# +# Hardware Monitoring support +# +# CONFIG_HWMON is not set +# CONFIG_HWMON_VID is not set + +# +# Misc devices +# + +# +# Multimedia Capabilities Port drivers +# +CONFIG_MCP=y +CONFIG_MCP_SA11X0=y +CONFIG_MCP_UCB1200=y +# CONFIG_MCP_UCB1200_AUDIO is not set +CONFIG_MCP_UCB1200_TS=m + +# +# Multi-Function Devices +# + +# +# LED devices +# +CONFIG_NEW_LEDS=y +CONFIG_LEDS_CLASS=y + +# +# LED drivers +# +# CONFIG_LEDS_LOCOMO is not set + +# +# LED Triggers +# +CONFIG_LEDS_TRIGGERS=y +CONFIG_LEDS_TRIGGER_TIMER=m +# CONFIG_LEDS_TRIGGER_IDE_DISK is not set + +# +# Multimedia devices +# +CONFIG_VIDEO_DEV=m +CONFIG_VIDEO_V4L1=y +CONFIG_VIDEO_V4L1_COMPAT=y +CONFIG_VIDEO_V4L2=y + +# +# Video Capture Adapters +# + +# +# Video Capture Adapters +# +# CONFIG_VIDEO_ADV_DEBUG is not set +# CONFIG_VIDEO_VIVI is not set +# CONFIG_VIDEO_PMS is not set +# CONFIG_VIDEO_CPIA is not set +# CONFIG_VIDEO_CPIA2 is not set +# CONFIG_VIDEO_SAA5246A is not set +# CONFIG_VIDEO_SAA5249 is not set +# CONFIG_TUNER_3036 is not set +# CONFIG_VIDEO_OVCAMCHIP is not set + +# +# Encoders and Decoders +# +# CONFIG_VIDEO_MSP3400 is not set +# CONFIG_VIDEO_CS53L32A is not set +# CONFIG_VIDEO_WM8775 is not set +# CONFIG_VIDEO_WM8739 is not set +# CONFIG_VIDEO_CX25840 is not set +# CONFIG_VIDEO_SAA711X is not set +# CONFIG_VIDEO_SAA7127 is not set +# CONFIG_VIDEO_UPD64031A is not set +# CONFIG_VIDEO_UPD64083 is not set + +# +# V4L USB devices +# +# CONFIG_VIDEO_EM28XX is not set +# CONFIG_USB_DSBR is not set +# CONFIG_USB_VICAM is not set +# CONFIG_USB_IBMCAM is not set +# CONFIG_USB_KONICAWC is not set +# CONFIG_USB_ET61X251 is not set +# CONFIG_USB_OV511 is not set +# CONFIG_USB_SE401 is not set +# CONFIG_USB_SN9C102 is not set +# CONFIG_USB_STV680 is not set +# CONFIG_USB_W9968CF is not set +# CONFIG_USB_ZC0301 is not set +# CONFIG_USB_PWC is not set + +# +# Radio Adapters +# +# CONFIG_RADIO_CADET is not set +# CONFIG_RADIO_RTRACK is not set +# CONFIG_RADIO_RTRACK2 is not set +# CONFIG_RADIO_AZTECH is not set +# CONFIG_RADIO_GEMTEK is not set +# CONFIG_RADIO_MAESTRO is not set +# CONFIG_RADIO_SF16FMI is not set +# CONFIG_RADIO_SF16FMR2 is not set +# CONFIG_RADIO_TERRATEC is not set +# CONFIG_RADIO_TRUST is not set +# CONFIG_RADIO_TYPHOON is not set +# CONFIG_RADIO_ZOLTRIX is not set + +# +# Digital Video Broadcasting Devices +# +# CONFIG_DVB is not set +# CONFIG_USB_DABUSB is not set + +# +# Graphics support +# +CONFIG_FB=y +CONFIG_FB_CFB_FILLRECT=y +CONFIG_FB_CFB_COPYAREA=y +CONFIG_FB_CFB_IMAGEBLIT=y +# CONFIG_FB_MACMODES is not set +# CONFIG_FB_FIRMWARE_EDID is not set +# CONFIG_FB_MODE_HELPERS is not set +# CONFIG_FB_TILEBLITTING is not set +CONFIG_FB_SA1100=y +# CONFIG_FB_S1D13XXX is not set +# CONFIG_FB_VIRTUAL is not set + +# +# Console display driver support +# +# CONFIG_VGA_CONSOLE is not set +# CONFIG_MDA_CONSOLE is not set +CONFIG_DUMMY_CONSOLE=y +CONFIG_FRAMEBUFFER_CONSOLE=y +CONFIG_FRAMEBUFFER_CONSOLE_ROTATION=y +CONFIG_FONTS=y +CONFIG_FONT_8x8=y +# CONFIG_FONT_8x16 is not set +# CONFIG_FONT_6x11 is not set +# CONFIG_FONT_7x14 is not set +# CONFIG_FONT_PEARL_8x8 is not set +# CONFIG_FONT_ACORN_8x8 is not set +# CONFIG_FONT_MINI_4x6 is not set +# CONFIG_FONT_SUN8x16 is not set +# CONFIG_FONT_SUN12x22 is not set +# CONFIG_FONT_10x18 is not set + +# +# Logo configuration +# +CONFIG_LOGO=y +# CONFIG_LOGO_LINUX_MONO is not set +# CONFIG_LOGO_LINUX_VGA16 is not set +# CONFIG_LOGO_LINUX_CLUT224 is not set +# CONFIG_LOGO_OHAND_CLUT224 is not set +CONFIG_LOGO_OZ240_CLUT224=y +# CONFIG_LOGO_OZ480_CLUT224 is not set +# CONFIG_LOGO_OZ640_CLUT224 is not set +CONFIG_BACKLIGHT_LCD_SUPPORT=y +CONFIG_BACKLIGHT_CLASS_DEVICE=y +CONFIG_BACKLIGHT_DEVICE=y +# CONFIG_LCD_CLASS_DEVICE is not set +CONFIG_BACKLIGHT_LOCOMO=y + +# +# Sound +# +CONFIG_SOUND=m + +# +# Advanced Linux Sound Architecture +# +CONFIG_SND=m +CONFIG_SND_TIMER=m +CONFIG_SND_PCM=m +# CONFIG_SND_SEQUENCER is not set +CONFIG_SND_OSSEMUL=y +CONFIG_SND_MIXER_OSS=m +CONFIG_SND_PCM_OSS=m +CONFIG_SND_PCM_OSS_PLUGINS=y +# CONFIG_SND_DYNAMIC_MINORS is not set +CONFIG_SND_SUPPORT_OLD_API=y +CONFIG_SND_VERBOSE_PROCFS=y +# CONFIG_SND_VERBOSE_PRINTK is not set +# CONFIG_SND_DEBUG is not set + +# +# Generic devices +# +CONFIG_SND_DUMMY=m +# CONFIG_SND_MTPAV is not set +# CONFIG_SND_SERIAL_U16550 is not set +# CONFIG_SND_MPU401 is not set + +# +# ALSA ARM devices +# + +# +# USB devices +# +# CONFIG_SND_USB_AUDIO is not set + +# +# PCMCIA devices +# +# CONFIG_SND_VXPOCKET is not set +# CONFIG_SND_PDAUDIOCF is not set + +# +# SoC audio support +# +# CONFIG_SND_SOC is not set + +# +# Open Sound System +# +# CONFIG_SOUND_PRIME is not set + +# +# USB support +# +CONFIG_USB_ARCH_HAS_HCD=y +# CONFIG_USB_ARCH_HAS_OHCI is not set +# CONFIG_USB_ARCH_HAS_EHCI is not set +CONFIG_USB=m +# CONFIG_USB_DEBUG is not set + +# +# Miscellaneous USB options +# +CONFIG_USB_DEVICEFS=y +# CONFIG_USB_BANDWIDTH is not set +# CONFIG_USB_DYNAMIC_MINORS is not set +# CONFIG_USB_SUSPEND is not set +# CONFIG_USB_OTG is not set + +# +# USB Host Controller Drivers +# +# CONFIG_USB_ISP116X_HCD is not set +CONFIG_USB_SL811_HCD=m +CONFIG_USB_SL811_CS=m + +# +# USB Device Class drivers +# +# CONFIG_USB_ACM is not set +# CONFIG_USB_PRINTER is not set + +# +# NOTE: USB_STORAGE enables SCSI, and 'SCSI disk support' +# + +# +# may also be needed; see USB_STORAGE Help for more information +# +CONFIG_USB_STORAGE=m +# CONFIG_USB_STORAGE_DEBUG is not set +# CONFIG_USB_STORAGE_DATAFAB is not set +# CONFIG_USB_STORAGE_FREECOM is not set +# CONFIG_USB_STORAGE_ISD200 is not set +# CONFIG_USB_STORAGE_DPCM is not set +# CONFIG_USB_STORAGE_USBAT is not set +# CONFIG_USB_STORAGE_SDDR09 is not set +# CONFIG_USB_STORAGE_SDDR55 is not set +# CONFIG_USB_STORAGE_JUMPSHOT is not set +# CONFIG_USB_STORAGE_ALAUDA is not set +# CONFIG_USB_LIBUSUAL is not set + +# +# USB Input Devices +# +CONFIG_USB_HID=m +CONFIG_USB_HIDINPUT=y +# CONFIG_USB_HIDINPUT_POWERBOOK is not set +# CONFIG_HID_FF is not set +# CONFIG_USB_HIDDEV is not set + +# +# USB HID Boot Protocol drivers +# +# CONFIG_USB_KBD is not set +# CONFIG_USB_MOUSE is not set +# CONFIG_USB_AIPTEK is not set +# CONFIG_USB_WACOM is not set +# CONFIG_USB_ACECAD is not set +# CONFIG_USB_KBTAB is not set +# CONFIG_USB_POWERMATE is not set +# CONFIG_USB_TOUCHSCREEN is not set +# CONFIG_USB_YEALINK is not set +# CONFIG_USB_XPAD is not set +# CONFIG_USB_ATI_REMOTE is not set +# CONFIG_USB_ATI_REMOTE2 is not set +# CONFIG_USB_KEYSPAN_REMOTE is not set +# CONFIG_USB_APPLETOUCH is not set + +# +# USB Imaging devices +# +# CONFIG_USB_MDC800 is not set +# CONFIG_USB_MICROTEK is not set + +# +# USB Network Adapters +# +# CONFIG_USB_CATC is not set +# CONFIG_USB_KAWETH is not set +# CONFIG_USB_PEGASUS is not set +# CONFIG_USB_RTL8150 is not set +CONFIG_USB_USBNET=m +CONFIG_USB_NET_AX8817X=m +CONFIG_USB_NET_CDCETHER=m +# CONFIG_USB_NET_GL620A is not set +CONFIG_USB_NET_NET1080=m +# CONFIG_USB_NET_PLUSB is not set +# CONFIG_USB_NET_RNDIS_HOST is not set +CONFIG_USB_NET_CDC_SUBSET=m +# CONFIG_USB_ALI_M5632 is not set +# CONFIG_USB_AN2720 is not set +CONFIG_USB_BELKIN=y +CONFIG_USB_ARMLINUX=y +# CONFIG_USB_EPSON2888 is not set +CONFIG_USB_NET_ZAURUS=m +# CONFIG_USB_ZD1201 is not set +CONFIG_USB_MON=y + +# +# USB port drivers +# + +# +# USB Serial Converter support +# +CONFIG_USB_SERIAL=m +CONFIG_USB_SERIAL_GENERIC=y +CONFIG_USB_SERIAL_AIRPRIME=m +CONFIG_USB_SERIAL_ANYDATA=m +CONFIG_USB_SERIAL_ARK3116=m +CONFIG_USB_SERIAL_BELKIN=m +CONFIG_USB_SERIAL_WHITEHEAT=m +CONFIG_USB_SERIAL_DIGI_ACCELEPORT=m +CONFIG_USB_SERIAL_CP2101=m +CONFIG_USB_SERIAL_CYPRESS_M8=m +CONFIG_USB_SERIAL_EMPEG=m +CONFIG_USB_SERIAL_FTDI_SIO=m +CONFIG_USB_SERIAL_FUNSOFT=m +CONFIG_USB_SERIAL_VISOR=m +CONFIG_USB_SERIAL_IPAQ=m +CONFIG_USB_SERIAL_IR=m +CONFIG_USB_SERIAL_EDGEPORT=m +CONFIG_USB_SERIAL_EDGEPORT_TI=m +CONFIG_USB_SERIAL_GARMIN=m +CONFIG_USB_SERIAL_IPW=m +CONFIG_USB_SERIAL_KEYSPAN_PDA=m +CONFIG_USB_SERIAL_KEYSPAN=m +# CONFIG_USB_SERIAL_KEYSPAN_MPR is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XA is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XB is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA18X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19QW is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19QI is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49WLC is not set +CONFIG_USB_SERIAL_KLSI=m +CONFIG_USB_SERIAL_KOBIL_SCT=m +CONFIG_USB_SERIAL_MCT_U232=m +# CONFIG_USB_SERIAL_NAVMAN is not set +CONFIG_USB_SERIAL_PL2303=m +CONFIG_USB_SERIAL_HP4X=m +CONFIG_USB_SERIAL_SAFE=m +# CONFIG_USB_SERIAL_SAFE_PADDED is not set +CONFIG_USB_SERIAL_TI=m +CONFIG_USB_SERIAL_CYBERJACK=m +CONFIG_USB_SERIAL_XIRCOM=m +# CONFIG_USB_SERIAL_OMNINET is not set +CONFIG_USB_EZUSB=y + +# +# USB Miscellaneous drivers +# +# CONFIG_USB_EMI62 is not set +# CONFIG_USB_EMI26 is not set +# CONFIG_USB_AUERSWALD is not set +# CONFIG_USB_RIO500 is not set +# CONFIG_USB_LEGOTOWER is not set +# CONFIG_USB_LCD is not set +# CONFIG_USB_LED is not set +# CONFIG_USB_CYTHERM is not set +# CONFIG_USB_PHIDGETKIT is not set +# CONFIG_USB_PHIDGETSERVO is not set +# CONFIG_USB_IDMOUSE is not set +# CONFIG_USB_LD is not set +# CONFIG_USB_TEST is not set + +# +# USB DSL modem support +# + +# +# USB Gadget Support +# +CONFIG_USB_GADGET=y +# CONFIG_USB_GADGET_DEBUG_FILES is not set +CONFIG_USB_GADGET_SELECTED=y +# CONFIG_USB_GADGET_NET2280 is not set +# CONFIG_USB_GADGET_PXA2XX is not set +# CONFIG_USB_GADGET_PXA27X is not set +# CONFIG_USB_GADGET_GOKU is not set +# CONFIG_USB_GADGET_LH7A40X is not set +# CONFIG_USB_GADGET_OMAP is not set +# CONFIG_USB_GADGET_AT91 is not set +CONFIG_USB_GADGET_SA1100=y +CONFIG_USB_SA1100=y +# CONFIG_USB_GADGET_DUMMY_HCD is not set +# CONFIG_USB_GADGET_DUALSPEED is not set +CONFIG_USB_ZERO=m +CONFIG_USB_ETH=m +CONFIG_USB_ETH_RNDIS=y +CONFIG_USB_GADGETFS=m +CONFIG_USB_FILE_STORAGE=m +CONFIG_USB_FILE_STORAGE_TEST=y +CONFIG_USB_G_SERIAL=m + +# +# MMC/SD Card support +# +CONFIG_MMC=y +# CONFIG_MMC_DEBUG is not set +CONFIG_MMC_BLOCK=y + +# +# Real Time Clock +# +CONFIG_RTC_LIB=y +CONFIG_RTC_CLASS=y +CONFIG_RTC_HCTOSYS=y +CONFIG_RTC_HCTOSYS_DEVICE="rtc0" + +# +# RTC interfaces +# +CONFIG_RTC_INTF_SYSFS=y +CONFIG_RTC_INTF_PROC=y +CONFIG_RTC_INTF_DEV=y + +# +# RTC drivers +# +# CONFIG_RTC_DRV_X1205 is not set +# CONFIG_RTC_DRV_DS1672 is not set +# CONFIG_RTC_DRV_PCF8563 is not set +# CONFIG_RTC_DRV_RS5C372 is not set +# CONFIG_RTC_DRV_M48T86 is not set +CONFIG_RTC_DRV_SA1100=y +# CONFIG_RTC_DRV_TEST is not set + +# +# File systems +# +CONFIG_EXT2_FS=y +# CONFIG_EXT2_FS_XATTR is not set +# CONFIG_EXT2_FS_XIP is not set +# CONFIG_EXT3_FS is not set +# CONFIG_REISERFS_FS is not set +# CONFIG_JFS_FS is not set +# CONFIG_FS_POSIX_ACL is not set +# CONFIG_XFS_FS is not set +# CONFIG_OCFS2_FS is not set +# CONFIG_MINIX_FS is not set +# CONFIG_ROMFS_FS is not set +CONFIG_INOTIFY=y +# CONFIG_QUOTA is not set +CONFIG_DNOTIFY=y +# CONFIG_AUTOFS_FS is not set +# CONFIG_AUTOFS4_FS is not set +CONFIG_FUSE_FS=m + +# +# CD-ROM/DVD Filesystems +# +# CONFIG_ISO9660_FS is not set +# CONFIG_UDF_FS is not set + +# +# DOS/FAT/NT Filesystems +# +CONFIG_FAT_FS=m +CONFIG_MSDOS_FS=m +CONFIG_VFAT_FS=m +CONFIG_FAT_DEFAULT_CODEPAGE=437 +CONFIG_FAT_DEFAULT_IOCHARSET="iso8859-1" +# CONFIG_NTFS_FS is not set + +# +# Pseudo filesystems +# +CONFIG_PROC_FS=y +CONFIG_SYSFS=y +CONFIG_TMPFS=y +# CONFIG_HUGETLB_PAGE is not set +CONFIG_RAMFS=y +# CONFIG_CONFIGFS_FS is not set + +# +# Miscellaneous filesystems +# +# CONFIG_ADFS_FS is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_HFSPLUS_FS is not set +# CONFIG_BEFS_FS is not set +# CONFIG_BFS_FS is not set +# CONFIG_EFS_FS is not set +# CONFIG_JFFS_FS is not set +CONFIG_JFFS2_FS=y +CONFIG_JFFS2_FS_DEBUG=0 +CONFIG_JFFS2_FS_WRITEBUFFER=y +# CONFIG_JFFS2_SUMMARY is not set +# CONFIG_JFFS2_COMPRESSION_OPTIONS is not set +CONFIG_JFFS2_ZLIB=y +CONFIG_JFFS2_RTIME=y +# CONFIG_JFFS2_RUBIN is not set +CONFIG_CRAMFS=m +CONFIG_SQUASHFS=m +# CONFIG_SQUASHFS_EMBEDDED is not set +CONFIG_SQUASHFS_FRAGMENT_CACHE_SIZE=3 +# CONFIG_SQUASHFS_VMALLOC is not set +# CONFIG_VXFS_FS is not set +# CONFIG_HPFS_FS is not set +# CONFIG_QNX4FS_FS is not set +# CONFIG_SYSV_FS is not set +# CONFIG_UFS_FS is not set + +# +# Network File Systems +# +CONFIG_NFS_FS=m +CONFIG_NFS_V3=y +# CONFIG_NFS_V3_ACL is not set +CONFIG_NFS_V4=y +# CONFIG_NFS_DIRECTIO is not set +# CONFIG_NFSD is not set +CONFIG_LOCKD=m +CONFIG_LOCKD_V4=y +CONFIG_NFS_COMMON=y +CONFIG_SUNRPC=m +CONFIG_SUNRPC_GSS=m +CONFIG_RPCSEC_GSS_KRB5=m +# CONFIG_RPCSEC_GSS_SPKM3 is not set +CONFIG_SMB_FS=m +CONFIG_SMB_NLS_DEFAULT=y +CONFIG_SMB_NLS_REMOTE="cp437" +CONFIG_CIFS=m +# CONFIG_CIFS_STATS is not set +# CONFIG_CIFS_XATTR is not set +# CONFIG_CIFS_EXPERIMENTAL is not set +# CONFIG_NCP_FS is not set +# CONFIG_CODA_FS is not set +# CONFIG_AFS_FS is not set +# CONFIG_9P_FS is not set + +# +# Partition Types +# +CONFIG_PARTITION_ADVANCED=y +# CONFIG_ACORN_PARTITION is not set +# CONFIG_OSF_PARTITION is not set +# CONFIG_AMIGA_PARTITION is not set +# CONFIG_ATARI_PARTITION is not set +# CONFIG_MAC_PARTITION is not set +CONFIG_MSDOS_PARTITION=y +# CONFIG_BSD_DISKLABEL is not set +# CONFIG_MINIX_SUBPARTITION is not set +# CONFIG_SOLARIS_X86_PARTITION is not set +# CONFIG_UNIXWARE_DISKLABEL is not set +# CONFIG_LDM_PARTITION is not set +# CONFIG_SGI_PARTITION is not set +# CONFIG_ULTRIX_PARTITION is not set +# CONFIG_SUN_PARTITION is not set +# CONFIG_KARMA_PARTITION is not set +# CONFIG_EFI_PARTITION is not set + +# +# Native Language Support +# +CONFIG_NLS=y +CONFIG_NLS_DEFAULT="cp437" +CONFIG_NLS_CODEPAGE_437=y +CONFIG_NLS_CODEPAGE_737=m +CONFIG_NLS_CODEPAGE_775=m +CONFIG_NLS_CODEPAGE_850=m +CONFIG_NLS_CODEPAGE_852=m +CONFIG_NLS_CODEPAGE_855=m +CONFIG_NLS_CODEPAGE_857=m +CONFIG_NLS_CODEPAGE_860=m +CONFIG_NLS_CODEPAGE_861=m +CONFIG_NLS_CODEPAGE_862=m +CONFIG_NLS_CODEPAGE_863=m +CONFIG_NLS_CODEPAGE_864=m +CONFIG_NLS_CODEPAGE_865=m +CONFIG_NLS_CODEPAGE_866=m +CONFIG_NLS_CODEPAGE_869=m +CONFIG_NLS_CODEPAGE_936=m +CONFIG_NLS_CODEPAGE_950=m +CONFIG_NLS_CODEPAGE_932=m +CONFIG_NLS_CODEPAGE_949=m +CONFIG_NLS_CODEPAGE_874=m +CONFIG_NLS_ISO8859_8=m +CONFIG_NLS_CODEPAGE_1250=m +CONFIG_NLS_CODEPAGE_1251=m +CONFIG_NLS_ASCII=m +CONFIG_NLS_ISO8859_1=y +CONFIG_NLS_ISO8859_2=m +CONFIG_NLS_ISO8859_3=m +CONFIG_NLS_ISO8859_4=m +CONFIG_NLS_ISO8859_5=m +CONFIG_NLS_ISO8859_6=m +CONFIG_NLS_ISO8859_7=m +CONFIG_NLS_ISO8859_9=m +CONFIG_NLS_ISO8859_13=m +CONFIG_NLS_ISO8859_14=m +CONFIG_NLS_ISO8859_15=m +CONFIG_NLS_KOI8_R=m +CONFIG_NLS_KOI8_U=m +CONFIG_NLS_UTF8=y + +# +# Profiling support +# +# CONFIG_PROFILING is not set + +# +# Kernel hacking +# +# CONFIG_PRINTK_TIME is not set +CONFIG_MAGIC_SYSRQ=y +CONFIG_DEBUG_KERNEL=y +CONFIG_LOG_BUF_SHIFT=14 +CONFIG_DETECT_SOFTLOCKUP=y +# CONFIG_SCHEDSTATS is not set +# CONFIG_DEBUG_SLAB is not set +# CONFIG_DEBUG_PREEMPT is not set +# CONFIG_DEBUG_MUTEXES is not set +# CONFIG_DEBUG_SPINLOCK is not set +# CONFIG_DEBUG_SPINLOCK_SLEEP is not set +# CONFIG_DEBUG_KOBJECT is not set +# CONFIG_DEBUG_BUGVERBOSE is not set +# CONFIG_DEBUG_INFO is not set +# CONFIG_DEBUG_FS is not set +# CONFIG_DEBUG_VM is not set +CONFIG_FRAME_POINTER=y +# CONFIG_UNWIND_INFO is not set +# CONFIG_FORCED_INLINING is not set +# CONFIG_RCU_TORTURE_TEST is not set +# CONFIG_DEBUG_USER is not set +# CONFIG_DEBUG_WAITQ is not set +CONFIG_DEBUG_ERRORS=y +# CONFIG_DEBUG_LL is not set + +# +# Security options +# +# CONFIG_KEYS is not set +# CONFIG_SECURITY is not set + +# +# Cryptographic options +# +CONFIG_CRYPTO=y +CONFIG_CRYPTO_HMAC=y +CONFIG_CRYPTO_NULL=m +CONFIG_CRYPTO_MD4=m +CONFIG_CRYPTO_MD5=m +CONFIG_CRYPTO_SHA1=m +CONFIG_CRYPTO_SHA256=m +CONFIG_CRYPTO_SHA512=m +CONFIG_CRYPTO_WP512=m +# CONFIG_CRYPTO_TGR192 is not set +CONFIG_CRYPTO_DES=m +CONFIG_CRYPTO_BLOWFISH=m +CONFIG_CRYPTO_TWOFISH=m +CONFIG_CRYPTO_SERPENT=m +CONFIG_CRYPTO_AES=m +CONFIG_CRYPTO_CAST5=m +CONFIG_CRYPTO_CAST6=m +CONFIG_CRYPTO_TEA=m +CONFIG_CRYPTO_ARC4=m +CONFIG_CRYPTO_KHAZAD=m +CONFIG_CRYPTO_ANUBIS=m +CONFIG_CRYPTO_DEFLATE=m +CONFIG_CRYPTO_MICHAEL_MIC=m +CONFIG_CRYPTO_CRC32C=m +CONFIG_CRYPTO_TEST=m + +# +# Hardware crypto devices +# + +# +# Library routines +# +CONFIG_CRC_CCITT=m +# CONFIG_CRC16 is not set +CONFIG_CRC32=y +CONFIG_LIBCRC32C=m +CONFIG_ZLIB_INFLATE=y +CONFIG_ZLIB_DEFLATE=y diff --git a/packages/linux/linux-rp-2.6.20/defconfig-hx2000 b/packages/linux/linux-rp-2.6.20/defconfig-hx2000 new file mode 100644 index 0000000000..296c462641 --- /dev/null +++ b/packages/linux/linux-rp-2.6.20/defconfig-hx2000 @@ -0,0 +1,1161 @@ +# +# Automatically generated make config: don't edit +# Linux kernel version: 2.6.20-rc2 +# Mon Jan 1 01:49:04 2007 +# +CONFIG_ARM=y +# CONFIG_GENERIC_TIME is not set +CONFIG_MMU=y +CONFIG_GENERIC_HARDIRQS=y +CONFIG_TRACE_IRQFLAGS_SUPPORT=y +CONFIG_HARDIRQS_SW_RESEND=y +CONFIG_GENERIC_IRQ_PROBE=y +CONFIG_RWSEM_GENERIC_SPINLOCK=y +# CONFIG_ARCH_HAS_ILOG2_U32 is not set +# CONFIG_ARCH_HAS_ILOG2_U64 is not set +CONFIG_GENERIC_HWEIGHT=y +CONFIG_GENERIC_CALIBRATE_DELAY=y +CONFIG_ARCH_MTD_XIP=y +CONFIG_VECTORS_BASE=0xffff0000 +CONFIG_DEFCONFIG_LIST="/lib/modules/$UNAME_RELEASE/.config" + +# +# Code maturity level options +# +CONFIG_EXPERIMENTAL=y +CONFIG_BROKEN_ON_SMP=y +CONFIG_LOCK_KERNEL=y +CONFIG_INIT_ENV_ARG_LIMIT=32 + +# +# General setup +# +CONFIG_LOCALVERSION="" +CONFIG_LOCALVERSION_AUTO=y +# CONFIG_SWAP is not set +CONFIG_SYSVIPC=y +# CONFIG_IPC_NS is not set +# CONFIG_POSIX_MQUEUE is not set +# CONFIG_BSD_PROCESS_ACCT is not set +# CONFIG_TASKSTATS is not set +# CONFIG_UTS_NS is not set +# CONFIG_AUDIT is not set +# CONFIG_IKCONFIG is not set +CONFIG_SYSFS_DEPRECATED=y +# CONFIG_RELAY is not set +CONFIG_INITRAMFS_SOURCE="" +CONFIG_CC_OPTIMIZE_FOR_SIZE=y +CONFIG_SYSCTL=y +CONFIG_EMBEDDED=y +CONFIG_UID16=y +CONFIG_SYSCTL_SYSCALL=y +CONFIG_KALLSYMS=y +# CONFIG_KALLSYMS_ALL is not set +# CONFIG_KALLSYMS_EXTRA_PASS is not set +CONFIG_HOTPLUG=y +CONFIG_PRINTK=y +CONFIG_BUG=y +CONFIG_ELF_CORE=y +CONFIG_BASE_FULL=y +CONFIG_FUTEX=y +CONFIG_EPOLL=y +CONFIG_SHMEM=y +CONFIG_SLAB=y +CONFIG_VM_EVENT_COUNTERS=y +CONFIG_RT_MUTEXES=y +# CONFIG_TINY_SHMEM is not set +CONFIG_BASE_SMALL=0 +# CONFIG_SLOB is not set + +# +# Loadable module support +# +CONFIG_MODULES=y +CONFIG_MODULE_UNLOAD=y +# CONFIG_MODULE_FORCE_UNLOAD is not set +# CONFIG_MODVERSIONS is not set +# CONFIG_MODULE_SRCVERSION_ALL is not set +# CONFIG_KMOD is not set + +# +# Block layer +# +CONFIG_BLOCK=y +# CONFIG_LBD is not set +# CONFIG_BLK_DEV_IO_TRACE is not set +# CONFIG_LSF is not set + +# +# IO Schedulers +# +CONFIG_IOSCHED_NOOP=y +CONFIG_IOSCHED_AS=y +CONFIG_IOSCHED_DEADLINE=m +CONFIG_IOSCHED_CFQ=m +CONFIG_DEFAULT_AS=y +# CONFIG_DEFAULT_DEADLINE is not set +# CONFIG_DEFAULT_CFQ is not set +# CONFIG_DEFAULT_NOOP is not set +CONFIG_DEFAULT_IOSCHED="anticipatory" + +# +# System Type +# +# CONFIG_ARCH_AAEC2000 is not set +# CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_REALVIEW is not set +# CONFIG_ARCH_VERSATILE is not set +# CONFIG_ARCH_AT91 is not set +# CONFIG_ARCH_CLPS7500 is not set +# CONFIG_ARCH_CLPS711X is not set +# CONFIG_ARCH_CO285 is not set +# CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_EP93XX is not set +# CONFIG_ARCH_FOOTBRIDGE is not set +# CONFIG_ARCH_NETX is not set +# CONFIG_ARCH_H720X is not set +# CONFIG_ARCH_IMX is not set +# CONFIG_ARCH_IOP32X is not set +# CONFIG_ARCH_IOP33X is not set +# CONFIG_ARCH_IOP13XX is not set +# CONFIG_ARCH_IXP4XX is not set +# CONFIG_ARCH_IXP2000 is not set +# CONFIG_ARCH_IXP23XX is not set +# CONFIG_ARCH_L7200 is not set +# CONFIG_ARCH_PNX4008 is not set +CONFIG_ARCH_PXA=y +# CONFIG_ARCH_RPC is not set +# CONFIG_ARCH_SA1100 is not set +# CONFIG_ARCH_S3C2410 is not set +# CONFIG_ARCH_SHARK is not set +# CONFIG_ARCH_LH7A40X is not set +# CONFIG_ARCH_OMAP is not set + +# +# Intel PXA2xx Implementations +# +# CONFIG_ARCH_LUBBOCK is not set +# CONFIG_MACH_LOGICPD_PXA270 is not set +# CONFIG_MACH_MAINSTONE is not set +# CONFIG_ARCH_PXA_IDP is not set +# CONFIG_PXA_SHARPSL is not set +# CONFIG_MACH_TRIZEPS4 is not set +CONFIG_MACH_HX2750=y +CONFIG_PXA27x=y +CONFIG_PXA_SSP=y +CONFIG_PXA_KEYS=y + +# +# Processor Type +# +CONFIG_CPU_32=y +CONFIG_CPU_XSCALE=y +CONFIG_CPU_32v5=y +CONFIG_CPU_ABRT_EV5T=y +CONFIG_CPU_CACHE_VIVT=y +CONFIG_CPU_TLB_V4WBI=y +CONFIG_CPU_CP15=y +CONFIG_CPU_CP15_MMU=y + +# +# Processor Features +# +CONFIG_ARM_THUMB=y +# CONFIG_CPU_DCACHE_DISABLE is not set +CONFIG_IWMMXT=y +CONFIG_XSCALE_PMU=y + +# +# Bus support +# + +# +# PCCARD (PCMCIA/CardBus) support +# +CONFIG_PCCARD=y +# CONFIG_PCMCIA_DEBUG is not set +CONFIG_PCMCIA=y +CONFIG_PCMCIA_LOAD_CIS=y +CONFIG_PCMCIA_IOCTL=y + +# +# PC-card bridges +# +CONFIG_PCMCIA_PXA2XX=y + +# +# Kernel Features +# +CONFIG_PREEMPT=y +CONFIG_NO_IDLE_HZ=y +CONFIG_HZ=100 +CONFIG_AEABI=y +CONFIG_OABI_COMPAT=y +# CONFIG_ARCH_DISCONTIGMEM_ENABLE is not set +CONFIG_SELECT_MEMORY_MODEL=y +CONFIG_FLATMEM_MANUAL=y +# CONFIG_DISCONTIGMEM_MANUAL is not set +# CONFIG_SPARSEMEM_MANUAL is not set +CONFIG_FLATMEM=y +CONFIG_FLAT_NODE_MEM_MAP=y +# CONFIG_SPARSEMEM_STATIC is not set +CONFIG_SPLIT_PTLOCK_CPUS=4096 +# CONFIG_RESOURCES_64BIT is not set +CONFIG_ALIGNMENT_TRAP=y + +# +# Boot options +# +CONFIG_ZBOOT_ROM_TEXT=0x0 +CONFIG_ZBOOT_ROM_BSS=0x0 +CONFIG_CMDLINE="console=ttyS0,115200n8 console=tty1 noinitrd root=/dev/mtdblock2 rootfstype=jffs2 dyntick=enable debug" +# CONFIG_XIP_KERNEL is not set +# CONFIG_KEXEC is not set + +# +# CPU Frequency scaling +# +# CONFIG_CPU_FREQ is not set + +# +# Floating point emulation +# + +# +# At least one emulation must be selected +# +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_NWFPE_XP is not set +# CONFIG_FPE_FASTFPE is not set + +# +# Userspace binary formats +# +CONFIG_BINFMT_ELF=y +# CONFIG_BINFMT_AOUT is not set +# CONFIG_BINFMT_MISC is not set + +# +# Power management options +# +CONFIG_PM=y +# CONFIG_PM_LEGACY is not set +CONFIG_PM_DEBUG=y +# CONFIG_DISABLE_CONSOLE_SUSPEND is not set +# CONFIG_PM_SYSFS_DEPRECATED is not set +CONFIG_APM=y + +# +# Networking +# +CONFIG_NET=y + +# +# Networking options +# +# CONFIG_NETDEBUG is not set +CONFIG_PACKET=y +CONFIG_PACKET_MMAP=y +CONFIG_UNIX=y +CONFIG_XFRM=y +# CONFIG_XFRM_USER is not set +# CONFIG_XFRM_SUB_POLICY is not set +# CONFIG_NET_KEY is not set +CONFIG_INET=y +# CONFIG_IP_MULTICAST is not set +# CONFIG_IP_ADVANCED_ROUTER is not set +CONFIG_IP_FIB_HASH=y +# CONFIG_IP_PNP is not set +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set +CONFIG_SYN_COOKIES=y +# CONFIG_INET_AH is not set +# CONFIG_INET_ESP is not set +# CONFIG_INET_IPCOMP is not set +# CONFIG_INET_XFRM_TUNNEL is not set +# CONFIG_INET_TUNNEL is not set +CONFIG_INET_XFRM_MODE_TRANSPORT=y +CONFIG_INET_XFRM_MODE_TUNNEL=y +CONFIG_INET_XFRM_MODE_BEET=y +CONFIG_INET_DIAG=m +CONFIG_INET_TCP_DIAG=m +# CONFIG_TCP_CONG_ADVANCED is not set +CONFIG_TCP_CONG_CUBIC=y +CONFIG_DEFAULT_TCP_CONG="cubic" +# CONFIG_TCP_MD5SIG is not set +# CONFIG_IPV6 is not set +# CONFIG_INET6_XFRM_TUNNEL is not set +# CONFIG_INET6_TUNNEL is not set +# CONFIG_NETWORK_SECMARK is not set +# CONFIG_NETFILTER is not set + +# +# DCCP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_DCCP is not set + +# +# SCTP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_SCTP is not set + +# +# TIPC Configuration (EXPERIMENTAL) +# +# CONFIG_TIPC is not set +# CONFIG_ATM is not set +# CONFIG_BRIDGE is not set +# CONFIG_VLAN_8021Q is not set +# CONFIG_DECNET is not set +# CONFIG_LLC2 is not set +# CONFIG_IPX is not set +# CONFIG_ATALK is not set +# CONFIG_X25 is not set +# CONFIG_LAPB is not set +# CONFIG_ECONET is not set +# CONFIG_WAN_ROUTER is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network testing +# +# CONFIG_NET_PKTGEN is not set +# CONFIG_HAMRADIO is not set +# CONFIG_IRDA is not set +# CONFIG_BT is not set +CONFIG_IEEE80211=m +# CONFIG_IEEE80211_DEBUG is not set +CONFIG_IEEE80211_CRYPT_WEP=m +# CONFIG_IEEE80211_CRYPT_CCMP is not set +# CONFIG_IEEE80211_CRYPT_TKIP is not set +# CONFIG_IEEE80211_SOFTMAC is not set +CONFIG_WIRELESS_EXT=y + +# +# Device Drivers +# + +# +# Generic Driver Options +# +CONFIG_STANDALONE=y +# CONFIG_PREVENT_FIRMWARE_BUILD is not set +CONFIG_FW_LOADER=y +# CONFIG_DEBUG_DRIVER is not set +# CONFIG_SYS_HYPERVISOR is not set + +# +# Connector - unified userspace <-> kernelspace linker +# +# CONFIG_CONNECTOR is not set + +# +# Memory Technology Devices (MTD) +# +CONFIG_MTD=y +# CONFIG_MTD_DEBUG is not set +CONFIG_MTD_CONCAT=y +CONFIG_MTD_PARTITIONS=y +# CONFIG_MTD_REDBOOT_PARTS is not set +CONFIG_MTD_CMDLINE_PARTS=y +# CONFIG_MTD_AFS_PARTS is not set + +# +# User Modules And Translation Layers +# +CONFIG_MTD_CHAR=y +CONFIG_MTD_BLOCK=y +# CONFIG_FTL is not set +# CONFIG_NFTL is not set +# CONFIG_INFTL is not set +# CONFIG_RFD_FTL is not set +# CONFIG_SSFDC is not set + +# +# RAM/ROM/Flash chip drivers +# +CONFIG_MTD_CFI=y +# CONFIG_MTD_JEDECPROBE is not set +CONFIG_MTD_GEN_PROBE=y +CONFIG_MTD_CFI_ADV_OPTIONS=y +CONFIG_MTD_CFI_NOSWAP=y +# CONFIG_MTD_CFI_BE_BYTE_SWAP is not set +# CONFIG_MTD_CFI_LE_BYTE_SWAP is not set +CONFIG_MTD_CFI_GEOMETRY=y +CONFIG_MTD_MAP_BANK_WIDTH_1=y +CONFIG_MTD_MAP_BANK_WIDTH_2=y +CONFIG_MTD_MAP_BANK_WIDTH_4=y +# CONFIG_MTD_MAP_BANK_WIDTH_8 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_16 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_32 is not set +CONFIG_MTD_CFI_I1=y +CONFIG_MTD_CFI_I2=y +# CONFIG_MTD_CFI_I4 is not set +# CONFIG_MTD_CFI_I8 is not set +# CONFIG_MTD_OTP is not set +CONFIG_MTD_CFI_INTELEXT=y +# CONFIG_MTD_CFI_AMDSTD is not set +# CONFIG_MTD_CFI_STAA is not set +CONFIG_MTD_CFI_UTIL=y +# CONFIG_MTD_RAM is not set +# CONFIG_MTD_ROM is not set +# CONFIG_MTD_ABSENT is not set +# CONFIG_MTD_OBSOLETE_CHIPS is not set +# CONFIG_MTD_XIP is not set + +# +# Mapping drivers for chip access +# +# CONFIG_MTD_COMPLEX_MAPPINGS is not set +# CONFIG_MTD_PHYSMAP is not set +# CONFIG_MTD_ARM_INTEGRATOR is not set +# CONFIG_MTD_SHARP_SL is not set +# CONFIG_MTD_PLATRAM is not set + +# +# Self-contained MTD device drivers +# +# CONFIG_MTD_SLRAM is not set +# CONFIG_MTD_PHRAM is not set +# CONFIG_MTD_MTDRAM is not set +# CONFIG_MTD_BLOCK2MTD is not set + +# +# Disk-On-Chip Device Drivers +# +# CONFIG_MTD_DOC2000 is not set +# CONFIG_MTD_DOC2001 is not set +# CONFIG_MTD_DOC2001PLUS is not set + +# +# NAND Flash Device Drivers +# +# CONFIG_MTD_NAND is not set + +# +# OneNAND Flash Device Drivers +# +# CONFIG_MTD_ONENAND is not set + +# +# Parallel port support +# +# CONFIG_PARPORT is not set + +# +# Plug and Play support +# + +# +# Block devices +# +# CONFIG_BLK_DEV_COW_COMMON is not set +CONFIG_BLK_DEV_LOOP=y +# CONFIG_BLK_DEV_CRYPTOLOOP is not set +# CONFIG_BLK_DEV_NBD is not set +CONFIG_BLK_DEV_RAM=y +CONFIG_BLK_DEV_RAM_COUNT=16 +CONFIG_BLK_DEV_RAM_SIZE=8192 +CONFIG_BLK_DEV_RAM_BLOCKSIZE=1024 +CONFIG_BLK_DEV_INITRD=y +# CONFIG_CDROM_PKTCDVD is not set +# CONFIG_ATA_OVER_ETH is not set + +# +# ATA/ATAPI/MFM/RLL support +# +CONFIG_IDE=y +CONFIG_IDE_MAX_HWIFS=4 +CONFIG_BLK_DEV_IDE=y + +# +# Please see Documentation/ide.txt for help/info on IDE drives +# +# CONFIG_BLK_DEV_IDE_SATA is not set +CONFIG_BLK_DEV_IDEDISK=y +# CONFIG_IDEDISK_MULTI_MODE is not set +CONFIG_BLK_DEV_IDECS=y +# CONFIG_BLK_DEV_IDECD is not set +# CONFIG_BLK_DEV_IDETAPE is not set +# CONFIG_BLK_DEV_IDEFLOPPY is not set +# CONFIG_IDE_TASK_IOCTL is not set + +# +# IDE chipset support/bugfixes +# +CONFIG_IDE_GENERIC=y +# CONFIG_IDE_ARM is not set +# CONFIG_BLK_DEV_IDEDMA is not set +# CONFIG_IDEDMA_AUTO is not set +# CONFIG_BLK_DEV_HD is not set + +# +# SCSI device support +# +# CONFIG_RAID_ATTRS is not set +# CONFIG_SCSI is not set +# CONFIG_SCSI_NETLINK is not set + +# +# Serial ATA (prod) and Parallel ATA (experimental) drivers +# +# CONFIG_ATA is not set + +# +# Multi-device support (RAID and LVM) +# +# CONFIG_MD is not set + +# +# Fusion MPT device support +# +# CONFIG_FUSION is not set + +# +# IEEE 1394 (FireWire) support +# + +# +# I2O device support +# + +# +# Network device support +# +CONFIG_NETDEVICES=y +# CONFIG_DUMMY is not set +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +CONFIG_TUN=m + +# +# PHY device support +# +# CONFIG_PHYLIB is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +CONFIG_MII=m +# CONFIG_SMC91X is not set +# CONFIG_DM9000 is not set +# CONFIG_SMC911X is not set + +# +# Ethernet (1000 Mbit) +# + +# +# Ethernet (10000 Mbit) +# + +# +# Token Ring devices +# + +# +# Wireless LAN (non-hamradio) +# +CONFIG_NET_RADIO=y +# CONFIG_NET_WIRELESS_RTNETLINK is not set + +# +# Obsolete Wireless cards support (pre-802.11) +# +# CONFIG_STRIP is not set +# CONFIG_PCMCIA_WAVELAN is not set +# CONFIG_PCMCIA_NETWAVE is not set + +# +# Wireless 802.11 Frequency Hopping cards support +# +# CONFIG_PCMCIA_RAYCS is not set + +# +# Wireless 802.11b ISA/PCI cards support +# +CONFIG_HERMES=m +# CONFIG_ATMEL is not set + +# +# Wireless 802.11b Pcmcia/Cardbus cards support +# +CONFIG_PCMCIA_HERMES=m +CONFIG_PCMCIA_SPECTRUM=m +# CONFIG_AIRO_CS is not set +# CONFIG_PCMCIA_WL3501 is not set +CONFIG_HOSTAP=m +CONFIG_HOSTAP_FIRMWARE=y +# CONFIG_HOSTAP_FIRMWARE_NVRAM is not set +CONFIG_HOSTAP_CS=m +CONFIG_NET_WIRELESS=y + +# +# PCMCIA network device support +# +CONFIG_NET_PCMCIA=y +# CONFIG_PCMCIA_3C589 is not set +# CONFIG_PCMCIA_3C574 is not set +# CONFIG_PCMCIA_FMVJ18X is not set +CONFIG_PCMCIA_PCNET=m +# CONFIG_PCMCIA_NMCLAN is not set +# CONFIG_PCMCIA_SMC91C92 is not set +# CONFIG_PCMCIA_XIRC2PS is not set +# CONFIG_PCMCIA_AXNET is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set +CONFIG_PPP=m +# CONFIG_PPP_MULTILINK is not set +# CONFIG_PPP_FILTER is not set +CONFIG_PPP_ASYNC=m +# CONFIG_PPP_SYNC_TTY is not set +CONFIG_PPP_DEFLATE=m +CONFIG_PPP_BSDCOMP=m +# CONFIG_PPP_MPPE is not set +# CONFIG_PPPOE is not set +# CONFIG_SLIP is not set +CONFIG_SLHC=m +# CONFIG_SHAPER is not set +# CONFIG_NETCONSOLE is not set +# CONFIG_NETPOLL is not set +# CONFIG_NET_POLL_CONTROLLER is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Input device support +# +CONFIG_INPUT=y +# CONFIG_INPUT_FF_MEMLESS is not set + +# +# Userland interfaces +# +CONFIG_INPUT_MOUSEDEV=m +# CONFIG_INPUT_MOUSEDEV_PSAUX is not set +CONFIG_INPUT_MOUSEDEV_SCREEN_X=240 +CONFIG_INPUT_MOUSEDEV_SCREEN_Y=320 +# CONFIG_INPUT_JOYDEV is not set +# CONFIG_INPUT_TSDEV is not set +CONFIG_INPUT_EVDEV=y +# CONFIG_INPUT_EVBUG is not set +CONFIG_INPUT_POWER=y + +# +# Input Device Drivers +# +# CONFIG_INPUT_KEYBOARD is not set +# CONFIG_INPUT_MOUSE is not set +# CONFIG_INPUT_JOYSTICK is not set +CONFIG_INPUT_TOUCHSCREEN=y +# CONFIG_TOUCHSCREEN_GUNZE is not set +# CONFIG_TOUCHSCREEN_ELO is not set +# CONFIG_TOUCHSCREEN_MTOUCH is not set +# CONFIG_TOUCHSCREEN_MK712 is not set +# CONFIG_TOUCHSCREEN_PENMOUNT is not set +# CONFIG_TOUCHSCREEN_TOUCHRIGHT is not set +# CONFIG_TOUCHSCREEN_TOUCHWIN is not set +# CONFIG_TOUCHSCREEN_UCB1400 is not set +CONFIG_TOUCHSCREEN_TSC2101=y +# CONFIG_INPUT_MISC is not set + +# +# Hardware I/O ports +# +# CONFIG_SERIO is not set +# CONFIG_GAMEPORT is not set + +# +# Character devices +# +CONFIG_VT=y +CONFIG_VT_CONSOLE=y +CONFIG_HW_CONSOLE=y +# CONFIG_VT_HW_CONSOLE_BINDING is not set +# CONFIG_SERIAL_NONSTANDARD is not set + +# +# Serial drivers +# +# CONFIG_SERIAL_8250 is not set + +# +# Non-8250 serial port support +# +CONFIG_SERIAL_PXA=y +CONFIG_SERIAL_PXA_CONSOLE=y +CONFIG_SERIAL_CORE=y +CONFIG_SERIAL_CORE_CONSOLE=y +CONFIG_UNIX98_PTYS=y +# CONFIG_LEGACY_PTYS is not set + +# +# IPMI +# +# CONFIG_IPMI_HANDLER is not set + +# +# Watchdog Cards +# +# CONFIG_WATCHDOG is not set +CONFIG_HW_RANDOM=m +# CONFIG_NVRAM is not set +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set + +# +# PCMCIA character devices +# +# CONFIG_SYNCLINK_CS is not set +# CONFIG_CARDMAN_4000 is not set +# CONFIG_CARDMAN_4040 is not set +# CONFIG_RAW_DRIVER is not set + +# +# TPM devices +# +# CONFIG_TCG_TPM is not set + +# +# I2C support +# +# CONFIG_I2C is not set + +# +# SPI support +# +# CONFIG_SPI is not set +# CONFIG_SPI_MASTER is not set + +# +# Dallas's 1-wire bus +# +# CONFIG_W1 is not set + +# +# Hardware Monitoring support +# +# CONFIG_HWMON is not set +# CONFIG_HWMON_VID is not set + +# +# Misc devices +# +# CONFIG_TIFM_CORE is not set + +# +# Multi-Function Devices +# +CONFIG_MFD_TSC2101=y + +# +# LED devices +# +# CONFIG_NEW_LEDS is not set + +# +# LED drivers +# + +# +# LED Triggers +# + +# +# Multimedia devices +# +# CONFIG_VIDEO_DEV is not set + +# +# Digital Video Broadcasting Devices +# +# CONFIG_DVB is not set + +# +# Graphics support +# +CONFIG_FIRMWARE_EDID=y +CONFIG_FB=y +CONFIG_FB_CFB_FILLRECT=y +CONFIG_FB_CFB_COPYAREA=y +CONFIG_FB_CFB_IMAGEBLIT=y +# CONFIG_FB_MACMODES is not set +# CONFIG_FB_BACKLIGHT is not set +# CONFIG_FB_MODE_HELPERS is not set +# CONFIG_FB_TILEBLITTING is not set +# CONFIG_FB_S1D13XXX is not set +CONFIG_FB_PXA=y +CONFIG_FB_PXA_LCD_QVGA=y +# CONFIG_FB_PXA_LCD_VGA is not set +# CONFIG_FB_PXA_OVERLAY is not set +# CONFIG_FB_PXA_PARAMETERS is not set +# CONFIG_FB_MBX is not set +# CONFIG_FB_VIRTUAL is not set + +# +# Console display driver support +# +# CONFIG_VGA_CONSOLE is not set +CONFIG_DUMMY_CONSOLE=y +CONFIG_FRAMEBUFFER_CONSOLE=y +# CONFIG_FRAMEBUFFER_CONSOLE_ROTATION is not set +# CONFIG_FONTS is not set +CONFIG_FONT_8x8=y +CONFIG_FONT_8x16=y + +# +# Logo configuration +# +CONFIG_LOGO=y +CONFIG_LOGO_LINUX_MONO=y +CONFIG_LOGO_LINUX_VGA16=y +# CONFIG_LOGO_LINUX_CLUT224 is not set +CONFIG_LOGO_OHAND_CLUT224=y +# CONFIG_LOGO_OZ240_CLUT224 is not set +# CONFIG_LOGO_OZ480_CLUT224 is not set +# CONFIG_LOGO_OZ640_CLUT224 is not set +CONFIG_BACKLIGHT_LCD_SUPPORT=y +CONFIG_BACKLIGHT_CLASS_DEVICE=y +CONFIG_BACKLIGHT_DEVICE=y +CONFIG_LCD_CLASS_DEVICE=y +CONFIG_LCD_DEVICE=y +CONFIG_BACKLIGHT_HX2750=y + +# +# Sound +# +# CONFIG_SOUND is not set + +# +# HID Devices +# +CONFIG_HID=y + +# +# USB support +# +CONFIG_USB_ARCH_HAS_HCD=y +CONFIG_USB_ARCH_HAS_OHCI=y +# CONFIG_USB_ARCH_HAS_EHCI is not set +# CONFIG_USB is not set + +# +# NOTE: USB_STORAGE enables SCSI, and 'SCSI disk support' +# + +# +# USB Gadget Support +# +# CONFIG_USB_GADGET is not set + +# +# MMC/SD Card support +# +CONFIG_MMC=y +# CONFIG_MMC_DEBUG is not set +CONFIG_MMC_BLOCK=y +CONFIG_MMC_PXA=y +# CONFIG_MMC_TIFM_SD is not set + +# +# Real Time Clock +# +CONFIG_RTC_LIB=y +CONFIG_RTC_CLASS=y +CONFIG_RTC_HCTOSYS=y +CONFIG_RTC_HCTOSYS_DEVICE="rtc0" +# CONFIG_RTC_DEBUG is not set + +# +# RTC interfaces +# +CONFIG_RTC_INTF_SYSFS=y +CONFIG_RTC_INTF_PROC=y +CONFIG_RTC_INTF_DEV=y +# CONFIG_RTC_INTF_DEV_UIE_EMUL is not set + +# +# RTC drivers +# +# CONFIG_RTC_DRV_DS1553 is not set +# CONFIG_RTC_DRV_DS1742 is not set +# CONFIG_RTC_DRV_M48T86 is not set +CONFIG_RTC_DRV_SA1100=y +# CONFIG_RTC_DRV_TEST is not set +# CONFIG_RTC_DRV_V3020 is not set + +# +# File systems +# +CONFIG_EXT2_FS=y +# CONFIG_EXT2_FS_XATTR is not set +# CONFIG_EXT2_FS_XIP is not set +# CONFIG_EXT3_FS is not set +# CONFIG_EXT4DEV_FS is not set +# CONFIG_REISERFS_FS is not set +# CONFIG_JFS_FS is not set +# CONFIG_FS_POSIX_ACL is not set +# CONFIG_XFS_FS is not set +# CONFIG_GFS2_FS is not set +# CONFIG_OCFS2_FS is not set +# CONFIG_MINIX_FS is not set +# CONFIG_ROMFS_FS is not set +CONFIG_INOTIFY=y +CONFIG_INOTIFY_USER=y +# CONFIG_QUOTA is not set +CONFIG_DNOTIFY=y +# CONFIG_AUTOFS_FS is not set +# CONFIG_AUTOFS4_FS is not set +# CONFIG_FUSE_FS is not set + +# +# CD-ROM/DVD Filesystems +# +# CONFIG_ISO9660_FS is not set +# CONFIG_UDF_FS is not set + +# +# DOS/FAT/NT Filesystems +# +CONFIG_FAT_FS=y +# CONFIG_MSDOS_FS is not set +CONFIG_VFAT_FS=y +CONFIG_FAT_DEFAULT_CODEPAGE=437 +CONFIG_FAT_DEFAULT_IOCHARSET="iso8859-1" +# CONFIG_NTFS_FS is not set + +# +# Pseudo filesystems +# +CONFIG_PROC_FS=y +CONFIG_PROC_SYSCTL=y +CONFIG_SYSFS=y +CONFIG_TMPFS=y +# CONFIG_TMPFS_POSIX_ACL is not set +# CONFIG_HUGETLB_PAGE is not set +CONFIG_RAMFS=y +# CONFIG_CONFIGFS_FS is not set + +# +# Miscellaneous filesystems +# +# CONFIG_ADFS_FS is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_HFSPLUS_FS is not set +# CONFIG_BEFS_FS is not set +# CONFIG_BFS_FS is not set +# CONFIG_EFS_FS is not set +CONFIG_JFFS2_FS=y +CONFIG_JFFS2_FS_DEBUG=0 +CONFIG_JFFS2_FS_WRITEBUFFER=y +# CONFIG_JFFS2_SUMMARY is not set +# CONFIG_JFFS2_FS_XATTR is not set +# CONFIG_JFFS2_COMPRESSION_OPTIONS is not set +CONFIG_JFFS2_ZLIB=y +CONFIG_JFFS2_RTIME=y +# CONFIG_JFFS2_RUBIN is not set +CONFIG_CRAMFS=y +CONFIG_SQUASHFS=m +# CONFIG_SQUASHFS_EMBEDDED is not set +CONFIG_SQUASHFS_FRAGMENT_CACHE_SIZE=3 +# CONFIG_SQUASHFS_VMALLOC is not set +# CONFIG_VXFS_FS is not set +# CONFIG_HPFS_FS is not set +# CONFIG_QNX4FS_FS is not set +# CONFIG_SYSV_FS is not set +# CONFIG_UFS_FS is not set + +# +# Network File Systems +# +# CONFIG_NFS_FS is not set +# CONFIG_NFSD is not set +# CONFIG_SMB_FS is not set +# CONFIG_CIFS is not set +# CONFIG_NCP_FS is not set +# CONFIG_CODA_FS is not set +# CONFIG_AFS_FS is not set +# CONFIG_9P_FS is not set + +# +# Partition Types +# +CONFIG_PARTITION_ADVANCED=y +# CONFIG_ACORN_PARTITION is not set +# CONFIG_OSF_PARTITION is not set +# CONFIG_AMIGA_PARTITION is not set +# CONFIG_ATARI_PARTITION is not set +# CONFIG_MAC_PARTITION is not set +CONFIG_MSDOS_PARTITION=y +# CONFIG_BSD_DISKLABEL is not set +# CONFIG_MINIX_SUBPARTITION is not set +# CONFIG_SOLARIS_X86_PARTITION is not set +# CONFIG_UNIXWARE_DISKLABEL is not set +# CONFIG_LDM_PARTITION is not set +# CONFIG_SGI_PARTITION is not set +# CONFIG_ULTRIX_PARTITION is not set +# CONFIG_SUN_PARTITION is not set +# CONFIG_KARMA_PARTITION is not set +# CONFIG_EFI_PARTITION is not set + +# +# Native Language Support +# +CONFIG_NLS=y +CONFIG_NLS_DEFAULT="cp437" +CONFIG_NLS_CODEPAGE_437=y +# CONFIG_NLS_CODEPAGE_737 is not set +# CONFIG_NLS_CODEPAGE_775 is not set +# CONFIG_NLS_CODEPAGE_850 is not set +# CONFIG_NLS_CODEPAGE_852 is not set +# CONFIG_NLS_CODEPAGE_855 is not set +# CONFIG_NLS_CODEPAGE_857 is not set +# CONFIG_NLS_CODEPAGE_860 is not set +# CONFIG_NLS_CODEPAGE_861 is not set +# CONFIG_NLS_CODEPAGE_862 is not set +# CONFIG_NLS_CODEPAGE_863 is not set +# CONFIG_NLS_CODEPAGE_864 is not set +# CONFIG_NLS_CODEPAGE_865 is not set +# CONFIG_NLS_CODEPAGE_866 is not set +# CONFIG_NLS_CODEPAGE_869 is not set +# CONFIG_NLS_CODEPAGE_936 is not set +# CONFIG_NLS_CODEPAGE_950 is not set +# CONFIG_NLS_CODEPAGE_932 is not set +# CONFIG_NLS_CODEPAGE_949 is not set +# CONFIG_NLS_CODEPAGE_874 is not set +# CONFIG_NLS_ISO8859_8 is not set +# CONFIG_NLS_CODEPAGE_1250 is not set +# CONFIG_NLS_CODEPAGE_1251 is not set +# CONFIG_NLS_ASCII is not set +CONFIG_NLS_ISO8859_1=y +# CONFIG_NLS_ISO8859_2 is not set +# CONFIG_NLS_ISO8859_3 is not set +# CONFIG_NLS_ISO8859_4 is not set +# CONFIG_NLS_ISO8859_5 is not set +# CONFIG_NLS_ISO8859_6 is not set +# CONFIG_NLS_ISO8859_7 is not set +# CONFIG_NLS_ISO8859_9 is not set +# CONFIG_NLS_ISO8859_13 is not set +# CONFIG_NLS_ISO8859_14 is not set +# CONFIG_NLS_ISO8859_15 is not set +# CONFIG_NLS_KOI8_R is not set +# CONFIG_NLS_KOI8_U is not set +CONFIG_NLS_UTF8=y + +# +# Distributed Lock Manager +# +# CONFIG_DLM is not set + +# +# Profiling support +# +# CONFIG_PROFILING is not set + +# +# Kernel hacking +# +# CONFIG_PRINTK_TIME is not set +CONFIG_ENABLE_MUST_CHECK=y +# CONFIG_MAGIC_SYSRQ is not set +# CONFIG_UNUSED_SYMBOLS is not set +# CONFIG_DEBUG_FS is not set +# CONFIG_HEADERS_CHECK is not set +CONFIG_DEBUG_KERNEL=y +CONFIG_LOG_BUF_SHIFT=14 +CONFIG_DETECT_SOFTLOCKUP=y +# CONFIG_SCHEDSTATS is not set +# CONFIG_DEBUG_SLAB is not set +CONFIG_DEBUG_PREEMPT=y +# CONFIG_DEBUG_RT_MUTEXES is not set +# CONFIG_RT_MUTEX_TESTER is not set +# CONFIG_DEBUG_SPINLOCK is not set +# CONFIG_DEBUG_MUTEXES is not set +# CONFIG_DEBUG_RWSEMS is not set +# CONFIG_DEBUG_SPINLOCK_SLEEP is not set +# CONFIG_DEBUG_LOCKING_API_SELFTESTS is not set +# CONFIG_DEBUG_KOBJECT is not set +# CONFIG_DEBUG_BUGVERBOSE is not set +# CONFIG_DEBUG_INFO is not set +# CONFIG_DEBUG_VM is not set +# CONFIG_DEBUG_LIST is not set +CONFIG_FRAME_POINTER=y +CONFIG_FORCED_INLINING=y +# CONFIG_RCU_TORTURE_TEST is not set +# CONFIG_DEBUG_USER is not set +CONFIG_DEBUG_ERRORS=y +CONFIG_DEBUG_LL=y +# CONFIG_DEBUG_ICEDCC is not set + +# +# Security options +# +# CONFIG_KEYS is not set +# CONFIG_SECURITY is not set + +# +# Cryptographic options +# +CONFIG_CRYPTO=y +CONFIG_CRYPTO_ALGAPI=y +CONFIG_CRYPTO_BLKCIPHER=m +CONFIG_CRYPTO_MANAGER=m +# CONFIG_CRYPTO_HMAC is not set +# CONFIG_CRYPTO_XCBC is not set +# CONFIG_CRYPTO_NULL is not set +# CONFIG_CRYPTO_MD4 is not set +# CONFIG_CRYPTO_MD5 is not set +# CONFIG_CRYPTO_SHA1 is not set +# CONFIG_CRYPTO_SHA256 is not set +# CONFIG_CRYPTO_SHA512 is not set +# CONFIG_CRYPTO_WP512 is not set +# CONFIG_CRYPTO_TGR192 is not set +# CONFIG_CRYPTO_GF128MUL is not set +CONFIG_CRYPTO_ECB=m +CONFIG_CRYPTO_CBC=m +# CONFIG_CRYPTO_LRW is not set +# CONFIG_CRYPTO_DES is not set +# CONFIG_CRYPTO_BLOWFISH is not set +# CONFIG_CRYPTO_TWOFISH is not set +# CONFIG_CRYPTO_SERPENT is not set +# CONFIG_CRYPTO_AES is not set +# CONFIG_CRYPTO_CAST5 is not set +# CONFIG_CRYPTO_CAST6 is not set +# CONFIG_CRYPTO_TEA is not set +CONFIG_CRYPTO_ARC4=m +# CONFIG_CRYPTO_KHAZAD is not set +# CONFIG_CRYPTO_ANUBIS is not set +CONFIG_CRYPTO_DEFLATE=y +# CONFIG_CRYPTO_MICHAEL_MIC is not set +CONFIG_CRYPTO_CRC32C=y +# CONFIG_CRYPTO_TEST is not set + +# +# Hardware crypto devices +# + +# +# Library routines +# +CONFIG_BITREVERSE=y +CONFIG_CRC_CCITT=m +# CONFIG_CRC16 is not set +CONFIG_CRC32=y +CONFIG_LIBCRC32C=y +CONFIG_ZLIB_INFLATE=y +CONFIG_ZLIB_DEFLATE=y +CONFIG_PLIST=y +CONFIG_IOMAP_COPY=y diff --git a/packages/linux/linux-rp-2.6.20/defconfig-ipaq-pxa270 b/packages/linux/linux-rp-2.6.20/defconfig-ipaq-pxa270 new file mode 100644 index 0000000000..cb7cce7bdd --- /dev/null +++ b/packages/linux/linux-rp-2.6.20/defconfig-ipaq-pxa270 @@ -0,0 +1,1021 @@ +# +# Automatically generated make config: don't edit +# Linux kernel version: 2.6.15-rc1-git7 +# Sat Nov 19 23:13:51 2005 +# +CONFIG_ARM=y +CONFIG_MMU=y +CONFIG_UID16=y +CONFIG_RWSEM_GENERIC_SPINLOCK=y +CONFIG_GENERIC_CALIBRATE_DELAY=y + +# +# Code maturity level options +# +CONFIG_EXPERIMENTAL=y +# CONFIG_CLEAN_COMPILE is not set +CONFIG_BROKEN=y +CONFIG_BROKEN_ON_SMP=y +CONFIG_LOCK_KERNEL=y +CONFIG_INIT_ENV_ARG_LIMIT=32 + +# +# General setup +# +CONFIG_LOCALVERSION="" +CONFIG_LOCALVERSION_AUTO=y +# CONFIG_SWAP is not set +# CONFIG_SYSVIPC is not set +# CONFIG_POSIX_MQUEUE is not set +# CONFIG_BSD_PROCESS_ACCT is not set +CONFIG_SYSCTL=y +# CONFIG_AUDIT is not set +CONFIG_HOTPLUG=y +CONFIG_KOBJECT_UEVENT=y +# CONFIG_IKCONFIG is not set +CONFIG_INITRAMFS_SOURCE="" +CONFIG_EMBEDDED=y +CONFIG_KALLSYMS=y +# CONFIG_KALLSYMS_ALL is not set +# CONFIG_KALLSYMS_EXTRA_PASS is not set +CONFIG_PRINTK=y +CONFIG_BUG=y +CONFIG_BASE_FULL=y +CONFIG_FUTEX=y +CONFIG_EPOLL=y +CONFIG_CC_OPTIMIZE_FOR_SIZE=y +CONFIG_SHMEM=y +CONFIG_CC_ALIGN_FUNCTIONS=0 +CONFIG_CC_ALIGN_LABELS=0 +CONFIG_CC_ALIGN_LOOPS=0 +CONFIG_CC_ALIGN_JUMPS=0 +# CONFIG_TINY_SHMEM is not set +CONFIG_BASE_SMALL=0 + +# +# Loadable module support +# +CONFIG_MODULES=y +CONFIG_MODULE_UNLOAD=y +# CONFIG_MODULE_FORCE_UNLOAD is not set +CONFIG_OBSOLETE_MODPARM=y +# CONFIG_MODVERSIONS is not set +# CONFIG_MODULE_SRCVERSION_ALL is not set +# CONFIG_KMOD is not set + +# +# Block layer +# + +# +# IO Schedulers +# +CONFIG_IOSCHED_NOOP=y +CONFIG_IOSCHED_AS=y +CONFIG_IOSCHED_DEADLINE=m +CONFIG_IOSCHED_CFQ=m +CONFIG_DEFAULT_AS=y +# CONFIG_DEFAULT_DEADLINE is not set +# CONFIG_DEFAULT_CFQ is not set +# CONFIG_DEFAULT_NOOP is not set +CONFIG_DEFAULT_IOSCHED="anticipatory" + +# +# System Type +# +# CONFIG_ARCH_CLPS7500 is not set +# CONFIG_ARCH_CLPS711X is not set +# CONFIG_ARCH_CO285 is not set +# CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_CAMELOT is not set +# CONFIG_ARCH_FOOTBRIDGE is not set +# CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_IOP3XX is not set +# CONFIG_ARCH_IXP4XX is not set +# CONFIG_ARCH_IXP2000 is not set +# CONFIG_ARCH_L7200 is not set +CONFIG_ARCH_PXA=y +# CONFIG_ARCH_RPC is not set +# CONFIG_ARCH_SA1100 is not set +# CONFIG_ARCH_S3C2410 is not set +# CONFIG_ARCH_SHARK is not set +# CONFIG_ARCH_LH7A40X is not set +# CONFIG_ARCH_OMAP is not set +# CONFIG_ARCH_VERSATILE is not set +# CONFIG_ARCH_REALVIEW is not set +# CONFIG_ARCH_IMX is not set +# CONFIG_ARCH_H720X is not set +# CONFIG_ARCH_AAEC2000 is not set + +# +# Intel PXA2xx Implementations +# +# CONFIG_ARCH_LUBBOCK is not set +# CONFIG_MACH_MAINSTONE is not set +# CONFIG_ARCH_PXA_IDP is not set +# CONFIG_PXA_SHARPSL is not set +CONFIG_MACH_HX2750=y +CONFIG_PXA27x=y +CONFIG_PXA_KEYS=y +CONFIG_PXA_SSP=y + +# +# Processor Type +# +CONFIG_CPU_32=y +CONFIG_CPU_XSCALE=y +CONFIG_CPU_32v5=y +CONFIG_CPU_ABRT_EV5T=y +CONFIG_CPU_CACHE_VIVT=y +CONFIG_CPU_TLB_V4WBI=y + +# +# Processor Features +# +CONFIG_ARM_THUMB=y +CONFIG_XSCALE_PMU=y + +# +# Bus support +# +CONFIG_ISA_DMA_API=y + +# +# PCCARD (PCMCIA/CardBus) support +# +CONFIG_PCCARD=y +# CONFIG_PCMCIA_DEBUG is not set +CONFIG_PCMCIA=y +CONFIG_PCMCIA_LOAD_CIS=y +CONFIG_PCMCIA_IOCTL=y + +# +# PC-card bridges +# +CONFIG_PCMCIA_PXA2XX=y + +# +# Kernel Features +# +CONFIG_PREEMPT=y +CONFIG_NO_IDLE_HZ=y +# CONFIG_ARCH_DISCONTIGMEM_ENABLE is not set +CONFIG_SELECT_MEMORY_MODEL=y +CONFIG_FLATMEM_MANUAL=y +# CONFIG_DISCONTIGMEM_MANUAL is not set +# CONFIG_SPARSEMEM_MANUAL is not set +CONFIG_FLATMEM=y +CONFIG_FLAT_NODE_MEM_MAP=y +# CONFIG_SPARSEMEM_STATIC is not set +CONFIG_SPLIT_PTLOCK_CPUS=4096 +CONFIG_ALIGNMENT_TRAP=y + +# +# Boot options +# +CONFIG_ZBOOT_ROM_TEXT=0x0 +CONFIG_ZBOOT_ROM_BSS=0x0 +# CONFIG_XIP_KERNEL is not set + +# +# CPU Frequency scaling +# +# CONFIG_CPU_FREQ is not set + +# +# Floating point emulation +# + +# +# At least one emulation must be selected +# +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_NWFPE_XP is not set +# CONFIG_FPE_FASTFPE is not set + +# +# Userspace binary formats +# +CONFIG_BINFMT_ELF=y +# CONFIG_BINFMT_AOUT is not set +# CONFIG_BINFMT_MISC is not set +# CONFIG_ARTHUR is not set + +# +# Power management options +# +CONFIG_PM=y +# CONFIG_PM_LEGACY is not set +# CONFIG_PM_DEBUG is not set +CONFIG_APM=y + +# +# Networking +# +CONFIG_NET=y + +# +# Networking options +# +CONFIG_PACKET=y +CONFIG_PACKET_MMAP=y +CONFIG_UNIX=y +# CONFIG_NET_KEY is not set +CONFIG_INET=y +# CONFIG_IP_MULTICAST is not set +# CONFIG_IP_ADVANCED_ROUTER is not set +CONFIG_IP_FIB_HASH=y +# CONFIG_IP_PNP is not set +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set +CONFIG_SYN_COOKIES=y +# CONFIG_INET_AH is not set +# CONFIG_INET_ESP is not set +# CONFIG_INET_IPCOMP is not set +# CONFIG_INET_TUNNEL is not set +CONFIG_INET_DIAG=m +CONFIG_INET_TCP_DIAG=m +# CONFIG_TCP_CONG_ADVANCED is not set +CONFIG_TCP_CONG_BIC=y +# CONFIG_IPV6 is not set +# CONFIG_NETFILTER is not set + +# +# DCCP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_DCCP is not set + +# +# SCTP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_SCTP is not set +# CONFIG_ATM is not set +# CONFIG_BRIDGE is not set +# CONFIG_VLAN_8021Q is not set +# CONFIG_DECNET is not set +# CONFIG_LLC2 is not set +# CONFIG_IPX is not set +# CONFIG_ATALK is not set +# CONFIG_X25 is not set +# CONFIG_LAPB is not set +# CONFIG_NET_DIVERT is not set +# CONFIG_ECONET is not set +# CONFIG_WAN_ROUTER is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network testing +# +# CONFIG_NET_PKTGEN is not set +# CONFIG_HAMRADIO is not set +# CONFIG_IRDA is not set +# CONFIG_BT is not set +CONFIG_IEEE80211=m +# CONFIG_IEEE80211_DEBUG is not set +CONFIG_IEEE80211_CRYPT_WEP=m +# CONFIG_IEEE80211_CRYPT_CCMP is not set +# CONFIG_IEEE80211_CRYPT_TKIP is not set + +# +# Device Drivers +# + +# +# Generic Driver Options +# +CONFIG_STANDALONE=y +# CONFIG_PREVENT_FIRMWARE_BUILD is not set +CONFIG_FW_LOADER=y +# CONFIG_DEBUG_DRIVER is not set + +# +# Connector - unified userspace <-> kernelspace linker +# +# CONFIG_CONNECTOR is not set + +# +# Memory Technology Devices (MTD) +# +CONFIG_MTD=y +# CONFIG_MTD_DEBUG is not set +CONFIG_MTD_CONCAT=y +CONFIG_MTD_PARTITIONS=y +# CONFIG_MTD_REDBOOT_PARTS is not set +CONFIG_MTD_CMDLINE_PARTS=y +# CONFIG_MTD_AFS_PARTS is not set + +# +# User Modules And Translation Layers +# +CONFIG_MTD_CHAR=y +CONFIG_MTD_BLOCK=y +# CONFIG_FTL is not set +# CONFIG_NFTL is not set +# CONFIG_INFTL is not set +# CONFIG_RFD_FTL is not set + +# +# RAM/ROM/Flash chip drivers +# +CONFIG_MTD_CFI=y +# CONFIG_MTD_JEDECPROBE is not set +CONFIG_MTD_GEN_PROBE=y +CONFIG_MTD_CFI_ADV_OPTIONS=y +CONFIG_MTD_CFI_NOSWAP=y +# CONFIG_MTD_CFI_BE_BYTE_SWAP is not set +# CONFIG_MTD_CFI_LE_BYTE_SWAP is not set +CONFIG_MTD_CFI_GEOMETRY=y +CONFIG_MTD_MAP_BANK_WIDTH_1=y +CONFIG_MTD_MAP_BANK_WIDTH_2=y +CONFIG_MTD_MAP_BANK_WIDTH_4=y +# CONFIG_MTD_MAP_BANK_WIDTH_8 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_16 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_32 is not set +CONFIG_MTD_CFI_I1=y +CONFIG_MTD_CFI_I2=y +# CONFIG_MTD_CFI_I4 is not set +# CONFIG_MTD_CFI_I8 is not set +# CONFIG_MTD_OTP is not set +CONFIG_MTD_CFI_INTELEXT=y +# CONFIG_MTD_CFI_AMDSTD is not set +# CONFIG_MTD_CFI_STAA is not set +CONFIG_MTD_CFI_UTIL=y +# CONFIG_MTD_RAM is not set +# CONFIG_MTD_ROM is not set +# CONFIG_MTD_ABSENT is not set +# CONFIG_MTD_OBSOLETE_CHIPS is not set +# CONFIG_MTD_XIP is not set + +# +# Mapping drivers for chip access +# +# CONFIG_MTD_COMPLEX_MAPPINGS is not set +# CONFIG_MTD_PHYSMAP is not set +# CONFIG_MTD_ARM_INTEGRATOR is not set +# CONFIG_MTD_SHARP_SL is not set +# CONFIG_MTD_PLATRAM is not set + +# +# Self-contained MTD device drivers +# +# CONFIG_MTD_SLRAM is not set +# CONFIG_MTD_PHRAM is not set +# CONFIG_MTD_MTDRAM is not set +# CONFIG_MTD_BLKMTD is not set +# CONFIG_MTD_BLOCK2MTD is not set + +# +# Disk-On-Chip Device Drivers +# +# CONFIG_MTD_DOC2000 is not set +# CONFIG_MTD_DOC2001 is not set +# CONFIG_MTD_DOC2001PLUS is not set + +# +# NAND Flash Device Drivers +# +# CONFIG_MTD_NAND is not set + +# +# OneNAND Flash Device Drivers +# +# CONFIG_MTD_ONENAND is not set + +# +# Parallel port support +# +# CONFIG_PARPORT is not set + +# +# Plug and Play support +# + +# +# Block devices +# +# CONFIG_BLK_DEV_COW_COMMON is not set +CONFIG_BLK_DEV_LOOP=y +# CONFIG_BLK_DEV_CRYPTOLOOP is not set +# CONFIG_BLK_DEV_NBD is not set +CONFIG_BLK_DEV_RAM=y +CONFIG_BLK_DEV_RAM_COUNT=16 +CONFIG_BLK_DEV_RAM_SIZE=8192 +CONFIG_BLK_DEV_INITRD=y +# CONFIG_CDROM_PKTCDVD is not set +# CONFIG_ATA_OVER_ETH is not set + +# +# ATA/ATAPI/MFM/RLL support +# +CONFIG_IDE=y +CONFIG_BLK_DEV_IDE=y + +# +# Please see Documentation/ide.txt for help/info on IDE drives +# +# CONFIG_BLK_DEV_IDE_SATA is not set +CONFIG_BLK_DEV_IDEDISK=y +# CONFIG_IDEDISK_MULTI_MODE is not set +CONFIG_BLK_DEV_IDECS=y +# CONFIG_BLK_DEV_IDECD is not set +# CONFIG_BLK_DEV_IDETAPE is not set +# CONFIG_BLK_DEV_IDEFLOPPY is not set +# CONFIG_IDE_TASK_IOCTL is not set + +# +# IDE chipset support/bugfixes +# +CONFIG_IDE_GENERIC=y +# CONFIG_IDE_ARM is not set +# CONFIG_BLK_DEV_IDEDMA is not set +# CONFIG_IDEDMA_AUTO is not set +# CONFIG_BLK_DEV_HD is not set + +# +# SCSI device support +# +# CONFIG_RAID_ATTRS is not set +# CONFIG_SCSI is not set + +# +# Multi-device support (RAID and LVM) +# +# CONFIG_MD is not set + +# +# Fusion MPT device support +# +# CONFIG_FUSION is not set + +# +# IEEE 1394 (FireWire) support +# +# CONFIG_IEEE1394 is not set + +# +# I2O device support +# + +# +# Network device support +# +CONFIG_NETDEVICES=y +# CONFIG_DUMMY is not set +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +CONFIG_TUN=m + +# +# PHY device support +# +# CONFIG_PHYLIB is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +CONFIG_MII=m +# CONFIG_SMC91X is not set +# CONFIG_DM9000 is not set + +# +# Ethernet (1000 Mbit) +# + +# +# Ethernet (10000 Mbit) +# + +# +# Token Ring devices +# + +# +# Wireless LAN (non-hamradio) +# +CONFIG_NET_RADIO=y + +# +# Obsolete Wireless cards support (pre-802.11) +# +# CONFIG_STRIP is not set +# CONFIG_PCMCIA_WAVELAN is not set +# CONFIG_PCMCIA_NETWAVE is not set + +# +# Wireless 802.11 Frequency Hopping cards support +# +# CONFIG_PCMCIA_RAYCS is not set + +# +# Wireless 802.11b ISA/PCI cards support +# +# CONFIG_AIRO is not set +CONFIG_HERMES=m +# CONFIG_ATMEL is not set + +# +# Wireless 802.11b Pcmcia/Cardbus cards support +# +CONFIG_PCMCIA_HERMES=m +CONFIG_PCMCIA_SPECTRUM=m +# CONFIG_AIRO_CS is not set +# CONFIG_PCMCIA_WL3501 is not set +CONFIG_HOSTAP=m +CONFIG_HOSTAP_FIRMWARE=y +CONFIG_HOSTAP_CS=m +CONFIG_NET_WIRELESS=y + +# +# PCMCIA network device support +# +CONFIG_NET_PCMCIA=y +# CONFIG_PCMCIA_3C589 is not set +# CONFIG_PCMCIA_3C574 is not set +# CONFIG_PCMCIA_FMVJ18X is not set +CONFIG_PCMCIA_PCNET=m +# CONFIG_PCMCIA_NMCLAN is not set +# CONFIG_PCMCIA_SMC91C92 is not set +# CONFIG_PCMCIA_XIRC2PS is not set +# CONFIG_PCMCIA_AXNET is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set +CONFIG_PPP=m +# CONFIG_PPP_MULTILINK is not set +# CONFIG_PPP_FILTER is not set +CONFIG_PPP_ASYNC=m +# CONFIG_PPP_SYNC_TTY is not set +CONFIG_PPP_DEFLATE=m +CONFIG_PPP_BSDCOMP=m +# CONFIG_PPP_MPPE is not set +# CONFIG_PPPOE is not set +# CONFIG_SLIP is not set +# CONFIG_SHAPER is not set +# CONFIG_NETCONSOLE is not set +# CONFIG_NETPOLL is not set +# CONFIG_NET_POLL_CONTROLLER is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Input device support +# +CONFIG_INPUT=y + +# +# Userland interfaces +# +CONFIG_INPUT_MOUSEDEV=y +# CONFIG_INPUT_MOUSEDEV_PSAUX is not set +CONFIG_INPUT_MOUSEDEV_SCREEN_X=240 +CONFIG_INPUT_MOUSEDEV_SCREEN_Y=320 +# CONFIG_INPUT_JOYDEV is not set +# CONFIG_INPUT_TSDEV is not set +CONFIG_INPUT_EVDEV=y +# CONFIG_INPUT_EVBUG is not set +# CONFIG_INPUT_POWER is not set + +# +# Input Device Drivers +# +# CONFIG_INPUT_KEYBOARD is not set +# CONFIG_INPUT_MOUSE is not set +# CONFIG_INPUT_JOYSTICK is not set +CONFIG_INPUT_TOUCHSCREEN=y +# CONFIG_TOUCHSCREEN_GUNZE is not set +# CONFIG_TOUCHSCREEN_ELO is not set +# CONFIG_TOUCHSCREEN_MTOUCH is not set +# CONFIG_TOUCHSCREEN_MK712 is not set +CONFIG_TOUCHSCREEN_TSC2101=y +# CONFIG_INPUT_MISC is not set + +# +# Hardware I/O ports +# +# CONFIG_SERIO is not set +# CONFIG_GAMEPORT is not set + +# +# Character devices +# +CONFIG_VT=y +CONFIG_VT_CONSOLE=y +CONFIG_HW_CONSOLE=y +# CONFIG_SERIAL_NONSTANDARD is not set + +# +# Serial drivers +# +# CONFIG_SERIAL_8250 is not set + +# +# Non-8250 serial port support +# +CONFIG_SERIAL_PXA=y +CONFIG_SERIAL_PXA_CONSOLE=y +CONFIG_SERIAL_CORE=y +CONFIG_SERIAL_CORE_CONSOLE=y +CONFIG_UNIX98_PTYS=y +# CONFIG_LEGACY_PTYS is not set + +# +# IPMI +# +# CONFIG_IPMI_HANDLER is not set + +# +# Watchdog Cards +# +# CONFIG_WATCHDOG is not set +# CONFIG_NVRAM is not set +# CONFIG_SA1100_RTC is not set +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set + +# +# Ftape, the floppy tape device driver +# + +# +# PCMCIA character devices +# +# CONFIG_SYNCLINK_CS is not set +# CONFIG_CARDMAN_4000 is not set +# CONFIG_CARDMAN_4040 is not set +# CONFIG_RAW_DRIVER is not set + +# +# TPM devices +# +# CONFIG_TCG_TPM is not set +# CONFIG_TELCLOCK is not set + +# +# I2C support +# +# CONFIG_I2C is not set + +# +# Hardware Monitoring support +# +# CONFIG_HWMON is not set + +# +# Misc devices +# + +# +# Multimedia Capabilities Port drivers +# + +# +# Multi-Function Devices +# +CONFIG_MFD_TSC2101=y + +# +# Multimedia devices +# +# CONFIG_VIDEO_DEV is not set + +# +# Digital Video Broadcasting Devices +# +# CONFIG_DVB is not set + +# +# Graphics support +# +CONFIG_FB=y +CONFIG_FB_CFB_FILLRECT=y +CONFIG_FB_CFB_COPYAREA=y +CONFIG_FB_CFB_IMAGEBLIT=y +# CONFIG_FB_MACMODES is not set +# CONFIG_FB_MODE_HELPERS is not set +# CONFIG_FB_TILEBLITTING is not set +# CONFIG_FB_S1D13XXX is not set +CONFIG_FB_PXA=y +# CONFIG_FB_PXA_PARAMETERS is not set +# CONFIG_FB_VIRTUAL is not set + +# +# Console display driver support +# +# CONFIG_VGA_CONSOLE is not set +CONFIG_DUMMY_CONSOLE=y +CONFIG_FRAMEBUFFER_CONSOLE=y +# CONFIG_FRAMEBUFFER_CONSOLE_ROTATION is not set +# CONFIG_FONTS is not set +CONFIG_FONT_8x8=y +CONFIG_FONT_8x16=y + +# +# Logo configuration +# +# CONFIG_LOGO is not set +CONFIG_BACKLIGHT_LCD_SUPPORT=y +CONFIG_BACKLIGHT_CLASS_DEVICE=y +CONFIG_BACKLIGHT_DEVICE=y +CONFIG_LCD_CLASS_DEVICE=y +CONFIG_LCD_DEVICE=y +CONFIG_BACKLIGHT_HX2750=y + +# +# Sound +# +# CONFIG_SOUND is not set + +# +# USB support +# +CONFIG_USB_ARCH_HAS_HCD=y +CONFIG_USB_ARCH_HAS_OHCI=y +# CONFIG_USB is not set + +# +# NOTE: USB_STORAGE enables SCSI, and 'SCSI disk support' +# + +# +# USB Gadget Support +# +# CONFIG_USB_GADGET is not set + +# +# MMC/SD Card support +# +CONFIG_MMC=y +# CONFIG_MMC_DEBUG is not set +CONFIG_MMC_BLOCK=y +CONFIG_MMC_PXA=y +# CONFIG_MMC_WBSD is not set + +# +# Real Time Clock +# +CONFIG_RTC_CLASS=y +CONFIG_RTC_HCTOSYS=y +CONFIG_RTC_HCTOSYS_DEVICE="rtc0" + +# +# RTC interfaces +# +CONFIG_RTC_INTF_SYSFS=y +CONFIG_RTC_INTF_PROC=y +CONFIG_RTC_INTF_DEV=y + +# +# RTC drivers +# +# CONFIG_RTC_DRV_X1205 is not set +# CONFIG_RTC_DRV_DS1672 is not set +# CONFIG_RTC_DRV_PCF8563 is not set +CONFIG_RTC_DRV_SA1100=y +# CONFIG_RTC_DRV_TEST is not set + +# +# File systems +# +CONFIG_EXT2_FS=y +# CONFIG_EXT2_FS_XATTR is not set +# CONFIG_EXT2_FS_XIP is not set +# CONFIG_EXT3_FS is not set +# CONFIG_JBD is not set +# CONFIG_REISERFS_FS is not set +# CONFIG_JFS_FS is not set +# CONFIG_FS_POSIX_ACL is not set +# CONFIG_XFS_FS is not set +# CONFIG_MINIX_FS is not set +# CONFIG_ROMFS_FS is not set +CONFIG_INOTIFY=y +# CONFIG_QUOTA is not set +CONFIG_DNOTIFY=y +# CONFIG_AUTOFS_FS is not set +# CONFIG_AUTOFS4_FS is not set +# CONFIG_FUSE_FS is not set + +# +# CD-ROM/DVD Filesystems +# +# CONFIG_ISO9660_FS is not set +# CONFIG_UDF_FS is not set + +# +# DOS/FAT/NT Filesystems +# +CONFIG_FAT_FS=y +# CONFIG_MSDOS_FS is not set +CONFIG_VFAT_FS=y +CONFIG_FAT_DEFAULT_CODEPAGE=437 +CONFIG_FAT_DEFAULT_IOCHARSET="iso8859-1" +# CONFIG_NTFS_FS is not set + +# +# Pseudo filesystems +# +CONFIG_PROC_FS=y +CONFIG_SYSFS=y +# CONFIG_TMPFS is not set +# CONFIG_HUGETLBFS is not set +# CONFIG_HUGETLB_PAGE is not set +CONFIG_RAMFS=y +# CONFIG_RELAYFS_FS is not set + +# +# Miscellaneous filesystems +# +# CONFIG_ADFS_FS is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_HFSPLUS_FS is not set +# CONFIG_BEFS_FS is not set +# CONFIG_BFS_FS is not set +# CONFIG_EFS_FS is not set +# CONFIG_JFFS_FS is not set +CONFIG_JFFS2_FS=y +CONFIG_JFFS2_FS_DEBUG=0 +CONFIG_JFFS2_FS_WRITEBUFFER=y +# CONFIG_JFFS2_SUMMARY is not set +# CONFIG_JFFS2_COMPRESSION_OPTIONS is not set +CONFIG_JFFS2_ZLIB=y +CONFIG_JFFS2_RTIME=y +CONFIG_JFFS2_SUMMARY=y +# CONFIG_JFFS2_RUBIN is not set +CONFIG_CRAMFS=y +CONFIG_SQUASHFS=m +# CONFIG_SQUASHFS_EMBEDDED is not set +CONFIG_SQUASHFS_FRAGMENT_CACHE_SIZE=3 +# CONFIG_SQUASHFS_VMALLOC is not set +# CONFIG_VXFS_FS is not set +# CONFIG_HPFS_FS is not set +# CONFIG_QNX4FS_FS is not set +# CONFIG_SYSV_FS is not set +# CONFIG_UFS_FS is not set + +# +# Network File Systems +# +# CONFIG_NFS_FS is not set +# CONFIG_NFSD is not set +# CONFIG_SMB_FS is not set +# CONFIG_CIFS is not set +# CONFIG_NCP_FS is not set +# CONFIG_CODA_FS is not set +# CONFIG_AFS_FS is not set +# CONFIG_9P_FS is not set + +# +# Partition Types +# +CONFIG_PARTITION_ADVANCED=y +# CONFIG_ACORN_PARTITION is not set +# CONFIG_OSF_PARTITION is not set +# CONFIG_AMIGA_PARTITION is not set +# CONFIG_ATARI_PARTITION is not set +# CONFIG_MAC_PARTITION is not set +CONFIG_MSDOS_PARTITION=y +# CONFIG_BSD_DISKLABEL is not set +# CONFIG_MINIX_SUBPARTITION is not set +# CONFIG_SOLARIS_X86_PARTITION is not set +# CONFIG_UNIXWARE_DISKLABEL is not set +# CONFIG_LDM_PARTITION is not set +# CONFIG_SGI_PARTITION is not set +# CONFIG_ULTRIX_PARTITION is not set +# CONFIG_SUN_PARTITION is not set +# CONFIG_EFI_PARTITION is not set + +# +# Native Language Support +# +CONFIG_NLS=y +CONFIG_NLS_DEFAULT="cp437" +CONFIG_NLS_CODEPAGE_437=y +# CONFIG_NLS_CODEPAGE_737 is not set +# CONFIG_NLS_CODEPAGE_775 is not set +# CONFIG_NLS_CODEPAGE_850 is not set +# CONFIG_NLS_CODEPAGE_852 is not set +# CONFIG_NLS_CODEPAGE_855 is not set +# CONFIG_NLS_CODEPAGE_857 is not set +# CONFIG_NLS_CODEPAGE_860 is not set +# CONFIG_NLS_CODEPAGE_861 is not set +# CONFIG_NLS_CODEPAGE_862 is not set +# CONFIG_NLS_CODEPAGE_863 is not set +# CONFIG_NLS_CODEPAGE_864 is not set +# CONFIG_NLS_CODEPAGE_865 is not set +# CONFIG_NLS_CODEPAGE_866 is not set +# CONFIG_NLS_CODEPAGE_869 is not set +# CONFIG_NLS_CODEPAGE_936 is not set +# CONFIG_NLS_CODEPAGE_950 is not set +# CONFIG_NLS_CODEPAGE_932 is not set +# CONFIG_NLS_CODEPAGE_949 is not set +# CONFIG_NLS_CODEPAGE_874 is not set +# CONFIG_NLS_ISO8859_8 is not set +# CONFIG_NLS_CODEPAGE_1250 is not set +# CONFIG_NLS_CODEPAGE_1251 is not set +# CONFIG_NLS_ASCII is not set +CONFIG_NLS_ISO8859_1=y +# CONFIG_NLS_ISO8859_2 is not set +# CONFIG_NLS_ISO8859_3 is not set +# CONFIG_NLS_ISO8859_4 is not set +# CONFIG_NLS_ISO8859_5 is not set +# CONFIG_NLS_ISO8859_6 is not set +# CONFIG_NLS_ISO8859_7 is not set +# CONFIG_NLS_ISO8859_9 is not set +# CONFIG_NLS_ISO8859_13 is not set +# CONFIG_NLS_ISO8859_14 is not set +# CONFIG_NLS_ISO8859_15 is not set +# CONFIG_NLS_KOI8_R is not set +# CONFIG_NLS_KOI8_U is not set +CONFIG_NLS_UTF8=y + +# +# Profiling support +# +# CONFIG_PROFILING is not set + +# +# Kernel hacking +# +# CONFIG_PRINTK_TIME is not set +CONFIG_DEBUG_KERNEL=y +# CONFIG_MAGIC_SYSRQ is not set +CONFIG_LOG_BUF_SHIFT=14 +CONFIG_DETECT_SOFTLOCKUP=y +# CONFIG_SCHEDSTATS is not set +# CONFIG_DEBUG_SLAB is not set +CONFIG_DEBUG_PREEMPT=y +# CONFIG_DEBUG_SPINLOCK is not set +# CONFIG_DEBUG_SPINLOCK_SLEEP is not set +# CONFIG_DEBUG_KOBJECT is not set +# CONFIG_DEBUG_BUGVERBOSE is not set +# CONFIG_DEBUG_INFO is not set +# CONFIG_DEBUG_FS is not set +# CONFIG_DEBUG_VM is not set +CONFIG_FRAME_POINTER=y +# CONFIG_RCU_TORTURE_TEST is not set +# CONFIG_DEBUG_USER is not set +# CONFIG_DEBUG_WAITQ is not set +CONFIG_DEBUG_ERRORS=y +CONFIG_DEBUG_LL=y +# CONFIG_DEBUG_ICEDCC is not set + +# +# Security options +# +# CONFIG_KEYS is not set +# CONFIG_SECURITY is not set + +# +# Cryptographic options +# +CONFIG_CRYPTO=y +# CONFIG_CRYPTO_HMAC is not set +# CONFIG_CRYPTO_NULL is not set +# CONFIG_CRYPTO_MD4 is not set +# CONFIG_CRYPTO_MD5 is not set +# CONFIG_CRYPTO_SHA1 is not set +# CONFIG_CRYPTO_SHA256 is not set +# CONFIG_CRYPTO_SHA512 is not set +# CONFIG_CRYPTO_WP512 is not set +# CONFIG_CRYPTO_TGR192 is not set +# CONFIG_CRYPTO_DES is not set +# CONFIG_CRYPTO_BLOWFISH is not set +# CONFIG_CRYPTO_TWOFISH is not set +# CONFIG_CRYPTO_SERPENT is not set +# CONFIG_CRYPTO_AES is not set +# CONFIG_CRYPTO_CAST5 is not set +# CONFIG_CRYPTO_CAST6 is not set +# CONFIG_CRYPTO_TEA is not set +CONFIG_CRYPTO_ARC4=m +# CONFIG_CRYPTO_KHAZAD is not set +# CONFIG_CRYPTO_ANUBIS is not set +CONFIG_CRYPTO_DEFLATE=y +# CONFIG_CRYPTO_MICHAEL_MIC is not set +CONFIG_CRYPTO_CRC32C=y +# CONFIG_CRYPTO_TEST is not set + +# +# Hardware crypto devices +# + +# +# Library routines +# +CONFIG_CRC_CCITT=m +# CONFIG_CRC16 is not set +CONFIG_CRC32=y +CONFIG_LIBCRC32C=y +CONFIG_ZLIB_INFLATE=y +CONFIG_ZLIB_DEFLATE=y + diff --git a/packages/linux/linux-rp-2.6.20/defconfig-poodle b/packages/linux/linux-rp-2.6.20/defconfig-poodle new file mode 100644 index 0000000000..deacd170f2 --- /dev/null +++ b/packages/linux/linux-rp-2.6.20/defconfig-poodle @@ -0,0 +1,1655 @@ +# +# Automatically generated make config: don't edit +# Linux kernel version: 2.6.17 +# Mon Jul 10 23:38:56 2006 +# +CONFIG_ARM=y +CONFIG_MMU=y +CONFIG_RWSEM_GENERIC_SPINLOCK=y +CONFIG_GENERIC_HWEIGHT=y +CONFIG_GENERIC_CALIBRATE_DELAY=y +CONFIG_ARCH_MTD_XIP=y +CONFIG_VECTORS_BASE=0xffff0000 + +# +# Code maturity level options +# +CONFIG_EXPERIMENTAL=y +CONFIG_BROKEN_ON_SMP=y +CONFIG_LOCK_KERNEL=y +CONFIG_INIT_ENV_ARG_LIMIT=32 + +# +# General setup +# +CONFIG_LOCALVERSION="" +CONFIG_LOCALVERSION_AUTO=y +CONFIG_SWAP=y +CONFIG_SYSVIPC=y +# CONFIG_POSIX_MQUEUE is not set +CONFIG_BSD_PROCESS_ACCT=y +# CONFIG_BSD_PROCESS_ACCT_V3 is not set +CONFIG_SYSCTL=y +# CONFIG_AUDIT is not set +# CONFIG_IKCONFIG is not set +# CONFIG_RELAY is not set +CONFIG_INITRAMFS_SOURCE="" +CONFIG_UID16=y +CONFIG_CC_OPTIMIZE_FOR_SIZE=y +CONFIG_EMBEDDED=y +CONFIG_KALLSYMS=y +# CONFIG_KALLSYMS_ALL is not set +# CONFIG_KALLSYMS_EXTRA_PASS is not set +CONFIG_HOTPLUG=y +CONFIG_PRINTK=y +CONFIG_BUG=y +CONFIG_ELF_CORE=y +CONFIG_BASE_FULL=y +CONFIG_FUTEX=y +CONFIG_EPOLL=y +CONFIG_SHMEM=y +CONFIG_SLAB=y +# CONFIG_TINY_SHMEM is not set +CONFIG_BASE_SMALL=0 +# CONFIG_SLOB is not set + +# +# Loadable module support +# +CONFIG_MODULES=y +CONFIG_MODULE_UNLOAD=y +CONFIG_MODULE_FORCE_UNLOAD=y +# CONFIG_MODVERSIONS is not set +# CONFIG_MODULE_SRCVERSION_ALL is not set +CONFIG_KMOD=y + +# +# Block layer +# +# CONFIG_BLK_DEV_IO_TRACE is not set + +# +# IO Schedulers +# +CONFIG_IOSCHED_NOOP=y +CONFIG_IOSCHED_AS=y +CONFIG_IOSCHED_DEADLINE=m +CONFIG_IOSCHED_CFQ=m +CONFIG_DEFAULT_AS=y +# CONFIG_DEFAULT_DEADLINE is not set +# CONFIG_DEFAULT_CFQ is not set +# CONFIG_DEFAULT_NOOP is not set +CONFIG_DEFAULT_IOSCHED="anticipatory" + +# +# System Type +# +# CONFIG_ARCH_CLPS7500 is not set +# CONFIG_ARCH_CLPS711X is not set +# CONFIG_ARCH_CO285 is not set +# CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_EP93XX is not set +# CONFIG_ARCH_FOOTBRIDGE is not set +# CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_IOP3XX is not set +# CONFIG_ARCH_IXP4XX is not set +# CONFIG_ARCH_IXP2000 is not set +# CONFIG_ARCH_IXP23XX is not set +# CONFIG_ARCH_L7200 is not set +CONFIG_ARCH_PXA=y +# CONFIG_ARCH_RPC is not set +# CONFIG_ARCH_SA1100 is not set +# CONFIG_ARCH_S3C2410 is not set +# CONFIG_ARCH_SHARK is not set +# CONFIG_ARCH_LH7A40X is not set +# CONFIG_ARCH_OMAP is not set +# CONFIG_ARCH_VERSATILE is not set +# CONFIG_ARCH_REALVIEW is not set +# CONFIG_ARCH_IMX is not set +# CONFIG_ARCH_H720X is not set +# CONFIG_ARCH_AAEC2000 is not set +# CONFIG_ARCH_AT91RM9200 is not set + +# +# Intel PXA2xx Implementations +# +# CONFIG_ARCH_LUBBOCK is not set +# CONFIG_MACH_LOGICPD_PXA270 is not set +# CONFIG_MACH_MAINSTONE is not set +# CONFIG_ARCH_PXA_IDP is not set +CONFIG_PXA_SHARPSL=y +# CONFIG_MACH_HX2750 is not set +CONFIG_PXA_SHARPSL_25x=y +# CONFIG_PXA_SHARPSL_27x is not set +CONFIG_MACH_POODLE=y +# CONFIG_MACH_CORGI is not set +# CONFIG_MACH_SHEPHERD is not set +# CONFIG_MACH_HUSKY is not set +# CONFIG_MACH_TOSA is not set +CONFIG_PXA25x=y +# CONFIG_PXA_KEYS is not set +CONFIG_PXA_SSP=y + +# +# Processor Type +# +CONFIG_CPU_32=y +CONFIG_CPU_XSCALE=y +CONFIG_CPU_32v5=y +CONFIG_CPU_ABRT_EV5T=y +CONFIG_CPU_CACHE_VIVT=y +CONFIG_CPU_TLB_V4WBI=y + +# +# Processor Features +# +CONFIG_ARM_THUMB=y +CONFIG_XSCALE_PMU=y +CONFIG_KEXEC=y +CONFIG_SHARP_LOCOMO=y +CONFIG_SHARP_PARAM=y +CONFIG_SHARPSL_PM=y +CONFIG_SHARP_SCOOP=y + +# +# Bus support +# + +# +# PCCARD (PCMCIA/CardBus) support +# +CONFIG_PCCARD=y +# CONFIG_PCMCIA_DEBUG is not set +CONFIG_PCMCIA=y +CONFIG_PCMCIA_LOAD_CIS=y +CONFIG_PCMCIA_IOCTL=y + +# +# PC-card bridges +# +CONFIG_PCMCIA_PXA2XX=y + +# +# Kernel Features +# +CONFIG_PREEMPT=y +CONFIG_NO_IDLE_HZ=y +CONFIG_HZ=100 +# CONFIG_AEABI is not set +# CONFIG_ARCH_DISCONTIGMEM_ENABLE is not set +CONFIG_SELECT_MEMORY_MODEL=y +CONFIG_FLATMEM_MANUAL=y +# CONFIG_DISCONTIGMEM_MANUAL is not set +# CONFIG_SPARSEMEM_MANUAL is not set +CONFIG_FLATMEM=y +CONFIG_FLAT_NODE_MEM_MAP=y +# CONFIG_SPARSEMEM_STATIC is not set +CONFIG_SPLIT_PTLOCK_CPUS=4096 +CONFIG_ALIGNMENT_TRAP=y + +# +# Boot options +# +CONFIG_ZBOOT_ROM_TEXT=0x0 +CONFIG_ZBOOT_ROM_BSS=0x0 +CONFIG_CMDLINE="console=ttyS0,115200n8 console=tty1 noinitrd root=/dev/mtdblock2 rootfstype=jffs2 fbcon=rotate:1 dyntick=enable debug" +# CONFIG_XIP_KERNEL is not set + +# +# CPU Frequency scaling +# +CONFIG_CPU_FREQ=y +CONFIG_CPU_FREQ_TABLE=y +CONFIG_CPU_FREQ_DEBUG=y +CONFIG_CPU_FREQ_STAT=y +# CONFIG_CPU_FREQ_STAT_DETAILS is not set +CONFIG_CPU_FREQ_DEFAULT_GOV_PERFORMANCE=y +# CONFIG_CPU_FREQ_DEFAULT_GOV_USERSPACE is not set +CONFIG_CPU_FREQ_GOV_PERFORMANCE=y +CONFIG_CPU_FREQ_GOV_POWERSAVE=y +CONFIG_CPU_FREQ_GOV_USERSPACE=y +CONFIG_CPU_FREQ_GOV_ONDEMAND=y +CONFIG_CPU_FREQ_GOV_CONSERVATIVE=y +CONFIG_CPU_FREQ_PXA25x=y + +# +# Floating point emulation +# + +# +# At least one emulation must be selected +# +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_NWFPE_XP is not set +# CONFIG_FPE_FASTFPE is not set + +# +# Userspace binary formats +# +CONFIG_BINFMT_ELF=y +CONFIG_BINFMT_AOUT=m +CONFIG_BINFMT_MISC=m +# CONFIG_ARTHUR is not set + +# +# Power management options +# +CONFIG_PM=y +# CONFIG_PM_LEGACY is not set +# CONFIG_PM_DEBUG is not set +CONFIG_APM=y + +# +# Networking +# +CONFIG_NET=y + +# +# Networking options +# +# CONFIG_NETDEBUG is not set +CONFIG_PACKET=y +CONFIG_PACKET_MMAP=y +CONFIG_UNIX=y +CONFIG_XFRM=y +# CONFIG_XFRM_USER is not set +# CONFIG_NET_KEY is not set +CONFIG_INET=y +# CONFIG_IP_MULTICAST is not set +# CONFIG_IP_ADVANCED_ROUTER is not set +CONFIG_IP_FIB_HASH=y +# CONFIG_IP_PNP is not set +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set +CONFIG_SYN_COOKIES=y +# CONFIG_INET_AH is not set +# CONFIG_INET_ESP is not set +# CONFIG_INET_IPCOMP is not set +# CONFIG_INET_XFRM_TUNNEL is not set +# CONFIG_INET_TUNNEL is not set +CONFIG_INET_DIAG=m +CONFIG_INET_TCP_DIAG=m +# CONFIG_TCP_CONG_ADVANCED is not set +CONFIG_TCP_CONG_BIC=y + +# +# IP: Virtual Server Configuration +# +# CONFIG_IP_VS is not set +CONFIG_IPV6=m +# CONFIG_IPV6_PRIVACY is not set +# CONFIG_IPV6_ROUTER_PREF is not set +CONFIG_INET6_AH=m +CONFIG_INET6_ESP=m +CONFIG_INET6_IPCOMP=m +CONFIG_INET6_XFRM_TUNNEL=m +CONFIG_INET6_TUNNEL=m +CONFIG_IPV6_TUNNEL=m +CONFIG_NETFILTER=y +# CONFIG_NETFILTER_DEBUG is not set + +# +# Core Netfilter Configuration +# +# CONFIG_NETFILTER_NETLINK is not set +# CONFIG_NETFILTER_XTABLES is not set + +# +# IP: Netfilter Configuration +# +CONFIG_IP_NF_CONNTRACK=m +# CONFIG_IP_NF_CT_ACCT is not set +# CONFIG_IP_NF_CONNTRACK_MARK is not set +# CONFIG_IP_NF_CONNTRACK_EVENTS is not set +CONFIG_IP_NF_CT_PROTO_SCTP=m +CONFIG_IP_NF_FTP=m +CONFIG_IP_NF_IRC=m +# CONFIG_IP_NF_NETBIOS_NS is not set +CONFIG_IP_NF_TFTP=m +CONFIG_IP_NF_AMANDA=m +# CONFIG_IP_NF_PPTP is not set +# CONFIG_IP_NF_H323 is not set +CONFIG_IP_NF_QUEUE=m + +# +# IPv6: Netfilter Configuration (EXPERIMENTAL) +# +# CONFIG_IP6_NF_QUEUE is not set + +# +# DCCP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_DCCP is not set + +# +# SCTP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_SCTP is not set + +# +# TIPC Configuration (EXPERIMENTAL) +# +# CONFIG_TIPC is not set +# CONFIG_ATM is not set +# CONFIG_BRIDGE is not set +# CONFIG_VLAN_8021Q is not set +# CONFIG_DECNET is not set +# CONFIG_LLC2 is not set +# CONFIG_IPX is not set +# CONFIG_ATALK is not set +# CONFIG_X25 is not set +# CONFIG_LAPB is not set +# CONFIG_NET_DIVERT is not set +# CONFIG_ECONET is not set +# CONFIG_WAN_ROUTER is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network testing +# +# CONFIG_NET_PKTGEN is not set +# CONFIG_HAMRADIO is not set +CONFIG_IRDA=m + +# +# IrDA protocols +# +CONFIG_IRLAN=m +CONFIG_IRNET=m +CONFIG_IRCOMM=m +# CONFIG_IRDA_ULTRA is not set + +# +# IrDA options +# +# CONFIG_IRDA_CACHE_LAST_LSAP is not set +# CONFIG_IRDA_FAST_RR is not set +# CONFIG_IRDA_DEBUG is not set + +# +# Infrared-port device drivers +# + +# +# SIR device drivers +# +# CONFIG_IRTTY_SIR is not set + +# +# Dongle support +# + +# +# Old SIR device drivers +# +# CONFIG_IRPORT_SIR is not set + +# +# Old Serial dongle support +# + +# +# FIR device drivers +# +# CONFIG_USB_IRDA is not set +# CONFIG_SIGMATEL_FIR is not set +CONFIG_PXA_FICP=m +CONFIG_BT=m +CONFIG_BT_L2CAP=m +CONFIG_BT_SCO=m +CONFIG_BT_RFCOMM=m +CONFIG_BT_RFCOMM_TTY=y +CONFIG_BT_BNEP=m +CONFIG_BT_BNEP_MC_FILTER=y +CONFIG_BT_BNEP_PROTO_FILTER=y +CONFIG_BT_HIDP=m + +# +# Bluetooth device drivers +# +CONFIG_BT_HCIUSB=m +# CONFIG_BT_HCIUSB_SCO is not set +CONFIG_BT_HCIUART=m +CONFIG_BT_HCIUART_H4=y +CONFIG_BT_HCIUART_BCSP=y +CONFIG_BT_HCIBCM203X=m +CONFIG_BT_HCIBPA10X=m +CONFIG_BT_HCIBFUSB=m +CONFIG_BT_HCIDTL1=m +CONFIG_BT_HCIBT3C=m +CONFIG_BT_HCIBLUECARD=m +CONFIG_BT_HCIBTUART=m +CONFIG_BT_HCIVHCI=m +CONFIG_IEEE80211=m +# CONFIG_IEEE80211_DEBUG is not set +CONFIG_IEEE80211_CRYPT_WEP=m +CONFIG_IEEE80211_CRYPT_CCMP=m +CONFIG_IEEE80211_CRYPT_TKIP=m +# CONFIG_IEEE80211_SOFTMAC is not set +CONFIG_WIRELESS_EXT=y + +# +# Device Drivers +# + +# +# Generic Driver Options +# +CONFIG_STANDALONE=y +CONFIG_PREVENT_FIRMWARE_BUILD=y +CONFIG_FW_LOADER=y +# CONFIG_DEBUG_DRIVER is not set + +# +# Connector - unified userspace <-> kernelspace linker +# +# CONFIG_CONNECTOR is not set + +# +# Memory Technology Devices (MTD) +# +CONFIG_MTD=y +# CONFIG_MTD_DEBUG is not set +# CONFIG_MTD_CONCAT is not set +CONFIG_MTD_PARTITIONS=y +# CONFIG_MTD_REDBOOT_PARTS is not set +CONFIG_MTD_CMDLINE_PARTS=y +# CONFIG_MTD_AFS_PARTS is not set + +# +# User Modules And Translation Layers +# +CONFIG_MTD_CHAR=y +CONFIG_MTD_BLOCK=y +# CONFIG_FTL is not set +# CONFIG_NFTL is not set +# CONFIG_INFTL is not set +# CONFIG_RFD_FTL is not set + +# +# RAM/ROM/Flash chip drivers +# +# CONFIG_MTD_CFI is not set +# CONFIG_MTD_JEDECPROBE is not set +CONFIG_MTD_MAP_BANK_WIDTH_1=y +CONFIG_MTD_MAP_BANK_WIDTH_2=y +CONFIG_MTD_MAP_BANK_WIDTH_4=y +# CONFIG_MTD_MAP_BANK_WIDTH_8 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_16 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_32 is not set +CONFIG_MTD_CFI_I1=y +CONFIG_MTD_CFI_I2=y +# CONFIG_MTD_CFI_I4 is not set +# CONFIG_MTD_CFI_I8 is not set +# CONFIG_MTD_RAM is not set +CONFIG_MTD_ROM=y +# CONFIG_MTD_ABSENT is not set +# CONFIG_MTD_OBSOLETE_CHIPS is not set + +# +# Mapping drivers for chip access +# +CONFIG_MTD_COMPLEX_MAPPINGS=y +CONFIG_MTD_SHARP_SL=y +# CONFIG_MTD_PLATRAM is not set + +# +# Self-contained MTD device drivers +# +# CONFIG_MTD_SLRAM is not set +# CONFIG_MTD_PHRAM is not set +# CONFIG_MTD_MTDRAM is not set +# CONFIG_MTD_BLOCK2MTD is not set + +# +# Disk-On-Chip Device Drivers +# +# CONFIG_MTD_DOC2000 is not set +# CONFIG_MTD_DOC2001 is not set +# CONFIG_MTD_DOC2001PLUS is not set + +# +# NAND Flash Device Drivers +# +CONFIG_MTD_NAND=y +CONFIG_MTD_NAND_VERIFY_WRITE=y +# CONFIG_MTD_NAND_H1900 is not set +CONFIG_MTD_NAND_IDS=y +# CONFIG_MTD_NAND_DISKONCHIP is not set +CONFIG_MTD_NAND_SHARPSL=y +# CONFIG_MTD_NAND_NANDSIM is not set + +# +# OneNAND Flash Device Drivers +# +# CONFIG_MTD_ONENAND is not set + +# +# Parallel port support +# +# CONFIG_PARPORT is not set + +# +# Plug and Play support +# + +# +# Block devices +# +# CONFIG_BLK_DEV_COW_COMMON is not set +CONFIG_BLK_DEV_LOOP=y +# CONFIG_BLK_DEV_CRYPTOLOOP is not set +# CONFIG_BLK_DEV_NBD is not set +# CONFIG_BLK_DEV_UB is not set +# CONFIG_BLK_DEV_RAM is not set +# CONFIG_BLK_DEV_INITRD is not set +# CONFIG_CDROM_PKTCDVD is not set +# CONFIG_ATA_OVER_ETH is not set + +# +# ATA/ATAPI/MFM/RLL support +# +CONFIG_IDE=y +CONFIG_BLK_DEV_IDE=y + +# +# Please see Documentation/ide.txt for help/info on IDE drives +# +# CONFIG_BLK_DEV_IDE_SATA is not set +CONFIG_BLK_DEV_IDEDISK=y +# CONFIG_IDEDISK_MULTI_MODE is not set +CONFIG_BLK_DEV_IDECS=y +# CONFIG_BLK_DEV_IDECD is not set +# CONFIG_BLK_DEV_IDETAPE is not set +# CONFIG_BLK_DEV_IDEFLOPPY is not set +# CONFIG_BLK_DEV_IDESCSI is not set +# CONFIG_IDE_TASK_IOCTL is not set + +# +# IDE chipset support/bugfixes +# +CONFIG_IDE_GENERIC=y +# CONFIG_IDE_ARM is not set +# CONFIG_BLK_DEV_IDEDMA is not set +# CONFIG_IDEDMA_AUTO is not set +# CONFIG_BLK_DEV_HD is not set + +# +# SCSI device support +# +# CONFIG_RAID_ATTRS is not set +CONFIG_SCSI=m +CONFIG_SCSI_PROC_FS=y + +# +# SCSI support type (disk, tape, CD-ROM) +# +CONFIG_BLK_DEV_SD=m +CONFIG_CHR_DEV_ST=m +CONFIG_CHR_DEV_OSST=m +CONFIG_BLK_DEV_SR=m +# CONFIG_BLK_DEV_SR_VENDOR is not set +CONFIG_CHR_DEV_SG=m +# CONFIG_CHR_DEV_SCH is not set + +# +# Some SCSI devices (e.g. CD jukebox) support multiple LUNs +# +CONFIG_SCSI_MULTI_LUN=y +# CONFIG_SCSI_CONSTANTS is not set +# CONFIG_SCSI_LOGGING is not set + +# +# SCSI Transport Attributes +# +# CONFIG_SCSI_SPI_ATTRS is not set +# CONFIG_SCSI_FC_ATTRS is not set +# CONFIG_SCSI_ISCSI_ATTRS is not set +# CONFIG_SCSI_SAS_ATTRS is not set + +# +# SCSI low-level drivers +# +# CONFIG_ISCSI_TCP is not set +# CONFIG_SCSI_SATA is not set +# CONFIG_SCSI_DEBUG is not set + +# +# PCMCIA SCSI adapter support +# +# CONFIG_PCMCIA_AHA152X is not set +# CONFIG_PCMCIA_FDOMAIN is not set +# CONFIG_PCMCIA_NINJA_SCSI is not set +# CONFIG_PCMCIA_QLOGIC is not set +# CONFIG_PCMCIA_SYM53C500 is not set + +# +# Multi-device support (RAID and LVM) +# +CONFIG_MD=y +CONFIG_BLK_DEV_DM=m +CONFIG_DM_CRYPT=m +CONFIG_DM_SNAPSHOT=m +CONFIG_DM_MIRROR=m +CONFIG_DM_ZERO=m +CONFIG_DM_MULTIPATH=m +CONFIG_DM_MULTIPATH_EMC=m + +# +# Fusion MPT device support +# +# CONFIG_FUSION is not set + +# +# IEEE 1394 (FireWire) support +# + +# +# I2O device support +# + +# +# Network device support +# +CONFIG_NETDEVICES=y +# CONFIG_DUMMY is not set +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +CONFIG_TUN=m + +# +# PHY device support +# +# CONFIG_PHYLIB is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +CONFIG_MII=m +# CONFIG_SMC91X is not set +# CONFIG_DM9000 is not set + +# +# Ethernet (1000 Mbit) +# + +# +# Ethernet (10000 Mbit) +# + +# +# Token Ring devices +# + +# +# Wireless LAN (non-hamradio) +# +CONFIG_NET_RADIO=y +# CONFIG_NET_WIRELESS_RTNETLINK is not set + +# +# Obsolete Wireless cards support (pre-802.11) +# +# CONFIG_STRIP is not set +# CONFIG_PCMCIA_WAVELAN is not set +# CONFIG_PCMCIA_NETWAVE is not set + +# +# Wireless 802.11 Frequency Hopping cards support +# +# CONFIG_PCMCIA_RAYCS is not set + +# +# Wireless 802.11b ISA/PCI cards support +# +CONFIG_HERMES=m +# CONFIG_ATMEL is not set + +# +# Wireless 802.11b Pcmcia/Cardbus cards support +# +CONFIG_PCMCIA_HERMES=m +CONFIG_PCMCIA_SPECTRUM=m +# CONFIG_AIRO_CS is not set +# CONFIG_PCMCIA_WL3501 is not set +CONFIG_HOSTAP=m +CONFIG_HOSTAP_FIRMWARE=y +# CONFIG_HOSTAP_FIRMWARE_NVRAM is not set +CONFIG_HOSTAP_CS=m +CONFIG_NET_WIRELESS=y + +# +# PCMCIA network device support +# +CONFIG_NET_PCMCIA=y +# CONFIG_PCMCIA_3C589 is not set +# CONFIG_PCMCIA_3C574 is not set +# CONFIG_PCMCIA_FMVJ18X is not set +CONFIG_PCMCIA_PCNET=m +# CONFIG_PCMCIA_NMCLAN is not set +# CONFIG_PCMCIA_SMC91C92 is not set +# CONFIG_PCMCIA_XIRC2PS is not set +# CONFIG_PCMCIA_AXNET is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set +CONFIG_PPP=m +# CONFIG_PPP_MULTILINK is not set +# CONFIG_PPP_FILTER is not set +CONFIG_PPP_ASYNC=m +# CONFIG_PPP_SYNC_TTY is not set +CONFIG_PPP_DEFLATE=m +CONFIG_PPP_BSDCOMP=m +# CONFIG_PPP_MPPE is not set +# CONFIG_PPPOE is not set +# CONFIG_SLIP is not set +# CONFIG_SHAPER is not set +# CONFIG_NETCONSOLE is not set +# CONFIG_NETPOLL is not set +# CONFIG_NET_POLL_CONTROLLER is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Input device support +# +CONFIG_INPUT=y + +# +# Userland interfaces +# +CONFIG_INPUT_MOUSEDEV=m +# CONFIG_INPUT_MOUSEDEV_PSAUX is not set +CONFIG_INPUT_MOUSEDEV_SCREEN_X=240 +CONFIG_INPUT_MOUSEDEV_SCREEN_Y=320 +# CONFIG_INPUT_JOYDEV is not set +# CONFIG_INPUT_TSDEV is not set +CONFIG_INPUT_EVDEV=y +# CONFIG_INPUT_EVBUG is not set +CONFIG_INPUT_POWER=y + +# +# Input Device Drivers +# +CONFIG_INPUT_KEYBOARD=y +# CONFIG_KEYBOARD_ATKBD is not set +# CONFIG_KEYBOARD_SUNKBD is not set +# CONFIG_KEYBOARD_LKKBD is not set +CONFIG_KEYBOARD_LOCOMO=y +# CONFIG_KEYBOARD_XTKBD is not set +# CONFIG_KEYBOARD_NEWTON is not set +# CONFIG_KEYBOARD_CORGI is not set +# CONFIG_KEYBOARD_SPITZ is not set +# CONFIG_INPUT_MOUSE is not set +# CONFIG_INPUT_JOYSTICK is not set +CONFIG_INPUT_TOUCHSCREEN=y +CONFIG_TOUCHSCREEN_CORGI=y +# CONFIG_TOUCHSCREEN_GUNZE is not set +# CONFIG_TOUCHSCREEN_ELO is not set +# CONFIG_TOUCHSCREEN_MTOUCH is not set +# CONFIG_TOUCHSCREEN_MK712 is not set +CONFIG_INPUT_MISC=y +CONFIG_INPUT_UINPUT=m + +# +# Hardware I/O ports +# +# CONFIG_SERIO is not set +# CONFIG_GAMEPORT is not set + +# +# Character devices +# +CONFIG_VT=y +CONFIG_VT_CONSOLE=y +CONFIG_HW_CONSOLE=y +# CONFIG_SERIAL_NONSTANDARD is not set + +# +# Serial drivers +# +CONFIG_SERIAL_8250=m +CONFIG_SERIAL_8250_CS=m +CONFIG_SERIAL_8250_NR_UARTS=4 +CONFIG_SERIAL_8250_RUNTIME_UARTS=4 +# CONFIG_SERIAL_8250_EXTENDED is not set + +# +# Non-8250 serial port support +# +CONFIG_SERIAL_PXA=y +CONFIG_SERIAL_PXA_CONSOLE=y +CONFIG_SERIAL_CORE=y +CONFIG_SERIAL_CORE_CONSOLE=y +CONFIG_UNIX98_PTYS=y +# CONFIG_LEGACY_PTYS is not set + +# +# IPMI +# +# CONFIG_IPMI_HANDLER is not set + +# +# Watchdog Cards +# +# CONFIG_WATCHDOG is not set +# CONFIG_NVRAM is not set +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set + +# +# Ftape, the floppy tape device driver +# + +# +# PCMCIA character devices +# +# CONFIG_SYNCLINK_CS is not set +# CONFIG_CARDMAN_4000 is not set +# CONFIG_CARDMAN_4040 is not set +# CONFIG_RAW_DRIVER is not set + +# +# TPM devices +# +# CONFIG_TCG_TPM is not set +# CONFIG_TELCLOCK is not set + +# +# I2C support +# +CONFIG_I2C=y +# CONFIG_I2C_CHARDEV is not set + +# +# I2C Algorithms +# +CONFIG_I2C_ALGOBIT=y +# CONFIG_I2C_ALGOPCF is not set +# CONFIG_I2C_ALGOPCA is not set + +# +# I2C Hardware Bus support +# +CONFIG_I2C_PXA=y +# CONFIG_I2C_PXA_SLAVE is not set +# CONFIG_I2C_PARPORT_LIGHT is not set +# CONFIG_I2C_STUB is not set +# CONFIG_I2C_PCA_ISA is not set + +# +# Miscellaneous I2C Chip support +# +# CONFIG_SENSORS_DS1337 is not set +# CONFIG_SENSORS_DS1374 is not set +# CONFIG_SENSORS_EEPROM is not set +# CONFIG_SENSORS_PCF8574 is not set +# CONFIG_SENSORS_PCA9539 is not set +# CONFIG_SENSORS_PCF8591 is not set +# CONFIG_SENSORS_MAX6875 is not set +# CONFIG_I2C_DEBUG_CORE is not set +# CONFIG_I2C_DEBUG_ALGO is not set +# CONFIG_I2C_DEBUG_BUS is not set +# CONFIG_I2C_DEBUG_CHIP is not set + +# +# SPI support +# +# CONFIG_SPI is not set +# CONFIG_SPI_MASTER is not set + +# +# Dallas's 1-wire bus +# +# CONFIG_W1 is not set + +# +# Hardware Monitoring support +# +# CONFIG_HWMON is not set +# CONFIG_HWMON_VID is not set + +# +# Misc devices +# + +# +# Multi-Function Devices +# + +# +# LED devices +# +CONFIG_NEW_LEDS=y +CONFIG_LEDS_CLASS=y + +# +# LED drivers +# +CONFIG_LEDS_LOCOMO=y +# CONFIG_LEDS_TOSA is not set + +# +# LED Triggers +# +CONFIG_LEDS_TRIGGERS=y +CONFIG_LEDS_TRIGGER_TIMER=y +CONFIG_LEDS_TRIGGER_IDE_DISK=y + +# +# Multimedia devices +# +CONFIG_VIDEO_DEV=m +CONFIG_VIDEO_V4L1=y +CONFIG_VIDEO_V4L1_COMPAT=y +CONFIG_VIDEO_V4L2=y + +# +# Video Capture Adapters +# + +# +# Video Capture Adapters +# +# CONFIG_VIDEO_ADV_DEBUG is not set +# CONFIG_VIDEO_VIVI is not set +# CONFIG_VIDEO_CPIA is not set +# CONFIG_VIDEO_CPIA2 is not set +# CONFIG_VIDEO_SAA5246A is not set +# CONFIG_VIDEO_SAA5249 is not set +# CONFIG_TUNER_3036 is not set +# CONFIG_VIDEO_OVCAMCHIP is not set + +# +# Encoders and Decoders +# +# CONFIG_VIDEO_MSP3400 is not set +# CONFIG_VIDEO_CS53L32A is not set +# CONFIG_VIDEO_WM8775 is not set +# CONFIG_VIDEO_WM8739 is not set +# CONFIG_VIDEO_CX25840 is not set +# CONFIG_VIDEO_SAA711X is not set +# CONFIG_VIDEO_SAA7127 is not set +# CONFIG_VIDEO_UPD64031A is not set +# CONFIG_VIDEO_UPD64083 is not set + +# +# V4L USB devices +# +# CONFIG_VIDEO_EM28XX is not set +CONFIG_USB_DSBR=m +CONFIG_VIDEO_USBVIDEO=m +CONFIG_USB_VICAM=m +CONFIG_USB_IBMCAM=m +CONFIG_USB_KONICAWC=m +# CONFIG_USB_ET61X251 is not set +CONFIG_USB_OV511=m +CONFIG_USB_SE401=m +CONFIG_USB_SN9C102=m +CONFIG_USB_STV680=m +# CONFIG_USB_W9968CF is not set +# CONFIG_USB_ZC0301 is not set +# CONFIG_USB_PWC is not set + +# +# Radio Adapters +# +# CONFIG_RADIO_MAESTRO is not set + +# +# Digital Video Broadcasting Devices +# +# CONFIG_DVB is not set +CONFIG_USB_DABUSB=m + +# +# Graphics support +# +CONFIG_FB=y +CONFIG_FB_CFB_FILLRECT=y +CONFIG_FB_CFB_COPYAREA=y +CONFIG_FB_CFB_IMAGEBLIT=y +# CONFIG_FB_MACMODES is not set +CONFIG_FB_FIRMWARE_EDID=y +CONFIG_FB_MODE_HELPERS=y +# CONFIG_FB_TILEBLITTING is not set +# CONFIG_FB_S1D13XXX is not set +CONFIG_FB_PXA=y +# CONFIG_FB_PXA_PARAMETERS is not set +# CONFIG_FB_W100 is not set +# CONFIG_FB_VIRTUAL is not set + +# +# Console display driver support +# +# CONFIG_VGA_CONSOLE is not set +CONFIG_DUMMY_CONSOLE=y +CONFIG_FRAMEBUFFER_CONSOLE=y +CONFIG_FRAMEBUFFER_CONSOLE_ROTATION=y +CONFIG_FONTS=y +# CONFIG_FONT_8x8 is not set +# CONFIG_FONT_8x16 is not set +# CONFIG_FONT_6x11 is not set +# CONFIG_FONT_7x14 is not set +# CONFIG_FONT_PEARL_8x8 is not set +# CONFIG_FONT_ACORN_8x8 is not set +CONFIG_FONT_MINI_4x6=y +# CONFIG_FONT_SUN8x16 is not set +# CONFIG_FONT_SUN12x22 is not set +# CONFIG_FONT_10x18 is not set + +# +# Logo configuration +# +CONFIG_LOGO=y +CONFIG_LOGO_LINUX_MONO=y +CONFIG_LOGO_LINUX_VGA16=y +# CONFIG_LOGO_LINUX_CLUT224 is not set +# CONFIG_LOGO_OHAND_CLUT224 is not set +CONFIG_LOGO_OZ240_CLUT224=y +# CONFIG_LOGO_OZ480_CLUT224 is not set +# CONFIG_LOGO_OZ640_CLUT224 is not set +CONFIG_BACKLIGHT_LCD_SUPPORT=y +CONFIG_BACKLIGHT_CLASS_DEVICE=y +CONFIG_BACKLIGHT_DEVICE=y +CONFIG_LCD_CLASS_DEVICE=m +CONFIG_LCD_DEVICE=y +# CONFIG_BACKLIGHT_CORGI is not set +CONFIG_BACKLIGHT_LOCOMO=y + +# +# Sound +# +CONFIG_SOUND=m + +# +# Advanced Linux Sound Architecture +# +CONFIG_SND=m +CONFIG_SND_TIMER=m +CONFIG_SND_PCM=m +# CONFIG_SND_SEQUENCER is not set +CONFIG_SND_OSSEMUL=y +CONFIG_SND_MIXER_OSS=m +CONFIG_SND_PCM_OSS=m +CONFIG_SND_PCM_OSS_PLUGINS=y +# CONFIG_SND_DYNAMIC_MINORS is not set +CONFIG_SND_SUPPORT_OLD_API=y +CONFIG_SND_VERBOSE_PROCFS=y +# CONFIG_SND_VERBOSE_PRINTK is not set +# CONFIG_SND_DEBUG is not set + +# +# Generic devices +# +# CONFIG_SND_DUMMY is not set +# CONFIG_SND_MTPAV is not set +# CONFIG_SND_SERIAL_U16550 is not set +# CONFIG_SND_MPU401 is not set + +# +# ALSA ARM devices +# +# CONFIG_SND_PXA2XX_AC97 is not set + +# +# USB devices +# +# CONFIG_SND_USB_AUDIO is not set + +# +# PCMCIA devices +# +# CONFIG_SND_VXPOCKET is not set +# CONFIG_SND_PDAUDIOCF is not set + +# +# SoC audio support +# +CONFIG_SND_SOC=m + +# +# Soc Platforms +# + +# +# SoC Audio for the Intel PXA2xx +# +CONFIG_SND_PXA2xx_SOC=m +# CONFIG_SND_PXA2xx_SOC_MAINSTONE is not set +# CONFIG_SND_PXA2xx_SOC_MAINSTONE_WM8731 is not set +# CONFIG_SND_PXA2xx_SOC_MAINSTONE_WM8753 is not set +# CONFIG_SND_PXA2xx_SOC_MAINSTONE_WM8974 is not set +# CONFIG_SND_PXA2xx_SOC_MAINSTONE_WM9713 is not set +# CONFIG_SND_MAINSTONE_BASEBAND is not set +# CONFIG_SND_MAINSTONE_BLUETOOTH is not set +# CONFIG_SND_PXA2xx_SOC_MAINSTONE_WM9712 is not set +# CONFIG_SND_PXA2xx_SOC_CORGI is not set +# CONFIG_SND_PXA2xx_SOC_SPITZ is not set +CONFIG_SND_PXA2xx_SOC_POODLE=m +# CONFIG_SND_PXA2xx_SOC_TOSA is not set + +# +# Soc Codecs +# +# CONFIG_SND_SOC_AC97_CODEC is not set +CONFIG_SND_SOC_WM8731=m +# CONFIG_SND_SOC_WM8750 is not set +# CONFIG_SND_SOC_WM8753 is not set +# CONFIG_SND_SOC_WM8772 is not set +# CONFIG_SND_SOC_WM8971 is not set +# CONFIG_SND_SOC_WM8974 is not set +# CONFIG_SND_SOC_WM9713 is not set +# CONFIG_SND_SOC_WM9712 is not set +# CONFIG_SND_SOC_UDA1380 is not set +# CONFIG_SND_SOC_AK4535 is not set + +# +# Open Sound System +# +# CONFIG_SOUND_PRIME is not set + +# +# USB support +# +CONFIG_USB_ARCH_HAS_HCD=y +# CONFIG_USB_ARCH_HAS_OHCI is not set +# CONFIG_USB_ARCH_HAS_EHCI is not set +CONFIG_USB=m +# CONFIG_USB_DEBUG is not set + +# +# Miscellaneous USB options +# +CONFIG_USB_DEVICEFS=y +# CONFIG_USB_BANDWIDTH is not set +# CONFIG_USB_DYNAMIC_MINORS is not set +# CONFIG_USB_SUSPEND is not set +# CONFIG_USB_OTG is not set + +# +# USB Host Controller Drivers +# +# CONFIG_USB_ISP116X_HCD is not set +CONFIG_USB_SL811_HCD=m +CONFIG_USB_SL811_CS=m + +# +# USB Device Class drivers +# +CONFIG_USB_ACM=m +CONFIG_USB_PRINTER=m + +# +# NOTE: USB_STORAGE enables SCSI, and 'SCSI disk support' +# + +# +# may also be needed; see USB_STORAGE Help for more information +# +CONFIG_USB_STORAGE=m +# CONFIG_USB_STORAGE_DEBUG is not set +# CONFIG_USB_STORAGE_DATAFAB is not set +# CONFIG_USB_STORAGE_FREECOM is not set +# CONFIG_USB_STORAGE_ISD200 is not set +# CONFIG_USB_STORAGE_DPCM is not set +# CONFIG_USB_STORAGE_USBAT is not set +# CONFIG_USB_STORAGE_SDDR09 is not set +# CONFIG_USB_STORAGE_SDDR55 is not set +# CONFIG_USB_STORAGE_JUMPSHOT is not set +# CONFIG_USB_STORAGE_ALAUDA is not set +# CONFIG_USB_LIBUSUAL is not set + +# +# USB Input Devices +# +CONFIG_USB_HID=m +CONFIG_USB_HIDINPUT=y +# CONFIG_USB_HIDINPUT_POWERBOOK is not set +# CONFIG_HID_FF is not set +# CONFIG_USB_HIDDEV is not set + +# +# USB HID Boot Protocol drivers +# +CONFIG_USB_KBD=m +CONFIG_USB_MOUSE=m +CONFIG_USB_AIPTEK=m +CONFIG_USB_WACOM=m +# CONFIG_USB_ACECAD is not set +CONFIG_USB_KBTAB=m +CONFIG_USB_POWERMATE=m +# CONFIG_USB_TOUCHSCREEN is not set +# CONFIG_USB_YEALINK is not set +CONFIG_USB_XPAD=m +CONFIG_USB_ATI_REMOTE=m +# CONFIG_USB_ATI_REMOTE2 is not set +# CONFIG_USB_KEYSPAN_REMOTE is not set +# CONFIG_USB_APPLETOUCH is not set + +# +# USB Imaging devices +# +CONFIG_USB_MDC800=m +CONFIG_USB_MICROTEK=m + +# +# USB Network Adapters +# +CONFIG_USB_CATC=m +CONFIG_USB_KAWETH=m +CONFIG_USB_PEGASUS=m +CONFIG_USB_RTL8150=m +CONFIG_USB_USBNET=m +CONFIG_USB_NET_AX8817X=m +CONFIG_USB_NET_CDCETHER=m +# CONFIG_USB_NET_GL620A is not set +CONFIG_USB_NET_NET1080=m +# CONFIG_USB_NET_PLUSB is not set +# CONFIG_USB_NET_RNDIS_HOST is not set +# CONFIG_USB_NET_CDC_SUBSET is not set +CONFIG_USB_NET_ZAURUS=m +# CONFIG_USB_ZD1201 is not set +CONFIG_USB_MON=y + +# +# USB port drivers +# + +# +# USB Serial Converter support +# +CONFIG_USB_SERIAL=m +CONFIG_USB_SERIAL_GENERIC=y +# CONFIG_USB_SERIAL_AIRPRIME is not set +# CONFIG_USB_SERIAL_ANYDATA is not set +# CONFIG_USB_SERIAL_ARK3116 is not set +CONFIG_USB_SERIAL_BELKIN=m +# CONFIG_USB_SERIAL_WHITEHEAT is not set +CONFIG_USB_SERIAL_DIGI_ACCELEPORT=m +# CONFIG_USB_SERIAL_CP2101 is not set +CONFIG_USB_SERIAL_CYPRESS_M8=m +CONFIG_USB_SERIAL_EMPEG=m +CONFIG_USB_SERIAL_FTDI_SIO=m +# CONFIG_USB_SERIAL_FUNSOFT is not set +CONFIG_USB_SERIAL_VISOR=m +CONFIG_USB_SERIAL_IPAQ=m +CONFIG_USB_SERIAL_IR=m +CONFIG_USB_SERIAL_EDGEPORT=m +CONFIG_USB_SERIAL_EDGEPORT_TI=m +CONFIG_USB_SERIAL_GARMIN=m +CONFIG_USB_SERIAL_IPW=m +CONFIG_USB_SERIAL_KEYSPAN_PDA=m +CONFIG_USB_SERIAL_KEYSPAN=m +# CONFIG_USB_SERIAL_KEYSPAN_MPR is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XA is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XB is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA18X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19QW is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19QI is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49WLC is not set +CONFIG_USB_SERIAL_KLSI=m +CONFIG_USB_SERIAL_KOBIL_SCT=m +CONFIG_USB_SERIAL_MCT_U232=m +# CONFIG_USB_SERIAL_NAVMAN is not set +CONFIG_USB_SERIAL_PL2303=m +# CONFIG_USB_SERIAL_HP4X is not set +CONFIG_USB_SERIAL_SAFE=m +# CONFIG_USB_SERIAL_SAFE_PADDED is not set +CONFIG_USB_SERIAL_TI=m +CONFIG_USB_SERIAL_CYBERJACK=m +CONFIG_USB_SERIAL_XIRCOM=m +CONFIG_USB_SERIAL_OMNINET=m +CONFIG_USB_EZUSB=y + +# +# USB Miscellaneous drivers +# +CONFIG_USB_EMI62=m +CONFIG_USB_EMI26=m +CONFIG_USB_AUERSWALD=m +CONFIG_USB_RIO500=m +CONFIG_USB_LEGOTOWER=m +CONFIG_USB_LCD=m +CONFIG_USB_LED=m +CONFIG_USB_CYTHERM=m +CONFIG_USB_PHIDGETKIT=m +CONFIG_USB_PHIDGETSERVO=m +CONFIG_USB_IDMOUSE=m +# CONFIG_USB_LD is not set +# CONFIG_USB_TEST is not set + +# +# USB DSL modem support +# + +# +# USB Gadget Support +# +CONFIG_USB_GADGET=y +# CONFIG_USB_GADGET_DEBUG_FILES is not set +CONFIG_USB_GADGET_SELECTED=y +# CONFIG_USB_GADGET_NET2280 is not set +CONFIG_USB_GADGET_PXA2XX=y +CONFIG_USB_PXA2XX=y +# CONFIG_USB_PXA2XX_SMALL is not set +# CONFIG_USB_GADGET_PXA27X is not set +# CONFIG_USB_GADGET_GOKU is not set +# CONFIG_USB_GADGET_LH7A40X is not set +# CONFIG_USB_GADGET_OMAP is not set +# CONFIG_USB_GADGET_AT91 is not set +# CONFIG_USB_GADGET_DUMMY_HCD is not set +# CONFIG_USB_GADGET_DUALSPEED is not set +CONFIG_USB_ZERO=m +CONFIG_USB_ETH=m +CONFIG_USB_ETH_RNDIS=y +CONFIG_USB_GADGETFS=m +CONFIG_USB_FILE_STORAGE=m +# CONFIG_USB_FILE_STORAGE_TEST is not set +CONFIG_USB_G_SERIAL=m + +# +# MMC/SD Card support +# +CONFIG_MMC=y +# CONFIG_MMC_DEBUG is not set +CONFIG_MMC_BLOCK=y +CONFIG_MMC_PXA=y + +# +# Real Time Clock +# +CONFIG_RTC_LIB=y +CONFIG_RTC_CLASS=y +CONFIG_RTC_HCTOSYS=y +CONFIG_RTC_HCTOSYS_DEVICE="rtc0" + +# +# RTC interfaces +# +CONFIG_RTC_INTF_SYSFS=y +CONFIG_RTC_INTF_PROC=y +CONFIG_RTC_INTF_DEV=y + +# +# RTC drivers +# +# CONFIG_RTC_DRV_X1205 is not set +# CONFIG_RTC_DRV_DS1672 is not set +# CONFIG_RTC_DRV_PCF8563 is not set +# CONFIG_RTC_DRV_RS5C372 is not set +# CONFIG_RTC_DRV_M48T86 is not set +CONFIG_RTC_DRV_SA1100=y +# CONFIG_RTC_DRV_TEST is not set + +# +# File systems +# +CONFIG_EXT2_FS=y +# CONFIG_EXT2_FS_XATTR is not set +# CONFIG_EXT2_FS_XIP is not set +# CONFIG_EXT3_FS is not set +# CONFIG_REISERFS_FS is not set +# CONFIG_JFS_FS is not set +# CONFIG_FS_POSIX_ACL is not set +# CONFIG_XFS_FS is not set +# CONFIG_OCFS2_FS is not set +# CONFIG_MINIX_FS is not set +# CONFIG_ROMFS_FS is not set +CONFIG_INOTIFY=y +# CONFIG_QUOTA is not set +CONFIG_DNOTIFY=y +# CONFIG_AUTOFS_FS is not set +# CONFIG_AUTOFS4_FS is not set +# CONFIG_FUSE_FS is not set + +# +# CD-ROM/DVD Filesystems +# +# CONFIG_ISO9660_FS is not set +# CONFIG_UDF_FS is not set + +# +# DOS/FAT/NT Filesystems +# +CONFIG_FAT_FS=y +# CONFIG_MSDOS_FS is not set +CONFIG_VFAT_FS=y +CONFIG_FAT_DEFAULT_CODEPAGE=437 +CONFIG_FAT_DEFAULT_IOCHARSET="iso8859-1" +# CONFIG_NTFS_FS is not set + +# +# Pseudo filesystems +# +CONFIG_PROC_FS=y +CONFIG_SYSFS=y +CONFIG_TMPFS=y +# CONFIG_HUGETLB_PAGE is not set +CONFIG_RAMFS=y +# CONFIG_CONFIGFS_FS is not set + +# +# Miscellaneous filesystems +# +# CONFIG_ADFS_FS is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_HFSPLUS_FS is not set +# CONFIG_BEFS_FS is not set +# CONFIG_BFS_FS is not set +# CONFIG_EFS_FS is not set +# CONFIG_JFFS_FS is not set +CONFIG_JFFS2_FS=y +CONFIG_JFFS2_FS_DEBUG=0 +CONFIG_JFFS2_FS_WRITEBUFFER=y +CONFIG_JFFS2_SUMMARY=y +CONFIG_JFFS2_COMPRESSION_OPTIONS=y +CONFIG_JFFS2_ZLIB=y +CONFIG_JFFS2_RTIME=y +CONFIG_JFFS2_RUBIN=y +# CONFIG_JFFS2_CMODE_NONE is not set +CONFIG_JFFS2_CMODE_PRIORITY=y +# CONFIG_JFFS2_CMODE_SIZE is not set +CONFIG_CRAMFS=m +CONFIG_SQUASHFS=m +# CONFIG_SQUASHFS_EMBEDDED is not set +CONFIG_SQUASHFS_FRAGMENT_CACHE_SIZE=3 +# CONFIG_SQUASHFS_VMALLOC is not set +# CONFIG_VXFS_FS is not set +# CONFIG_HPFS_FS is not set +# CONFIG_QNX4FS_FS is not set +# CONFIG_SYSV_FS is not set +# CONFIG_UFS_FS is not set + +# +# Network File Systems +# +CONFIG_NFS_FS=m +CONFIG_NFS_V3=y +# CONFIG_NFS_V3_ACL is not set +CONFIG_NFS_V4=y +# CONFIG_NFS_DIRECTIO is not set +# CONFIG_NFSD is not set +CONFIG_LOCKD=m +CONFIG_LOCKD_V4=y +CONFIG_NFS_COMMON=y +CONFIG_SUNRPC=m +CONFIG_SUNRPC_GSS=m +CONFIG_RPCSEC_GSS_KRB5=m +# CONFIG_RPCSEC_GSS_SPKM3 is not set +CONFIG_SMB_FS=m +CONFIG_SMB_NLS_DEFAULT=y +CONFIG_SMB_NLS_REMOTE="cp437" +CONFIG_CIFS=m +# CONFIG_CIFS_STATS is not set +# CONFIG_CIFS_XATTR is not set +# CONFIG_CIFS_EXPERIMENTAL is not set +# CONFIG_NCP_FS is not set +# CONFIG_CODA_FS is not set +# CONFIG_AFS_FS is not set +# CONFIG_9P_FS is not set + +# +# Partition Types +# +CONFIG_PARTITION_ADVANCED=y +# CONFIG_ACORN_PARTITION is not set +# CONFIG_OSF_PARTITION is not set +# CONFIG_AMIGA_PARTITION is not set +# CONFIG_ATARI_PARTITION is not set +# CONFIG_MAC_PARTITION is not set +CONFIG_MSDOS_PARTITION=y +# CONFIG_BSD_DISKLABEL is not set +# CONFIG_MINIX_SUBPARTITION is not set +# CONFIG_SOLARIS_X86_PARTITION is not set +# CONFIG_UNIXWARE_DISKLABEL is not set +# CONFIG_LDM_PARTITION is not set +# CONFIG_SGI_PARTITION is not set +# CONFIG_ULTRIX_PARTITION is not set +# CONFIG_SUN_PARTITION is not set +# CONFIG_KARMA_PARTITION is not set +# CONFIG_EFI_PARTITION is not set + +# +# Native Language Support +# +CONFIG_NLS=y +CONFIG_NLS_DEFAULT="cp437" +CONFIG_NLS_CODEPAGE_437=y +CONFIG_NLS_CODEPAGE_737=m +CONFIG_NLS_CODEPAGE_775=m +CONFIG_NLS_CODEPAGE_850=m +CONFIG_NLS_CODEPAGE_852=m +CONFIG_NLS_CODEPAGE_855=m +CONFIG_NLS_CODEPAGE_857=m +CONFIG_NLS_CODEPAGE_860=m +CONFIG_NLS_CODEPAGE_861=m +CONFIG_NLS_CODEPAGE_862=m +CONFIG_NLS_CODEPAGE_863=m +CONFIG_NLS_CODEPAGE_864=m +CONFIG_NLS_CODEPAGE_865=m +CONFIG_NLS_CODEPAGE_866=m +CONFIG_NLS_CODEPAGE_869=m +CONFIG_NLS_CODEPAGE_936=m +CONFIG_NLS_CODEPAGE_950=m +CONFIG_NLS_CODEPAGE_932=m +CONFIG_NLS_CODEPAGE_949=m +CONFIG_NLS_CODEPAGE_874=m +CONFIG_NLS_ISO8859_8=m +CONFIG_NLS_CODEPAGE_1250=m +CONFIG_NLS_CODEPAGE_1251=m +CONFIG_NLS_ASCII=m +CONFIG_NLS_ISO8859_1=y +CONFIG_NLS_ISO8859_2=m +CONFIG_NLS_ISO8859_3=m +CONFIG_NLS_ISO8859_4=m +CONFIG_NLS_ISO8859_5=m +CONFIG_NLS_ISO8859_6=m +CONFIG_NLS_ISO8859_7=m +CONFIG_NLS_ISO8859_9=m +CONFIG_NLS_ISO8859_13=m +CONFIG_NLS_ISO8859_14=m +CONFIG_NLS_ISO8859_15=m +CONFIG_NLS_KOI8_R=m +CONFIG_NLS_KOI8_U=m +CONFIG_NLS_UTF8=y + +# +# Profiling support +# +CONFIG_PROFILING=y +CONFIG_OPROFILE=m + +# +# Kernel hacking +# +# CONFIG_PRINTK_TIME is not set +CONFIG_MAGIC_SYSRQ=y +CONFIG_DEBUG_KERNEL=y +CONFIG_LOG_BUF_SHIFT=14 +CONFIG_DETECT_SOFTLOCKUP=y +# CONFIG_SCHEDSTATS is not set +# CONFIG_DEBUG_SLAB is not set +# CONFIG_DEBUG_PREEMPT is not set +# CONFIG_DEBUG_MUTEXES is not set +# CONFIG_DEBUG_SPINLOCK is not set +# CONFIG_DEBUG_SPINLOCK_SLEEP is not set +# CONFIG_DEBUG_KOBJECT is not set +CONFIG_DEBUG_BUGVERBOSE=y +# CONFIG_DEBUG_INFO is not set +# CONFIG_DEBUG_FS is not set +# CONFIG_DEBUG_VM is not set +CONFIG_FRAME_POINTER=y +# CONFIG_UNWIND_INFO is not set +# CONFIG_FORCED_INLINING is not set +# CONFIG_RCU_TORTURE_TEST is not set +# CONFIG_DEBUG_USER is not set +# CONFIG_DEBUG_WAITQ is not set +CONFIG_DEBUG_ERRORS=y +# CONFIG_DEBUG_LL is not set + +# +# Security options +# +# CONFIG_KEYS is not set +# CONFIG_SECURITY is not set + +# +# Cryptographic options +# +CONFIG_CRYPTO=y +CONFIG_CRYPTO_HMAC=y +CONFIG_CRYPTO_NULL=m +CONFIG_CRYPTO_MD4=m +CONFIG_CRYPTO_MD5=m +CONFIG_CRYPTO_SHA1=m +CONFIG_CRYPTO_SHA256=m +CONFIG_CRYPTO_SHA512=m +CONFIG_CRYPTO_WP512=m +# CONFIG_CRYPTO_TGR192 is not set +CONFIG_CRYPTO_DES=m +CONFIG_CRYPTO_BLOWFISH=m +CONFIG_CRYPTO_TWOFISH=m +CONFIG_CRYPTO_SERPENT=m +CONFIG_CRYPTO_AES=m +CONFIG_CRYPTO_CAST5=m +CONFIG_CRYPTO_CAST6=m +CONFIG_CRYPTO_TEA=m +CONFIG_CRYPTO_ARC4=m +CONFIG_CRYPTO_KHAZAD=m +CONFIG_CRYPTO_ANUBIS=m +CONFIG_CRYPTO_DEFLATE=m +CONFIG_CRYPTO_MICHAEL_MIC=m +CONFIG_CRYPTO_CRC32C=m +CONFIG_CRYPTO_TEST=m + +# +# Hardware crypto devices +# + +# +# Library routines +# +CONFIG_CRC_CCITT=y +# CONFIG_CRC16 is not set +CONFIG_CRC32=y +CONFIG_LIBCRC32C=m +CONFIG_ZLIB_INFLATE=y +CONFIG_ZLIB_DEFLATE=y diff --git a/packages/linux/linux-rp-2.6.20/defconfig-qemuarm b/packages/linux/linux-rp-2.6.20/defconfig-qemuarm new file mode 100644 index 0000000000..e34fe5c090 --- /dev/null +++ b/packages/linux/linux-rp-2.6.20/defconfig-qemuarm @@ -0,0 +1,1190 @@ +# +# Automatically generated make config: don't edit +# Linux kernel version: 2.6.17 +# Sat Aug 26 22:45:02 2006 +# +CONFIG_ARM=y +CONFIG_MMU=y +CONFIG_RWSEM_GENERIC_SPINLOCK=y +CONFIG_GENERIC_HWEIGHT=y +CONFIG_GENERIC_CALIBRATE_DELAY=y +CONFIG_VECTORS_BASE=0xffff0000 + +# +# Code maturity level options +# +CONFIG_EXPERIMENTAL=y +CONFIG_BROKEN_ON_SMP=y +CONFIG_INIT_ENV_ARG_LIMIT=32 + +# +# General setup +# +CONFIG_LOCALVERSION="" +CONFIG_LOCALVERSION_AUTO=y +CONFIG_SWAP=y +CONFIG_SYSVIPC=y +# CONFIG_POSIX_MQUEUE is not set +# CONFIG_BSD_PROCESS_ACCT is not set +CONFIG_SYSCTL=y +# CONFIG_AUDIT is not set +CONFIG_IKCONFIG=y +CONFIG_IKCONFIG_PROC=y +# CONFIG_RELAY is not set +CONFIG_INITRAMFS_SOURCE="" +CONFIG_UID16=y +CONFIG_CC_OPTIMIZE_FOR_SIZE=y +# CONFIG_EMBEDDED is not set +CONFIG_KALLSYMS=y +# CONFIG_KALLSYMS_ALL is not set +# CONFIG_KALLSYMS_EXTRA_PASS is not set +CONFIG_HOTPLUG=y +CONFIG_PRINTK=y +CONFIG_BUG=y +CONFIG_ELF_CORE=y +CONFIG_BASE_FULL=y +CONFIG_FUTEX=y +CONFIG_EPOLL=y +CONFIG_SHMEM=y +CONFIG_SLAB=y +# CONFIG_TINY_SHMEM is not set +CONFIG_BASE_SMALL=0 +# CONFIG_SLOB is not set +CONFIG_OBSOLETE_INTERMODULE=y + +# +# Loadable module support +# +CONFIG_MODULES=y +CONFIG_MODULE_UNLOAD=y +# CONFIG_MODULE_FORCE_UNLOAD is not set +# CONFIG_MODVERSIONS is not set +# CONFIG_MODULE_SRCVERSION_ALL is not set +CONFIG_KMOD=y + +# +# Block layer +# +# CONFIG_BLK_DEV_IO_TRACE is not set + +# +# IO Schedulers +# +CONFIG_IOSCHED_NOOP=y +CONFIG_IOSCHED_AS=y +CONFIG_IOSCHED_DEADLINE=y +CONFIG_IOSCHED_CFQ=y +CONFIG_DEFAULT_AS=y +# CONFIG_DEFAULT_DEADLINE is not set +# CONFIG_DEFAULT_CFQ is not set +# CONFIG_DEFAULT_NOOP is not set +CONFIG_DEFAULT_IOSCHED="anticipatory" + +# +# System Type +# +# CONFIG_ARCH_CLPS7500 is not set +# CONFIG_ARCH_CLPS711X is not set +# CONFIG_ARCH_CO285 is not set +# CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_EP93XX is not set +# CONFIG_ARCH_FOOTBRIDGE is not set +# CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_IOP3XX is not set +# CONFIG_ARCH_IXP4XX is not set +# CONFIG_ARCH_IXP2000 is not set +# CONFIG_ARCH_IXP23XX is not set +# CONFIG_ARCH_L7200 is not set +# CONFIG_ARCH_PXA is not set +# CONFIG_ARCH_RPC is not set +# CONFIG_ARCH_SA1100 is not set +# CONFIG_ARCH_S3C2410 is not set +# CONFIG_ARCH_SHARK is not set +# CONFIG_ARCH_LH7A40X is not set +# CONFIG_ARCH_OMAP is not set +CONFIG_ARCH_VERSATILE=y +# CONFIG_ARCH_REALVIEW is not set +# CONFIG_ARCH_IMX is not set +# CONFIG_ARCH_H720X is not set +# CONFIG_ARCH_AAEC2000 is not set +# CONFIG_ARCH_AT91RM9200 is not set + +# +# Versatile platform type +# +CONFIG_ARCH_VERSATILE_PB=y +# CONFIG_MACH_VERSATILE_AB is not set + +# +# Processor Type +# +CONFIG_CPU_32=y +CONFIG_CPU_ARM926T=y +CONFIG_CPU_32v5=y +CONFIG_CPU_ABRT_EV5TJ=y +CONFIG_CPU_CACHE_VIVT=y +CONFIG_CPU_COPY_V4WB=y +CONFIG_CPU_TLB_V4WBI=y + +# +# Processor Features +# +CONFIG_ARM_THUMB=y +# CONFIG_CPU_ICACHE_DISABLE is not set +# CONFIG_CPU_DCACHE_DISABLE is not set +# CONFIG_CPU_DCACHE_WRITETHROUGH is not set +# CONFIG_CPU_CACHE_ROUND_ROBIN is not set +# CONFIG_KEXEC is not set +CONFIG_ARM_VIC=y +CONFIG_ICST307=y + +# +# Bus support +# +CONFIG_ARM_AMBA=y +CONFIG_PCI=y +# CONFIG_PCI_DEBUG is not set + +# +# PCCARD (PCMCIA/CardBus) support +# +# CONFIG_PCCARD is not set + +# +# Kernel Features +# +# CONFIG_PREEMPT is not set +# CONFIG_NO_IDLE_HZ is not set +CONFIG_HZ=100 +# CONFIG_AEABI is not set +# CONFIG_ARCH_DISCONTIGMEM_ENABLE is not set +CONFIG_SELECT_MEMORY_MODEL=y +CONFIG_FLATMEM_MANUAL=y +# CONFIG_DISCONTIGMEM_MANUAL is not set +# CONFIG_SPARSEMEM_MANUAL is not set +CONFIG_FLATMEM=y +CONFIG_FLAT_NODE_MEM_MAP=y +# CONFIG_SPARSEMEM_STATIC is not set +CONFIG_SPLIT_PTLOCK_CPUS=4096 +CONFIG_LEDS=y +CONFIG_LEDS_TIMER=y +CONFIG_LEDS_CPU=y +CONFIG_ALIGNMENT_TRAP=y + +# +# Boot options +# +CONFIG_ZBOOT_ROM_TEXT=0x0 +CONFIG_ZBOOT_ROM_BSS=0x0 +CONFIG_CMDLINE="console=ttyS0,115200n8 console=tty1 noinitrd root=/dev/mtdblock2 rootfstype=jffs2 dyntick=enable debug" +# CONFIG_XIP_KERNEL is not set + +# +# Floating point emulation +# + +# +# At least one emulation must be selected +# +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_NWFPE_XP is not set +# CONFIG_FPE_FASTFPE is not set +CONFIG_VFP=y + +# +# Userspace binary formats +# +CONFIG_BINFMT_ELF=y +# CONFIG_BINFMT_AOUT is not set +# CONFIG_BINFMT_MISC is not set +# CONFIG_ARTHUR is not set + +# +# Power management options +# +CONFIG_PM=y +# CONFIG_PM_LEGACY is not set +# CONFIG_PM_DEBUG is not set +# CONFIG_APM is not set + +# +# Networking +# +CONFIG_NET=y + +# +# Networking options +# +# CONFIG_NETDEBUG is not set +CONFIG_PACKET=y +CONFIG_PACKET_MMAP=y +CONFIG_UNIX=y +# CONFIG_NET_KEY is not set +CONFIG_INET=y +CONFIG_IP_MULTICAST=y +# CONFIG_IP_ADVANCED_ROUTER is not set +CONFIG_IP_FIB_HASH=y +CONFIG_IP_PNP=y +CONFIG_IP_PNP_DHCP=y +CONFIG_IP_PNP_BOOTP=y +# CONFIG_IP_PNP_RARP is not set +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_IP_MROUTE is not set +# CONFIG_ARPD is not set +# CONFIG_SYN_COOKIES is not set +# CONFIG_INET_AH is not set +# CONFIG_INET_ESP is not set +# CONFIG_INET_IPCOMP is not set +# CONFIG_INET_XFRM_TUNNEL is not set +# CONFIG_INET_TUNNEL is not set +CONFIG_INET_DIAG=y +CONFIG_INET_TCP_DIAG=y +# CONFIG_TCP_CONG_ADVANCED is not set +CONFIG_TCP_CONG_BIC=y +# CONFIG_IPV6 is not set +# CONFIG_INET6_XFRM_TUNNEL is not set +# CONFIG_INET6_TUNNEL is not set +# CONFIG_NETFILTER is not set + +# +# DCCP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_DCCP is not set + +# +# SCTP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_SCTP is not set + +# +# TIPC Configuration (EXPERIMENTAL) +# +# CONFIG_TIPC is not set +# CONFIG_ATM is not set +# CONFIG_BRIDGE is not set +# CONFIG_VLAN_8021Q is not set +# CONFIG_DECNET is not set +# CONFIG_LLC2 is not set +# CONFIG_IPX is not set +# CONFIG_ATALK is not set +# CONFIG_X25 is not set +# CONFIG_LAPB is not set +# CONFIG_NET_DIVERT is not set +# CONFIG_ECONET is not set +# CONFIG_WAN_ROUTER is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network testing +# +# CONFIG_NET_PKTGEN is not set +# CONFIG_HAMRADIO is not set +# CONFIG_IRDA is not set +# CONFIG_BT is not set +# CONFIG_IEEE80211 is not set + +# +# Device Drivers +# + +# +# Generic Driver Options +# +CONFIG_STANDALONE=y +CONFIG_PREVENT_FIRMWARE_BUILD=y +# CONFIG_FW_LOADER is not set +# CONFIG_DEBUG_DRIVER is not set + +# +# Connector - unified userspace <-> kernelspace linker +# +# CONFIG_CONNECTOR is not set + +# +# Memory Technology Devices (MTD) +# +CONFIG_MTD=y +# CONFIG_MTD_DEBUG is not set +# CONFIG_MTD_CONCAT is not set +CONFIG_MTD_PARTITIONS=y +# CONFIG_MTD_REDBOOT_PARTS is not set +CONFIG_MTD_CMDLINE_PARTS=y +CONFIG_MTD_AFS_PARTS=y + +# +# User Modules And Translation Layers +# +CONFIG_MTD_CHAR=y +CONFIG_MTD_BLOCK=y +# CONFIG_FTL is not set +# CONFIG_NFTL is not set +# CONFIG_INFTL is not set +# CONFIG_RFD_FTL is not set + +# +# RAM/ROM/Flash chip drivers +# +CONFIG_MTD_CFI=y +# CONFIG_MTD_JEDECPROBE is not set +CONFIG_MTD_GEN_PROBE=y +CONFIG_MTD_CFI_ADV_OPTIONS=y +CONFIG_MTD_CFI_NOSWAP=y +# CONFIG_MTD_CFI_BE_BYTE_SWAP is not set +# CONFIG_MTD_CFI_LE_BYTE_SWAP is not set +# CONFIG_MTD_CFI_GEOMETRY is not set +CONFIG_MTD_MAP_BANK_WIDTH_1=y +CONFIG_MTD_MAP_BANK_WIDTH_2=y +CONFIG_MTD_MAP_BANK_WIDTH_4=y +# CONFIG_MTD_MAP_BANK_WIDTH_8 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_16 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_32 is not set +CONFIG_MTD_CFI_I1=y +CONFIG_MTD_CFI_I2=y +# CONFIG_MTD_CFI_I4 is not set +# CONFIG_MTD_CFI_I8 is not set +# CONFIG_MTD_OTP is not set +CONFIG_MTD_CFI_INTELEXT=y +# CONFIG_MTD_CFI_AMDSTD is not set +# CONFIG_MTD_CFI_STAA is not set +CONFIG_MTD_CFI_UTIL=y +# CONFIG_MTD_RAM is not set +# CONFIG_MTD_ROM is not set +# CONFIG_MTD_ABSENT is not set +# CONFIG_MTD_OBSOLETE_CHIPS is not set + +# +# Mapping drivers for chip access +# +# CONFIG_MTD_COMPLEX_MAPPINGS is not set +# CONFIG_MTD_PHYSMAP is not set +# CONFIG_MTD_ARM_INTEGRATOR is not set +# CONFIG_MTD_PLATRAM is not set + +# +# Self-contained MTD device drivers +# +# CONFIG_MTD_PMC551 is not set +# CONFIG_MTD_SLRAM is not set +# CONFIG_MTD_PHRAM is not set +# CONFIG_MTD_MTDRAM is not set +# CONFIG_MTD_BLOCK2MTD is not set + +# +# Disk-On-Chip Device Drivers +# +# CONFIG_MTD_DOC2000 is not set +# CONFIG_MTD_DOC2001 is not set +# CONFIG_MTD_DOC2001PLUS is not set + +# +# NAND Flash Device Drivers +# +# CONFIG_MTD_NAND is not set + +# +# OneNAND Flash Device Drivers +# +# CONFIG_MTD_ONENAND is not set + +# +# Parallel port support +# +# CONFIG_PARPORT is not set + +# +# Plug and Play support +# + +# +# Block devices +# +# CONFIG_BLK_CPQ_DA is not set +# CONFIG_BLK_CPQ_CISS_DA is not set +# CONFIG_BLK_DEV_DAC960 is not set +# CONFIG_BLK_DEV_UMEM is not set +# CONFIG_BLK_DEV_COW_COMMON is not set +CONFIG_BLK_DEV_LOOP=y +# CONFIG_BLK_DEV_CRYPTOLOOP is not set +# CONFIG_BLK_DEV_NBD is not set +# CONFIG_BLK_DEV_SX8 is not set +# CONFIG_BLK_DEV_UB is not set +CONFIG_BLK_DEV_RAM=y +CONFIG_BLK_DEV_RAM_COUNT=16 +CONFIG_BLK_DEV_RAM_SIZE=8192 +CONFIG_BLK_DEV_INITRD=y +# CONFIG_CDROM_PKTCDVD is not set +# CONFIG_ATA_OVER_ETH is not set + +# +# SCSI device support +# +# CONFIG_RAID_ATTRS is not set +CONFIG_SCSI=y +CONFIG_SCSI_PROC_FS=y + +# +# SCSI support type (disk, tape, CD-ROM) +# +CONFIG_BLK_DEV_SD=y +# CONFIG_CHR_DEV_ST is not set +# CONFIG_CHR_DEV_OSST is not set +# CONFIG_BLK_DEV_SR is not set +# CONFIG_CHR_DEV_SG is not set +# CONFIG_CHR_DEV_SCH is not set + +# +# Some SCSI devices (e.g. CD jukebox) support multiple LUNs +# +# CONFIG_SCSI_MULTI_LUN is not set +# CONFIG_SCSI_CONSTANTS is not set +# CONFIG_SCSI_LOGGING is not set + +# +# SCSI Transport Attributes +# +CONFIG_SCSI_SPI_ATTRS=y +# CONFIG_SCSI_FC_ATTRS is not set +# CONFIG_SCSI_ISCSI_ATTRS is not set +# CONFIG_SCSI_SAS_ATTRS is not set + +# +# SCSI low-level drivers +# +# CONFIG_ISCSI_TCP is not set +# CONFIG_BLK_DEV_3W_XXXX_RAID is not set +# CONFIG_SCSI_3W_9XXX is not set +# CONFIG_SCSI_ACARD is not set +# CONFIG_SCSI_AACRAID is not set +# CONFIG_SCSI_AIC7XXX is not set +# CONFIG_SCSI_AIC7XXX_OLD is not set +# CONFIG_SCSI_AIC79XX is not set +# CONFIG_SCSI_DPT_I2O is not set +# CONFIG_MEGARAID_NEWGEN is not set +# CONFIG_MEGARAID_LEGACY is not set +# CONFIG_MEGARAID_SAS is not set +# CONFIG_SCSI_SATA is not set +# CONFIG_SCSI_DMX3191D is not set +# CONFIG_SCSI_FUTURE_DOMAIN is not set +# CONFIG_SCSI_IPS is not set +# CONFIG_SCSI_INITIO is not set +# CONFIG_SCSI_INIA100 is not set +CONFIG_SCSI_SYM53C8XX_2=y +CONFIG_SCSI_SYM53C8XX_DMA_ADDRESSING_MODE=1 +CONFIG_SCSI_SYM53C8XX_DEFAULT_TAGS=16 +CONFIG_SCSI_SYM53C8XX_MAX_TAGS=64 +CONFIG_SCSI_SYM53C8XX_MMIO=y +# CONFIG_SCSI_IPR is not set +# CONFIG_SCSI_QLOGIC_1280 is not set +# CONFIG_SCSI_QLA_FC is not set +# CONFIG_SCSI_LPFC is not set +# CONFIG_SCSI_DC395x is not set +# CONFIG_SCSI_DC390T is not set +# CONFIG_SCSI_NSP32 is not set +# CONFIG_SCSI_DEBUG is not set + +# +# Multi-device support (RAID and LVM) +# +CONFIG_MD=y +CONFIG_BLK_DEV_DM=m +CONFIG_DM_CRYPT=m +CONFIG_DM_SNAPSHOT=m +CONFIG_DM_MIRROR=m +CONFIG_DM_ZERO=m +CONFIG_DM_MULTIPATH=m +CONFIG_DM_MULTIPATH_EMC=m + +# +# Fusion MPT device support +# +# CONFIG_FUSION is not set +# CONFIG_FUSION_SPI is not set +# CONFIG_FUSION_FC is not set +# CONFIG_FUSION_SAS is not set + +# +# IEEE 1394 (FireWire) support +# +# CONFIG_IEEE1394 is not set + +# +# I2O device support +# +# CONFIG_I2O is not set + +# +# Network device support +# +CONFIG_NETDEVICES=y +# CONFIG_DUMMY is not set +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +CONFIG_TUN=m + +# +# ARCnet devices +# +# CONFIG_ARCNET is not set + +# +# PHY device support +# +# CONFIG_PHYLIB is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +CONFIG_MII=y +# CONFIG_HAPPYMEAL is not set +# CONFIG_SUNGEM is not set +# CONFIG_CASSINI is not set +# CONFIG_NET_VENDOR_3COM is not set +CONFIG_SMC91X=y +# CONFIG_DM9000 is not set + +# +# Tulip family network device support +# +# CONFIG_NET_TULIP is not set +# CONFIG_HP100 is not set +# CONFIG_NET_PCI is not set + +# +# Ethernet (1000 Mbit) +# +# CONFIG_ACENIC is not set +# CONFIG_DL2K is not set +# CONFIG_E1000 is not set +# CONFIG_NS83820 is not set +# CONFIG_HAMACHI is not set +# CONFIG_YELLOWFIN is not set +# CONFIG_R8169 is not set +# CONFIG_SIS190 is not set +# CONFIG_SKGE is not set +# CONFIG_SKY2 is not set +# CONFIG_SK98LIN is not set +# CONFIG_TIGON3 is not set +# CONFIG_BNX2 is not set + +# +# Ethernet (10000 Mbit) +# +# CONFIG_CHELSIO_T1 is not set +# CONFIG_IXGB is not set +# CONFIG_S2IO is not set + +# +# Token Ring devices +# +# CONFIG_TR is not set + +# +# Wireless LAN (non-hamradio) +# +# CONFIG_NET_RADIO is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set +# CONFIG_FDDI is not set +# CONFIG_HIPPI is not set +# CONFIG_PPP is not set +# CONFIG_SLIP is not set +# CONFIG_NET_FC is not set +# CONFIG_SHAPER is not set +# CONFIG_NETCONSOLE is not set +# CONFIG_NETPOLL is not set +# CONFIG_NET_POLL_CONTROLLER is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Input device support +# +CONFIG_INPUT=y + +# +# Userland interfaces +# +# CONFIG_INPUT_MOUSEDEV is not set +# CONFIG_INPUT_JOYDEV is not set +# CONFIG_INPUT_TSDEV is not set +CONFIG_INPUT_EVDEV=y +# CONFIG_INPUT_EVBUG is not set +# CONFIG_INPUT_POWER is not set + +# +# Input Device Drivers +# +CONFIG_INPUT_KEYBOARD=y +CONFIG_KEYBOARD_ATKBD=y +# CONFIG_KEYBOARD_SUNKBD is not set +# CONFIG_KEYBOARD_LKKBD is not set +# CONFIG_KEYBOARD_XTKBD is not set +# CONFIG_KEYBOARD_NEWTON is not set +CONFIG_INPUT_MOUSE=y +CONFIG_MOUSE_PS2=y +# CONFIG_MOUSE_SERIAL is not set +# CONFIG_MOUSE_VSXXXAA is not set +# CONFIG_INPUT_JOYSTICK is not set +# CONFIG_INPUT_TOUCHSCREEN is not set +CONFIG_INPUT_MISC=y +# CONFIG_INPUT_UINPUT is not set + +# +# Hardware I/O ports +# +CONFIG_SERIO=y +# CONFIG_SERIO_SERPORT is not set +CONFIG_SERIO_AMBAKMI=y +# CONFIG_SERIO_PCIPS2 is not set +CONFIG_SERIO_LIBPS2=y +# CONFIG_SERIO_RAW is not set +# CONFIG_GAMEPORT is not set + +# +# Character devices +# +CONFIG_VT=y +CONFIG_VT_CONSOLE=y +CONFIG_HW_CONSOLE=y +# CONFIG_SERIAL_NONSTANDARD is not set + +# +# Serial drivers +# +# CONFIG_SERIAL_8250 is not set + +# +# Non-8250 serial port support +# +CONFIG_SERIAL_AMBA_PL011=y +CONFIG_SERIAL_AMBA_PL011_CONSOLE=y +CONFIG_SERIAL_CORE=y +CONFIG_SERIAL_CORE_CONSOLE=y +# CONFIG_SERIAL_JSM is not set +CONFIG_UNIX98_PTYS=y +# CONFIG_LEGACY_PTYS is not set + +# +# IPMI +# +# CONFIG_IPMI_HANDLER is not set + +# +# Watchdog Cards +# +# CONFIG_WATCHDOG is not set +# CONFIG_NVRAM is not set +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set +# CONFIG_APPLICOM is not set + +# +# Ftape, the floppy tape device driver +# +# CONFIG_DRM is not set +# CONFIG_RAW_DRIVER is not set + +# +# TPM devices +# +# CONFIG_TCG_TPM is not set +# CONFIG_TELCLOCK is not set + +# +# I2C support +# +# CONFIG_I2C is not set + +# +# SPI support +# +# CONFIG_SPI is not set +# CONFIG_SPI_MASTER is not set + +# +# Dallas's 1-wire bus +# +# CONFIG_W1 is not set + +# +# Hardware Monitoring support +# +CONFIG_HWMON=y +# CONFIG_HWMON_VID is not set +# CONFIG_SENSORS_F71805F is not set +# CONFIG_HWMON_DEBUG_CHIP is not set + +# +# Misc devices +# + +# +# Multi-Function Devices +# + +# +# LED devices +# +CONFIG_NEW_LEDS=y +# CONFIG_LEDS_CLASS is not set + +# +# LED drivers +# + +# +# LED Triggers +# +# CONFIG_LEDS_TRIGGERS is not set + +# +# Multimedia devices +# +# CONFIG_VIDEO_DEV is not set +CONFIG_VIDEO_V4L2=y + +# +# Digital Video Broadcasting Devices +# +# CONFIG_DVB is not set +# CONFIG_USB_DABUSB is not set + +# +# Graphics support +# +CONFIG_FB=y +CONFIG_FB_CFB_FILLRECT=y +CONFIG_FB_CFB_COPYAREA=y +CONFIG_FB_CFB_IMAGEBLIT=y +# CONFIG_FB_MACMODES is not set +CONFIG_FB_FIRMWARE_EDID=y +CONFIG_FB_MODE_HELPERS=y +CONFIG_FB_TILEBLITTING=y +# CONFIG_FB_CIRRUS is not set +# CONFIG_FB_PM2 is not set +CONFIG_FB_ARMCLCD=y +# CONFIG_FB_CYBER2000 is not set +# CONFIG_FB_ASILIANT is not set +# CONFIG_FB_IMSTT is not set +# CONFIG_FB_S1D13XXX is not set +# CONFIG_FB_NVIDIA is not set +# CONFIG_FB_RIVA is not set +# CONFIG_FB_MATROX is not set +# CONFIG_FB_RADEON is not set +# CONFIG_FB_ATY128 is not set +# CONFIG_FB_ATY is not set +# CONFIG_FB_SAVAGE is not set +# CONFIG_FB_SIS is not set +# CONFIG_FB_NEOMAGIC is not set +# CONFIG_FB_KYRO is not set +# CONFIG_FB_3DFX is not set +# CONFIG_FB_VOODOO1 is not set +# CONFIG_FB_TRIDENT is not set +# CONFIG_FB_VIRTUAL is not set + +# +# Console display driver support +# +CONFIG_DUMMY_CONSOLE=y +CONFIG_FRAMEBUFFER_CONSOLE=y +# CONFIG_FRAMEBUFFER_CONSOLE_ROTATION is not set +# CONFIG_FONTS is not set +CONFIG_FONT_8x8=y +CONFIG_FONT_8x16=y + +# +# Logo configuration +# +CONFIG_LOGO=y +CONFIG_LOGO_LINUX_MONO=y +CONFIG_LOGO_LINUX_VGA16=y +# CONFIG_LOGO_LINUX_CLUT224 is not set +CONFIG_LOGO_OHAND_CLUT224=y +# CONFIG_LOGO_OZ240_CLUT224 is not set +# CONFIG_LOGO_OZ480_CLUT224 is not set +# CONFIG_LOGO_OZ640_CLUT224 is not set +# CONFIG_BACKLIGHT_LCD_SUPPORT is not set + +# +# Sound +# +# CONFIG_SOUND is not set + +# +# USB support +# +CONFIG_USB_ARCH_HAS_HCD=y +CONFIG_USB_ARCH_HAS_OHCI=y +CONFIG_USB_ARCH_HAS_EHCI=y +CONFIG_USB=y +# CONFIG_USB_DEBUG is not set + +# +# Miscellaneous USB options +# +# CONFIG_USB_DEVICEFS is not set +# CONFIG_USB_BANDWIDTH is not set +# CONFIG_USB_DYNAMIC_MINORS is not set +# CONFIG_USB_SUSPEND is not set +# CONFIG_USB_OTG is not set + +# +# USB Host Controller Drivers +# +# CONFIG_USB_EHCI_HCD is not set +# CONFIG_USB_ISP116X_HCD is not set +CONFIG_USB_OHCI_HCD=y +# CONFIG_USB_OHCI_BIG_ENDIAN is not set +CONFIG_USB_OHCI_LITTLE_ENDIAN=y +# CONFIG_USB_UHCI_HCD is not set +# CONFIG_USB_SL811_HCD is not set + +# +# USB Device Class drivers +# +# CONFIG_USB_ACM is not set +# CONFIG_USB_PRINTER is not set + +# +# NOTE: USB_STORAGE enables SCSI, and 'SCSI disk support' +# + +# +# may also be needed; see USB_STORAGE Help for more information +# +# CONFIG_USB_STORAGE is not set +# CONFIG_USB_LIBUSUAL is not set + +# +# USB Input Devices +# +CONFIG_USB_HID=y + +# +# USB HID Boot Protocol drivers +# +# CONFIG_USB_KBD is not set +# CONFIG_USB_MOUSE is not set +# CONFIG_USB_AIPTEK is not set +CONFIG_USB_WACOM=y +# CONFIG_USB_ACECAD is not set +# CONFIG_USB_KBTAB is not set +# CONFIG_USB_POWERMATE is not set +# CONFIG_USB_TOUCHSCREEN is not set +# CONFIG_USB_YEALINK is not set +# CONFIG_USB_XPAD is not set +# CONFIG_USB_ATI_REMOTE is not set +# CONFIG_USB_ATI_REMOTE2 is not set +# CONFIG_USB_KEYSPAN_REMOTE is not set +# CONFIG_USB_APPLETOUCH is not set + +# +# USB Imaging devices +# +# CONFIG_USB_MDC800 is not set +# CONFIG_USB_MICROTEK is not set + +# +# USB Network Adapters +# +# CONFIG_USB_CATC is not set +# CONFIG_USB_KAWETH is not set +# CONFIG_USB_PEGASUS is not set +# CONFIG_USB_RTL8150 is not set +# CONFIG_USB_USBNET is not set +CONFIG_USB_MON=y + +# +# USB port drivers +# + +# +# USB Serial Converter support +# +# CONFIG_USB_SERIAL is not set + +# +# USB Miscellaneous drivers +# +# CONFIG_USB_EMI62 is not set +# CONFIG_USB_EMI26 is not set +# CONFIG_USB_AUERSWALD is not set +# CONFIG_USB_RIO500 is not set +# CONFIG_USB_LEGOTOWER is not set +# CONFIG_USB_LCD is not set +# CONFIG_USB_LED is not set +# CONFIG_USB_CYTHERM is not set +# CONFIG_USB_PHIDGETKIT is not set +# CONFIG_USB_PHIDGETSERVO is not set +# CONFIG_USB_IDMOUSE is not set +# CONFIG_USB_LD is not set + +# +# USB DSL modem support +# + +# +# USB Gadget Support +# +# CONFIG_USB_GADGET is not set + +# +# MMC/SD Card support +# +# CONFIG_MMC is not set + +# +# Real Time Clock +# +CONFIG_RTC_LIB=y +CONFIG_RTC_CLASS=y +CONFIG_RTC_HCTOSYS=y +CONFIG_RTC_HCTOSYS_DEVICE="rtc0" + +# +# RTC interfaces +# +CONFIG_RTC_INTF_SYSFS=y +CONFIG_RTC_INTF_PROC=y +CONFIG_RTC_INTF_DEV=y + +# +# RTC drivers +# +# CONFIG_RTC_DRV_M48T86 is not set +# CONFIG_RTC_DRV_TEST is not set + +# +# File systems +# +CONFIG_EXT2_FS=y +# CONFIG_EXT2_FS_XATTR is not set +# CONFIG_EXT2_FS_XIP is not set +# CONFIG_EXT3_FS is not set +# CONFIG_REISERFS_FS is not set +# CONFIG_JFS_FS is not set +# CONFIG_FS_POSIX_ACL is not set +# CONFIG_XFS_FS is not set +# CONFIG_OCFS2_FS is not set +# CONFIG_MINIX_FS is not set +# CONFIG_ROMFS_FS is not set +CONFIG_INOTIFY=y +# CONFIG_QUOTA is not set +CONFIG_DNOTIFY=y +# CONFIG_AUTOFS_FS is not set +# CONFIG_AUTOFS4_FS is not set +# CONFIG_FUSE_FS is not set + +# +# CD-ROM/DVD Filesystems +# +# CONFIG_ISO9660_FS is not set +# CONFIG_UDF_FS is not set + +# +# DOS/FAT/NT Filesystems +# +# CONFIG_MSDOS_FS is not set +# CONFIG_VFAT_FS is not set +# CONFIG_NTFS_FS is not set + +# +# Pseudo filesystems +# +CONFIG_PROC_FS=y +CONFIG_SYSFS=y +CONFIG_TMPFS=y +# CONFIG_HUGETLB_PAGE is not set +CONFIG_RAMFS=y +# CONFIG_CONFIGFS_FS is not set + +# +# Miscellaneous filesystems +# +# CONFIG_ADFS_FS is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_HFSPLUS_FS is not set +# CONFIG_BEFS_FS is not set +# CONFIG_BFS_FS is not set +# CONFIG_EFS_FS is not set +# CONFIG_JFFS_FS is not set +CONFIG_JFFS2_FS=y +CONFIG_JFFS2_FS_DEBUG=0 +CONFIG_JFFS2_FS_WRITEBUFFER=y +# CONFIG_JFFS2_SUMMARY is not set +# CONFIG_JFFS2_COMPRESSION_OPTIONS is not set +CONFIG_JFFS2_ZLIB=y +CONFIG_JFFS2_RTIME=y +# CONFIG_JFFS2_RUBIN is not set +CONFIG_CRAMFS=y +# CONFIG_SQUASHFS is not set +# CONFIG_VXFS_FS is not set +# CONFIG_HPFS_FS is not set +# CONFIG_QNX4FS_FS is not set +# CONFIG_SYSV_FS is not set +# CONFIG_UFS_FS is not set + +# +# Network File Systems +# +CONFIG_NFS_FS=y +CONFIG_NFS_V3=y +# CONFIG_NFS_V3_ACL is not set +# CONFIG_NFS_V4 is not set +# CONFIG_NFS_DIRECTIO is not set +CONFIG_NFSD=y +CONFIG_NFSD_V3=y +# CONFIG_NFSD_V3_ACL is not set +# CONFIG_NFSD_V4 is not set +# CONFIG_NFSD_TCP is not set +CONFIG_ROOT_NFS=y +CONFIG_LOCKD=y +CONFIG_LOCKD_V4=y +CONFIG_EXPORTFS=y +CONFIG_NFS_COMMON=y +CONFIG_SUNRPC=y +# CONFIG_RPCSEC_GSS_KRB5 is not set +# CONFIG_RPCSEC_GSS_SPKM3 is not set +CONFIG_SMB_FS=y +# CONFIG_SMB_NLS_DEFAULT is not set +# CONFIG_CIFS is not set +# CONFIG_NCP_FS is not set +# CONFIG_CODA_FS is not set +# CONFIG_AFS_FS is not set +# CONFIG_9P_FS is not set + +# +# Partition Types +# +CONFIG_PARTITION_ADVANCED=y +# CONFIG_ACORN_PARTITION is not set +# CONFIG_OSF_PARTITION is not set +# CONFIG_AMIGA_PARTITION is not set +# CONFIG_ATARI_PARTITION is not set +# CONFIG_MAC_PARTITION is not set +CONFIG_MSDOS_PARTITION=y +# CONFIG_BSD_DISKLABEL is not set +# CONFIG_MINIX_SUBPARTITION is not set +# CONFIG_SOLARIS_X86_PARTITION is not set +# CONFIG_UNIXWARE_DISKLABEL is not set +# CONFIG_LDM_PARTITION is not set +# CONFIG_SGI_PARTITION is not set +# CONFIG_ULTRIX_PARTITION is not set +# CONFIG_SUN_PARTITION is not set +# CONFIG_KARMA_PARTITION is not set +# CONFIG_EFI_PARTITION is not set + +# +# Native Language Support +# +CONFIG_NLS=y +CONFIG_NLS_DEFAULT="iso8859-1" +# CONFIG_NLS_CODEPAGE_437 is not set +# CONFIG_NLS_CODEPAGE_737 is not set +# CONFIG_NLS_CODEPAGE_775 is not set +# CONFIG_NLS_CODEPAGE_850 is not set +# CONFIG_NLS_CODEPAGE_852 is not set +# CONFIG_NLS_CODEPAGE_855 is not set +# CONFIG_NLS_CODEPAGE_857 is not set +# CONFIG_NLS_CODEPAGE_860 is not set +# CONFIG_NLS_CODEPAGE_861 is not set +# CONFIG_NLS_CODEPAGE_862 is not set +# CONFIG_NLS_CODEPAGE_863 is not set +# CONFIG_NLS_CODEPAGE_864 is not set +# CONFIG_NLS_CODEPAGE_865 is not set +# CONFIG_NLS_CODEPAGE_866 is not set +# CONFIG_NLS_CODEPAGE_869 is not set +# CONFIG_NLS_CODEPAGE_936 is not set +# CONFIG_NLS_CODEPAGE_950 is not set +# CONFIG_NLS_CODEPAGE_932 is not set +# CONFIG_NLS_CODEPAGE_949 is not set +# CONFIG_NLS_CODEPAGE_874 is not set +# CONFIG_NLS_ISO8859_8 is not set +# CONFIG_NLS_CODEPAGE_1250 is not set +# CONFIG_NLS_CODEPAGE_1251 is not set +# CONFIG_NLS_ASCII is not set +# CONFIG_NLS_ISO8859_1 is not set +# CONFIG_NLS_ISO8859_2 is not set +# CONFIG_NLS_ISO8859_3 is not set +# CONFIG_NLS_ISO8859_4 is not set +# CONFIG_NLS_ISO8859_5 is not set +# CONFIG_NLS_ISO8859_6 is not set +# CONFIG_NLS_ISO8859_7 is not set +# CONFIG_NLS_ISO8859_9 is not set +# CONFIG_NLS_ISO8859_13 is not set +# CONFIG_NLS_ISO8859_14 is not set +# CONFIG_NLS_ISO8859_15 is not set +# CONFIG_NLS_KOI8_R is not set +# CONFIG_NLS_KOI8_U is not set +# CONFIG_NLS_UTF8 is not set + +# +# Profiling support +# +CONFIG_PROFILING=y +CONFIG_OPROFILE=m + +# +# Kernel hacking +# +# CONFIG_PRINTK_TIME is not set +CONFIG_MAGIC_SYSRQ=y +CONFIG_DEBUG_KERNEL=y +CONFIG_LOG_BUF_SHIFT=14 +CONFIG_DETECT_SOFTLOCKUP=y +# CONFIG_SCHEDSTATS is not set +# CONFIG_DEBUG_SLAB is not set +# CONFIG_DEBUG_MUTEXES is not set +# CONFIG_DEBUG_SPINLOCK is not set +# CONFIG_DEBUG_SPINLOCK_SLEEP is not set +# CONFIG_DEBUG_KOBJECT is not set +CONFIG_DEBUG_BUGVERBOSE=y +CONFIG_DEBUG_INFO=y +# CONFIG_DEBUG_FS is not set +# CONFIG_DEBUG_VM is not set +CONFIG_FRAME_POINTER=y +# CONFIG_UNWIND_INFO is not set +CONFIG_FORCED_INLINING=y +# CONFIG_RCU_TORTURE_TEST is not set +# CONFIG_DEBUG_USER is not set +# CONFIG_DEBUG_WAITQ is not set +CONFIG_DEBUG_ERRORS=y +# CONFIG_DEBUG_LL is not set + +# +# Security options +# +# CONFIG_KEYS is not set +# CONFIG_SECURITY is not set + +# +# Cryptographic options +# +# CONFIG_CRYPTO is not set + +# +# Hardware crypto devices +# + +# +# Library routines +# +# CONFIG_CRC_CCITT is not set +# CONFIG_CRC16 is not set +CONFIG_CRC32=y +# CONFIG_LIBCRC32C is not set +CONFIG_ZLIB_INFLATE=y +CONFIG_ZLIB_DEFLATE=y diff --git a/packages/linux/linux-rp-2.6.20/defconfig-qemux86 b/packages/linux/linux-rp-2.6.20/defconfig-qemux86 new file mode 100644 index 0000000000..6fe280985f --- /dev/null +++ b/packages/linux/linux-rp-2.6.20/defconfig-qemux86 @@ -0,0 +1,1562 @@ +# +# Automatically generated make config: don't edit +# Linux kernel version: 2.6.17 +# Mon Oct 16 19:42:42 2006 +# +CONFIG_X86_32=y +CONFIG_SEMAPHORE_SLEEPERS=y +CONFIG_X86=y +CONFIG_MMU=y +CONFIG_GENERIC_ISA_DMA=y +CONFIG_GENERIC_IOMAP=y +CONFIG_GENERIC_HWEIGHT=y +CONFIG_ARCH_MAY_HAVE_PC_FDC=y +CONFIG_DMI=y + +# +# Code maturity level options +# +CONFIG_EXPERIMENTAL=y +CONFIG_LOCK_KERNEL=y +CONFIG_INIT_ENV_ARG_LIMIT=32 + +# +# General setup +# +CONFIG_LOCALVERSION="" +CONFIG_LOCALVERSION_AUTO=y +CONFIG_SWAP=y +CONFIG_SYSVIPC=y +CONFIG_POSIX_MQUEUE=y +# CONFIG_BSD_PROCESS_ACCT is not set +CONFIG_SYSCTL=y +CONFIG_AUDIT=y +CONFIG_AUDITSYSCALL=y +# CONFIG_IKCONFIG is not set +# CONFIG_CPUSETS is not set +# CONFIG_RELAY is not set +CONFIG_INITRAMFS_SOURCE="" +CONFIG_UID16=y +CONFIG_VM86=y +# CONFIG_CC_OPTIMIZE_FOR_SIZE is not set +CONFIG_EMBEDDED=y +CONFIG_KALLSYMS=y +# CONFIG_KALLSYMS_EXTRA_PASS is not set +CONFIG_HOTPLUG=y +CONFIG_PRINTK=y +CONFIG_BUG=y +CONFIG_ELF_CORE=y +CONFIG_BASE_FULL=y +CONFIG_FUTEX=y +CONFIG_EPOLL=y +CONFIG_SHMEM=y +CONFIG_SLAB=y +# CONFIG_TINY_SHMEM is not set +CONFIG_BASE_SMALL=0 +# CONFIG_SLOB is not set + +# +# Loadable module support +# +CONFIG_MODULES=y +# CONFIG_MODULE_UNLOAD is not set +# CONFIG_MODVERSIONS is not set +# CONFIG_MODULE_SRCVERSION_ALL is not set +CONFIG_KMOD=y + +# +# Block layer +# +CONFIG_LBD=y +# CONFIG_BLK_DEV_IO_TRACE is not set +# CONFIG_LSF is not set + +# +# IO Schedulers +# +CONFIG_IOSCHED_NOOP=y +CONFIG_IOSCHED_AS=y +CONFIG_IOSCHED_DEADLINE=y +CONFIG_IOSCHED_CFQ=y +CONFIG_DEFAULT_AS=y +# CONFIG_DEFAULT_DEADLINE is not set +# CONFIG_DEFAULT_CFQ is not set +# CONFIG_DEFAULT_NOOP is not set +CONFIG_DEFAULT_IOSCHED="anticipatory" + +# +# Processor type and features +# +CONFIG_SMP=y +CONFIG_X86_PC=y +# CONFIG_X86_ELAN is not set +# CONFIG_X86_VOYAGER is not set +# CONFIG_X86_NUMAQ is not set +# CONFIG_X86_SUMMIT is not set +# CONFIG_X86_BIGSMP is not set +# CONFIG_X86_VISWS is not set +# CONFIG_X86_GENERICARCH is not set +# CONFIG_X86_ES7000 is not set +CONFIG_M386=y +# CONFIG_M486 is not set +# CONFIG_M586 is not set +# CONFIG_M586TSC is not set +# CONFIG_M586MMX is not set +# CONFIG_M686 is not set +# CONFIG_MPENTIUMII is not set +# CONFIG_MPENTIUMIII is not set +# CONFIG_MPENTIUMM is not set +# CONFIG_MPENTIUM4 is not set +# CONFIG_MK6 is not set +# CONFIG_MK7 is not set +# CONFIG_MK8 is not set +# CONFIG_MCRUSOE is not set +# CONFIG_MEFFICEON is not set +# CONFIG_MWINCHIPC6 is not set +# CONFIG_MWINCHIP2 is not set +# CONFIG_MWINCHIP3D is not set +# CONFIG_MGEODEGX1 is not set +# CONFIG_MGEODE_LX is not set +# CONFIG_MCYRIXIII is not set +# CONFIG_MVIAC3_2 is not set +CONFIG_X86_GENERIC=y +CONFIG_X86_L1_CACHE_SHIFT=7 +CONFIG_RWSEM_GENERIC_SPINLOCK=y +CONFIG_GENERIC_CALIBRATE_DELAY=y +CONFIG_X86_PPRO_FENCE=y +CONFIG_X86_F00F_BUG=y +CONFIG_X86_INTEL_USERCOPY=y +# CONFIG_HPET_TIMER is not set +CONFIG_NR_CPUS=8 +CONFIG_SCHED_SMT=y +CONFIG_SCHED_MC=y +CONFIG_PREEMPT_NONE=y +# CONFIG_PREEMPT_VOLUNTARY is not set +# CONFIG_PREEMPT is not set +CONFIG_PREEMPT_BKL=y +CONFIG_X86_LOCAL_APIC=y +CONFIG_X86_IO_APIC=y +CONFIG_X86_MCE=y +CONFIG_X86_MCE_NONFATAL=y +CONFIG_X86_MCE_P4THERMAL=y +# CONFIG_TOSHIBA is not set +# CONFIG_I8K is not set +# CONFIG_X86_REBOOTFIXUPS is not set +# CONFIG_MICROCODE is not set +# CONFIG_X86_MSR is not set +# CONFIG_X86_CPUID is not set + +# +# Firmware Drivers +# +# CONFIG_EDD is not set +# CONFIG_DELL_RBU is not set +# CONFIG_DCDBAS is not set +CONFIG_NOHIGHMEM=y +# CONFIG_HIGHMEM4G is not set +# CONFIG_HIGHMEM64G is not set +CONFIG_PAGE_OFFSET=0xC0000000 +CONFIG_ARCH_FLATMEM_ENABLE=y +CONFIG_ARCH_SPARSEMEM_ENABLE=y +CONFIG_ARCH_SELECT_MEMORY_MODEL=y +CONFIG_SELECT_MEMORY_MODEL=y +CONFIG_FLATMEM_MANUAL=y +# CONFIG_DISCONTIGMEM_MANUAL is not set +# CONFIG_SPARSEMEM_MANUAL is not set +CONFIG_FLATMEM=y +CONFIG_FLAT_NODE_MEM_MAP=y +CONFIG_SPARSEMEM_STATIC=y +CONFIG_SPLIT_PTLOCK_CPUS=4 +# CONFIG_MATH_EMULATION is not set +CONFIG_MTRR=y +# CONFIG_EFI is not set +CONFIG_IRQBALANCE=y +# CONFIG_REGPARM is not set +CONFIG_SECCOMP=y +# CONFIG_HZ_100 is not set +CONFIG_HZ_250=y +# CONFIG_HZ_1000 is not set +CONFIG_HZ=250 +# CONFIG_KEXEC is not set +CONFIG_PHYSICAL_START=0x100000 +# CONFIG_HOTPLUG_CPU is not set + +# +# Power management options (ACPI, APM) +# +CONFIG_PM=y +CONFIG_PM_LEGACY=y +# CONFIG_PM_DEBUG is not set + +# +# ACPI (Advanced Configuration and Power Interface) Support +# +CONFIG_ACPI=y +CONFIG_ACPI_AC=y +CONFIG_ACPI_BATTERY=y +CONFIG_ACPI_BUTTON=y +CONFIG_ACPI_VIDEO=m +# CONFIG_ACPI_HOTKEY is not set +CONFIG_ACPI_FAN=y +CONFIG_ACPI_PROCESSOR=y +CONFIG_ACPI_THERMAL=y +# CONFIG_ACPI_ASUS is not set +CONFIG_ACPI_IBM=m +# CONFIG_ACPI_IBM_DOCK is not set +# CONFIG_ACPI_TOSHIBA is not set +CONFIG_ACPI_BLACKLIST_YEAR=0 +# CONFIG_ACPI_DEBUG is not set +CONFIG_ACPI_EC=y +CONFIG_ACPI_POWER=y +CONFIG_ACPI_SYSTEM=y +CONFIG_X86_PM_TIMER=y +# CONFIG_ACPI_CONTAINER is not set + +# +# APM (Advanced Power Management) BIOS Support +# +# CONFIG_APM is not set + +# +# CPU Frequency scaling +# +# CONFIG_CPU_FREQ is not set + +# +# Bus options (PCI, PCMCIA, EISA, MCA, ISA) +# +CONFIG_PCI=y +# CONFIG_PCI_GOBIOS is not set +# CONFIG_PCI_GOMMCONFIG is not set +# CONFIG_PCI_GODIRECT is not set +CONFIG_PCI_GOANY=y +CONFIG_PCI_BIOS=y +CONFIG_PCI_DIRECT=y +CONFIG_PCI_MMCONFIG=y +# CONFIG_PCIEPORTBUS is not set +# CONFIG_PCI_MSI is not set +CONFIG_ISA_DMA_API=y +CONFIG_ISA=y +# CONFIG_EISA is not set +# CONFIG_MCA is not set +# CONFIG_SCx200 is not set + +# +# PCCARD (PCMCIA/CardBus) support +# +# CONFIG_PCCARD is not set + +# +# PCI Hotplug Support +# +# CONFIG_HOTPLUG_PCI is not set + +# +# Executable file formats +# +CONFIG_BINFMT_ELF=y +CONFIG_BINFMT_AOUT=y +CONFIG_BINFMT_MISC=y + +# +# Networking +# +CONFIG_NET=y + +# +# Networking options +# +# CONFIG_NETDEBUG is not set +CONFIG_PACKET=y +# CONFIG_PACKET_MMAP is not set +CONFIG_UNIX=y +# CONFIG_NET_KEY is not set +CONFIG_INET=y +CONFIG_IP_MULTICAST=y +# CONFIG_IP_ADVANCED_ROUTER is not set +CONFIG_IP_FIB_HASH=y +CONFIG_IP_PNP=y +CONFIG_IP_PNP_DHCP=y +CONFIG_IP_PNP_BOOTP=y +# CONFIG_IP_PNP_RARP is not set +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_IP_MROUTE is not set +# CONFIG_ARPD is not set +# CONFIG_SYN_COOKIES is not set +# CONFIG_INET_AH is not set +# CONFIG_INET_ESP is not set +# CONFIG_INET_IPCOMP is not set +# CONFIG_INET_XFRM_TUNNEL is not set +# CONFIG_INET_TUNNEL is not set +CONFIG_INET_DIAG=y +CONFIG_INET_TCP_DIAG=y +# CONFIG_TCP_CONG_ADVANCED is not set +CONFIG_TCP_CONG_BIC=y + +# +# IP: Virtual Server Configuration +# +# CONFIG_IP_VS is not set +# CONFIG_IPV6 is not set +# CONFIG_INET6_XFRM_TUNNEL is not set +# CONFIG_INET6_TUNNEL is not set +CONFIG_NETFILTER=y +# CONFIG_NETFILTER_DEBUG is not set + +# +# Core Netfilter Configuration +# +# CONFIG_NETFILTER_NETLINK is not set +# CONFIG_NETFILTER_XTABLES is not set + +# +# IP: Netfilter Configuration +# +CONFIG_IP_NF_CONNTRACK=y +# CONFIG_IP_NF_CT_ACCT is not set +# CONFIG_IP_NF_CONNTRACK_MARK is not set +# CONFIG_IP_NF_CONNTRACK_EVENTS is not set +# CONFIG_IP_NF_CT_PROTO_SCTP is not set +# CONFIG_IP_NF_FTP is not set +# CONFIG_IP_NF_IRC is not set +# CONFIG_IP_NF_NETBIOS_NS is not set +# CONFIG_IP_NF_TFTP is not set +# CONFIG_IP_NF_AMANDA is not set +# CONFIG_IP_NF_PPTP is not set +# CONFIG_IP_NF_H323 is not set +CONFIG_IP_NF_QUEUE=y + +# +# DCCP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_DCCP is not set + +# +# SCTP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_SCTP is not set + +# +# TIPC Configuration (EXPERIMENTAL) +# +# CONFIG_TIPC is not set +# CONFIG_ATM is not set +# CONFIG_BRIDGE is not set +# CONFIG_VLAN_8021Q is not set +# CONFIG_DECNET is not set +# CONFIG_LLC2 is not set +# CONFIG_IPX is not set +# CONFIG_ATALK is not set +# CONFIG_X25 is not set +# CONFIG_LAPB is not set +# CONFIG_NET_DIVERT is not set +# CONFIG_ECONET is not set +# CONFIG_WAN_ROUTER is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network testing +# +# CONFIG_NET_PKTGEN is not set +# CONFIG_HAMRADIO is not set +# CONFIG_IRDA is not set +# CONFIG_BT is not set +# CONFIG_IEEE80211 is not set + +# +# Device Drivers +# + +# +# Generic Driver Options +# +CONFIG_STANDALONE=y +CONFIG_PREVENT_FIRMWARE_BUILD=y +CONFIG_FW_LOADER=m + +# +# Connector - unified userspace <-> kernelspace linker +# +# CONFIG_CONNECTOR is not set + +# +# Memory Technology Devices (MTD) +# +# CONFIG_MTD is not set + +# +# Parallel port support +# +CONFIG_PARPORT=y +CONFIG_PARPORT_PC=y +# CONFIG_PARPORT_SERIAL is not set +# CONFIG_PARPORT_PC_FIFO is not set +# CONFIG_PARPORT_PC_SUPERIO is not set +# CONFIG_PARPORT_GSC is not set +# CONFIG_PARPORT_1284 is not set + +# +# Plug and Play support +# +CONFIG_PNP=y +# CONFIG_PNP_DEBUG is not set + +# +# Protocols +# +# CONFIG_ISAPNP is not set +# CONFIG_PNPBIOS is not set +CONFIG_PNPACPI=y + +# +# Block devices +# +CONFIG_BLK_DEV_FD=y +# CONFIG_BLK_DEV_XD is not set +# CONFIG_PARIDE is not set +# CONFIG_BLK_CPQ_DA is not set +# CONFIG_BLK_CPQ_CISS_DA is not set +# CONFIG_BLK_DEV_DAC960 is not set +# CONFIG_BLK_DEV_UMEM is not set +# CONFIG_BLK_DEV_COW_COMMON is not set +# CONFIG_BLK_DEV_LOOP is not set +# CONFIG_BLK_DEV_NBD is not set +# CONFIG_BLK_DEV_SX8 is not set +# CONFIG_BLK_DEV_UB is not set +# CONFIG_BLK_DEV_RAM is not set +# CONFIG_BLK_DEV_INITRD is not set +# CONFIG_CDROM_PKTCDVD is not set +# CONFIG_ATA_OVER_ETH is not set + +# +# ATA/ATAPI/MFM/RLL support +# +CONFIG_IDE=y +CONFIG_BLK_DEV_IDE=y + +# +# Please see Documentation/ide.txt for help/info on IDE drives +# +# CONFIG_BLK_DEV_IDE_SATA is not set +# CONFIG_BLK_DEV_HD_IDE is not set +CONFIG_BLK_DEV_IDEDISK=y +CONFIG_IDEDISK_MULTI_MODE=y +CONFIG_BLK_DEV_IDECD=y +# CONFIG_BLK_DEV_IDETAPE is not set +# CONFIG_BLK_DEV_IDEFLOPPY is not set +# CONFIG_BLK_DEV_IDESCSI is not set +# CONFIG_IDE_TASK_IOCTL is not set + +# +# IDE chipset support/bugfixes +# +CONFIG_IDE_GENERIC=y +CONFIG_BLK_DEV_CMD640=y +# CONFIG_BLK_DEV_CMD640_ENHANCED is not set +# CONFIG_BLK_DEV_IDEPNP is not set +CONFIG_BLK_DEV_IDEPCI=y +CONFIG_IDEPCI_SHARE_IRQ=y +# CONFIG_BLK_DEV_OFFBOARD is not set +CONFIG_BLK_DEV_GENERIC=y +# CONFIG_BLK_DEV_OPTI621 is not set +CONFIG_BLK_DEV_RZ1000=y +CONFIG_BLK_DEV_IDEDMA_PCI=y +# CONFIG_BLK_DEV_IDEDMA_FORCED is not set +CONFIG_IDEDMA_PCI_AUTO=y +# CONFIG_IDEDMA_ONLYDISK is not set +# CONFIG_BLK_DEV_AEC62XX is not set +# CONFIG_BLK_DEV_ALI15X3 is not set +# CONFIG_BLK_DEV_AMD74XX is not set +# CONFIG_BLK_DEV_ATIIXP is not set +# CONFIG_BLK_DEV_CMD64X is not set +# CONFIG_BLK_DEV_TRIFLEX is not set +# CONFIG_BLK_DEV_CY82C693 is not set +# CONFIG_BLK_DEV_CS5520 is not set +# CONFIG_BLK_DEV_CS5530 is not set +# CONFIG_BLK_DEV_CS5535 is not set +# CONFIG_BLK_DEV_HPT34X is not set +# CONFIG_BLK_DEV_HPT366 is not set +# CONFIG_BLK_DEV_SC1200 is not set +CONFIG_BLK_DEV_PIIX=y +# CONFIG_BLK_DEV_IT821X is not set +# CONFIG_BLK_DEV_NS87415 is not set +# CONFIG_BLK_DEV_PDC202XX_OLD is not set +# CONFIG_BLK_DEV_PDC202XX_NEW is not set +# CONFIG_BLK_DEV_SVWKS is not set +# CONFIG_BLK_DEV_SIIMAGE is not set +# CONFIG_BLK_DEV_SIS5513 is not set +# CONFIG_BLK_DEV_SLC90E66 is not set +# CONFIG_BLK_DEV_TRM290 is not set +# CONFIG_BLK_DEV_VIA82CXXX is not set +# CONFIG_IDE_ARM is not set +# CONFIG_IDE_CHIPSETS is not set +CONFIG_BLK_DEV_IDEDMA=y +# CONFIG_IDEDMA_IVB is not set +CONFIG_IDEDMA_AUTO=y +# CONFIG_BLK_DEV_HD is not set + +# +# SCSI device support +# +# CONFIG_RAID_ATTRS is not set +CONFIG_SCSI=y +CONFIG_SCSI_PROC_FS=y + +# +# SCSI support type (disk, tape, CD-ROM) +# +CONFIG_BLK_DEV_SD=y +# CONFIG_CHR_DEV_ST is not set +# CONFIG_CHR_DEV_OSST is not set +# CONFIG_BLK_DEV_SR is not set +CONFIG_CHR_DEV_SG=y +# CONFIG_CHR_DEV_SCH is not set + +# +# Some SCSI devices (e.g. CD jukebox) support multiple LUNs +# +# CONFIG_SCSI_MULTI_LUN is not set +# CONFIG_SCSI_CONSTANTS is not set +# CONFIG_SCSI_LOGGING is not set + +# +# SCSI Transport Attributes +# +# CONFIG_SCSI_SPI_ATTRS is not set +# CONFIG_SCSI_FC_ATTRS is not set +# CONFIG_SCSI_ISCSI_ATTRS is not set +# CONFIG_SCSI_SAS_ATTRS is not set + +# +# SCSI low-level drivers +# +# CONFIG_ISCSI_TCP is not set +# CONFIG_BLK_DEV_3W_XXXX_RAID is not set +# CONFIG_SCSI_3W_9XXX is not set +# CONFIG_SCSI_7000FASST is not set +# CONFIG_SCSI_ACARD is not set +# CONFIG_SCSI_AHA152X is not set +# CONFIG_SCSI_AHA1542 is not set +# CONFIG_SCSI_AACRAID is not set +# CONFIG_SCSI_AIC7XXX is not set +# CONFIG_SCSI_AIC7XXX_OLD is not set +# CONFIG_SCSI_AIC79XX is not set +CONFIG_SCSI_DPT_I2O=m +# CONFIG_SCSI_ADVANSYS is not set +# CONFIG_SCSI_IN2000 is not set +# CONFIG_MEGARAID_NEWGEN is not set +# CONFIG_MEGARAID_LEGACY is not set +# CONFIG_MEGARAID_SAS is not set +CONFIG_SCSI_SATA=y +# CONFIG_SCSI_SATA_AHCI is not set +# CONFIG_SCSI_SATA_SVW is not set +CONFIG_SCSI_ATA_PIIX=y +# CONFIG_SCSI_SATA_MV is not set +# CONFIG_SCSI_SATA_NV is not set +# CONFIG_SCSI_PDC_ADMA is not set +# CONFIG_SCSI_SATA_QSTOR is not set +# CONFIG_SCSI_SATA_PROMISE is not set +CONFIG_SCSI_SATA_SX4=m +# CONFIG_SCSI_SATA_SIL is not set +# CONFIG_SCSI_SATA_SIL24 is not set +CONFIG_SCSI_SATA_SIS=m +# CONFIG_SCSI_SATA_ULI is not set +# CONFIG_SCSI_SATA_VIA is not set +# CONFIG_SCSI_SATA_VITESSE is not set +CONFIG_SCSI_SATA_INTEL_COMBINED=y +# CONFIG_SCSI_BUSLOGIC is not set +# CONFIG_SCSI_DMX3191D is not set +# CONFIG_SCSI_DTC3280 is not set +# CONFIG_SCSI_EATA is not set +# CONFIG_SCSI_FUTURE_DOMAIN is not set +# CONFIG_SCSI_GDTH is not set +# CONFIG_SCSI_GENERIC_NCR5380 is not set +# CONFIG_SCSI_GENERIC_NCR5380_MMIO is not set +# CONFIG_SCSI_IPS is not set +# CONFIG_SCSI_INITIO is not set +# CONFIG_SCSI_INIA100 is not set +# CONFIG_SCSI_PPA is not set +# CONFIG_SCSI_IMM is not set +# CONFIG_SCSI_NCR53C406A is not set +# CONFIG_SCSI_SYM53C8XX_2 is not set +CONFIG_SCSI_IPR=m +# CONFIG_SCSI_IPR_TRACE is not set +# CONFIG_SCSI_IPR_DUMP is not set +# CONFIG_SCSI_PAS16 is not set +# CONFIG_SCSI_PSI240I is not set +# CONFIG_SCSI_QLOGIC_FAS is not set +# CONFIG_SCSI_QLOGIC_1280 is not set +# CONFIG_SCSI_QLA_FC is not set +# CONFIG_SCSI_LPFC is not set +# CONFIG_SCSI_SYM53C416 is not set +# CONFIG_SCSI_DC395x is not set +# CONFIG_SCSI_DC390T is not set +# CONFIG_SCSI_T128 is not set +# CONFIG_SCSI_U14_34F is not set +# CONFIG_SCSI_ULTRASTOR is not set +# CONFIG_SCSI_NSP32 is not set +# CONFIG_SCSI_DEBUG is not set + +# +# Old CD-ROM drivers (not SCSI, not IDE) +# +# CONFIG_CD_NO_IDESCSI is not set + +# +# Multi-device support (RAID and LVM) +# +CONFIG_MD=y +CONFIG_BLK_DEV_DM=m +CONFIG_DM_CRYPT=m +CONFIG_DM_SNAPSHOT=m +CONFIG_DM_MIRROR=m +CONFIG_DM_ZERO=m +CONFIG_DM_MULTIPATH=m +CONFIG_DM_MULTIPATH_EMC=m + +# +# Fusion MPT device support +# +# CONFIG_FUSION is not set +# CONFIG_FUSION_SPI is not set +# CONFIG_FUSION_FC is not set +# CONFIG_FUSION_SAS is not set + +# +# IEEE 1394 (FireWire) support +# +CONFIG_IEEE1394=y + +# +# Subsystem Options +# +# CONFIG_IEEE1394_VERBOSEDEBUG is not set +# CONFIG_IEEE1394_OUI_DB is not set +# CONFIG_IEEE1394_EXTRA_CONFIG_ROMS is not set +# CONFIG_IEEE1394_EXPORT_FULL_API is not set + +# +# Device Drivers +# + +# +# Texas Instruments PCILynx requires I2C +# +CONFIG_IEEE1394_OHCI1394=y + +# +# Protocol Drivers +# +# CONFIG_IEEE1394_VIDEO1394 is not set +# CONFIG_IEEE1394_SBP2 is not set +# CONFIG_IEEE1394_ETH1394 is not set +# CONFIG_IEEE1394_DV1394 is not set +CONFIG_IEEE1394_RAWIO=y + +# +# I2O device support +# +# CONFIG_I2O is not set + +# +# Network device support +# +CONFIG_NETDEVICES=y +CONFIG_DUMMY=m +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +# CONFIG_TUN is not set +# CONFIG_NET_SB1000 is not set + +# +# ARCnet devices +# +# CONFIG_ARCNET is not set + +# +# PHY device support +# +# CONFIG_PHYLIB is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +CONFIG_MII=y +# CONFIG_HAPPYMEAL is not set +# CONFIG_SUNGEM is not set +# CONFIG_CASSINI is not set +# CONFIG_NET_VENDOR_3COM is not set +# CONFIG_LANCE is not set +# CONFIG_NET_VENDOR_SMC is not set +# CONFIG_NET_VENDOR_RACAL is not set + +# +# Tulip family network device support +# +# CONFIG_NET_TULIP is not set +# CONFIG_AT1700 is not set +# CONFIG_DEPCA is not set +# CONFIG_HP100 is not set +CONFIG_NET_ISA=y +# CONFIG_E2100 is not set +# CONFIG_EWRK3 is not set +# CONFIG_EEXPRESS is not set +# CONFIG_EEXPRESS_PRO is not set +# CONFIG_HPLAN_PLUS is not set +# CONFIG_HPLAN is not set +# CONFIG_LP486E is not set +# CONFIG_ETH16I is not set +CONFIG_NE2000=y +# CONFIG_ZNET is not set +# CONFIG_SEEQ8005 is not set +CONFIG_NET_PCI=y +# CONFIG_PCNET32 is not set +# CONFIG_AMD8111_ETH is not set +# CONFIG_ADAPTEC_STARFIRE is not set +# CONFIG_AC3200 is not set +# CONFIG_APRICOT is not set +# CONFIG_B44 is not set +# CONFIG_FORCEDETH is not set +# CONFIG_CS89x0 is not set +# CONFIG_DGRS is not set +# CONFIG_EEPRO100 is not set +# CONFIG_E100 is not set +# CONFIG_FEALNX is not set +# CONFIG_NATSEMI is not set +CONFIG_NE2K_PCI=y +# CONFIG_8139CP is not set +CONFIG_8139TOO=y +CONFIG_8139TOO_PIO=y +# CONFIG_8139TOO_TUNE_TWISTER is not set +# CONFIG_8139TOO_8129 is not set +# CONFIG_8139_OLD_RX_RESET is not set +# CONFIG_SIS900 is not set +# CONFIG_EPIC100 is not set +# CONFIG_SUNDANCE is not set +# CONFIG_TLAN is not set +# CONFIG_VIA_RHINE is not set +# CONFIG_NET_POCKET is not set + +# +# Ethernet (1000 Mbit) +# +# CONFIG_ACENIC is not set +# CONFIG_DL2K is not set +# CONFIG_E1000 is not set +# CONFIG_NS83820 is not set +# CONFIG_HAMACHI is not set +# CONFIG_YELLOWFIN is not set +# CONFIG_R8169 is not set +# CONFIG_SIS190 is not set +# CONFIG_SKGE is not set +# CONFIG_SKY2 is not set +# CONFIG_SK98LIN is not set +# CONFIG_VIA_VELOCITY is not set +# CONFIG_TIGON3 is not set +# CONFIG_BNX2 is not set + +# +# Ethernet (10000 Mbit) +# +# CONFIG_CHELSIO_T1 is not set +# CONFIG_IXGB is not set +CONFIG_S2IO=m +# CONFIG_S2IO_NAPI is not set + +# +# Token Ring devices +# +# CONFIG_TR is not set + +# +# Wireless LAN (non-hamradio) +# +# CONFIG_NET_RADIO is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set +# CONFIG_FDDI is not set +# CONFIG_HIPPI is not set +# CONFIG_PLIP is not set +# CONFIG_PPP is not set +# CONFIG_SLIP is not set +# CONFIG_NET_FC is not set +# CONFIG_SHAPER is not set +# CONFIG_NETCONSOLE is not set +# CONFIG_NETPOLL is not set +# CONFIG_NET_POLL_CONTROLLER is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Telephony Support +# +# CONFIG_PHONE is not set + +# +# Input device support +# +CONFIG_INPUT=y + +# +# Userland interfaces +# +# CONFIG_INPUT_MOUSEDEV is not set +# CONFIG_INPUT_JOYDEV is not set +# CONFIG_INPUT_TSDEV is not set +CONFIG_INPUT_EVDEV=y +# CONFIG_INPUT_EVBUG is not set +# CONFIG_INPUT_POWER is not set + +# +# Input Device Drivers +# +CONFIG_INPUT_KEYBOARD=y +CONFIG_KEYBOARD_ATKBD=y +# CONFIG_KEYBOARD_SUNKBD is not set +# CONFIG_KEYBOARD_LKKBD is not set +# CONFIG_KEYBOARD_XTKBD is not set +# CONFIG_KEYBOARD_NEWTON is not set +# CONFIG_INPUT_MOUSE is not set +# CONFIG_MOUSE_PS2 is not set +# CONFIG_MOUSE_SERIAL is not set +# CONFIG_MOUSE_INPORT is not set +# CONFIG_MOUSE_LOGIBM is not set +# CONFIG_MOUSE_PC110PAD is not set +# CONFIG_MOUSE_VSXXXAA is not set +# CONFIG_INPUT_JOYSTICK is not set +# CONFIG_INPUT_TOUCHSCREEN is not set +# CONFIG_INPUT_MISC is not set + +# +# Hardware I/O ports +# +CONFIG_SERIO=y +CONFIG_SERIO_I8042=y +# CONFIG_SERIO_SERPORT is not set +# CONFIG_SERIO_CT82C710 is not set +# CONFIG_SERIO_PARKBD is not set +# CONFIG_SERIO_PCIPS2 is not set +CONFIG_SERIO_LIBPS2=y +# CONFIG_SERIO_RAW is not set +# CONFIG_GAMEPORT is not set + +# +# Character devices +# +CONFIG_VT=y +CONFIG_VT_CONSOLE=y +CONFIG_HW_CONSOLE=y +# CONFIG_SERIAL_NONSTANDARD is not set + +# +# Serial drivers +# +CONFIG_SERIAL_8250=y +# CONFIG_SERIAL_8250_CONSOLE is not set +CONFIG_SERIAL_8250_PCI=y +CONFIG_SERIAL_8250_PNP=y +CONFIG_SERIAL_8250_NR_UARTS=4 +CONFIG_SERIAL_8250_RUNTIME_UARTS=4 +# CONFIG_SERIAL_8250_EXTENDED is not set + +# +# Non-8250 serial port support +# +CONFIG_SERIAL_CORE=y +# CONFIG_SERIAL_JSM is not set +CONFIG_UNIX98_PTYS=y +# CONFIG_LEGACY_PTYS is not set +CONFIG_PRINTER=y +# CONFIG_LP_CONSOLE is not set +# CONFIG_PPDEV is not set +# CONFIG_TIPAR is not set + +# +# IPMI +# +# CONFIG_IPMI_HANDLER is not set + +# +# Watchdog Cards +# +# CONFIG_WATCHDOG is not set +# CONFIG_HW_RANDOM is not set +# CONFIG_NVRAM is not set +# CONFIG_RTC is not set +# CONFIG_GEN_RTC is not set +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set +# CONFIG_APPLICOM is not set +# CONFIG_SONYPI is not set + +# +# Ftape, the floppy tape device driver +# +CONFIG_AGP=y +# CONFIG_AGP_ALI is not set +# CONFIG_AGP_ATI is not set +# CONFIG_AGP_AMD is not set +# CONFIG_AGP_AMD64 is not set +CONFIG_AGP_INTEL=y +# CONFIG_AGP_NVIDIA is not set +# CONFIG_AGP_SIS is not set +# CONFIG_AGP_SWORKS is not set +# CONFIG_AGP_VIA is not set +# CONFIG_AGP_EFFICEON is not set +CONFIG_DRM=y +# CONFIG_DRM_TDFX is not set +# CONFIG_DRM_R128 is not set +# CONFIG_DRM_RADEON is not set +# CONFIG_DRM_I810 is not set +# CONFIG_DRM_I830 is not set +# CONFIG_DRM_I915 is not set +# CONFIG_DRM_MGA is not set +# CONFIG_DRM_SIS is not set +# CONFIG_DRM_VIA is not set +# CONFIG_DRM_SAVAGE is not set +# CONFIG_MWAVE is not set +# CONFIG_CS5535_GPIO is not set +# CONFIG_RAW_DRIVER is not set +# CONFIG_HPET is not set +# CONFIG_HANGCHECK_TIMER is not set + +# +# TPM devices +# +# CONFIG_TCG_TPM is not set +# CONFIG_TELCLOCK is not set + +# +# I2C support +# +# CONFIG_I2C is not set + +# +# SPI support +# +# CONFIG_SPI is not set +# CONFIG_SPI_MASTER is not set + +# +# Dallas's 1-wire bus +# +# CONFIG_W1 is not set + +# +# Hardware Monitoring support +# +CONFIG_HWMON=y +# CONFIG_HWMON_VID is not set +# CONFIG_SENSORS_F71805F is not set +# CONFIG_SENSORS_HDAPS is not set +# CONFIG_HWMON_DEBUG_CHIP is not set + +# +# Misc devices +# +# CONFIG_IBM_ASM is not set + +# +# Multi-Function Devices +# + +# +# Multimedia devices +# +# CONFIG_VIDEO_DEV is not set +CONFIG_VIDEO_V4L2=y + +# +# Digital Video Broadcasting Devices +# +# CONFIG_DVB is not set +# CONFIG_USB_DABUSB is not set + +# +# Graphics support +# +CONFIG_FB=y +CONFIG_FB_CFB_FILLRECT=y +CONFIG_FB_CFB_COPYAREA=y +CONFIG_FB_CFB_IMAGEBLIT=y +# CONFIG_FB_MACMODES is not set +CONFIG_FB_FIRMWARE_EDID=y +CONFIG_FB_MODE_HELPERS=y +# CONFIG_FB_TILEBLITTING is not set +# CONFIG_FB_CIRRUS is not set +# CONFIG_FB_PM2 is not set +# CONFIG_FB_CYBER2000 is not set +# CONFIG_FB_ARC is not set +# CONFIG_FB_ASILIANT is not set +# CONFIG_FB_IMSTT is not set +CONFIG_FB_VGA16=y +CONFIG_FB_VESA=y +# CONFIG_FB_VESA_STD is not set +CONFIG_FB_VESA_TNG=y +CONFIG_FB_VESA_DEFAULT_MODE="640x480-32@60" +CONFIG_VIDEO_SELECT=y +# CONFIG_FB_HGA is not set +# CONFIG_FB_S1D13XXX is not set +# CONFIG_FB_NVIDIA is not set +# CONFIG_FB_RIVA is not set +# CONFIG_FB_I810 is not set +# CONFIG_FB_INTEL is not set +# CONFIG_FB_MATROX is not set +# CONFIG_FB_RADEON is not set +# CONFIG_FB_ATY128 is not set +# CONFIG_FB_ATY is not set +# CONFIG_FB_SAVAGE is not set +# CONFIG_FB_SIS is not set +# CONFIG_FB_NEOMAGIC is not set +# CONFIG_FB_KYRO is not set +# CONFIG_FB_3DFX is not set +# CONFIG_FB_VOODOO1 is not set +# CONFIG_FB_CYBLA is not set +# CONFIG_FB_TRIDENT is not set +# CONFIG_FB_GEODE is not set +# CONFIG_FB_VIRTUAL is not set + +# +# Console display driver support +# +CONFIG_VGA_CONSOLE=y +# CONFIG_VGACON_SOFT_SCROLLBACK is not set +# CONFIG_MDA_CONSOLE is not set +CONFIG_DUMMY_CONSOLE=y +CONFIG_FRAMEBUFFER_CONSOLE=y +# CONFIG_FRAMEBUFFER_CONSOLE_ROTATION is not set +CONFIG_FONTS=y +CONFIG_FONT_8x8=y +CONFIG_FONT_8x16=y +# CONFIG_FONT_6x11 is not set +# CONFIG_FONT_7x14 is not set +# CONFIG_FONT_PEARL_8x8 is not set +# CONFIG_FONT_ACORN_8x8 is not set +# CONFIG_FONT_MINI_4x6 is not set +# CONFIG_FONT_SUN8x16 is not set +# CONFIG_FONT_SUN12x22 is not set +# CONFIG_FONT_10x18 is not set + +# +# Logo configuration +# +CONFIG_LOGO=y +# CONFIG_LOGO_LINUX_MONO is not set +# CONFIG_LOGO_LINUX_VGA16 is not set +# CONFIG_LOGO_LINUX_CLUT224 is not set +CONFIG_LOGO_OHAND_CLUT224=y +# CONFIG_LOGO_OZ240_CLUT224 is not set +# CONFIG_LOGO_OZ480_CLUT224 is not set +# CONFIG_LOGO_OZ640_CLUT224 is not set +# CONFIG_BACKLIGHT_LCD_SUPPORT is not set + +# +# Sound +# +CONFIG_SOUND=y + +# +# Advanced Linux Sound Architecture +# +CONFIG_SND=y +CONFIG_SND_TIMER=y +CONFIG_SND_PCM=y +CONFIG_SND_SEQUENCER=y +# CONFIG_SND_SEQ_DUMMY is not set +CONFIG_SND_OSSEMUL=y +CONFIG_SND_MIXER_OSS=y +CONFIG_SND_PCM_OSS=y +CONFIG_SND_PCM_OSS_PLUGINS=y +CONFIG_SND_SEQUENCER_OSS=y +# CONFIG_SND_DYNAMIC_MINORS is not set +CONFIG_SND_SUPPORT_OLD_API=y +CONFIG_SND_VERBOSE_PROCFS=y +# CONFIG_SND_VERBOSE_PRINTK is not set +# CONFIG_SND_DEBUG is not set + +# +# Generic devices +# +CONFIG_SND_AC97_CODEC=y +CONFIG_SND_AC97_BUS=y +# CONFIG_SND_DUMMY is not set +# CONFIG_SND_VIRMIDI is not set +# CONFIG_SND_MTPAV is not set +# CONFIG_SND_SERIAL_U16550 is not set +# CONFIG_SND_MPU401 is not set + +# +# ISA devices +# +# CONFIG_SND_ADLIB is not set +# CONFIG_SND_AD1816A is not set +# CONFIG_SND_AD1848 is not set +# CONFIG_SND_ALS100 is not set +# CONFIG_SND_AZT2320 is not set +# CONFIG_SND_CMI8330 is not set +# CONFIG_SND_CS4231 is not set +# CONFIG_SND_CS4232 is not set +# CONFIG_SND_CS4236 is not set +# CONFIG_SND_DT019X is not set +# CONFIG_SND_ES968 is not set +# CONFIG_SND_ES1688 is not set +# CONFIG_SND_ES18XX is not set +# CONFIG_SND_GUSCLASSIC is not set +# CONFIG_SND_GUSEXTREME is not set +# CONFIG_SND_GUSMAX is not set +# CONFIG_SND_INTERWAVE is not set +# CONFIG_SND_INTERWAVE_STB is not set +# CONFIG_SND_OPL3SA2 is not set +# CONFIG_SND_OPTI92X_AD1848 is not set +# CONFIG_SND_OPTI92X_CS4231 is not set +# CONFIG_SND_OPTI93X is not set +# CONFIG_SND_MIRO is not set +# CONFIG_SND_SB8 is not set +# CONFIG_SND_SB16 is not set +# CONFIG_SND_SBAWE is not set +# CONFIG_SND_SGALAXY is not set +# CONFIG_SND_SSCAPE is not set +# CONFIG_SND_WAVEFRONT is not set + +# +# PCI devices +# +# CONFIG_SND_AD1889 is not set +# CONFIG_SND_ALS300 is not set +# CONFIG_SND_ALS4000 is not set +# CONFIG_SND_ALI5451 is not set +# CONFIG_SND_ATIIXP is not set +# CONFIG_SND_ATIIXP_MODEM is not set +# CONFIG_SND_AU8810 is not set +# CONFIG_SND_AU8820 is not set +# CONFIG_SND_AU8830 is not set +# CONFIG_SND_AZT3328 is not set +# CONFIG_SND_BT87X is not set +# CONFIG_SND_CA0106 is not set +# CONFIG_SND_CMIPCI is not set +# CONFIG_SND_CS4281 is not set +# CONFIG_SND_CS46XX is not set +# CONFIG_SND_CS5535AUDIO is not set +# CONFIG_SND_EMU10K1 is not set +# CONFIG_SND_EMU10K1X is not set +# CONFIG_SND_ENS1370 is not set +# CONFIG_SND_ENS1371 is not set +# CONFIG_SND_ES1938 is not set +# CONFIG_SND_ES1968 is not set +# CONFIG_SND_FM801 is not set +# CONFIG_SND_HDA_INTEL is not set +# CONFIG_SND_HDSP is not set +# CONFIG_SND_HDSPM is not set +# CONFIG_SND_ICE1712 is not set +# CONFIG_SND_ICE1724 is not set +CONFIG_SND_INTEL8X0=y +# CONFIG_SND_INTEL8X0M is not set +# CONFIG_SND_KORG1212 is not set +# CONFIG_SND_MAESTRO3 is not set +# CONFIG_SND_MIXART is not set +# CONFIG_SND_NM256 is not set +# CONFIG_SND_PCXHR is not set +# CONFIG_SND_RIPTIDE is not set +# CONFIG_SND_RME32 is not set +# CONFIG_SND_RME96 is not set +# CONFIG_SND_RME9652 is not set +# CONFIG_SND_SONICVIBES is not set +# CONFIG_SND_TRIDENT is not set +# CONFIG_SND_VIA82XX is not set +# CONFIG_SND_VIA82XX_MODEM is not set +# CONFIG_SND_VX222 is not set +# CONFIG_SND_YMFPCI is not set + +# +# USB devices +# +# CONFIG_SND_USB_AUDIO is not set +# CONFIG_SND_USB_USX2Y is not set + +# +# SoC audio support +# +# CONFIG_SND_SOC is not set + +# +# Open Sound System +# +# CONFIG_SOUND_PRIME is not set + +# +# USB support +# +CONFIG_USB_ARCH_HAS_HCD=y +CONFIG_USB_ARCH_HAS_OHCI=y +CONFIG_USB_ARCH_HAS_EHCI=y +CONFIG_USB=y +# CONFIG_USB_DEBUG is not set + +# +# Miscellaneous USB options +# +CONFIG_USB_DEVICEFS=y +# CONFIG_USB_BANDWIDTH is not set +# CONFIG_USB_DYNAMIC_MINORS is not set +# CONFIG_USB_SUSPEND is not set +# CONFIG_USB_OTG is not set + +# +# USB Host Controller Drivers +# +CONFIG_USB_EHCI_HCD=y +# CONFIG_USB_EHCI_SPLIT_ISO is not set +# CONFIG_USB_EHCI_ROOT_HUB_TT is not set +# CONFIG_USB_ISP116X_HCD is not set +CONFIG_USB_OHCI_HCD=y +CONFIG_USB_UHCI_HCD=y +# CONFIG_USB_SL811_HCD is not set + +# +# USB Device Class drivers +# +# CONFIG_USB_ACM is not set +CONFIG_USB_PRINTER=y + +# +# NOTE: USB_STORAGE enables SCSI, and 'SCSI disk support' +# + +# +# may also be needed; see USB_STORAGE Help for more information +# +CONFIG_USB_STORAGE=y +# CONFIG_USB_STORAGE_DEBUG is not set +# CONFIG_USB_STORAGE_DATAFAB is not set +# CONFIG_USB_STORAGE_FREECOM is not set +# CONFIG_USB_STORAGE_ISD200 is not set +# CONFIG_USB_STORAGE_DPCM is not set +# CONFIG_USB_STORAGE_USBAT is not set +# CONFIG_USB_STORAGE_SDDR09 is not set +# CONFIG_USB_STORAGE_SDDR55 is not set +# CONFIG_USB_STORAGE_JUMPSHOT is not set +# CONFIG_USB_STORAGE_ALAUDA is not set +# CONFIG_USB_LIBUSUAL is not set + +# +# USB Input Devices +# +CONFIG_USB_HID=y +CONFIG_USB_HIDINPUT=y +# CONFIG_USB_HIDINPUT_POWERBOOK is not set +# CONFIG_HID_FF is not set +# CONFIG_USB_HIDDEV is not set +# CONFIG_USB_AIPTEK is not set +CONFIG_USB_WACOM=y +# CONFIG_USB_ACECAD is not set +# CONFIG_USB_KBTAB is not set +# CONFIG_USB_POWERMATE is not set +# CONFIG_USB_TOUCHSCREEN is not set +# CONFIG_USB_YEALINK is not set +# CONFIG_USB_XPAD is not set +# CONFIG_USB_ATI_REMOTE is not set +# CONFIG_USB_ATI_REMOTE2 is not set +# CONFIG_USB_KEYSPAN_REMOTE is not set +# CONFIG_USB_APPLETOUCH is not set + +# +# USB Imaging devices +# +# CONFIG_USB_MDC800 is not set +# CONFIG_USB_MICROTEK is not set + +# +# USB Network Adapters +# +# CONFIG_USB_CATC is not set +# CONFIG_USB_KAWETH is not set +# CONFIG_USB_PEGASUS is not set +# CONFIG_USB_RTL8150 is not set +# CONFIG_USB_USBNET is not set +CONFIG_USB_MON=y + +# +# USB port drivers +# +# CONFIG_USB_USS720 is not set + +# +# USB Serial Converter support +# +# CONFIG_USB_SERIAL is not set + +# +# USB Miscellaneous drivers +# +# CONFIG_USB_EMI62 is not set +# CONFIG_USB_EMI26 is not set +# CONFIG_USB_AUERSWALD is not set +# CONFIG_USB_RIO500 is not set +# CONFIG_USB_LEGOTOWER is not set +# CONFIG_USB_LCD is not set +# CONFIG_USB_LED is not set +CONFIG_USB_CYTHERM=m +# CONFIG_USB_PHIDGETKIT is not set +CONFIG_USB_PHIDGETSERVO=m +# CONFIG_USB_IDMOUSE is not set +# CONFIG_USB_SISUSBVGA is not set +# CONFIG_USB_LD is not set +# CONFIG_USB_TEST is not set + +# +# USB DSL modem support +# + +# +# USB Gadget Support +# +# CONFIG_USB_GADGET is not set + +# +# MMC/SD Card support +# +# CONFIG_MMC is not set + +# +# LED devices +# +# CONFIG_NEW_LEDS is not set + +# +# LED drivers +# + +# +# LED Triggers +# + +# +# InfiniBand support +# +# CONFIG_INFINIBAND is not set + +# +# EDAC - error detection and reporting (RAS) (EXPERIMENTAL) +# +# CONFIG_EDAC is not set + +# +# Real Time Clock +# +# CONFIG_RTC_CLASS is not set + +# +# File systems +# +CONFIG_EXT2_FS=y +# CONFIG_EXT2_FS_XATTR is not set +# CONFIG_EXT2_FS_XIP is not set +CONFIG_EXT3_FS=y +CONFIG_EXT3_FS_XATTR=y +# CONFIG_EXT3_FS_POSIX_ACL is not set +# CONFIG_EXT3_FS_SECURITY is not set +CONFIG_JBD=y +# CONFIG_JBD_DEBUG is not set +CONFIG_FS_MBCACHE=y +# CONFIG_REISERFS_FS is not set +# CONFIG_JFS_FS is not set +# CONFIG_FS_POSIX_ACL is not set +# CONFIG_XFS_FS is not set +# CONFIG_OCFS2_FS is not set +# CONFIG_MINIX_FS is not set +# CONFIG_ROMFS_FS is not set +CONFIG_INOTIFY=y +# CONFIG_QUOTA is not set +CONFIG_DNOTIFY=y +# CONFIG_AUTOFS_FS is not set +CONFIG_AUTOFS4_FS=y +# CONFIG_FUSE_FS is not set + +# +# CD-ROM/DVD Filesystems +# +CONFIG_ISO9660_FS=y +CONFIG_JOLIET=y +# CONFIG_ZISOFS is not set +CONFIG_UDF_FS=y +CONFIG_UDF_NLS=y + +# +# DOS/FAT/NT Filesystems +# +CONFIG_FAT_FS=y +CONFIG_MSDOS_FS=y +CONFIG_VFAT_FS=y +CONFIG_FAT_DEFAULT_CODEPAGE=437 +CONFIG_FAT_DEFAULT_IOCHARSET="iso8859-1" +# CONFIG_NTFS_FS is not set + +# +# Pseudo filesystems +# +CONFIG_PROC_FS=y +CONFIG_PROC_KCORE=y +CONFIG_SYSFS=y +CONFIG_TMPFS=y +# CONFIG_HUGETLBFS is not set +# CONFIG_HUGETLB_PAGE is not set +CONFIG_RAMFS=y +# CONFIG_CONFIGFS_FS is not set + +# +# Miscellaneous filesystems +# +# CONFIG_ADFS_FS is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_HFSPLUS_FS is not set +# CONFIG_BEFS_FS is not set +# CONFIG_BFS_FS is not set +# CONFIG_EFS_FS is not set +# CONFIG_CRAMFS is not set +# CONFIG_SQUASHFS is not set +# CONFIG_VXFS_FS is not set +# CONFIG_HPFS_FS is not set +# CONFIG_QNX4FS_FS is not set +# CONFIG_SYSV_FS is not set +# CONFIG_UFS_FS is not set + +# +# Network File Systems +# +CONFIG_NFS_FS=y +# CONFIG_NFS_V3 is not set +# CONFIG_NFS_V4 is not set +# CONFIG_NFS_DIRECTIO is not set +CONFIG_NFSD=y +# CONFIG_NFSD_V3 is not set +CONFIG_NFSD_TCP=y +CONFIG_ROOT_NFS=y +CONFIG_LOCKD=y +CONFIG_EXPORTFS=y +CONFIG_NFS_COMMON=y +CONFIG_SUNRPC=y +# CONFIG_RPCSEC_GSS_KRB5 is not set +# CONFIG_RPCSEC_GSS_SPKM3 is not set +# CONFIG_SMB_FS is not set +# CONFIG_CIFS is not set +# CONFIG_NCP_FS is not set +# CONFIG_CODA_FS is not set +# CONFIG_AFS_FS is not set +# CONFIG_9P_FS is not set + +# +# Partition Types +# +# CONFIG_PARTITION_ADVANCED is not set +CONFIG_MSDOS_PARTITION=y + +# +# Native Language Support +# +CONFIG_NLS=y +CONFIG_NLS_DEFAULT="iso8859-1" +CONFIG_NLS_CODEPAGE_437=y +# CONFIG_NLS_CODEPAGE_737 is not set +# CONFIG_NLS_CODEPAGE_775 is not set +# CONFIG_NLS_CODEPAGE_850 is not set +# CONFIG_NLS_CODEPAGE_852 is not set +# CONFIG_NLS_CODEPAGE_855 is not set +# CONFIG_NLS_CODEPAGE_857 is not set +# CONFIG_NLS_CODEPAGE_860 is not set +# CONFIG_NLS_CODEPAGE_861 is not set +# CONFIG_NLS_CODEPAGE_862 is not set +# CONFIG_NLS_CODEPAGE_863 is not set +# CONFIG_NLS_CODEPAGE_864 is not set +# CONFIG_NLS_CODEPAGE_865 is not set +# CONFIG_NLS_CODEPAGE_866 is not set +# CONFIG_NLS_CODEPAGE_869 is not set +# CONFIG_NLS_CODEPAGE_936 is not set +# CONFIG_NLS_CODEPAGE_950 is not set +# CONFIG_NLS_CODEPAGE_932 is not set +# CONFIG_NLS_CODEPAGE_949 is not set +# CONFIG_NLS_CODEPAGE_874 is not set +# CONFIG_NLS_ISO8859_8 is not set +# CONFIG_NLS_CODEPAGE_1250 is not set +# CONFIG_NLS_CODEPAGE_1251 is not set +# CONFIG_NLS_ASCII is not set +CONFIG_NLS_ISO8859_1=y +# CONFIG_NLS_ISO8859_2 is not set +# CONFIG_NLS_ISO8859_3 is not set +# CONFIG_NLS_ISO8859_4 is not set +# CONFIG_NLS_ISO8859_5 is not set +# CONFIG_NLS_ISO8859_6 is not set +# CONFIG_NLS_ISO8859_7 is not set +# CONFIG_NLS_ISO8859_9 is not set +# CONFIG_NLS_ISO8859_13 is not set +# CONFIG_NLS_ISO8859_14 is not set +# CONFIG_NLS_ISO8859_15 is not set +# CONFIG_NLS_KOI8_R is not set +# CONFIG_NLS_KOI8_U is not set +# CONFIG_NLS_UTF8 is not set + +# +# Instrumentation Support +# +CONFIG_PROFILING=y +CONFIG_OPROFILE=y +# CONFIG_KPROBES is not set + +# +# Kernel hacking +# +# CONFIG_PRINTK_TIME is not set +# CONFIG_MAGIC_SYSRQ is not set +# CONFIG_DEBUG_KERNEL is not set +CONFIG_LOG_BUF_SHIFT=15 +CONFIG_DEBUG_BUGVERBOSE=y +# CONFIG_DEBUG_FS is not set +# CONFIG_UNWIND_INFO is not set +CONFIG_EARLY_PRINTK=y +CONFIG_STACK_BACKTRACE_COLS=2 +CONFIG_X86_FIND_SMP_CONFIG=y +CONFIG_X86_MPPARSE=y +CONFIG_DOUBLEFAULT=y + +# +# Security options +# +# CONFIG_KEYS is not set +# CONFIG_SECURITY is not set + +# +# Cryptographic options +# +# CONFIG_CRYPTO is not set + +# +# Hardware crypto devices +# + +# +# Library routines +# +# CONFIG_CRC_CCITT is not set +# CONFIG_CRC16 is not set +CONFIG_CRC32=y +CONFIG_LIBCRC32C=m +CONFIG_GENERIC_HARDIRQS=y +CONFIG_GENERIC_IRQ_PROBE=y +CONFIG_GENERIC_PENDING_IRQ=y +CONFIG_X86_SMP=y +CONFIG_X86_HT=y +CONFIG_X86_BIOS_REBOOT=y +CONFIG_X86_TRAMPOLINE=y +CONFIG_KTIME_SCALAR=y diff --git a/packages/linux/linux-rp-2.6.20/defconfig-spitz b/packages/linux/linux-rp-2.6.20/defconfig-spitz new file mode 100644 index 0000000000..824fd57aaa --- /dev/null +++ b/packages/linux/linux-rp-2.6.20/defconfig-spitz @@ -0,0 +1,1603 @@ +# +# Automatically generated make config: don't edit +# Linux kernel version: 2.6.17 +# Sun Sep 3 23:29:17 2006 +# +CONFIG_ARM=y +CONFIG_MMU=y +CONFIG_RWSEM_GENERIC_SPINLOCK=y +CONFIG_GENERIC_HWEIGHT=y +CONFIG_GENERIC_CALIBRATE_DELAY=y +CONFIG_ARCH_MTD_XIP=y +CONFIG_VECTORS_BASE=0xffff0000 + +# +# Code maturity level options +# +CONFIG_EXPERIMENTAL=y +CONFIG_BROKEN_ON_SMP=y +CONFIG_LOCK_KERNEL=y +CONFIG_INIT_ENV_ARG_LIMIT=32 + +# +# General setup +# +CONFIG_LOCALVERSION="" +CONFIG_LOCALVERSION_AUTO=y +CONFIG_SWAP=y +CONFIG_SYSVIPC=y +# CONFIG_POSIX_MQUEUE is not set +CONFIG_BSD_PROCESS_ACCT=y +# CONFIG_BSD_PROCESS_ACCT_V3 is not set +CONFIG_SYSCTL=y +# CONFIG_AUDIT is not set +# CONFIG_IKCONFIG is not set +# CONFIG_RELAY is not set +CONFIG_INITRAMFS_SOURCE="" +CONFIG_UID16=y +CONFIG_CC_OPTIMIZE_FOR_SIZE=y +CONFIG_EMBEDDED=y +CONFIG_KALLSYMS=y +# CONFIG_KALLSYMS_ALL is not set +# CONFIG_KALLSYMS_EXTRA_PASS is not set +CONFIG_HOTPLUG=y +CONFIG_PRINTK=y +CONFIG_BUG=y +CONFIG_ELF_CORE=y +CONFIG_BASE_FULL=y +CONFIG_FUTEX=y +CONFIG_EPOLL=y +CONFIG_SHMEM=y +CONFIG_SLAB=y +# CONFIG_TINY_SHMEM is not set +CONFIG_BASE_SMALL=0 +# CONFIG_SLOB is not set + +# +# Loadable module support +# +CONFIG_MODULES=y +CONFIG_MODULE_UNLOAD=y +CONFIG_MODULE_FORCE_UNLOAD=y +# CONFIG_MODVERSIONS is not set +# CONFIG_MODULE_SRCVERSION_ALL is not set +CONFIG_KMOD=y + +# +# Block layer +# +# CONFIG_BLK_DEV_IO_TRACE is not set + +# +# IO Schedulers +# +CONFIG_IOSCHED_NOOP=y +CONFIG_IOSCHED_AS=y +CONFIG_IOSCHED_DEADLINE=m +CONFIG_IOSCHED_CFQ=m +CONFIG_DEFAULT_AS=y +# CONFIG_DEFAULT_DEADLINE is not set +# CONFIG_DEFAULT_CFQ is not set +# CONFIG_DEFAULT_NOOP is not set +CONFIG_DEFAULT_IOSCHED="anticipatory" + +# +# System Type +# +# CONFIG_ARCH_CLPS7500 is not set +# CONFIG_ARCH_CLPS711X is not set +# CONFIG_ARCH_CO285 is not set +# CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_EP93XX is not set +# CONFIG_ARCH_FOOTBRIDGE is not set +# CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_IOP3XX is not set +# CONFIG_ARCH_IXP4XX is not set +# CONFIG_ARCH_IXP2000 is not set +# CONFIG_ARCH_IXP23XX is not set +# CONFIG_ARCH_L7200 is not set +CONFIG_ARCH_PXA=y +# CONFIG_ARCH_RPC is not set +# CONFIG_ARCH_SA1100 is not set +# CONFIG_ARCH_S3C2410 is not set +# CONFIG_ARCH_SHARK is not set +# CONFIG_ARCH_LH7A40X is not set +# CONFIG_ARCH_OMAP is not set +# CONFIG_ARCH_VERSATILE is not set +# CONFIG_ARCH_REALVIEW is not set +# CONFIG_ARCH_IMX is not set +# CONFIG_ARCH_H720X is not set +# CONFIG_ARCH_AAEC2000 is not set +# CONFIG_ARCH_AT91RM9200 is not set + +# +# Intel PXA2xx Implementations +# +# CONFIG_ARCH_LUBBOCK is not set +# CONFIG_MACH_LOGICPD_PXA270 is not set +# CONFIG_MACH_MAINSTONE is not set +# CONFIG_ARCH_PXA_IDP is not set +CONFIG_PXA_SHARPSL=y +# CONFIG_MACH_HX2750 is not set +# CONFIG_PXA_SHARPSL_25x is not set +CONFIG_PXA_SHARPSL_27x=y +CONFIG_MACH_AKITA=y +CONFIG_MACH_SPITZ=y +CONFIG_MACH_BORZOI=y +CONFIG_PXA27x=y +# CONFIG_PXA_KEYS is not set +CONFIG_IWMMXT=y +CONFIG_PXA_SHARP_Cxx00=y +CONFIG_PXA_SSP=y + +# +# Processor Type +# +CONFIG_CPU_32=y +CONFIG_CPU_XSCALE=y +CONFIG_CPU_32v5=y +CONFIG_CPU_ABRT_EV5T=y +CONFIG_CPU_CACHE_VIVT=y +CONFIG_CPU_TLB_V4WBI=y + +# +# Processor Features +# +CONFIG_ARM_THUMB=y +CONFIG_XSCALE_PMU=y +CONFIG_KEXEC=y +CONFIG_SHARP_PARAM=y +CONFIG_SHARPSL_PM=y +CONFIG_SHARP_SCOOP=y + +# +# Bus support +# + +# +# PCCARD (PCMCIA/CardBus) support +# +CONFIG_PCCARD=y +# CONFIG_PCMCIA_DEBUG is not set +CONFIG_PCMCIA=y +CONFIG_PCMCIA_LOAD_CIS=y +CONFIG_PCMCIA_IOCTL=y + +# +# PC-card bridges +# +CONFIG_PCMCIA_PXA2XX=y + +# +# Kernel Features +# +CONFIG_PREEMPT=y +CONFIG_NO_IDLE_HZ=y +CONFIG_HZ=100 +# CONFIG_AEABI is not set +# CONFIG_ARCH_DISCONTIGMEM_ENABLE is not set +CONFIG_SELECT_MEMORY_MODEL=y +CONFIG_FLATMEM_MANUAL=y +# CONFIG_DISCONTIGMEM_MANUAL is not set +# CONFIG_SPARSEMEM_MANUAL is not set +CONFIG_FLATMEM=y +CONFIG_FLAT_NODE_MEM_MAP=y +# CONFIG_SPARSEMEM_STATIC is not set +CONFIG_SPLIT_PTLOCK_CPUS=4096 +CONFIG_ALIGNMENT_TRAP=y + +# +# Boot options +# +CONFIG_ZBOOT_ROM_TEXT=0x0 +CONFIG_ZBOOT_ROM_BSS=0x0 +CONFIG_CMDLINE="console=ttyS0,115200n8 console=tty1 noinitrd root=/dev/hda1 rootfstype=ext3 rw fbcon=rotate:1 dyntick=enable debug" +# CONFIG_XIP_KERNEL is not set + +# +# CPU Frequency scaling +# +# CONFIG_CPU_FREQ is not set + +# +# Floating point emulation +# + +# +# At least one emulation must be selected +# +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_NWFPE_XP is not set +# CONFIG_FPE_FASTFPE is not set + +# +# Userspace binary formats +# +CONFIG_BINFMT_ELF=y +CONFIG_BINFMT_AOUT=m +CONFIG_BINFMT_MISC=m +# CONFIG_ARTHUR is not set + +# +# Power management options +# +CONFIG_PM=y +# CONFIG_PM_LEGACY is not set +# CONFIG_PM_DEBUG is not set +CONFIG_APM=y + +# +# Networking +# +CONFIG_NET=y + +# +# Networking options +# +# CONFIG_NETDEBUG is not set +CONFIG_PACKET=y +CONFIG_PACKET_MMAP=y +CONFIG_UNIX=y +CONFIG_XFRM=y +# CONFIG_XFRM_USER is not set +# CONFIG_NET_KEY is not set +CONFIG_INET=y +# CONFIG_IP_MULTICAST is not set +# CONFIG_IP_ADVANCED_ROUTER is not set +CONFIG_IP_FIB_HASH=y +# CONFIG_IP_PNP is not set +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set +CONFIG_SYN_COOKIES=y +# CONFIG_INET_AH is not set +# CONFIG_INET_ESP is not set +# CONFIG_INET_IPCOMP is not set +# CONFIG_INET_XFRM_TUNNEL is not set +# CONFIG_INET_TUNNEL is not set +CONFIG_INET_DIAG=m +CONFIG_INET_TCP_DIAG=m +# CONFIG_TCP_CONG_ADVANCED is not set +CONFIG_TCP_CONG_BIC=y + +# +# IP: Virtual Server Configuration +# +# CONFIG_IP_VS is not set +CONFIG_IPV6=m +# CONFIG_IPV6_PRIVACY is not set +# CONFIG_IPV6_ROUTER_PREF is not set +CONFIG_INET6_AH=m +CONFIG_INET6_ESP=m +CONFIG_INET6_IPCOMP=m +CONFIG_INET6_XFRM_TUNNEL=m +CONFIG_INET6_TUNNEL=m +CONFIG_IPV6_TUNNEL=m +CONFIG_NETFILTER=y +# CONFIG_NETFILTER_DEBUG is not set + +# +# Core Netfilter Configuration +# +# CONFIG_NETFILTER_NETLINK is not set +# CONFIG_NETFILTER_XTABLES is not set + +# +# IP: Netfilter Configuration +# +CONFIG_IP_NF_CONNTRACK=m +# CONFIG_IP_NF_CT_ACCT is not set +# CONFIG_IP_NF_CONNTRACK_MARK is not set +# CONFIG_IP_NF_CONNTRACK_EVENTS is not set +CONFIG_IP_NF_CT_PROTO_SCTP=m +CONFIG_IP_NF_FTP=m +CONFIG_IP_NF_IRC=m +# CONFIG_IP_NF_NETBIOS_NS is not set +CONFIG_IP_NF_TFTP=m +CONFIG_IP_NF_AMANDA=m +# CONFIG_IP_NF_PPTP is not set +# CONFIG_IP_NF_H323 is not set +CONFIG_IP_NF_QUEUE=m + +# +# IPv6: Netfilter Configuration (EXPERIMENTAL) +# +# CONFIG_IP6_NF_QUEUE is not set + +# +# DCCP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_DCCP is not set + +# +# SCTP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_SCTP is not set + +# +# TIPC Configuration (EXPERIMENTAL) +# +# CONFIG_TIPC is not set +# CONFIG_ATM is not set +# CONFIG_BRIDGE is not set +# CONFIG_VLAN_8021Q is not set +# CONFIG_DECNET is not set +# CONFIG_LLC2 is not set +# CONFIG_IPX is not set +# CONFIG_ATALK is not set +# CONFIG_X25 is not set +# CONFIG_LAPB is not set +# CONFIG_NET_DIVERT is not set +# CONFIG_ECONET is not set +# CONFIG_WAN_ROUTER is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network testing +# +# CONFIG_NET_PKTGEN is not set +# CONFIG_HAMRADIO is not set +CONFIG_IRDA=m + +# +# IrDA protocols +# +CONFIG_IRLAN=m +CONFIG_IRNET=m +CONFIG_IRCOMM=m +# CONFIG_IRDA_ULTRA is not set + +# +# IrDA options +# +# CONFIG_IRDA_CACHE_LAST_LSAP is not set +# CONFIG_IRDA_FAST_RR is not set +# CONFIG_IRDA_DEBUG is not set + +# +# Infrared-port device drivers +# + +# +# SIR device drivers +# +# CONFIG_IRTTY_SIR is not set + +# +# Dongle support +# + +# +# Old SIR device drivers +# +# CONFIG_IRPORT_SIR is not set + +# +# Old Serial dongle support +# + +# +# FIR device drivers +# +# CONFIG_USB_IRDA is not set +# CONFIG_SIGMATEL_FIR is not set +CONFIG_PXA_FICP=m +CONFIG_BT=m +CONFIG_BT_L2CAP=m +CONFIG_BT_SCO=m +CONFIG_BT_RFCOMM=m +CONFIG_BT_RFCOMM_TTY=y +CONFIG_BT_BNEP=m +CONFIG_BT_BNEP_MC_FILTER=y +CONFIG_BT_BNEP_PROTO_FILTER=y +CONFIG_BT_HIDP=m + +# +# Bluetooth device drivers +# +CONFIG_BT_HCIUSB=m +# CONFIG_BT_HCIUSB_SCO is not set +CONFIG_BT_HCIUART=m +CONFIG_BT_HCIUART_H4=y +CONFIG_BT_HCIUART_BCSP=y +CONFIG_BT_HCIBCM203X=m +CONFIG_BT_HCIBPA10X=m +CONFIG_BT_HCIBFUSB=m +CONFIG_BT_HCIDTL1=m +CONFIG_BT_HCIBT3C=m +CONFIG_BT_HCIBLUECARD=m +CONFIG_BT_HCIBTUART=m +CONFIG_BT_HCIVHCI=m +CONFIG_IEEE80211=m +# CONFIG_IEEE80211_DEBUG is not set +CONFIG_IEEE80211_CRYPT_WEP=m +CONFIG_IEEE80211_CRYPT_CCMP=m +CONFIG_IEEE80211_CRYPT_TKIP=m +# CONFIG_IEEE80211_SOFTMAC is not set +CONFIG_WIRELESS_EXT=y + +# +# Device Drivers +# + +# +# Generic Driver Options +# +CONFIG_STANDALONE=y +CONFIG_PREVENT_FIRMWARE_BUILD=y +CONFIG_FW_LOADER=y +# CONFIG_DEBUG_DRIVER is not set + +# +# Connector - unified userspace <-> kernelspace linker +# +# CONFIG_CONNECTOR is not set + +# +# Memory Technology Devices (MTD) +# +CONFIG_MTD=y +# CONFIG_MTD_DEBUG is not set +# CONFIG_MTD_CONCAT is not set +CONFIG_MTD_PARTITIONS=y +# CONFIG_MTD_REDBOOT_PARTS is not set +CONFIG_MTD_CMDLINE_PARTS=y +# CONFIG_MTD_AFS_PARTS is not set + +# +# User Modules And Translation Layers +# +CONFIG_MTD_CHAR=y +CONFIG_MTD_BLOCK=y +# CONFIG_FTL is not set +# CONFIG_NFTL is not set +# CONFIG_INFTL is not set +# CONFIG_RFD_FTL is not set + +# +# RAM/ROM/Flash chip drivers +# +# CONFIG_MTD_CFI is not set +# CONFIG_MTD_JEDECPROBE is not set +CONFIG_MTD_MAP_BANK_WIDTH_1=y +CONFIG_MTD_MAP_BANK_WIDTH_2=y +CONFIG_MTD_MAP_BANK_WIDTH_4=y +# CONFIG_MTD_MAP_BANK_WIDTH_8 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_16 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_32 is not set +CONFIG_MTD_CFI_I1=y +CONFIG_MTD_CFI_I2=y +# CONFIG_MTD_CFI_I4 is not set +# CONFIG_MTD_CFI_I8 is not set +# CONFIG_MTD_RAM is not set +CONFIG_MTD_ROM=y +# CONFIG_MTD_ABSENT is not set +# CONFIG_MTD_OBSOLETE_CHIPS is not set + +# +# Mapping drivers for chip access +# +CONFIG_MTD_COMPLEX_MAPPINGS=y +CONFIG_MTD_SHARP_SL=y +# CONFIG_MTD_PLATRAM is not set + +# +# Self-contained MTD device drivers +# +# CONFIG_MTD_SLRAM is not set +# CONFIG_MTD_PHRAM is not set +# CONFIG_MTD_MTDRAM is not set +# CONFIG_MTD_BLOCK2MTD is not set + +# +# Disk-On-Chip Device Drivers +# +# CONFIG_MTD_DOC2000 is not set +# CONFIG_MTD_DOC2001 is not set +# CONFIG_MTD_DOC2001PLUS is not set + +# +# NAND Flash Device Drivers +# +CONFIG_MTD_NAND=y +CONFIG_MTD_NAND_VERIFY_WRITE=y +# CONFIG_MTD_NAND_H1900 is not set +CONFIG_MTD_NAND_IDS=y +# CONFIG_MTD_NAND_DISKONCHIP is not set +CONFIG_MTD_NAND_SHARPSL=y +# CONFIG_MTD_NAND_NANDSIM is not set + +# +# OneNAND Flash Device Drivers +# +# CONFIG_MTD_ONENAND is not set + +# +# Parallel port support +# +# CONFIG_PARPORT is not set + +# +# Plug and Play support +# + +# +# Block devices +# +# CONFIG_BLK_DEV_COW_COMMON is not set +CONFIG_BLK_DEV_LOOP=y +# CONFIG_BLK_DEV_CRYPTOLOOP is not set +# CONFIG_BLK_DEV_NBD is not set +# CONFIG_BLK_DEV_UB is not set +# CONFIG_BLK_DEV_RAM is not set +# CONFIG_BLK_DEV_INITRD is not set +# CONFIG_CDROM_PKTCDVD is not set +# CONFIG_ATA_OVER_ETH is not set + +# +# ATA/ATAPI/MFM/RLL support +# +CONFIG_IDE=y +CONFIG_BLK_DEV_IDE=y + +# +# Please see Documentation/ide.txt for help/info on IDE drives +# +# CONFIG_BLK_DEV_IDE_SATA is not set +CONFIG_BLK_DEV_IDEDISK=y +# CONFIG_IDEDISK_MULTI_MODE is not set +CONFIG_BLK_DEV_IDECS=y +# CONFIG_BLK_DEV_IDECD is not set +# CONFIG_BLK_DEV_IDETAPE is not set +# CONFIG_BLK_DEV_IDEFLOPPY is not set +# CONFIG_BLK_DEV_IDESCSI is not set +# CONFIG_IDE_TASK_IOCTL is not set + +# +# IDE chipset support/bugfixes +# +CONFIG_IDE_GENERIC=y +# CONFIG_IDE_ARM is not set +# CONFIG_BLK_DEV_IDEDMA is not set +# CONFIG_IDEDMA_AUTO is not set +# CONFIG_BLK_DEV_HD is not set + +# +# SCSI device support +# +# CONFIG_RAID_ATTRS is not set +CONFIG_SCSI=m +CONFIG_SCSI_PROC_FS=y + +# +# SCSI support type (disk, tape, CD-ROM) +# +CONFIG_BLK_DEV_SD=m +CONFIG_CHR_DEV_ST=m +CONFIG_CHR_DEV_OSST=m +CONFIG_BLK_DEV_SR=m +# CONFIG_BLK_DEV_SR_VENDOR is not set +CONFIG_CHR_DEV_SG=m +# CONFIG_CHR_DEV_SCH is not set + +# +# Some SCSI devices (e.g. CD jukebox) support multiple LUNs +# +CONFIG_SCSI_MULTI_LUN=y +# CONFIG_SCSI_CONSTANTS is not set +# CONFIG_SCSI_LOGGING is not set + +# +# SCSI Transport Attributes +# +# CONFIG_SCSI_SPI_ATTRS is not set +# CONFIG_SCSI_FC_ATTRS is not set +# CONFIG_SCSI_ISCSI_ATTRS is not set +# CONFIG_SCSI_SAS_ATTRS is not set + +# +# SCSI low-level drivers +# +# CONFIG_ISCSI_TCP is not set +# CONFIG_SCSI_SATA is not set +# CONFIG_SCSI_DEBUG is not set + +# +# PCMCIA SCSI adapter support +# +# CONFIG_PCMCIA_AHA152X is not set +# CONFIG_PCMCIA_FDOMAIN is not set +# CONFIG_PCMCIA_NINJA_SCSI is not set +# CONFIG_PCMCIA_QLOGIC is not set +# CONFIG_PCMCIA_SYM53C500 is not set + +# +# Multi-device support (RAID and LVM) +# +CONFIG_MD=y +CONFIG_BLK_DEV_DM=m +CONFIG_DM_CRYPT=m +CONFIG_DM_SNAPSHOT=m +CONFIG_DM_MIRROR=m +CONFIG_DM_ZERO=m +CONFIG_DM_MULTIPATH=m +CONFIG_DM_MULTIPATH_EMC=m + +# +# Fusion MPT device support +# +# CONFIG_FUSION is not set + +# +# IEEE 1394 (FireWire) support +# + +# +# I2O device support +# + +# +# Network device support +# +CONFIG_NETDEVICES=y +# CONFIG_DUMMY is not set +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +CONFIG_TUN=m + +# +# PHY device support +# +# CONFIG_PHYLIB is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +CONFIG_MII=m +# CONFIG_SMC91X is not set +# CONFIG_DM9000 is not set + +# +# Ethernet (1000 Mbit) +# + +# +# Ethernet (10000 Mbit) +# + +# +# Token Ring devices +# + +# +# Wireless LAN (non-hamradio) +# +CONFIG_NET_RADIO=y +# CONFIG_NET_WIRELESS_RTNETLINK is not set + +# +# Obsolete Wireless cards support (pre-802.11) +# +# CONFIG_STRIP is not set +# CONFIG_PCMCIA_WAVELAN is not set +# CONFIG_PCMCIA_NETWAVE is not set + +# +# Wireless 802.11 Frequency Hopping cards support +# +# CONFIG_PCMCIA_RAYCS is not set + +# +# Wireless 802.11b ISA/PCI cards support +# +CONFIG_HERMES=m +# CONFIG_ATMEL is not set + +# +# Wireless 802.11b Pcmcia/Cardbus cards support +# +CONFIG_PCMCIA_HERMES=m +CONFIG_PCMCIA_SPECTRUM=m +# CONFIG_AIRO_CS is not set +# CONFIG_PCMCIA_WL3501 is not set +CONFIG_HOSTAP=m +CONFIG_HOSTAP_FIRMWARE=y +# CONFIG_HOSTAP_FIRMWARE_NVRAM is not set +CONFIG_HOSTAP_CS=m +CONFIG_NET_WIRELESS=y + +# +# PCMCIA network device support +# +CONFIG_NET_PCMCIA=y +# CONFIG_PCMCIA_3C589 is not set +# CONFIG_PCMCIA_3C574 is not set +# CONFIG_PCMCIA_FMVJ18X is not set +CONFIG_PCMCIA_PCNET=m +# CONFIG_PCMCIA_NMCLAN is not set +# CONFIG_PCMCIA_SMC91C92 is not set +# CONFIG_PCMCIA_XIRC2PS is not set +# CONFIG_PCMCIA_AXNET is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set +CONFIG_PPP=m +# CONFIG_PPP_MULTILINK is not set +# CONFIG_PPP_FILTER is not set +CONFIG_PPP_ASYNC=m +# CONFIG_PPP_SYNC_TTY is not set +CONFIG_PPP_DEFLATE=m +CONFIG_PPP_BSDCOMP=m +# CONFIG_PPP_MPPE is not set +# CONFIG_PPPOE is not set +# CONFIG_SLIP is not set +# CONFIG_SHAPER is not set +# CONFIG_NETCONSOLE is not set +# CONFIG_NETPOLL is not set +# CONFIG_NET_POLL_CONTROLLER is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Input device support +# +CONFIG_INPUT=y + +# +# Userland interfaces +# +CONFIG_INPUT_MOUSEDEV=m +# CONFIG_INPUT_MOUSEDEV_PSAUX is not set +CONFIG_INPUT_MOUSEDEV_SCREEN_X=640 +CONFIG_INPUT_MOUSEDEV_SCREEN_Y=480 +# CONFIG_INPUT_JOYDEV is not set +# CONFIG_INPUT_TSDEV is not set +CONFIG_INPUT_EVDEV=y +# CONFIG_INPUT_EVBUG is not set +CONFIG_INPUT_POWER=y + +# +# Input Device Drivers +# +CONFIG_INPUT_KEYBOARD=y +# CONFIG_KEYBOARD_ATKBD is not set +# CONFIG_KEYBOARD_SUNKBD is not set +# CONFIG_KEYBOARD_LKKBD is not set +# CONFIG_KEYBOARD_XTKBD is not set +# CONFIG_KEYBOARD_NEWTON is not set +# CONFIG_KEYBOARD_CORGI is not set +CONFIG_KEYBOARD_SPITZ=y +# CONFIG_INPUT_MOUSE is not set +# CONFIG_INPUT_JOYSTICK is not set +CONFIG_INPUT_TOUCHSCREEN=y +CONFIG_TOUCHSCREEN_CORGI=y +# CONFIG_TOUCHSCREEN_GUNZE is not set +# CONFIG_TOUCHSCREEN_ELO is not set +# CONFIG_TOUCHSCREEN_MTOUCH is not set +# CONFIG_TOUCHSCREEN_MK712 is not set +CONFIG_INPUT_MISC=y +CONFIG_INPUT_UINPUT=m + +# +# Hardware I/O ports +# +# CONFIG_SERIO is not set +# CONFIG_GAMEPORT is not set + +# +# Character devices +# +CONFIG_VT=y +CONFIG_VT_CONSOLE=y +CONFIG_HW_CONSOLE=y +# CONFIG_SERIAL_NONSTANDARD is not set + +# +# Serial drivers +# +CONFIG_SERIAL_8250=m +CONFIG_SERIAL_8250_CS=m +CONFIG_SERIAL_8250_NR_UARTS=4 +CONFIG_SERIAL_8250_RUNTIME_UARTS=4 +# CONFIG_SERIAL_8250_EXTENDED is not set + +# +# Non-8250 serial port support +# +CONFIG_SERIAL_PXA=y +CONFIG_SERIAL_PXA_CONSOLE=y +CONFIG_SERIAL_CORE=y +CONFIG_SERIAL_CORE_CONSOLE=y +CONFIG_UNIX98_PTYS=y +# CONFIG_LEGACY_PTYS is not set + +# +# IPMI +# +# CONFIG_IPMI_HANDLER is not set + +# +# Watchdog Cards +# +# CONFIG_WATCHDOG is not set +# CONFIG_NVRAM is not set +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set + +# +# Ftape, the floppy tape device driver +# + +# +# PCMCIA character devices +# +# CONFIG_SYNCLINK_CS is not set +# CONFIG_CARDMAN_4000 is not set +# CONFIG_CARDMAN_4040 is not set +# CONFIG_RAW_DRIVER is not set + +# +# TPM devices +# +# CONFIG_TCG_TPM is not set +# CONFIG_TELCLOCK is not set + +# +# I2C support +# +CONFIG_I2C=y +# CONFIG_I2C_CHARDEV is not set + +# +# I2C Algorithms +# +# CONFIG_I2C_ALGOBIT is not set +# CONFIG_I2C_ALGOPCF is not set +# CONFIG_I2C_ALGOPCA is not set + +# +# I2C Hardware Bus support +# +CONFIG_I2C_PXA=y +# CONFIG_I2C_PXA_SLAVE is not set +# CONFIG_I2C_PARPORT_LIGHT is not set +# CONFIG_I2C_STUB is not set +# CONFIG_I2C_PCA_ISA is not set + +# +# Miscellaneous I2C Chip support +# +# CONFIG_SENSORS_DS1337 is not set +# CONFIG_SENSORS_DS1374 is not set +# CONFIG_SENSORS_EEPROM is not set +# CONFIG_SENSORS_PCF8574 is not set +# CONFIG_SENSORS_PCA9539 is not set +# CONFIG_SENSORS_PCF8591 is not set +# CONFIG_SENSORS_MAX6875 is not set +# CONFIG_I2C_DEBUG_CORE is not set +# CONFIG_I2C_DEBUG_ALGO is not set +# CONFIG_I2C_DEBUG_BUS is not set +# CONFIG_I2C_DEBUG_CHIP is not set + +# +# SPI support +# +# CONFIG_SPI is not set +# CONFIG_SPI_MASTER is not set + +# +# Dallas's 1-wire bus +# +# CONFIG_W1 is not set + +# +# Hardware Monitoring support +# +# CONFIG_HWMON is not set +# CONFIG_HWMON_VID is not set + +# +# Misc devices +# + +# +# Multi-Function Devices +# + +# +# LED devices +# +CONFIG_NEW_LEDS=y +CONFIG_LEDS_CLASS=y + +# +# LED drivers +# +CONFIG_LEDS_SPITZ=y +# CONFIG_LEDS_TOSA is not set + +# +# LED Triggers +# +CONFIG_LEDS_TRIGGERS=y +CONFIG_LEDS_TRIGGER_TIMER=y +CONFIG_LEDS_TRIGGER_IDE_DISK=y + +# +# Multimedia devices +# +# CONFIG_VIDEO_DEV is not set +CONFIG_VIDEO_V4L2=y + +# +# Digital Video Broadcasting Devices +# +# CONFIG_DVB is not set +CONFIG_USB_DABUSB=m + +# +# Graphics support +# +CONFIG_FB=y +CONFIG_FB_CFB_FILLRECT=y +CONFIG_FB_CFB_COPYAREA=y +CONFIG_FB_CFB_IMAGEBLIT=y +# CONFIG_FB_MACMODES is not set +CONFIG_FB_FIRMWARE_EDID=y +# CONFIG_FB_MODE_HELPERS is not set +# CONFIG_FB_TILEBLITTING is not set +# CONFIG_FB_S1D13XXX is not set +CONFIG_FB_PXA=y +CONFIG_FB_PXA_LCD_QVGA=y +# CONFIG_FB_PXA_LCD_VGA is not set +CONFIG_FB_PXA_OVERLAY=y +# CONFIG_FB_PXA_PARAMETERS is not set +# CONFIG_FB_W100 is not set +# CONFIG_FB_VIRTUAL is not set + +# +# Console display driver support +# +# CONFIG_VGA_CONSOLE is not set +CONFIG_DUMMY_CONSOLE=y +CONFIG_FRAMEBUFFER_CONSOLE=y +CONFIG_FRAMEBUFFER_CONSOLE_ROTATION=y +CONFIG_FONTS=y +# CONFIG_FONT_8x8 is not set +CONFIG_FONT_8x16=y +# CONFIG_FONT_6x11 is not set +# CONFIG_FONT_7x14 is not set +# CONFIG_FONT_PEARL_8x8 is not set +# CONFIG_FONT_ACORN_8x8 is not set +# CONFIG_FONT_MINI_4x6 is not set +# CONFIG_FONT_SUN8x16 is not set +# CONFIG_FONT_SUN12x22 is not set +# CONFIG_FONT_10x18 is not set + +# +# Logo configuration +# +CONFIG_LOGO=y +CONFIG_LOGO_LINUX_MONO=y +CONFIG_LOGO_LINUX_VGA16=y +# CONFIG_LOGO_LINUX_CLUT224 is not set +# CONFIG_LOGO_OHAND_CLUT224 is not set +# CONFIG_LOGO_OZ240_CLUT224 is not set +# CONFIG_LOGO_OZ480_CLUT224 is not set +CONFIG_LOGO_OZ640_CLUT224=y +CONFIG_BACKLIGHT_LCD_SUPPORT=y +CONFIG_BACKLIGHT_CLASS_DEVICE=y +CONFIG_BACKLIGHT_DEVICE=y +CONFIG_LCD_CLASS_DEVICE=y +CONFIG_LCD_DEVICE=y +CONFIG_BACKLIGHT_CORGI=y + +# +# Sound +# +CONFIG_SOUND=m + +# +# Advanced Linux Sound Architecture +# +CONFIG_SND=m +CONFIG_SND_TIMER=m +CONFIG_SND_PCM=m +CONFIG_SND_HWDEP=m +CONFIG_SND_RAWMIDI=m +CONFIG_SND_SEQUENCER=m +# CONFIG_SND_SEQ_DUMMY is not set +CONFIG_SND_OSSEMUL=y +CONFIG_SND_MIXER_OSS=m +CONFIG_SND_PCM_OSS=m +CONFIG_SND_PCM_OSS_PLUGINS=y +# CONFIG_SND_SEQUENCER_OSS is not set +# CONFIG_SND_DYNAMIC_MINORS is not set +CONFIG_SND_SUPPORT_OLD_API=y +CONFIG_SND_VERBOSE_PROCFS=y +CONFIG_SND_VERBOSE_PRINTK=y +CONFIG_SND_DEBUG=y +# CONFIG_SND_DEBUG_DETECT is not set +# CONFIG_SND_PCM_XRUN_DEBUG is not set + +# +# Generic devices +# +CONFIG_SND_AC97_CODEC=m +CONFIG_SND_AC97_BUS=m +# CONFIG_SND_DUMMY is not set +# CONFIG_SND_VIRMIDI is not set +# CONFIG_SND_MTPAV is not set +# CONFIG_SND_SERIAL_U16550 is not set +# CONFIG_SND_MPU401 is not set + +# +# ALSA ARM devices +# +CONFIG_SND_PXA2XX_PCM=m +CONFIG_SND_PXA2XX_AC97=m + +# +# USB devices +# +CONFIG_SND_USB_AUDIO=m + +# +# PCMCIA devices +# +# CONFIG_SND_VXPOCKET is not set +# CONFIG_SND_PDAUDIOCF is not set + +# +# SoC audio support +# +CONFIG_SND_SOC=m + +# +# Soc Platforms +# + +# +# SoC Audio for the Intel PXA2xx +# +CONFIG_SND_PXA2xx_SOC=m +CONFIG_SND_PXA2xx_SOC_I2S=m +CONFIG_SND_PXA2xx_SOC_SPITZ=m + +# +# SoC Audio for the Atmel AT91 +# + +# +# SoC Audio for the Freescale i.MX +# + +# +# Soc Codecs +# +# CONFIG_SND_SOC_AC97_CODEC is not set +# CONFIG_SND_SOC_WM8731 is not set +CONFIG_SND_SOC_WM8750=m +# CONFIG_SND_SOC_WM8753 is not set +# CONFIG_SND_SOC_WM8772 is not set +# CONFIG_SND_SOC_WM8971 is not set +# CONFIG_SND_SOC_WM8974 is not set +# CONFIG_SND_SOC_WM9713 is not set +# CONFIG_SND_SOC_WM9712 is not set +# CONFIG_SND_SOC_UDA1380 is not set +# CONFIG_SND_SOC_AK4535 is not set + +# +# Open Sound System +# +# CONFIG_SOUND_PRIME is not set + +# +# USB support +# +CONFIG_USB_ARCH_HAS_HCD=y +CONFIG_USB_ARCH_HAS_OHCI=y +# CONFIG_USB_ARCH_HAS_EHCI is not set +CONFIG_USB=m +# CONFIG_USB_DEBUG is not set + +# +# Miscellaneous USB options +# +CONFIG_USB_DEVICEFS=y +# CONFIG_USB_BANDWIDTH is not set +# CONFIG_USB_DYNAMIC_MINORS is not set +# CONFIG_USB_SUSPEND is not set +# CONFIG_USB_OTG is not set + +# +# USB Host Controller Drivers +# +# CONFIG_USB_ISP116X_HCD is not set +CONFIG_USB_OHCI_HCD=m +# CONFIG_USB_OHCI_BIG_ENDIAN is not set +CONFIG_USB_OHCI_LITTLE_ENDIAN=y +CONFIG_USB_SL811_HCD=m +CONFIG_USB_SL811_CS=m + +# +# USB Device Class drivers +# +CONFIG_USB_ACM=m +CONFIG_USB_PRINTER=m + +# +# NOTE: USB_STORAGE enables SCSI, and 'SCSI disk support' +# + +# +# may also be needed; see USB_STORAGE Help for more information +# +CONFIG_USB_STORAGE=m +# CONFIG_USB_STORAGE_DEBUG is not set +# CONFIG_USB_STORAGE_DATAFAB is not set +# CONFIG_USB_STORAGE_FREECOM is not set +# CONFIG_USB_STORAGE_ISD200 is not set +# CONFIG_USB_STORAGE_DPCM is not set +# CONFIG_USB_STORAGE_USBAT is not set +# CONFIG_USB_STORAGE_SDDR09 is not set +# CONFIG_USB_STORAGE_SDDR55 is not set +# CONFIG_USB_STORAGE_JUMPSHOT is not set +# CONFIG_USB_STORAGE_ALAUDA is not set +# CONFIG_USB_LIBUSUAL is not set + +# +# USB Input Devices +# +CONFIG_USB_HID=m +CONFIG_USB_HIDINPUT=y +# CONFIG_USB_HIDINPUT_POWERBOOK is not set +# CONFIG_HID_FF is not set +# CONFIG_USB_HIDDEV is not set + +# +# USB HID Boot Protocol drivers +# +CONFIG_USB_KBD=m +CONFIG_USB_MOUSE=m +CONFIG_USB_AIPTEK=m +CONFIG_USB_WACOM=m +# CONFIG_USB_ACECAD is not set +CONFIG_USB_KBTAB=m +CONFIG_USB_POWERMATE=m +# CONFIG_USB_TOUCHSCREEN is not set +# CONFIG_USB_YEALINK is not set +CONFIG_USB_XPAD=m +CONFIG_USB_ATI_REMOTE=m +# CONFIG_USB_ATI_REMOTE2 is not set +# CONFIG_USB_KEYSPAN_REMOTE is not set +# CONFIG_USB_APPLETOUCH is not set + +# +# USB Imaging devices +# +CONFIG_USB_MDC800=m +CONFIG_USB_MICROTEK=m + +# +# USB Network Adapters +# +CONFIG_USB_CATC=m +CONFIG_USB_KAWETH=m +CONFIG_USB_PEGASUS=m +CONFIG_USB_RTL8150=m +CONFIG_USB_USBNET=m +CONFIG_USB_NET_AX8817X=m +CONFIG_USB_NET_CDCETHER=m +# CONFIG_USB_NET_GL620A is not set +CONFIG_USB_NET_NET1080=m +# CONFIG_USB_NET_PLUSB is not set +# CONFIG_USB_NET_RNDIS_HOST is not set +# CONFIG_USB_NET_CDC_SUBSET is not set +CONFIG_USB_NET_ZAURUS=m +# CONFIG_USB_ZD1201 is not set +CONFIG_USB_MON=y + +# +# USB port drivers +# + +# +# USB Serial Converter support +# +CONFIG_USB_SERIAL=m +CONFIG_USB_SERIAL_GENERIC=y +# CONFIG_USB_SERIAL_AIRPRIME is not set +# CONFIG_USB_SERIAL_ANYDATA is not set +# CONFIG_USB_SERIAL_ARK3116 is not set +CONFIG_USB_SERIAL_BELKIN=m +# CONFIG_USB_SERIAL_WHITEHEAT is not set +CONFIG_USB_SERIAL_DIGI_ACCELEPORT=m +# CONFIG_USB_SERIAL_CP2101 is not set +CONFIG_USB_SERIAL_CYPRESS_M8=m +CONFIG_USB_SERIAL_EMPEG=m +CONFIG_USB_SERIAL_FTDI_SIO=m +# CONFIG_USB_SERIAL_FUNSOFT is not set +CONFIG_USB_SERIAL_VISOR=m +CONFIG_USB_SERIAL_IPAQ=m +CONFIG_USB_SERIAL_IR=m +CONFIG_USB_SERIAL_EDGEPORT=m +CONFIG_USB_SERIAL_EDGEPORT_TI=m +CONFIG_USB_SERIAL_GARMIN=m +CONFIG_USB_SERIAL_IPW=m +CONFIG_USB_SERIAL_KEYSPAN_PDA=m +CONFIG_USB_SERIAL_KEYSPAN=m +# CONFIG_USB_SERIAL_KEYSPAN_MPR is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XA is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XB is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA18X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19QW is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19QI is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49WLC is not set +CONFIG_USB_SERIAL_KLSI=m +CONFIG_USB_SERIAL_KOBIL_SCT=m +CONFIG_USB_SERIAL_MCT_U232=m +# CONFIG_USB_SERIAL_NAVMAN is not set +CONFIG_USB_SERIAL_PL2303=m +# CONFIG_USB_SERIAL_HP4X is not set +CONFIG_USB_SERIAL_SAFE=m +# CONFIG_USB_SERIAL_SAFE_PADDED is not set +CONFIG_USB_SERIAL_TI=m +CONFIG_USB_SERIAL_CYBERJACK=m +CONFIG_USB_SERIAL_XIRCOM=m +# CONFIG_USB_SERIAL_OPTION is not set +CONFIG_USB_SERIAL_OMNINET=m +CONFIG_USB_EZUSB=y + +# +# USB Miscellaneous drivers +# +CONFIG_USB_EMI62=m +CONFIG_USB_EMI26=m +CONFIG_USB_AUERSWALD=m +CONFIG_USB_RIO500=m +CONFIG_USB_LEGOTOWER=m +CONFIG_USB_LCD=m +CONFIG_USB_LED=m +CONFIG_USB_CYTHERM=m +CONFIG_USB_PHIDGETKIT=m +CONFIG_USB_PHIDGETSERVO=m +CONFIG_USB_IDMOUSE=m +# CONFIG_USB_LD is not set +# CONFIG_USB_TEST is not set + +# +# USB DSL modem support +# + +# +# USB Gadget Support +# +CONFIG_USB_GADGET=m +# CONFIG_USB_GADGET_DEBUG_FILES is not set +CONFIG_USB_GADGET_SELECTED=y +# CONFIG_USB_GADGET_NET2280 is not set +# CONFIG_USB_GADGET_PXA2XX is not set +CONFIG_USB_GADGET_PXA27X=y +CONFIG_USB_PXA27X=m +# CONFIG_USB_GADGET_GOKU is not set +# CONFIG_USB_GADGET_LH7A40X is not set +# CONFIG_USB_GADGET_OMAP is not set +# CONFIG_USB_GADGET_AT91 is not set +# CONFIG_USB_GADGET_DUMMY_HCD is not set +# CONFIG_USB_GADGET_DUALSPEED is not set +CONFIG_USB_ZERO=m +CONFIG_USB_ETH=m +CONFIG_USB_ETH_RNDIS=y +CONFIG_USB_GADGETFS=m +CONFIG_USB_FILE_STORAGE=m +# CONFIG_USB_FILE_STORAGE_TEST is not set +CONFIG_USB_G_SERIAL=m + +# +# MMC/SD Card support +# +CONFIG_MMC=y +# CONFIG_MMC_DEBUG is not set +CONFIG_MMC_BLOCK=y +CONFIG_MMC_PXA=y + +# +# Real Time Clock +# +CONFIG_RTC_LIB=y +CONFIG_RTC_CLASS=y +CONFIG_RTC_HCTOSYS=y +CONFIG_RTC_HCTOSYS_DEVICE="rtc0" + +# +# RTC interfaces +# +CONFIG_RTC_INTF_SYSFS=y +CONFIG_RTC_INTF_PROC=y +CONFIG_RTC_INTF_DEV=y + +# +# RTC drivers +# +# CONFIG_RTC_DRV_X1205 is not set +# CONFIG_RTC_DRV_DS1672 is not set +# CONFIG_RTC_DRV_PCF8563 is not set +# CONFIG_RTC_DRV_RS5C372 is not set +# CONFIG_RTC_DRV_M48T86 is not set +CONFIG_RTC_DRV_SA1100=y +# CONFIG_RTC_DRV_TEST is not set + +# +# File systems +# +CONFIG_EXT2_FS=y +# CONFIG_EXT2_FS_XATTR is not set +# CONFIG_EXT2_FS_XIP is not set +CONFIG_EXT3_FS=y +# CONFIG_EXT3_FS_XATTR is not set +CONFIG_JBD=y +# CONFIG_JBD_DEBUG is not set +# CONFIG_REISERFS_FS is not set +# CONFIG_JFS_FS is not set +# CONFIG_FS_POSIX_ACL is not set +# CONFIG_XFS_FS is not set +# CONFIG_OCFS2_FS is not set +# CONFIG_MINIX_FS is not set +# CONFIG_ROMFS_FS is not set +CONFIG_INOTIFY=y +# CONFIG_QUOTA is not set +CONFIG_DNOTIFY=y +# CONFIG_AUTOFS_FS is not set +# CONFIG_AUTOFS4_FS is not set +# CONFIG_FUSE_FS is not set + +# +# CD-ROM/DVD Filesystems +# +# CONFIG_ISO9660_FS is not set +# CONFIG_UDF_FS is not set + +# +# DOS/FAT/NT Filesystems +# +CONFIG_FAT_FS=y +# CONFIG_MSDOS_FS is not set +CONFIG_VFAT_FS=y +CONFIG_FAT_DEFAULT_CODEPAGE=437 +CONFIG_FAT_DEFAULT_IOCHARSET="iso8859-1" +# CONFIG_NTFS_FS is not set + +# +# Pseudo filesystems +# +CONFIG_PROC_FS=y +CONFIG_SYSFS=y +CONFIG_TMPFS=y +# CONFIG_HUGETLB_PAGE is not set +CONFIG_RAMFS=y +# CONFIG_CONFIGFS_FS is not set + +# +# Miscellaneous filesystems +# +# CONFIG_ADFS_FS is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_HFSPLUS_FS is not set +# CONFIG_BEFS_FS is not set +# CONFIG_BFS_FS is not set +# CONFIG_EFS_FS is not set +# CONFIG_JFFS_FS is not set +CONFIG_JFFS2_FS=m +CONFIG_JFFS2_FS_DEBUG=0 +CONFIG_JFFS2_FS_WRITEBUFFER=y +CONFIG_JFFS2_SUMMARY=y +CONFIG_JFFS2_COMPRESSION_OPTIONS=y +CONFIG_JFFS2_ZLIB=y +CONFIG_JFFS2_RTIME=y +CONFIG_JFFS2_RUBIN=y +# CONFIG_JFFS2_CMODE_NONE is not set +CONFIG_JFFS2_CMODE_PRIORITY=y +# CONFIG_JFFS2_CMODE_SIZE is not set +CONFIG_CRAMFS=m +CONFIG_SQUASHFS=m +# CONFIG_SQUASHFS_EMBEDDED is not set +CONFIG_SQUASHFS_FRAGMENT_CACHE_SIZE=3 +# CONFIG_SQUASHFS_VMALLOC is not set +# CONFIG_VXFS_FS is not set +# CONFIG_HPFS_FS is not set +# CONFIG_QNX4FS_FS is not set +# CONFIG_SYSV_FS is not set +# CONFIG_UFS_FS is not set + +# +# Network File Systems +# +CONFIG_NFS_FS=m +CONFIG_NFS_V3=y +# CONFIG_NFS_V3_ACL is not set +CONFIG_NFS_V4=y +# CONFIG_NFS_DIRECTIO is not set +# CONFIG_NFSD is not set +CONFIG_LOCKD=m +CONFIG_LOCKD_V4=y +CONFIG_NFS_COMMON=y +CONFIG_SUNRPC=m +CONFIG_SUNRPC_GSS=m +CONFIG_RPCSEC_GSS_KRB5=m +# CONFIG_RPCSEC_GSS_SPKM3 is not set +CONFIG_SMB_FS=m +CONFIG_SMB_NLS_DEFAULT=y +CONFIG_SMB_NLS_REMOTE="cp437" +CONFIG_CIFS=m +# CONFIG_CIFS_STATS is not set +# CONFIG_CIFS_XATTR is not set +# CONFIG_CIFS_EXPERIMENTAL is not set +# CONFIG_NCP_FS is not set +# CONFIG_CODA_FS is not set +# CONFIG_AFS_FS is not set +# CONFIG_9P_FS is not set + +# +# Partition Types +# +CONFIG_PARTITION_ADVANCED=y +# CONFIG_ACORN_PARTITION is not set +# CONFIG_OSF_PARTITION is not set +# CONFIG_AMIGA_PARTITION is not set +# CONFIG_ATARI_PARTITION is not set +# CONFIG_MAC_PARTITION is not set +CONFIG_MSDOS_PARTITION=y +# CONFIG_BSD_DISKLABEL is not set +# CONFIG_MINIX_SUBPARTITION is not set +# CONFIG_SOLARIS_X86_PARTITION is not set +# CONFIG_UNIXWARE_DISKLABEL is not set +# CONFIG_LDM_PARTITION is not set +# CONFIG_SGI_PARTITION is not set +# CONFIG_ULTRIX_PARTITION is not set +# CONFIG_SUN_PARTITION is not set +# CONFIG_KARMA_PARTITION is not set +# CONFIG_EFI_PARTITION is not set + +# +# Native Language Support +# +CONFIG_NLS=y +CONFIG_NLS_DEFAULT="cp437" +CONFIG_NLS_CODEPAGE_437=y +CONFIG_NLS_CODEPAGE_737=m +CONFIG_NLS_CODEPAGE_775=m +CONFIG_NLS_CODEPAGE_850=m +CONFIG_NLS_CODEPAGE_852=m +CONFIG_NLS_CODEPAGE_855=m +CONFIG_NLS_CODEPAGE_857=m +CONFIG_NLS_CODEPAGE_860=m +CONFIG_NLS_CODEPAGE_861=m +CONFIG_NLS_CODEPAGE_862=m +CONFIG_NLS_CODEPAGE_863=m +CONFIG_NLS_CODEPAGE_864=m +CONFIG_NLS_CODEPAGE_865=m +CONFIG_NLS_CODEPAGE_866=m +CONFIG_NLS_CODEPAGE_869=m +CONFIG_NLS_CODEPAGE_936=m +CONFIG_NLS_CODEPAGE_950=m +CONFIG_NLS_CODEPAGE_932=m +CONFIG_NLS_CODEPAGE_949=m +CONFIG_NLS_CODEPAGE_874=m +CONFIG_NLS_ISO8859_8=m +CONFIG_NLS_CODEPAGE_1250=m +CONFIG_NLS_CODEPAGE_1251=m +CONFIG_NLS_ASCII=m +CONFIG_NLS_ISO8859_1=y +CONFIG_NLS_ISO8859_2=m +CONFIG_NLS_ISO8859_3=m +CONFIG_NLS_ISO8859_4=m +CONFIG_NLS_ISO8859_5=m +CONFIG_NLS_ISO8859_6=m +CONFIG_NLS_ISO8859_7=m +CONFIG_NLS_ISO8859_9=m +CONFIG_NLS_ISO8859_13=m +CONFIG_NLS_ISO8859_14=m +CONFIG_NLS_ISO8859_15=m +CONFIG_NLS_KOI8_R=m +CONFIG_NLS_KOI8_U=m +CONFIG_NLS_UTF8=y + +# +# Profiling support +# +CONFIG_PROFILING=y +CONFIG_OPROFILE=m + +# +# Kernel hacking +# +# CONFIG_PRINTK_TIME is not set +CONFIG_MAGIC_SYSRQ=y +CONFIG_DEBUG_KERNEL=y +CONFIG_LOG_BUF_SHIFT=14 +# CONFIG_DETECT_SOFTLOCKUP is not set +# CONFIG_SCHEDSTATS is not set +# CONFIG_DEBUG_SLAB is not set +# CONFIG_DEBUG_PREEMPT is not set +# CONFIG_DEBUG_MUTEXES is not set +# CONFIG_DEBUG_SPINLOCK is not set +# CONFIG_DEBUG_SPINLOCK_SLEEP is not set +# CONFIG_DEBUG_KOBJECT is not set +CONFIG_DEBUG_BUGVERBOSE=y +# CONFIG_DEBUG_INFO is not set +# CONFIG_DEBUG_FS is not set +# CONFIG_DEBUG_VM is not set +CONFIG_FRAME_POINTER=y +# CONFIG_UNWIND_INFO is not set +# CONFIG_FORCED_INLINING is not set +# CONFIG_RCU_TORTURE_TEST is not set +# CONFIG_DEBUG_USER is not set +# CONFIG_DEBUG_WAITQ is not set +CONFIG_DEBUG_ERRORS=y +# CONFIG_DEBUG_LL is not set + +# +# Security options +# +# CONFIG_KEYS is not set +# CONFIG_SECURITY is not set + +# +# Cryptographic options +# +CONFIG_CRYPTO=y +CONFIG_CRYPTO_HMAC=y +CONFIG_CRYPTO_NULL=m +CONFIG_CRYPTO_MD4=m +CONFIG_CRYPTO_MD5=m +CONFIG_CRYPTO_SHA1=m +CONFIG_CRYPTO_SHA256=m +CONFIG_CRYPTO_SHA512=m +CONFIG_CRYPTO_WP512=m +# CONFIG_CRYPTO_TGR192 is not set +CONFIG_CRYPTO_DES=m +CONFIG_CRYPTO_BLOWFISH=m +CONFIG_CRYPTO_TWOFISH=m +CONFIG_CRYPTO_SERPENT=m +CONFIG_CRYPTO_AES=m +CONFIG_CRYPTO_CAST5=m +CONFIG_CRYPTO_CAST6=m +CONFIG_CRYPTO_TEA=m +CONFIG_CRYPTO_ARC4=m +CONFIG_CRYPTO_KHAZAD=m +CONFIG_CRYPTO_ANUBIS=m +CONFIG_CRYPTO_DEFLATE=m +CONFIG_CRYPTO_MICHAEL_MIC=m +CONFIG_CRYPTO_CRC32C=m +CONFIG_CRYPTO_TEST=m + +# +# Hardware crypto devices +# + +# +# Library routines +# +CONFIG_CRC_CCITT=y +# CONFIG_CRC16 is not set +CONFIG_CRC32=y +CONFIG_LIBCRC32C=m +CONFIG_ZLIB_INFLATE=m +CONFIG_ZLIB_DEFLATE=m diff --git a/packages/linux/linux-rp-2.6.20/defconfig-tosa b/packages/linux/linux-rp-2.6.20/defconfig-tosa new file mode 100644 index 0000000000..4144e5cf38 --- /dev/null +++ b/packages/linux/linux-rp-2.6.20/defconfig-tosa @@ -0,0 +1,1608 @@ +# +# Automatically generated make config: don't edit +# Linux kernel version: 2.6.16-rc5-git5 +# Tue Mar 14 09:05:26 2006 +# +CONFIG_ARM=y +CONFIG_MMU=y +CONFIG_RWSEM_GENERIC_SPINLOCK=y +CONFIG_GENERIC_CALIBRATE_DELAY=y +CONFIG_ARCH_MTD_XIP=y + +# +# Code maturity level options +# +CONFIG_EXPERIMENTAL=y +CONFIG_BROKEN_ON_SMP=y +CONFIG_LOCK_KERNEL=y +CONFIG_INIT_ENV_ARG_LIMIT=32 + +# +# General setup +# +CONFIG_LOCALVERSION="" +CONFIG_LOCALVERSION_AUTO=y +CONFIG_SWAP=y +CONFIG_SYSVIPC=y +# CONFIG_POSIX_MQUEUE is not set +CONFIG_BSD_PROCESS_ACCT=y +# CONFIG_BSD_PROCESS_ACCT_V3 is not set +CONFIG_SYSCTL=y +# CONFIG_AUDIT is not set +# CONFIG_IKCONFIG is not set +CONFIG_INITRAMFS_SOURCE="" +CONFIG_UID16=y +CONFIG_CC_OPTIMIZE_FOR_SIZE=y +CONFIG_EMBEDDED=y +CONFIG_KALLSYMS=y +# CONFIG_KALLSYMS_ALL is not set +# CONFIG_KALLSYMS_EXTRA_PASS is not set +CONFIG_HOTPLUG=y +CONFIG_PRINTK=y +CONFIG_BUG=y +CONFIG_ELF_CORE=y +CONFIG_BASE_FULL=y +CONFIG_FUTEX=y +CONFIG_EPOLL=y +CONFIG_SHMEM=y +CONFIG_CC_ALIGN_FUNCTIONS=0 +CONFIG_CC_ALIGN_LABELS=0 +CONFIG_CC_ALIGN_LOOPS=0 +CONFIG_CC_ALIGN_JUMPS=0 +CONFIG_SLAB=y +# CONFIG_TINY_SHMEM is not set +CONFIG_BASE_SMALL=0 +# CONFIG_SLOB is not set + +# +# Loadable module support +# +CONFIG_MODULES=y +CONFIG_MODULE_UNLOAD=y +CONFIG_MODULE_FORCE_UNLOAD=y +CONFIG_OBSOLETE_MODPARM=y +# CONFIG_MODVERSIONS is not set +# CONFIG_MODULE_SRCVERSION_ALL is not set +CONFIG_KMOD=y + +# +# Block layer +# + +# +# IO Schedulers +# +CONFIG_IOSCHED_NOOP=y +CONFIG_IOSCHED_AS=y +CONFIG_IOSCHED_DEADLINE=m +CONFIG_IOSCHED_CFQ=m +CONFIG_DEFAULT_AS=y +# CONFIG_DEFAULT_DEADLINE is not set +# CONFIG_DEFAULT_CFQ is not set +# CONFIG_DEFAULT_NOOP is not set +CONFIG_DEFAULT_IOSCHED="anticipatory" + +# +# System Type +# +# CONFIG_ARCH_CLPS7500 is not set +# CONFIG_ARCH_CLPS711X is not set +# CONFIG_ARCH_CO285 is not set +# CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_FOOTBRIDGE is not set +# CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_IOP3XX is not set +# CONFIG_ARCH_IXP4XX is not set +# CONFIG_ARCH_IXP2000 is not set +# CONFIG_ARCH_L7200 is not set +CONFIG_ARCH_PXA=y +# CONFIG_ARCH_RPC is not set +# CONFIG_ARCH_SA1100 is not set +# CONFIG_ARCH_S3C2410 is not set +# CONFIG_ARCH_SHARK is not set +# CONFIG_ARCH_LH7A40X is not set +# CONFIG_ARCH_OMAP is not set +# CONFIG_ARCH_VERSATILE is not set +# CONFIG_ARCH_REALVIEW is not set +# CONFIG_ARCH_IMX is not set +# CONFIG_ARCH_H720X is not set +# CONFIG_ARCH_AAEC2000 is not set +# CONFIG_ARCH_AT91RM9200 is not set + +# +# Intel PXA2xx Implementations +# +# CONFIG_ARCH_LUBBOCK is not set +# CONFIG_MACH_MAINSTONE is not set +# CONFIG_ARCH_PXA_IDP is not set +CONFIG_PXA_SHARPSL=y +# CONFIG_MACH_HX2750 is not set +CONFIG_PXA_SHARPSL_25x=y +# CONFIG_PXA_SHARPSL_27x is not set +# CONFIG_MACH_POODLE is not set +# CONFIG_MACH_CORGI is not set +# CONFIG_MACH_SHEPHERD is not set +# CONFIG_MACH_HUSKY is not set +CONFIG_MACH_TOSA=y +CONFIG_PXA25x=y +# CONFIG_PXA_KEYS is not set + +# +# Processor Type +# +CONFIG_CPU_32=y +CONFIG_CPU_XSCALE=y +CONFIG_CPU_32v5=y +CONFIG_CPU_ABRT_EV5T=y +CONFIG_CPU_CACHE_VIVT=y +CONFIG_CPU_TLB_V4WBI=y + +# +# Processor Features +# +CONFIG_ARM_THUMB=y +CONFIG_XSCALE_PMU=y +CONFIG_KEXEC=y +CONFIG_SHARP_PARAM=y +CONFIG_SHARPSL_PM=y +CONFIG_SHARP_SCOOP=y +CONFIG_TOSHIBA_TC6393XB=y + +# +# Bus support +# + +# +# PCCARD (PCMCIA/CardBus) support +# +CONFIG_PCCARD=y +# CONFIG_PCMCIA_DEBUG is not set +CONFIG_PCMCIA=y +CONFIG_PCMCIA_LOAD_CIS=y +CONFIG_PCMCIA_IOCTL=y + +# +# PC-card bridges +# +CONFIG_PCMCIA_PXA2XX=y + +# +# Kernel Features +# +CONFIG_PREEMPT=y +CONFIG_NO_IDLE_HZ=y +# CONFIG_AEABI is not set +# CONFIG_ARCH_DISCONTIGMEM_ENABLE is not set +CONFIG_SELECT_MEMORY_MODEL=y +CONFIG_FLATMEM_MANUAL=y +# CONFIG_DISCONTIGMEM_MANUAL is not set +# CONFIG_SPARSEMEM_MANUAL is not set +CONFIG_FLATMEM=y +CONFIG_FLAT_NODE_MEM_MAP=y +# CONFIG_SPARSEMEM_STATIC is not set +CONFIG_SPLIT_PTLOCK_CPUS=4096 +CONFIG_ALIGNMENT_TRAP=y + +# +# Boot options +# +CONFIG_ZBOOT_ROM_TEXT=0x0 +CONFIG_ZBOOT_ROM_BSS=0x0 +# CONFIG_XIP_KERNEL is not set + +# +# CPU Frequency scaling +# +CONFIG_CPU_FREQ=y +CONFIG_CPU_FREQ_TABLE=y +# CONFIG_CPU_FREQ_DEBUG is not set +CONFIG_CPU_FREQ_STAT=y +# CONFIG_CPU_FREQ_STAT_DETAILS is not set +CONFIG_CPU_FREQ_DEFAULT_GOV_PERFORMANCE=y +# CONFIG_CPU_FREQ_DEFAULT_GOV_USERSPACE is not set +CONFIG_CPU_FREQ_GOV_PERFORMANCE=y +CONFIG_CPU_FREQ_GOV_POWERSAVE=m +CONFIG_CPU_FREQ_GOV_USERSPACE=m +CONFIG_CPU_FREQ_GOV_ONDEMAND=m +CONFIG_CPU_FREQ_GOV_CONSERVATIVE=m +CONFIG_CPU_FREQ_PXA25x=y + +# +# Floating point emulation +# + +# +# At least one emulation must be selected +# +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_NWFPE_XP is not set +# CONFIG_FPE_FASTFPE is not set + +# +# Userspace binary formats +# +CONFIG_BINFMT_ELF=y +CONFIG_BINFMT_AOUT=m +CONFIG_BINFMT_MISC=m +# CONFIG_ARTHUR is not set + +# +# Power management options +# +CONFIG_PM=y +# CONFIG_PM_LEGACY is not set +# CONFIG_PM_DEBUG is not set +CONFIG_APM=y + +# +# Networking +# +CONFIG_NET=y + +# +# Networking options +# +# CONFIG_NETDEBUG is not set +CONFIG_PACKET=y +CONFIG_PACKET_MMAP=y +CONFIG_UNIX=y +CONFIG_XFRM=y +CONFIG_XFRM_USER=m +# CONFIG_NET_KEY is not set +CONFIG_INET=y +# CONFIG_IP_MULTICAST is not set +# CONFIG_IP_ADVANCED_ROUTER is not set +CONFIG_IP_FIB_HASH=y +# CONFIG_IP_PNP is not set +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set +CONFIG_SYN_COOKIES=y +# CONFIG_INET_AH is not set +# CONFIG_INET_ESP is not set +# CONFIG_INET_IPCOMP is not set +# CONFIG_INET_TUNNEL is not set +CONFIG_INET_DIAG=m +CONFIG_INET_TCP_DIAG=m +# CONFIG_TCP_CONG_ADVANCED is not set +CONFIG_TCP_CONG_BIC=y + +# +# IP: Virtual Server Configuration +# +# CONFIG_IP_VS is not set +CONFIG_IPV6=m +# CONFIG_IPV6_PRIVACY is not set +CONFIG_INET6_AH=m +CONFIG_INET6_ESP=m +CONFIG_INET6_IPCOMP=m +CONFIG_INET6_TUNNEL=m +CONFIG_IPV6_TUNNEL=m +CONFIG_NETFILTER=y +# CONFIG_NETFILTER_DEBUG is not set + +# +# Core Netfilter Configuration +# +# CONFIG_NETFILTER_NETLINK is not set +# CONFIG_NETFILTER_XTABLES is not set + +# +# IP: Netfilter Configuration +# +CONFIG_IP_NF_CONNTRACK=m +# CONFIG_IP_NF_CT_ACCT is not set +# CONFIG_IP_NF_CONNTRACK_MARK is not set +# CONFIG_IP_NF_CONNTRACK_EVENTS is not set +CONFIG_IP_NF_CT_PROTO_SCTP=m +CONFIG_IP_NF_FTP=m +CONFIG_IP_NF_IRC=m +# CONFIG_IP_NF_NETBIOS_NS is not set +CONFIG_IP_NF_TFTP=m +CONFIG_IP_NF_AMANDA=m +# CONFIG_IP_NF_PPTP is not set +CONFIG_IP_NF_QUEUE=m + +# +# IPv6: Netfilter Configuration (EXPERIMENTAL) +# +# CONFIG_IP6_NF_QUEUE is not set + +# +# DCCP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_DCCP is not set + +# +# SCTP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_SCTP is not set + +# +# TIPC Configuration (EXPERIMENTAL) +# +# CONFIG_TIPC is not set +# CONFIG_ATM is not set +# CONFIG_BRIDGE is not set +# CONFIG_VLAN_8021Q is not set +# CONFIG_DECNET is not set +# CONFIG_LLC2 is not set +# CONFIG_IPX is not set +# CONFIG_ATALK is not set +# CONFIG_X25 is not set +# CONFIG_LAPB is not set +# CONFIG_NET_DIVERT is not set +# CONFIG_ECONET is not set +# CONFIG_WAN_ROUTER is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network testing +# +# CONFIG_NET_PKTGEN is not set +# CONFIG_HAMRADIO is not set +CONFIG_IRDA=m + +# +# IrDA protocols +# +CONFIG_IRLAN=m +CONFIG_IRNET=m +CONFIG_IRCOMM=m +# CONFIG_IRDA_ULTRA is not set + +# +# IrDA options +# +# CONFIG_IRDA_CACHE_LAST_LSAP is not set +# CONFIG_IRDA_FAST_RR is not set +# CONFIG_IRDA_DEBUG is not set + +# +# Infrared-port device drivers +# + +# +# SIR device drivers +# +# CONFIG_IRTTY_SIR is not set + +# +# Dongle support +# + +# +# Old SIR device drivers +# +# CONFIG_IRPORT_SIR is not set + +# +# Old Serial dongle support +# + +# +# FIR device drivers +# +# CONFIG_USB_IRDA is not set +# CONFIG_SIGMATEL_FIR is not set +CONFIG_PXA_FICP=m +CONFIG_BT=m +CONFIG_BT_L2CAP=m +CONFIG_BT_SCO=m +CONFIG_BT_RFCOMM=m +CONFIG_BT_RFCOMM_TTY=y +CONFIG_BT_BNEP=m +CONFIG_BT_BNEP_MC_FILTER=y +CONFIG_BT_BNEP_PROTO_FILTER=y +CONFIG_BT_HIDP=m + +# +# Bluetooth device drivers +# +CONFIG_BT_HCIUSB=m +# CONFIG_BT_HCIUSB_SCO is not set +CONFIG_BT_HCIUART=m +CONFIG_BT_HCIUART_H4=y +CONFIG_BT_HCIUART_BCSP=y +CONFIG_BT_HCIBCM203X=m +CONFIG_BT_HCIBPA10X=m +CONFIG_BT_HCIBFUSB=m +CONFIG_BT_HCIDTL1=m +CONFIG_BT_HCIBT3C=m +CONFIG_BT_HCIBLUECARD=m +CONFIG_BT_HCIBTUART=m +CONFIG_BT_HCIVHCI=m +CONFIG_IEEE80211=m +# CONFIG_IEEE80211_DEBUG is not set +CONFIG_IEEE80211_CRYPT_WEP=m +CONFIG_IEEE80211_CRYPT_CCMP=m +CONFIG_IEEE80211_CRYPT_TKIP=m + +# +# Device Drivers +# + +# +# Generic Driver Options +# +CONFIG_STANDALONE=y +CONFIG_PREVENT_FIRMWARE_BUILD=y +CONFIG_FW_LOADER=y +# CONFIG_DEBUG_DRIVER is not set + +# +# Connector - unified userspace <-> kernelspace linker +# +# CONFIG_CONNECTOR is not set + +# +# Memory Technology Devices (MTD) +# +CONFIG_MTD=y +# CONFIG_MTD_DEBUG is not set +# CONFIG_MTD_CONCAT is not set +CONFIG_MTD_PARTITIONS=y +# CONFIG_MTD_REDBOOT_PARTS is not set +# CONFIG_MTD_CMDLINE_PARTS is not set +# CONFIG_MTD_AFS_PARTS is not set + +# +# User Modules And Translation Layers +# +CONFIG_MTD_CHAR=y +CONFIG_MTD_BLOCK=y +# CONFIG_FTL is not set +# CONFIG_NFTL is not set +# CONFIG_INFTL is not set +# CONFIG_RFD_FTL is not set + +# +# RAM/ROM/Flash chip drivers +# +# CONFIG_MTD_CFI is not set +# CONFIG_MTD_JEDECPROBE is not set +CONFIG_MTD_MAP_BANK_WIDTH_1=y +CONFIG_MTD_MAP_BANK_WIDTH_2=y +CONFIG_MTD_MAP_BANK_WIDTH_4=y +# CONFIG_MTD_MAP_BANK_WIDTH_8 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_16 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_32 is not set +CONFIG_MTD_CFI_I1=y +CONFIG_MTD_CFI_I2=y +# CONFIG_MTD_CFI_I4 is not set +# CONFIG_MTD_CFI_I8 is not set +# CONFIG_MTD_RAM is not set +CONFIG_MTD_ROM=y +# CONFIG_MTD_ABSENT is not set +# CONFIG_MTD_OBSOLETE_CHIPS is not set + +# +# Mapping drivers for chip access +# +# CONFIG_MTD_COMPLEX_MAPPINGS is not set +CONFIG_MTD_SHARP_SL=y +# CONFIG_MTD_PLATRAM is not set + +# +# Self-contained MTD device drivers +# +# CONFIG_MTD_SLRAM is not set +# CONFIG_MTD_PHRAM is not set +# CONFIG_MTD_MTDRAM is not set +# CONFIG_MTD_BLKMTD is not set +# CONFIG_MTD_BLOCK2MTD is not set + +# +# Disk-On-Chip Device Drivers +# +# CONFIG_MTD_DOC2000 is not set +# CONFIG_MTD_DOC2001 is not set +# CONFIG_MTD_DOC2001PLUS is not set + +# +# NAND Flash Device Drivers +# +CONFIG_MTD_NAND=y +CONFIG_MTD_NAND_VERIFY_WRITE=y +# CONFIG_MTD_NAND_H1900 is not set +CONFIG_MTD_NAND_TMIO=y +CONFIG_MTD_NAND_IDS=y +# CONFIG_MTD_NAND_DISKONCHIP is not set +# CONFIG_MTD_NAND_SHARPSL is not set +# CONFIG_MTD_NAND_NANDSIM is not set + +# +# OneNAND Flash Device Drivers +# +# CONFIG_MTD_ONENAND is not set + +# +# Parallel port support +# +# CONFIG_PARPORT is not set + +# +# Plug and Play support +# + +# +# Block devices +# +# CONFIG_BLK_DEV_COW_COMMON is not set +CONFIG_BLK_DEV_LOOP=m +# CONFIG_BLK_DEV_CRYPTOLOOP is not set +# CONFIG_BLK_DEV_NBD is not set +# CONFIG_BLK_DEV_UB is not set +# CONFIG_BLK_DEV_RAM is not set +CONFIG_BLK_DEV_RAM_COUNT=16 +# CONFIG_CDROM_PKTCDVD is not set +# CONFIG_ATA_OVER_ETH is not set + +# +# ATA/ATAPI/MFM/RLL support +# +CONFIG_IDE=y +CONFIG_BLK_DEV_IDE=y + +# +# Please see Documentation/ide.txt for help/info on IDE drives +# +# CONFIG_BLK_DEV_IDE_SATA is not set +CONFIG_BLK_DEV_IDEDISK=y +# CONFIG_IDEDISK_MULTI_MODE is not set +CONFIG_BLK_DEV_IDECS=y +# CONFIG_BLK_DEV_IDECD is not set +# CONFIG_BLK_DEV_IDETAPE is not set +# CONFIG_BLK_DEV_IDEFLOPPY is not set +# CONFIG_BLK_DEV_IDESCSI is not set +# CONFIG_IDE_TASK_IOCTL is not set + +# +# IDE chipset support/bugfixes +# +# CONFIG_IDE_GENERIC is not set +# CONFIG_IDE_ARM is not set +# CONFIG_BLK_DEV_IDEDMA is not set +# CONFIG_IDEDMA_AUTO is not set +# CONFIG_BLK_DEV_HD is not set + +# +# SCSI device support +# +# CONFIG_RAID_ATTRS is not set +CONFIG_SCSI=m +CONFIG_SCSI_PROC_FS=y + +# +# SCSI support type (disk, tape, CD-ROM) +# +CONFIG_BLK_DEV_SD=m +CONFIG_CHR_DEV_ST=m +CONFIG_CHR_DEV_OSST=m +CONFIG_BLK_DEV_SR=m +# CONFIG_BLK_DEV_SR_VENDOR is not set +CONFIG_CHR_DEV_SG=m +# CONFIG_CHR_DEV_SCH is not set + +# +# Some SCSI devices (e.g. CD jukebox) support multiple LUNs +# +CONFIG_SCSI_MULTI_LUN=y +# CONFIG_SCSI_CONSTANTS is not set +# CONFIG_SCSI_LOGGING is not set + +# +# SCSI Transport Attributes +# +# CONFIG_SCSI_SPI_ATTRS is not set +# CONFIG_SCSI_FC_ATTRS is not set +# CONFIG_SCSI_ISCSI_ATTRS is not set +# CONFIG_SCSI_SAS_ATTRS is not set + +# +# SCSI low-level drivers +# +# CONFIG_ISCSI_TCP is not set +# CONFIG_SCSI_SATA is not set +# CONFIG_SCSI_DEBUG is not set + +# +# PCMCIA SCSI adapter support +# +# CONFIG_PCMCIA_AHA152X is not set +# CONFIG_PCMCIA_FDOMAIN is not set +# CONFIG_PCMCIA_NINJA_SCSI is not set +# CONFIG_PCMCIA_QLOGIC is not set +# CONFIG_PCMCIA_SYM53C500 is not set + +# +# Multi-device support (RAID and LVM) +# +CONFIG_MD=y +CONFIG_BLK_DEV_DM=m +CONFIG_DM_CRYPT=m +CONFIG_DM_SNAPSHOT=m +CONFIG_DM_MIRROR=m +CONFIG_DM_ZERO=m +CONFIG_DM_MULTIPATH=m +CONFIG_DM_MULTIPATH_EMC=m + +# +# Fusion MPT device support +# +# CONFIG_FUSION is not set + +# +# IEEE 1394 (FireWire) support +# + +# +# I2O device support +# + +# +# Network device support +# +CONFIG_NETDEVICES=y +# CONFIG_DUMMY is not set +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +CONFIG_TUN=m + +# +# PHY device support +# +# CONFIG_PHYLIB is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +CONFIG_MII=m +# CONFIG_SMC91X is not set +# CONFIG_DM9000 is not set + +# +# Ethernet (1000 Mbit) +# + +# +# Ethernet (10000 Mbit) +# + +# +# Token Ring devices +# + +# +# Wireless LAN (non-hamradio) +# +CONFIG_NET_RADIO=y + +# +# Obsolete Wireless cards support (pre-802.11) +# +# CONFIG_STRIP is not set +# CONFIG_PCMCIA_WAVELAN is not set +# CONFIG_PCMCIA_NETWAVE is not set + +# +# Wireless 802.11 Frequency Hopping cards support +# +# CONFIG_PCMCIA_RAYCS is not set + +# +# Wireless 802.11b ISA/PCI cards support +# +CONFIG_HERMES=m +# CONFIG_ATMEL is not set + +# +# Wireless 802.11b Pcmcia/Cardbus cards support +# +CONFIG_PCMCIA_HERMES=m +CONFIG_PCMCIA_SPECTRUM=m +# CONFIG_AIRO_CS is not set +# CONFIG_PCMCIA_WL3501 is not set +CONFIG_HOSTAP=m +CONFIG_HOSTAP_FIRMWARE=y +# CONFIG_HOSTAP_FIRMWARE_NVRAM is not set +CONFIG_HOSTAP_CS=m +CONFIG_NET_WIRELESS=y + +# +# PCMCIA network device support +# +CONFIG_NET_PCMCIA=y +# CONFIG_PCMCIA_3C589 is not set +# CONFIG_PCMCIA_3C574 is not set +# CONFIG_PCMCIA_FMVJ18X is not set +CONFIG_PCMCIA_PCNET=m +# CONFIG_PCMCIA_NMCLAN is not set +# CONFIG_PCMCIA_SMC91C92 is not set +# CONFIG_PCMCIA_XIRC2PS is not set +# CONFIG_PCMCIA_AXNET is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set +CONFIG_PPP=m +# CONFIG_PPP_MULTILINK is not set +# CONFIG_PPP_FILTER is not set +CONFIG_PPP_ASYNC=m +# CONFIG_PPP_SYNC_TTY is not set +CONFIG_PPP_DEFLATE=m +CONFIG_PPP_BSDCOMP=m +# CONFIG_PPP_MPPE is not set +# CONFIG_PPPOE is not set +# CONFIG_SLIP is not set +# CONFIG_SHAPER is not set +# CONFIG_NETCONSOLE is not set +# CONFIG_NETPOLL is not set +# CONFIG_NET_POLL_CONTROLLER is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Input device support +# +CONFIG_INPUT=y + +# +# Userland interfaces +# +CONFIG_INPUT_MOUSEDEV=m +# CONFIG_INPUT_MOUSEDEV_PSAUX is not set +CONFIG_INPUT_MOUSEDEV_SCREEN_X=480 +CONFIG_INPUT_MOUSEDEV_SCREEN_Y=640 +# CONFIG_INPUT_JOYDEV is not set +# CONFIG_INPUT_TSDEV is not set +CONFIG_INPUT_EVDEV=y +# CONFIG_INPUT_EVBUG is not set +CONFIG_INPUT_POWER=y + +# +# Input Device Drivers +# +CONFIG_INPUT_KEYBOARD=y +# CONFIG_KEYBOARD_ATKBD is not set +# CONFIG_KEYBOARD_SUNKBD is not set +# CONFIG_KEYBOARD_LKKBD is not set +# CONFIG_KEYBOARD_XTKBD is not set +# CONFIG_KEYBOARD_NEWTON is not set +# CONFIG_KEYBOARD_CORGI is not set +# CONFIG_KEYBOARD_SPITZ is not set +CONFIG_KEYBOARD_TOSA=y +# CONFIG_INPUT_MOUSE is not set +# CONFIG_INPUT_JOYSTICK is not set +CONFIG_INPUT_TOUCHSCREEN=y +# CONFIG_TOUCHSCREEN_CORGI is not set +# CONFIG_TOUCHSCREEN_GUNZE is not set +# CONFIG_TOUCHSCREEN_ELO is not set +# CONFIG_TOUCHSCREEN_MTOUCH is not set +# CONFIG_TOUCHSCREEN_MK712 is not set +CONFIG_TOUCHSCREEN_WM97XX=y +# CONFIG_TOUCHSCREEN_WM9705 is not set +CONFIG_TOUCHSCREEN_WM9712=y +# CONFIG_TOUCHSCREEN_WM9713 is not set +# CONFIG_TOUCHSCREEN_WM97XX_PXA is not set +# CONFIG_INPUT_MISC is not set + +# +# Hardware I/O ports +# +# CONFIG_SERIO is not set +# CONFIG_GAMEPORT is not set + +# +# Character devices +# +CONFIG_VT=y +CONFIG_VT_CONSOLE=y +CONFIG_HW_CONSOLE=y +# CONFIG_SERIAL_NONSTANDARD is not set + +# +# Serial drivers +# +CONFIG_SERIAL_8250=m +CONFIG_SERIAL_8250_CS=m +CONFIG_SERIAL_8250_NR_UARTS=4 +CONFIG_SERIAL_8250_RUNTIME_UARTS=4 +# CONFIG_SERIAL_8250_EXTENDED is not set + +# +# Non-8250 serial port support +# +CONFIG_SERIAL_PXA=y +CONFIG_SERIAL_PXA_CONSOLE=y +CONFIG_SERIAL_CORE=y +CONFIG_SERIAL_CORE_CONSOLE=y +CONFIG_UNIX98_PTYS=y +# CONFIG_LEGACY_PTYS is not set + +# +# IPMI +# +# CONFIG_IPMI_HANDLER is not set + +# +# Watchdog Cards +# +# CONFIG_WATCHDOG is not set +# CONFIG_NVRAM is not set +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set + +# +# Ftape, the floppy tape device driver +# + +# +# PCMCIA character devices +# +# CONFIG_SYNCLINK_CS is not set +# CONFIG_CARDMAN_4000 is not set +# CONFIG_CARDMAN_4040 is not set +# CONFIG_RAW_DRIVER is not set + +# +# TPM devices +# +# CONFIG_TCG_TPM is not set +# CONFIG_TELCLOCK is not set + +# +# I2C support +# +CONFIG_I2C=y +# CONFIG_I2C_CHARDEV is not set + +# +# I2C Algorithms +# +# CONFIG_I2C_ALGOBIT is not set +# CONFIG_I2C_ALGOPCF is not set +# CONFIG_I2C_ALGOPCA is not set + +# +# I2C Hardware Bus support +# +CONFIG_I2C_PXA=y +# CONFIG_I2C_PXA_SLAVE is not set +# CONFIG_I2C_PARPORT_LIGHT is not set +# CONFIG_I2C_STUB is not set +# CONFIG_I2C_PCA_ISA is not set + +# +# Miscellaneous I2C Chip support +# +# CONFIG_SENSORS_DS1337 is not set +# CONFIG_SENSORS_DS1374 is not set +# CONFIG_SENSORS_EEPROM is not set +# CONFIG_SENSORS_PCF8574 is not set +# CONFIG_SENSORS_PCA9539 is not set +# CONFIG_SENSORS_PCF8591 is not set +# CONFIG_SENSORS_MAX6875 is not set +# CONFIG_I2C_DEBUG_CORE is not set +# CONFIG_I2C_DEBUG_ALGO is not set +# CONFIG_I2C_DEBUG_BUS is not set +# CONFIG_I2C_DEBUG_CHIP is not set + +# +# SPI support +# +# CONFIG_SPI is not set +# CONFIG_SPI_MASTER is not set + +# +# Hardware Monitoring support +# +# CONFIG_HWMON is not set +# CONFIG_HWMON_VID is not set + +# +# Misc devices +# + +# +# Multimedia Capabilities Port drivers +# + +# +# Multi-Function Devices +# + +# +# LED devices +# +CONFIG_NEW_LEDS=y +CONFIG_LEDS_CLASS=y +CONFIG_LEDS_TRIGGERS=y +CONFIG_LEDS_TOSA=y +CONFIG_LEDS_TRIGGER_TIMER=m +CONFIG_LEDS_TRIGGER_IDE_DISK=y + +# +# Multimedia devices +# +CONFIG_VIDEO_DEV=m + +# +# Video For Linux +# + +# +# Video Adapters +# +# CONFIG_VIDEO_ADV_DEBUG is not set +# CONFIG_VIDEO_CPIA is not set +# CONFIG_VIDEO_SAA5246A is not set +# CONFIG_VIDEO_SAA5249 is not set +# CONFIG_TUNER_3036 is not set +# CONFIG_VIDEO_EM28XX is not set +# CONFIG_VIDEO_OVCAMCHIP is not set +# CONFIG_VIDEO_AUDIO_DECODER is not set +# CONFIG_VIDEO_DECODER is not set + +# +# Radio Adapters +# +# CONFIG_RADIO_MAESTRO is not set + +# +# Digital Video Broadcasting Devices +# +# CONFIG_DVB is not set + +# +# Graphics support +# +CONFIG_FB=y +CONFIG_FB_CFB_FILLRECT=y +CONFIG_FB_CFB_COPYAREA=y +CONFIG_FB_CFB_IMAGEBLIT=y +# CONFIG_FB_MACMODES is not set +# CONFIG_FB_MODE_HELPERS is not set +# CONFIG_FB_TILEBLITTING is not set +# CONFIG_FB_S1D13XXX is not set +# CONFIG_FB_PXA is not set +# CONFIG_FB_W100 is not set +CONFIG_FB_TMIO=y +# CONFIG_FB_VIRTUAL is not set + +# +# Console display driver support +# +# CONFIG_VGA_CONSOLE is not set +CONFIG_DUMMY_CONSOLE=y +CONFIG_FRAMEBUFFER_CONSOLE=y +# CONFIG_FRAMEBUFFER_CONSOLE_ROTATION is not set +CONFIG_FONTS=y +CONFIG_FONT_8x8=y +# CONFIG_FONT_8x16 is not set +# CONFIG_FONT_6x11 is not set +# CONFIG_FONT_7x14 is not set +# CONFIG_FONT_PEARL_8x8 is not set +# CONFIG_FONT_ACORN_8x8 is not set +# CONFIG_FONT_MINI_4x6 is not set +# CONFIG_FONT_SUN8x16 is not set +# CONFIG_FONT_SUN12x22 is not set +# CONFIG_FONT_10x18 is not set + +# +# Logo configuration +# +CONFIG_LOGO=y +# CONFIG_LOGO_LINUX_MONO is not set +# CONFIG_LOGO_LINUX_VGA16 is not set +CONFIG_LOGO_LINUX_CLUT224=y +CONFIG_BACKLIGHT_LCD_SUPPORT=y +CONFIG_BACKLIGHT_CLASS_DEVICE=y +CONFIG_BACKLIGHT_DEVICE=y +# CONFIG_LCD_CLASS_DEVICE is not set +CONFIG_BACKLIGHT_CORGI=y +# CONFIG_BACKLIGHT_HP680 is not set + +# +# Sound +# +CONFIG_SOUND=y + +# +# Advanced Linux Sound Architecture +# +CONFIG_SND=y +CONFIG_SND_TIMER=y +CONFIG_SND_PCM=y +CONFIG_SND_HWDEP=m +CONFIG_SND_RAWMIDI=m +# CONFIG_SND_SEQUENCER is not set +CONFIG_SND_OSSEMUL=y +CONFIG_SND_MIXER_OSS=m +CONFIG_SND_PCM_OSS=m +# CONFIG_SND_DYNAMIC_MINORS is not set +CONFIG_SND_SUPPORT_OLD_API=y +# CONFIG_SND_VERBOSE_PRINTK is not set +# CONFIG_SND_DEBUG is not set + +# +# Generic devices +# +CONFIG_SND_AC97_BUS=y +CONFIG_SND_DUMMY=m +# CONFIG_SND_MTPAV is not set +# CONFIG_SND_SERIAL_U16550 is not set +# CONFIG_SND_MPU401 is not set + +# +# ALSA ARM devices +# +# CONFIG_SND_PXA2XX_AC97 is not set + +# +# USB devices +# +CONFIG_SND_USB_AUDIO=m + +# +# PCMCIA devices +# + +# +# SoC audio support +# +CONFIG_SND_SOC=y + +# +# Soc Platforms +# + +# +# SoC Audio for the Intel PXA2xx +# +CONFIG_SND_PXA2xx_SOC=y +CONFIG_SND_PXA2xx_SOC_AC97=y +# CONFIG_SND_PXA2xx_SOC_MAINSTONE is not set +# CONFIG_SND_PXA2xx_SOC_MAINSTONE_WM8753 is not set +# CONFIG_SND_PXA2xx_SOC_MAINSTONE_WM9713 is not set +# CONFIG_SND_PXA2xx_SOC_MAINSTONE_WM9712 is not set +# CONFIG_SND_PXA2xx_SOC_CORGI is not set +# CONFIG_SND_PXA2xx_SOC_SPITZ is not set +CONFIG_SND_PXA2xx_SOC_TOSA=y + +# +# Soc Codecs +# +# CONFIG_SND_SOC_AC97_CODEC is not set +# CONFIG_SND_SOC_WM8731 is not set +# CONFIG_SND_SOC_WM8750 is not set +# CONFIG_SND_SOC_WM8753 is not set +# CONFIG_SND_SOC_WM8772 is not set +# CONFIG_SND_SOC_WM8971 is not set +# CONFIG_SND_SOC_WM9713 is not set +CONFIG_SND_SOC_WM9712=y +# CONFIG_SND_SOC_UDA1380 is not set +# CONFIG_SND_SOC_AK4535 is not set + +# +# Open Sound System +# +# CONFIG_SOUND_PRIME is not set + +# +# USB support +# +CONFIG_USB_ARCH_HAS_HCD=y +CONFIG_USB_ARCH_HAS_OHCI=y +CONFIG_USB=m +# CONFIG_USB_DEBUG is not set + +# +# Miscellaneous USB options +# +CONFIG_USB_DEVICEFS=y +# CONFIG_USB_BANDWIDTH is not set +# CONFIG_USB_DYNAMIC_MINORS is not set +# CONFIG_USB_SUSPEND is not set +# CONFIG_USB_OTG is not set + +# +# USB Host Controller Drivers +# +# CONFIG_USB_ISP116X_HCD is not set +CONFIG_USB_OHCI_HCD=m +# CONFIG_USB_OHCI_BIG_ENDIAN is not set +CONFIG_USB_OHCI_LITTLE_ENDIAN=y +CONFIG_USB_SL811_HCD=m +CONFIG_USB_SL811_CS=m + +# +# USB Device Class drivers +# +# CONFIG_OBSOLETE_OSS_USB_DRIVER is not set +CONFIG_USB_ACM=m +CONFIG_USB_PRINTER=m + +# +# NOTE: USB_STORAGE enables SCSI, and 'SCSI disk support' +# + +# +# may also be needed; see USB_STORAGE Help for more information +# +CONFIG_USB_STORAGE=m +# CONFIG_USB_STORAGE_DEBUG is not set +# CONFIG_USB_STORAGE_DATAFAB is not set +# CONFIG_USB_STORAGE_FREECOM is not set +# CONFIG_USB_STORAGE_ISD200 is not set +# CONFIG_USB_STORAGE_DPCM is not set +# CONFIG_USB_STORAGE_USBAT is not set +# CONFIG_USB_STORAGE_SDDR09 is not set +# CONFIG_USB_STORAGE_SDDR55 is not set +# CONFIG_USB_STORAGE_JUMPSHOT is not set +# CONFIG_USB_STORAGE_ALAUDA is not set +# CONFIG_USB_LIBUSUAL is not set + +# +# USB Input Devices +# +CONFIG_USB_HID=m +CONFIG_USB_HIDINPUT=y +# CONFIG_USB_HIDINPUT_POWERBOOK is not set +# CONFIG_HID_FF is not set +# CONFIG_USB_HIDDEV is not set + +# +# USB HID Boot Protocol drivers +# +CONFIG_USB_KBD=m +CONFIG_USB_MOUSE=m +CONFIG_USB_AIPTEK=m +CONFIG_USB_WACOM=m +# CONFIG_USB_ACECAD is not set +CONFIG_USB_KBTAB=m +CONFIG_USB_POWERMATE=m +CONFIG_USB_MTOUCH=m +# CONFIG_USB_ITMTOUCH is not set +CONFIG_USB_EGALAX=m +# CONFIG_USB_YEALINK is not set +CONFIG_USB_XPAD=m +CONFIG_USB_ATI_REMOTE=m +# CONFIG_USB_ATI_REMOTE2 is not set +# CONFIG_USB_KEYSPAN_REMOTE is not set +# CONFIG_USB_APPLETOUCH is not set + +# +# USB Imaging devices +# +CONFIG_USB_MDC800=m +CONFIG_USB_MICROTEK=m + +# +# USB Multimedia devices +# +CONFIG_USB_DABUSB=m +CONFIG_USB_VICAM=m +CONFIG_USB_DSBR=m +# CONFIG_USB_ET61X251 is not set +CONFIG_USB_IBMCAM=m +CONFIG_USB_KONICAWC=m +CONFIG_USB_OV511=m +CONFIG_USB_SE401=m +CONFIG_USB_SN9C102=m +CONFIG_USB_STV680=m +# CONFIG_USB_PWC is not set + +# +# USB Network Adapters +# +CONFIG_USB_CATC=m +CONFIG_USB_KAWETH=m +CONFIG_USB_PEGASUS=m +CONFIG_USB_RTL8150=m +CONFIG_USB_USBNET=m +CONFIG_USB_NET_AX8817X=m +CONFIG_USB_NET_CDCETHER=m +CONFIG_USB_NET_GL620A=m +CONFIG_USB_NET_NET1080=m +CONFIG_USB_NET_PLUSB=m +# CONFIG_USB_NET_RNDIS_HOST is not set +# CONFIG_USB_NET_CDC_SUBSET is not set +# CONFIG_USB_NET_ZAURUS is not set +# CONFIG_USB_ZD1201 is not set +CONFIG_USB_MON=y + +# +# USB port drivers +# + +# +# USB Serial Converter support +# +CONFIG_USB_SERIAL=m +CONFIG_USB_SERIAL_GENERIC=y +# CONFIG_USB_SERIAL_AIRPRIME is not set +# CONFIG_USB_SERIAL_ANYDATA is not set +CONFIG_USB_SERIAL_BELKIN=m +# CONFIG_USB_SERIAL_WHITEHEAT is not set +CONFIG_USB_SERIAL_DIGI_ACCELEPORT=m +# CONFIG_USB_SERIAL_CP2101 is not set +CONFIG_USB_SERIAL_CYPRESS_M8=m +CONFIG_USB_SERIAL_EMPEG=m +CONFIG_USB_SERIAL_FTDI_SIO=m +CONFIG_USB_SERIAL_VISOR=m +CONFIG_USB_SERIAL_IPAQ=m +CONFIG_USB_SERIAL_IR=m +CONFIG_USB_SERIAL_EDGEPORT=m +CONFIG_USB_SERIAL_EDGEPORT_TI=m +CONFIG_USB_SERIAL_GARMIN=m +CONFIG_USB_SERIAL_IPW=m +CONFIG_USB_SERIAL_KEYSPAN_PDA=m +CONFIG_USB_SERIAL_KEYSPAN=m +# CONFIG_USB_SERIAL_KEYSPAN_MPR is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XA is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XB is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA18X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19QW is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19QI is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49WLC is not set +CONFIG_USB_SERIAL_KLSI=m +CONFIG_USB_SERIAL_KOBIL_SCT=m +CONFIG_USB_SERIAL_MCT_U232=m +CONFIG_USB_SERIAL_PL2303=m +# CONFIG_USB_SERIAL_HP4X is not set +CONFIG_USB_SERIAL_SAFE=m +# CONFIG_USB_SERIAL_SAFE_PADDED is not set +CONFIG_USB_SERIAL_TI=m +CONFIG_USB_SERIAL_CYBERJACK=m +CONFIG_USB_SERIAL_XIRCOM=m +# CONFIG_USB_SERIAL_OPTION is not set +CONFIG_USB_SERIAL_OMNINET=m +CONFIG_USB_EZUSB=y + +# +# USB Miscellaneous drivers +# +CONFIG_USB_EMI62=m +CONFIG_USB_EMI26=m +CONFIG_USB_AUERSWALD=m +CONFIG_USB_RIO500=m +CONFIG_USB_LEGOTOWER=m +CONFIG_USB_LCD=m +CONFIG_USB_LED=m +CONFIG_USB_CYTHERM=m +CONFIG_USB_PHIDGETKIT=m +CONFIG_USB_PHIDGETSERVO=m +CONFIG_USB_IDMOUSE=m +# CONFIG_USB_LD is not set +# CONFIG_USB_TEST is not set + +# +# USB DSL modem support +# + +# +# USB Gadget Support +# +CONFIG_USB_GADGET=y +# CONFIG_USB_GADGET_DEBUG_FILES is not set +CONFIG_USB_GADGET_SELECTED=y +# CONFIG_USB_GADGET_NET2280 is not set +CONFIG_USB_GADGET_PXA2XX=y +CONFIG_USB_PXA2XX=y +# CONFIG_USB_PXA2XX_SMALL is not set +# CONFIG_USB_GADGET_PXA27X is not set +# CONFIG_USB_GADGET_GOKU is not set +# CONFIG_USB_GADGET_LH7A40X is not set +# CONFIG_USB_GADGET_OMAP is not set +# CONFIG_USB_GADGET_DUMMY_HCD is not set +# CONFIG_USB_GADGET_DUALSPEED is not set +CONFIG_USB_ZERO=m +CONFIG_USB_ETH=m +CONFIG_USB_ETH_RNDIS=y +CONFIG_USB_GADGETFS=m +CONFIG_USB_FILE_STORAGE=m +# CONFIG_USB_FILE_STORAGE_TEST is not set +CONFIG_USB_G_SERIAL=m + +# +# MMC/SD Card support +# +CONFIG_MMC=y +# CONFIG_MMC_DEBUG is not set +CONFIG_MMC_BLOCK=y +CONFIG_MMC_PXA=y + +# +# Real Time Clock +# +CONFIG_RTC_CLASS=y +CONFIG_RTC_HCTOSYS=y +CONFIG_RTC_HCTOSYS_DEVICE="rtc0" + +# +# RTC interfaces +# +CONFIG_RTC_INTF_SYSFS=y +CONFIG_RTC_INTF_PROC=y +CONFIG_RTC_INTF_DEV=y + +# +# RTC drivers +# +# CONFIG_RTC_DRV_X1205 is not set +# CONFIG_RTC_DRV_DS1672 is not set +# CONFIG_RTC_DRV_PCF8563 is not set +# CONFIG_RTC_DRV_RS5C372 is not set +CONFIG_RTC_DRV_SA1100=y +# CONFIG_RTC_DRV_TEST is not set + +# +# File systems +# +CONFIG_EXT2_FS=y +# CONFIG_EXT2_FS_XATTR is not set +# CONFIG_EXT2_FS_XIP is not set +# CONFIG_EXT3_FS is not set +# CONFIG_REISERFS_FS is not set +# CONFIG_JFS_FS is not set +# CONFIG_FS_POSIX_ACL is not set +# CONFIG_XFS_FS is not set +# CONFIG_OCFS2_FS is not set +# CONFIG_MINIX_FS is not set +# CONFIG_ROMFS_FS is not set +CONFIG_INOTIFY=y +# CONFIG_QUOTA is not set +CONFIG_DNOTIFY=y +# CONFIG_AUTOFS_FS is not set +# CONFIG_AUTOFS4_FS is not set +CONFIG_FUSE_FS=m + +# +# CD-ROM/DVD Filesystems +# +# CONFIG_ISO9660_FS is not set +# CONFIG_UDF_FS is not set + +# +# DOS/FAT/NT Filesystems +# +CONFIG_FAT_FS=m +CONFIG_MSDOS_FS=m +CONFIG_VFAT_FS=m +CONFIG_FAT_DEFAULT_CODEPAGE=437 +CONFIG_FAT_DEFAULT_IOCHARSET="iso8859-1" +# CONFIG_NTFS_FS is not set + +# +# Pseudo filesystems +# +CONFIG_PROC_FS=y +CONFIG_SYSFS=y +CONFIG_TMPFS=y +# CONFIG_HUGETLB_PAGE is not set +CONFIG_RAMFS=y +# CONFIG_RELAYFS_FS is not set +# CONFIG_CONFIGFS_FS is not set + +# +# Miscellaneous filesystems +# +# CONFIG_ADFS_FS is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_HFSPLUS_FS is not set +# CONFIG_BEFS_FS is not set +# CONFIG_BFS_FS is not set +# CONFIG_EFS_FS is not set +# CONFIG_JFFS_FS is not set +CONFIG_JFFS2_FS=y +CONFIG_JFFS2_FS_DEBUG=0 +CONFIG_JFFS2_FS_WRITEBUFFER=y +CONFIG_JFFS2_SUMMARY=y +CONFIG_JFFS2_COMPRESSION_OPTIONS=y +CONFIG_JFFS2_ZLIB=y +CONFIG_JFFS2_RTIME=y +CONFIG_JFFS2_RUBIN=y +# CONFIG_JFFS2_CMODE_NONE is not set +CONFIG_JFFS2_CMODE_PRIORITY=y +# CONFIG_JFFS2_CMODE_SIZE is not set +CONFIG_CRAMFS=m +CONFIG_SQUASHFS=m +# CONFIG_SQUASHFS_EMBEDDED is not set +CONFIG_SQUASHFS_FRAGMENT_CACHE_SIZE=3 +# CONFIG_SQUASHFS_VMALLOC is not set +# CONFIG_VXFS_FS is not set +# CONFIG_HPFS_FS is not set +# CONFIG_QNX4FS_FS is not set +# CONFIG_SYSV_FS is not set +# CONFIG_UFS_FS is not set + +# +# Network File Systems +# +CONFIG_NFS_FS=m +CONFIG_NFS_V3=y +# CONFIG_NFS_V3_ACL is not set +CONFIG_NFS_V4=y +# CONFIG_NFS_DIRECTIO is not set +# CONFIG_NFSD is not set +CONFIG_LOCKD=m +CONFIG_LOCKD_V4=y +CONFIG_NFS_COMMON=y +CONFIG_SUNRPC=m +CONFIG_SUNRPC_GSS=m +CONFIG_RPCSEC_GSS_KRB5=m +# CONFIG_RPCSEC_GSS_SPKM3 is not set +CONFIG_SMB_FS=m +CONFIG_SMB_NLS_DEFAULT=y +CONFIG_SMB_NLS_REMOTE="cp437" +CONFIG_CIFS=m +# CONFIG_CIFS_STATS is not set +# CONFIG_CIFS_XATTR is not set +# CONFIG_CIFS_EXPERIMENTAL is not set +# CONFIG_NCP_FS is not set +# CONFIG_CODA_FS is not set +# CONFIG_AFS_FS is not set +# CONFIG_9P_FS is not set + +# +# Partition Types +# +CONFIG_PARTITION_ADVANCED=y +# CONFIG_ACORN_PARTITION is not set +# CONFIG_OSF_PARTITION is not set +# CONFIG_AMIGA_PARTITION is not set +# CONFIG_ATARI_PARTITION is not set +# CONFIG_MAC_PARTITION is not set +CONFIG_MSDOS_PARTITION=y +# CONFIG_BSD_DISKLABEL is not set +# CONFIG_MINIX_SUBPARTITION is not set +# CONFIG_SOLARIS_X86_PARTITION is not set +# CONFIG_UNIXWARE_DISKLABEL is not set +# CONFIG_LDM_PARTITION is not set +# CONFIG_SGI_PARTITION is not set +# CONFIG_ULTRIX_PARTITION is not set +# CONFIG_SUN_PARTITION is not set +# CONFIG_KARMA_PARTITION is not set +# CONFIG_EFI_PARTITION is not set + +# +# Native Language Support +# +CONFIG_NLS=y +CONFIG_NLS_DEFAULT="cp437" +CONFIG_NLS_CODEPAGE_437=y +CONFIG_NLS_CODEPAGE_737=m +CONFIG_NLS_CODEPAGE_775=m +CONFIG_NLS_CODEPAGE_850=m +CONFIG_NLS_CODEPAGE_852=m +CONFIG_NLS_CODEPAGE_855=m +CONFIG_NLS_CODEPAGE_857=m +CONFIG_NLS_CODEPAGE_860=m +CONFIG_NLS_CODEPAGE_861=m +CONFIG_NLS_CODEPAGE_862=m +CONFIG_NLS_CODEPAGE_863=m +CONFIG_NLS_CODEPAGE_864=m +CONFIG_NLS_CODEPAGE_865=m +CONFIG_NLS_CODEPAGE_866=m +CONFIG_NLS_CODEPAGE_869=m +CONFIG_NLS_CODEPAGE_936=m +CONFIG_NLS_CODEPAGE_950=m +CONFIG_NLS_CODEPAGE_932=m +CONFIG_NLS_CODEPAGE_949=m +CONFIG_NLS_CODEPAGE_874=m +CONFIG_NLS_ISO8859_8=m +CONFIG_NLS_CODEPAGE_1250=m +CONFIG_NLS_CODEPAGE_1251=m +CONFIG_NLS_ASCII=m +CONFIG_NLS_ISO8859_1=y +CONFIG_NLS_ISO8859_2=m +CONFIG_NLS_ISO8859_3=m +CONFIG_NLS_ISO8859_4=m +CONFIG_NLS_ISO8859_5=m +CONFIG_NLS_ISO8859_6=m +CONFIG_NLS_ISO8859_7=m +CONFIG_NLS_ISO8859_9=m +CONFIG_NLS_ISO8859_13=m +CONFIG_NLS_ISO8859_14=m +CONFIG_NLS_ISO8859_15=m +CONFIG_NLS_KOI8_R=m +CONFIG_NLS_KOI8_U=m +CONFIG_NLS_UTF8=y + +# +# Profiling support +# +# CONFIG_PROFILING is not set + +# +# Kernel hacking +# +# CONFIG_PRINTK_TIME is not set +CONFIG_MAGIC_SYSRQ=y +CONFIG_DEBUG_KERNEL=y +CONFIG_LOG_BUF_SHIFT=14 +CONFIG_DETECT_SOFTLOCKUP=y +# CONFIG_SCHEDSTATS is not set +# CONFIG_DEBUG_SLAB is not set +# CONFIG_DEBUG_PREEMPT is not set +# CONFIG_DEBUG_MUTEXES is not set +# CONFIG_DEBUG_SPINLOCK is not set +# CONFIG_DEBUG_SPINLOCK_SLEEP is not set +# CONFIG_DEBUG_KOBJECT is not set +# CONFIG_DEBUG_BUGVERBOSE is not set +# CONFIG_DEBUG_INFO is not set +# CONFIG_DEBUG_FS is not set +# CONFIG_DEBUG_VM is not set +CONFIG_FRAME_POINTER=y +# CONFIG_FORCED_INLINING is not set +# CONFIG_RCU_TORTURE_TEST is not set +# CONFIG_DEBUG_USER is not set +# CONFIG_DEBUG_WAITQ is not set +CONFIG_DEBUG_ERRORS=y +# CONFIG_DEBUG_LL is not set + +# +# Security options +# +# CONFIG_KEYS is not set +# CONFIG_SECURITY is not set + +# +# Cryptographic options +# +CONFIG_CRYPTO=y +CONFIG_CRYPTO_HMAC=y +CONFIG_CRYPTO_NULL=m +CONFIG_CRYPTO_MD4=m +CONFIG_CRYPTO_MD5=m +CONFIG_CRYPTO_SHA1=m +CONFIG_CRYPTO_SHA256=m +CONFIG_CRYPTO_SHA512=m +CONFIG_CRYPTO_WP512=m +# CONFIG_CRYPTO_TGR192 is not set +CONFIG_CRYPTO_DES=m +CONFIG_CRYPTO_BLOWFISH=m +CONFIG_CRYPTO_TWOFISH=m +CONFIG_CRYPTO_SERPENT=m +CONFIG_CRYPTO_AES=m +CONFIG_CRYPTO_CAST5=m +CONFIG_CRYPTO_CAST6=m +CONFIG_CRYPTO_TEA=m +CONFIG_CRYPTO_ARC4=m +CONFIG_CRYPTO_KHAZAD=m +CONFIG_CRYPTO_ANUBIS=m +CONFIG_CRYPTO_DEFLATE=m +CONFIG_CRYPTO_MICHAEL_MIC=m +CONFIG_CRYPTO_CRC32C=m +CONFIG_CRYPTO_TEST=m + +# +# Hardware crypto devices +# + +# +# Library routines +# +CONFIG_CRC_CCITT=m +# CONFIG_CRC16 is not set +CONFIG_CRC32=y +CONFIG_LIBCRC32C=m +CONFIG_ZLIB_INFLATE=y +CONFIG_ZLIB_DEFLATE=y +CONFIG_GENERIC_ALLOCATOR=y diff --git a/packages/linux/linux-rp-2.6.20/orinoco-remove-all-which-are-in-hostap-HACK.patch b/packages/linux/linux-rp-2.6.20/orinoco-remove-all-which-are-in-hostap-HACK.patch new file mode 100644 index 0000000000..380349f809 --- /dev/null +++ b/packages/linux/linux-rp-2.6.20/orinoco-remove-all-which-are-in-hostap-HACK.patch @@ -0,0 +1,88 @@ +This patch should resolve problem when people get eth0 (orinoco_cs) instead of wlan0 (hostap_cs) +with their WiFi cards. + +Patch will NEVER been accepted upstream. + +Signed-off-by: Marcin Juszkiewicz <openembedded@hrw.one.pl> + +Index: linux/drivers/net/wireless/orinoco_cs.c +=================================================================== +--- linux.orig/drivers/net/wireless/orinoco_cs.c 2006-08-23 16:04:10.000000000 +0200 ++++ linux/drivers/net/wireless/orinoco_cs.c 2006-08-23 16:17:43.000000000 +0200 +@@ -453,33 +453,21 @@ + "Pavel Roskin <proski@gnu.org>, et al)"; + + static struct pcmcia_device_id orinoco_cs_ids[] = { +- PCMCIA_DEVICE_MANF_CARD(0x000b, 0x7100), /* SonicWALL Long Range Wireless Card */ +- PCMCIA_DEVICE_MANF_CARD(0x000b, 0x7300), /* Sohoware NCP110, Philips 802.11b */ + PCMCIA_DEVICE_MANF_CARD(0x0089, 0x0002), /* AnyPoint(TM) Wireless II PC Card */ +- PCMCIA_DEVICE_MANF_CARD(0x0101, 0x0777), /* 3Com AirConnect PCI 777A */ +- PCMCIA_DEVICE_MANF_CARD(0x0126, 0x8000), /* PROXIM RangeLAN-DS/LAN PC CARD */ + PCMCIA_DEVICE_MANF_CARD(0x0138, 0x0002), /* Compaq WL100 11 Mbps Wireless Adapter */ + PCMCIA_DEVICE_MANF_CARD(0x0156, 0x0002), /* Lucent Orinoco and old Intersil */ + PCMCIA_DEVICE_MANF_CARD(0x016b, 0x0001), /* Ericsson WLAN Card C11 */ + PCMCIA_DEVICE_MANF_CARD(0x01eb, 0x080a), /* Nortel Networks eMobility 802.11 Wireless Adapter */ + PCMCIA_DEVICE_MANF_CARD(0x01ff, 0x0008), /* Intermec MobileLAN 11Mbps 802.11b WLAN Card */ +- PCMCIA_DEVICE_MANF_CARD(0x0250, 0x0002), /* Samsung SWL2000-N 11Mb/s WLAN Card */ + PCMCIA_DEVICE_MANF_CARD(0x0261, 0x0002), /* AirWay 802.11 Adapter (PCMCIA) */ + PCMCIA_DEVICE_MANF_CARD(0x0268, 0x0001), /* ARtem Onair */ + PCMCIA_DEVICE_MANF_CARD(0x026f, 0x0305), /* Buffalo WLI-PCM-S11 */ +- PCMCIA_DEVICE_MANF_CARD(0x0274, 0x1612), /* Linksys WPC11 Version 2.5 */ +- PCMCIA_DEVICE_MANF_CARD(0x0274, 0x1613), /* Linksys WPC11 Version 3 */ +- PCMCIA_DEVICE_MANF_CARD(0x028a, 0x0002), /* Compaq HNW-100 11 Mbps Wireless Adapter */ + PCMCIA_DEVICE_MANF_CARD(0x028a, 0x0673), /* Linksys WCF12 Wireless CompactFlash Card */ +- PCMCIA_DEVICE_MANF_CARD(0x02aa, 0x0002), /* ASUS SpaceLink WL-100 */ + PCMCIA_DEVICE_MANF_CARD(0x02ac, 0x0002), /* SpeedStream SS1021 Wireless Adapter */ + PCMCIA_DEVICE_MANF_CARD(0x14ea, 0xb001), /* PLANEX RoadLannerWave GW-NS11H */ +- PCMCIA_DEVICE_MANF_CARD(0x50c2, 0x7300), /* Airvast WN-100 */ + PCMCIA_DEVICE_MANF_CARD(0x9005, 0x0021), /* Adaptec Ultra Wireless ANW-8030 */ + PCMCIA_DEVICE_MANF_CARD(0xc001, 0x0008), /* CONTEC FLEXSCAN/FX-DDS110-PCC */ + PCMCIA_DEVICE_MANF_CARD(0xc250, 0x0002), /* Conceptronic CON11Cpro, EMTAC A2424i */ +- PCMCIA_DEVICE_MANF_CARD(0xd601, 0x0002), /* Safeway 802.11b, ZCOMAX AirRunner/XI-300 */ +- PCMCIA_DEVICE_MANF_CARD(0xd601, 0x0005), /* D-Link DCF660, Sandisk Connect SDWCFB-000 */ + PCMCIA_DEVICE_PROD_ID12(" ", "IEEE 802.11 Wireless LAN/PC Card", 0x3b6e20c8, 0xefccafe9), + PCMCIA_DEVICE_PROD_ID12("3Com", "3CRWE737A AirConnect Wireless LAN PC Card", 0x41240e5b, 0x56010af3), + PCMCIA_DEVICE_PROD_ID12("ACTIONTEC", "PRISM Wireless LAN PC Card", 0x393089da, 0xa71e69d5), +@@ -487,31 +475,25 @@ + PCMCIA_DEVICE_PROD_ID123("AIRVAST", "IEEE 802.11b Wireless PCMCIA Card", "HFA3863", 0xea569531, 0x4bcb9645, 0x355cb092), + PCMCIA_DEVICE_PROD_ID12("Allied Telesyn", "AT-WCL452 Wireless PCMCIA Radio", 0x5cd01705, 0x4271660f), + PCMCIA_DEVICE_PROD_ID12("ASUS", "802_11b_PC_CARD_25", 0x78fc06ee, 0xdb9aa842), +- PCMCIA_DEVICE_PROD_ID12("ASUS", "802_11B_CF_CARD_25", 0x78fc06ee, 0x45a50c1e), + PCMCIA_DEVICE_PROD_ID12("Avaya Communication", "Avaya Wireless PC Card", 0xd8a43b78, 0x0d341169), + PCMCIA_DEVICE_PROD_ID12("BENQ", "AWL100 PCMCIA ADAPTER", 0x35dadc74, 0x01f7fedb), + PCMCIA_DEVICE_PROD_ID12("BUFFALO", "WLI-PCM-L11G", 0x2decece3, 0xf57ca4b3), +- PCMCIA_DEVICE_PROD_ID12("BUFFALO", "WLI-CF-S11G", 0x2decece3, 0x82067c18), + PCMCIA_DEVICE_PROD_ID12("Cabletron", "RoamAbout 802.11 DS", 0x32d445f5, 0xedeffd90), +- PCMCIA_DEVICE_PROD_ID12("Compaq", "WL200_11Mbps_Wireless_PCI_Card", 0x54f7c49c, 0x15a75e5b), + PCMCIA_DEVICE_PROD_ID123("corega", "WL PCCL-11", "ISL37300P", 0x0a21501a, 0x59868926, 0xc9049a39), + PCMCIA_DEVICE_PROD_ID12("corega K.K.", "Wireless LAN PCC-11", 0x5261440f, 0xa6405584), + PCMCIA_DEVICE_PROD_ID12("corega K.K.", "Wireless LAN PCCA-11", 0x5261440f, 0xdf6115f9), + PCMCIA_DEVICE_PROD_ID12("corega_K.K.", "Wireless_LAN_PCCB-11", 0x29e33311, 0xee7a27ae), + PCMCIA_DEVICE_PROD_ID12("D", "Link DRC-650 11Mbps WLAN Card", 0x71b18589, 0xf144e3ac), +- PCMCIA_DEVICE_PROD_ID12("D", "Link DWL-650 11Mbps WLAN Card", 0x71b18589, 0xb6f1b0ab), + PCMCIA_DEVICE_PROD_ID12("D-Link Corporation", "D-Link DWL-650H 11Mbps WLAN Adapter", 0xef544d24, 0xcd8ea916), + PCMCIA_DEVICE_PROD_ID12("Digital Data Communications", "WPC-0100", 0xfdd73470, 0xe0b6f146), + PCMCIA_DEVICE_PROD_ID12("ELSA", "AirLancer MC-11", 0x4507a33a, 0xef54f0e3), + PCMCIA_DEVICE_PROD_ID12("HyperLink", "Wireless PC Card 11Mbps", 0x56cc3f1a, 0x0bcf220c), + PCMCIA_DEVICE_PROD_ID123("Instant Wireless ", " Network PC CARD", "Version 01.02", 0x11d901af, 0x6e9bd926, 0x4b74baa0), + PCMCIA_DEVICE_PROD_ID12("Intel", "PRO/Wireless 2011 LAN PC Card", 0x816cc815, 0x07f58077), +- PCMCIA_DEVICE_PROD_ID12("INTERSIL", "HFA384x/IEEE", 0x74c5e40d, 0xdb472a18), + PCMCIA_DEVICE_PROD_ID12("INTERSIL", "I-GATE 11M PC Card / PC Card plus", 0x74c5e40d, 0x8304ff77), + PCMCIA_DEVICE_PROD_ID12("Intersil", "PRISM 2_5 PCMCIA ADAPTER", 0x4b801a17, 0x6345a0bf), + PCMCIA_DEVICE_PROD_ID123("Intersil", "PRISM Freedom PCMCIA Adapter", "ISL37100P", 0x4b801a17, 0xf222ec2d, 0x630d52b2), + PCMCIA_DEVICE_PROD_ID12("LeArtery", "SYNCBYAIR 11Mbps Wireless LAN PC Card", 0x7e3b326a, 0x49893e92), +- PCMCIA_DEVICE_PROD_ID12("Linksys", "Wireless CompactFlash Card", 0x0733cc81, 0x0c52f395), + PCMCIA_DEVICE_PROD_ID12("Lucent Technologies", "WaveLAN/IEEE", 0x23eb9949, 0xc562e72a), + PCMCIA_DEVICE_PROD_ID12("MELCO", "WLI-PCM-L11", 0x481e0094, 0x7360e410), + PCMCIA_DEVICE_PROD_ID12("MELCO", "WLI-PCM-L11G", 0x481e0094, 0xf57ca4b3), +@@ -529,10 +511,8 @@ + PCMCIA_DEVICE_PROD_ID12("PROXIM", "LAN PCI CARD HARMONY 80211B", 0xc6536a5e, 0x9f494e26), + PCMCIA_DEVICE_PROD_ID12("SAMSUNG", "11Mbps WLAN Card", 0x43d74cb4, 0x579bd91b), + PCMCIA_DEVICE_PROD_ID12("SMC", "SMC2532W-B EliteConnect Wireless Adapter", 0xc4f8b18b, 0x196bd757), +- PCMCIA_DEVICE_PROD_ID12("SMC", "SMC2632W", 0xc4f8b18b, 0x474a1f2a), + PCMCIA_DEVICE_PROD_ID12("Symbol Technologies", "LA4111 Spectrum24 Wireless LAN PC Card", 0x3f02b4d6, 0x3663cb0e), + PCMCIA_DEVICE_PROD_ID123("The Linksys Group, Inc.", "Instant Wireless Network PC Card", "ISL37300P", 0xa5f472c2, 0x590eb502, 0xc9049a39), +- PCMCIA_DEVICE_PROD_ID12("ZoomAir 11Mbps High", "Rate wireless Networking", 0x273fe3db, 0x32a1eaee), + PCMCIA_DEVICE_NULL, + }; + MODULE_DEVICE_TABLE(pcmcia, orinoco_cs_ids); diff --git a/packages/linux/linux-rp-2.6.20/pxa-serial-hack.patch b/packages/linux/linux-rp-2.6.20/pxa-serial-hack.patch new file mode 100644 index 0000000000..9ece71331a --- /dev/null +++ b/packages/linux/linux-rp-2.6.20/pxa-serial-hack.patch @@ -0,0 +1,80 @@ +--- + drivers/serial/8250.c | 5 +++++ + drivers/serial/serial_core.c | 1 + + drivers/serial/serial_cs.c | 12 +++++++++--- + include/linux/serial_core.h | 1 + + 4 files changed, 16 insertions(+), 3 deletions(-) + +Index: git/drivers/serial/8250.c +=================================================================== +--- git.orig/drivers/serial/8250.c 2006-10-31 16:29:50.000000000 +0000 ++++ git/drivers/serial/8250.c 2006-10-31 16:29:53.000000000 +0000 +@@ -2429,7 +2429,12 @@ static struct uart_driver serial8250_reg + .driver_name = "serial", + .dev_name = "ttyS", + .major = TTY_MAJOR, ++#ifdef CONFIG_SERIAL_PXA ++ .minor = 64 + 3, ++ .name_base = 3, ++#else + .minor = 64, ++#endif + .nr = UART_NR, + .cons = SERIAL8250_CONSOLE, + }; +Index: git/drivers/serial/serial_core.c +=================================================================== +--- git.orig/drivers/serial/serial_core.c 2006-10-31 16:09:17.000000000 +0000 ++++ git/drivers/serial/serial_core.c 2006-10-31 16:29:53.000000000 +0000 +@@ -2183,6 +2183,7 @@ int uart_register_driver(struct uart_dri + normal->owner = drv->owner; + normal->driver_name = drv->driver_name; + normal->name = drv->dev_name; ++ normal->name_base = drv->name_base; + normal->major = drv->major; + normal->minor_start = drv->minor; + normal->type = TTY_DRIVER_TYPE_SERIAL; +Index: git/include/linux/serial_core.h +=================================================================== +--- git.orig/include/linux/serial_core.h 2006-10-31 16:09:45.000000000 +0000 ++++ git/include/linux/serial_core.h 2006-10-31 16:30:36.000000000 +0000 +@@ -339,6 +339,7 @@ struct uart_driver { + struct module *owner; + const char *driver_name; + const char *dev_name; ++ int name_base; + int major; + int minor; + int nr; +Index: git/drivers/serial/serial_cs.c +=================================================================== +--- git.orig/drivers/serial/serial_cs.c 2006-10-31 16:09:17.000000000 +0000 ++++ git/drivers/serial/serial_cs.c 2006-10-31 16:29:53.000000000 +0000 +@@ -390,7 +390,7 @@ static int setup_serial(struct pcmcia_de + kio_addr_t iobase, int irq) + { + struct uart_port port; +- int line; ++ int line, linestart; + + memset(&port, 0, sizeof (struct uart_port)); + port.iobase = iobase; +@@ -411,10 +411,16 @@ static int setup_serial(struct pcmcia_de + return -EINVAL; + } + ++#if CONFIG_SERIAL_PXA ++ linestart = 3; ++#else ++ linestart = 0; ++#endif ++ + info->line[info->ndev] = line; +- sprintf(info->node[info->ndev].dev_name, "ttyS%d", line); ++ sprintf(info->node[info->ndev].dev_name, "ttyS%d", line+linestart); + info->node[info->ndev].major = TTY_MAJOR; +- info->node[info->ndev].minor = 0x40 + line; ++ info->node[info->ndev].minor = 0x40 + line + linestart; + if (info->ndev > 0) + info->node[info->ndev - 1].next = &info->node[info->ndev]; + info->ndev++; diff --git a/packages/linux/linux-rp-2.6.20/serial-add-support-for-non-standard-xtals-to-16c950-driver.patch b/packages/linux/linux-rp-2.6.20/serial-add-support-for-non-standard-xtals-to-16c950-driver.patch new file mode 100644 index 0000000000..18bf4268fc --- /dev/null +++ b/packages/linux/linux-rp-2.6.20/serial-add-support-for-non-standard-xtals-to-16c950-driver.patch @@ -0,0 +1,155 @@ + +From: Petr Vandrovec <vandrove@vc.cvut.cz> + +Patch below adds support for using different prescaler than 16 for 16c950 +chips. This is needed for using Fujitsu-Siemens Connect2Air compact-flash +card, which comes (apparently) with 806kHz clocks, and so you have to +program prescaler for division by 7, and DLAB to 1, to get 115200Bd. + +To get card properly running you also have to add lines below to +/etc/pcmcia/serial.opts so kernel knows that base speed is not 115200 but +50400 (50400 * 16 = 806400; 806400 / 7 = 115200). As I've found no code +specifying baud_rate in serial_cs, I assume that specifying it in +serial.opts is right way to do this type of things. + +Patch also fixes problem that for UPF_MAGIC_MULTIPLIER maximum possible +baud rate passed to uart code was uartclk / 16 while correct value for +these devices (and for 16c950) is uartclk / 4. + +Patch also fixes problem that for UPF_MAGIC_MULTIPLIER devices with +baud_rate 19200 or 9600 spd_cust did not work correctly. Not that such +devices exist, but we should not ignore spd_cust, user probably knows why +he asked for spd_cust. + +serial.opts: + +case "$MANFID-$FUNCID-$PRODID_1-$PRODID_2-$PRODID_3-$PRODID_4" in +'0279,950b-2-GPRS Modem---') + SERIAL_OPTS="baud_base 50400" + ;; +esac + +Cc: David Woodhouse <dwmw2@infradead.org> +Signed-off-by: Andrew Morton <akpm@osdl.org> +--- + + drivers/serial/8250.c | 82 +++++++++++++++++++++++++++++++++++++++----------- + 1 files changed, 64 insertions(+), 18 deletions(-) + +diff -puN drivers/serial/8250.c~serial-add-support-for-non-standard-xtals-to-16c950-driver drivers/serial/8250.c +--- devel/drivers/serial/8250.c~serial-add-support-for-non-standard-xtals-to-16c950-driver 2005-09-12 03:34:57.000000000 -0700 ++++ devel-akpm/drivers/serial/8250.c 2005-09-12 03:34:57.000000000 -0700 +@@ -1653,24 +1653,58 @@ static void serial8250_shutdown(struct u + serial_unlink_irq_chain(up); + } + +-static unsigned int serial8250_get_divisor(struct uart_port *port, unsigned int baud) ++static unsigned int serial8250_get_divisor(struct uart_port *port, unsigned int baud, ++ unsigned int *prescaler) + { +- unsigned int quot; +- +- /* +- * Handle magic divisors for baud rates above baud_base on +- * SMSC SuperIO chips. ++ /* ++ * Use special handling only if user did not supply its own divider. ++ * spd_cust is defined in terms of baud_base, so always use default ++ * prescaler when spd_cust is requested. + */ +- if ((port->flags & UPF_MAGIC_MULTIPLIER) && +- baud == (port->uartclk/4)) +- quot = 0x8001; +- else if ((port->flags & UPF_MAGIC_MULTIPLIER) && +- baud == (port->uartclk/8)) +- quot = 0x8002; +- else +- quot = uart_get_divisor(port, baud); + +- return quot; ++ *prescaler = 16; ++ if (baud != 38400 || (port->flags & UPF_SPD_MASK) != UPF_SPD_CUST) { ++ unsigned int quot = port->uartclk / baud; ++ ++ /* ++ * Handle magic divisors for baud rates above baud_base on ++ * SMSC SuperIO chips. ++ */ ++ if (port->flags & UPF_MAGIC_MULTIPLIER) { ++ if (quot == 4) { ++ return 0x8001; ++ } else if (quot == 8) { ++ return 0x8002; ++ } ++ } ++ if (port->type == PORT_16C950) { ++ /* ++ * This computes TCR value (4 to 16), not CPR value (which can ++ * be between 1.000 and 31.875) - chip I have uses XTAL of ++ * 806400Hz, and so a division by 7 is required to get 115200Bd. ++ * I'm leaving CPR disabled for now, until someone will ++ * hit even more exotic XTAL (it is needed to get 500kbps ++ * or 1000kbps from 18.432MHz XTAL, but I have no device ++ * which would benefit from doing that). ++ * ++ * If we can use divide by 16, use it. Otherwise look for ++ * better prescaler, from 15 to 4. If quotient cannot ++ * be divided by any integer value between 4 and 15, use 4. ++ */ ++ if (quot & 0x0F) { ++ unsigned int div; ++ ++ for (div = 15; div > 4; div--) { ++ if (quot % div == 0) { ++ break; ++ } ++ } ++ *prescaler = div; ++ return quot / div; ++ } ++ } ++ } ++ return uart_get_divisor(port, baud); + } + + static void +@@ -1680,7 +1714,7 @@ serial8250_set_termios(struct uart_port + struct uart_8250_port *up = (struct uart_8250_port *)port; + unsigned char cval, fcr = 0; + unsigned long flags; +- unsigned int baud, quot; ++ unsigned int baud, quot, prescaler; + + switch (termios->c_cflag & CSIZE) { + case CS5: +@@ -1712,8 +1746,13 @@ serial8250_set_termios(struct uart_port + /* + * Ask the core to calculate the divisor for us. + */ +- baud = uart_get_baud_rate(port, termios, old, 0, port->uartclk/16); +- quot = serial8250_get_divisor(port, baud); ++ ++ if (port->type == PORT_16C950 || (port->flags & UPF_MAGIC_MULTIPLIER)) { ++ baud = uart_get_baud_rate(port, termios, old, 0, port->uartclk/4); ++ } else { ++ baud = uart_get_baud_rate(port, termios, old, 0, port->uartclk/16); ++ } ++ quot = serial8250_get_divisor(port, baud, &prescaler); + + /* + * Oxford Semi 952 rev B workaround +@@ -1817,6 +1856,13 @@ serial8250_set_termios(struct uart_port + serial_outp(up, UART_DLM, quot >> 8); /* MS of divisor */ + + /* ++ * Program prescaler for 16C950 chips. ++ */ ++ if (up->port.type == PORT_16C950) { ++ serial_icr_write(up, UART_TCR, prescaler == 16 ? 0 : prescaler); ++ } ++ ++ /* + * LCR DLAB must be set to enable 64-byte FIFO mode. If the FCR + * is written without DLAB set, this mode will be disabled. + */ +_ diff --git a/packages/linux/linux-rp-2.6.20/squashfs3.0-2.6.15.patch b/packages/linux/linux-rp-2.6.20/squashfs3.0-2.6.15.patch new file mode 100644 index 0000000000..405d4ec124 --- /dev/null +++ b/packages/linux/linux-rp-2.6.20/squashfs3.0-2.6.15.patch @@ -0,0 +1,4191 @@ + fs/Kconfig | 65 + + fs/Makefile | 1 + fs/squashfs/Makefile | 7 + fs/squashfs/inode.c | 2124 +++++++++++++++++++++++++++++++++++++++++ + fs/squashfs/squashfs.h | 86 + + fs/squashfs/squashfs2_0.c | 757 ++++++++++++++ + include/linux/squashfs_fs.h | 911 +++++++++++++++++ + include/linux/squashfs_fs_i.h | 45 + include/linux/squashfs_fs_sb.h | 74 + + init/do_mounts_rd.c | 13 + 10 files changed, 4083 insertions(+) + +Index: git/fs/Kconfig +=================================================================== +--- git.orig/fs/Kconfig 2006-10-31 16:09:22.000000000 +0000 ++++ git/fs/Kconfig 2006-10-31 21:26:45.000000000 +0000 +@@ -1407,6 +1407,71 @@ config CRAMFS + + If unsure, say N. + ++config SQUASHFS ++ tristate "SquashFS 3.0 - Squashed file system support" ++ select ZLIB_INFLATE ++ help ++ Saying Y here includes support for SquashFS 3.0 (a Compressed Read-Only File ++ System). Squashfs is a highly compressed read-only filesystem for Linux. ++ It uses zlib compression to compress both files, inodes and directories. ++ Inodes in the system are very small and all blocks are packed to minimise ++ data overhead. Block sizes greater than 4K are supported up to a maximum of 64K. ++ SquashFS 3.0 supports 64 bit filesystems and files (larger than 4GB), full ++ uid/gid information, hard links and timestamps. ++ ++ Squashfs is intended for general read-only filesystem use, for archival ++ use (i.e. in cases where a .tar.gz file may be used), and in embedded ++ systems where low overhead is needed. Further information and filesystem tools ++ are available from http://squashfs.sourceforge.net. ++ ++ If you want to compile this as a module ( = code which can be ++ inserted in and removed from the running kernel whenever you want), ++ say M here and read <file:Documentation/modules.txt>. The module ++ will be called squashfs. Note that the root file system (the one ++ containing the directory /) cannot be compiled as a module. ++ ++ If unsure, say N. ++ ++config SQUASHFS_EMBEDDED ++ ++ bool "Additional options for memory-constrained systems" ++ depends on SQUASHFS ++ default n ++ help ++ Saying Y here allows you to specify cache sizes and how Squashfs ++ allocates memory. This is only intended for memory constrained ++ systems. ++ ++ If unsure, say N. ++ ++config SQUASHFS_FRAGMENT_CACHE_SIZE ++ int "Number of fragments cached" if SQUASHFS_EMBEDDED ++ depends on SQUASHFS ++ default "3" ++ help ++ By default SquashFS caches the last 3 fragments read from ++ the filesystem. Increasing this amount may mean SquashFS ++ has to re-read fragments less often from disk, at the expense ++ of extra system memory. Decreasing this amount will mean ++ SquashFS uses less memory at the expense of extra reads from disk. ++ ++ Note there must be at least one cached fragment. Anything ++ much more than three will probably not make much difference. ++ ++config SQUASHFS_VMALLOC ++ bool "Use Vmalloc rather than Kmalloc" if SQUASHFS_EMBEDDED ++ depends on SQUASHFS ++ default n ++ help ++ By default SquashFS uses kmalloc to obtain fragment cache memory. ++ Kmalloc memory is the standard kernel allocator, but it can fail ++ on memory constrained systems. Because of the way Vmalloc works, ++ Vmalloc can succeed when kmalloc fails. Specifying this option ++ will make SquashFS always use Vmalloc to allocate the ++ fragment cache memory. ++ ++ If unsure, say N. ++ + config VXFS_FS + tristate "FreeVxFS file system support (VERITAS VxFS(TM) compatible)" + depends on BLOCK +Index: git/fs/Makefile +=================================================================== +--- git.orig/fs/Makefile 2006-10-31 16:09:22.000000000 +0000 ++++ git/fs/Makefile 2006-10-31 21:26:45.000000000 +0000 +@@ -67,6 +67,7 @@ obj-$(CONFIG_JBD) += jbd/ + obj-$(CONFIG_JBD2) += jbd2/ + obj-$(CONFIG_EXT2_FS) += ext2/ + obj-$(CONFIG_CRAMFS) += cramfs/ ++obj-$(CONFIG_SQUASHFS) += squashfs/ + obj-$(CONFIG_RAMFS) += ramfs/ + obj-$(CONFIG_HUGETLBFS) += hugetlbfs/ + obj-$(CONFIG_CODA_FS) += coda/ +Index: git/fs/squashfs/inode.c +=================================================================== +--- /dev/null 1970-01-01 00:00:00.000000000 +0000 ++++ git/fs/squashfs/inode.c 2006-11-01 18:01:32.000000000 +0000 +@@ -0,0 +1,2124 @@ ++/* ++ * Squashfs - a compressed read only filesystem for Linux ++ * ++ * Copyright (c) 2002, 2003, 2004, 2005, 2006 ++ * Phillip Lougher <phillip@lougher.org.uk> ++ * ++ * 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, ++ * or (at your option) any later version. ++ * ++ * This program is distributed in the hope that it will be useful, ++ * but WITHOUT ANY WARRANTY; without even the implied warranty of ++ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ++ * GNU General Public License for more details. ++ * ++ * You should have received a copy of the GNU General Public License ++ * along with this program; if not, write to the Free Software ++ * Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. ++ * ++ * inode.c ++ */ ++ ++#include <linux/types.h> ++#include <linux/squashfs_fs.h> ++#include <linux/module.h> ++#include <linux/errno.h> ++#include <linux/slab.h> ++#include <linux/fs.h> ++#include <linux/smp_lock.h> ++#include <linux/slab.h> ++#include <linux/squashfs_fs_sb.h> ++#include <linux/squashfs_fs_i.h> ++#include <linux/buffer_head.h> ++#include <linux/vfs.h> ++#include <linux/init.h> ++#include <linux/dcache.h> ++#include <linux/wait.h> ++#include <linux/zlib.h> ++#include <linux/blkdev.h> ++#include <linux/vmalloc.h> ++#include <asm/uaccess.h> ++#include <asm/semaphore.h> ++ ++#include "squashfs.h" ++ ++static void squashfs_put_super(struct super_block *); ++static int squashfs_statfs(struct dentry *, struct kstatfs *); ++static int squashfs_symlink_readpage(struct file *file, struct page *page); ++static int squashfs_readpage(struct file *file, struct page *page); ++static int squashfs_readpage4K(struct file *file, struct page *page); ++static int squashfs_readdir(struct file *, void *, filldir_t); ++static struct inode *squashfs_alloc_inode(struct super_block *sb); ++static void squashfs_destroy_inode(struct inode *inode); ++static int init_inodecache(void); ++static void destroy_inodecache(void); ++static struct dentry *squashfs_lookup(struct inode *, struct dentry *, ++ struct nameidata *); ++static struct inode *squashfs_iget(struct super_block *s, squashfs_inode_t inode); ++static long long read_blocklist(struct inode *inode, int index, ++ int readahead_blks, char *block_list, ++ unsigned short **block_p, unsigned int *bsize); ++static int squashfs_get_sb(struct file_system_type *, int, ++ const char *, void *, struct vfsmount *); ++ ++ ++static z_stream stream; ++ ++static struct file_system_type squashfs_fs_type = { ++ .owner = THIS_MODULE, ++ .name = "squashfs", ++ .get_sb = squashfs_get_sb, ++ .kill_sb = kill_block_super, ++ .fs_flags = FS_REQUIRES_DEV ++}; ++ ++static unsigned char squashfs_filetype_table[] = { ++ DT_UNKNOWN, DT_DIR, DT_REG, DT_LNK, DT_BLK, DT_CHR, DT_FIFO, DT_SOCK ++}; ++ ++static struct super_operations squashfs_ops = { ++ .alloc_inode = squashfs_alloc_inode, ++ .destroy_inode = squashfs_destroy_inode, ++ .statfs = squashfs_statfs, ++ .put_super = squashfs_put_super, ++}; ++ ++SQSH_EXTERN struct address_space_operations squashfs_symlink_aops = { ++ .readpage = squashfs_symlink_readpage ++}; ++ ++SQSH_EXTERN struct address_space_operations squashfs_aops = { ++ .readpage = squashfs_readpage ++}; ++ ++SQSH_EXTERN struct address_space_operations squashfs_aops_4K = { ++ .readpage = squashfs_readpage4K ++}; ++ ++static struct file_operations squashfs_dir_ops = { ++ .read = generic_read_dir, ++ .readdir = squashfs_readdir ++}; ++ ++SQSH_EXTERN struct inode_operations squashfs_dir_inode_ops = { ++ .lookup = squashfs_lookup ++}; ++ ++ ++static struct buffer_head *get_block_length(struct super_block *s, ++ int *cur_index, int *offset, int *c_byte) ++{ ++ struct squashfs_sb_info *msblk = s->s_fs_info; ++ unsigned short temp; ++ struct buffer_head *bh; ++ ++ if (!(bh = sb_bread(s, *cur_index))) ++ goto out; ++ ++ if (msblk->devblksize - *offset == 1) { ++ if (msblk->swap) ++ ((unsigned char *) &temp)[1] = *((unsigned char *) ++ (bh->b_data + *offset)); ++ else ++ ((unsigned char *) &temp)[0] = *((unsigned char *) ++ (bh->b_data + *offset)); ++ brelse(bh); ++ if (!(bh = sb_bread(s, ++(*cur_index)))) ++ goto out; ++ if (msblk->swap) ++ ((unsigned char *) &temp)[0] = *((unsigned char *) ++ bh->b_data); ++ else ++ ((unsigned char *) &temp)[1] = *((unsigned char *) ++ bh->b_data); ++ *c_byte = temp; ++ *offset = 1; ++ } else { ++ if (msblk->swap) { ++ ((unsigned char *) &temp)[1] = *((unsigned char *) ++ (bh->b_data + *offset)); ++ ((unsigned char *) &temp)[0] = *((unsigned char *) ++ (bh->b_data + *offset + 1)); ++ } else { ++ ((unsigned char *) &temp)[0] = *((unsigned char *) ++ (bh->b_data + *offset)); ++ ((unsigned char *) &temp)[1] = *((unsigned char *) ++ (bh->b_data + *offset + 1)); ++ } ++ *c_byte = temp; ++ *offset += 2; ++ } ++ ++ if (SQUASHFS_CHECK_DATA(msblk->sblk.flags)) { ++ if (*offset == msblk->devblksize) { ++ brelse(bh); ++ if (!(bh = sb_bread(s, ++(*cur_index)))) ++ goto out; ++ *offset = 0; ++ } ++ if (*((unsigned char *) (bh->b_data + *offset)) != ++ SQUASHFS_MARKER_BYTE) { ++ ERROR("Metadata block marker corrupt @ %x\n", ++ *cur_index); ++ brelse(bh); ++ goto out; ++ } ++ (*offset)++; ++ } ++ return bh; ++ ++out: ++ return NULL; ++} ++ ++ ++SQSH_EXTERN unsigned int squashfs_read_data(struct super_block *s, char *buffer, ++ long long index, unsigned int length, ++ long long *next_index) ++{ ++ struct squashfs_sb_info *msblk = s->s_fs_info; ++ struct buffer_head *bh[((SQUASHFS_FILE_MAX_SIZE - 1) >> ++ msblk->devblksize_log2) + 2]; ++ unsigned int offset = index & ((1 << msblk->devblksize_log2) - 1); ++ unsigned int cur_index = index >> msblk->devblksize_log2; ++ int bytes, avail_bytes, b = 0, k; ++ char *c_buffer; ++ unsigned int compressed; ++ unsigned int c_byte = length; ++ ++ if (c_byte) { ++ bytes = msblk->devblksize - offset; ++ compressed = SQUASHFS_COMPRESSED_BLOCK(c_byte); ++ c_buffer = compressed ? msblk->read_data : buffer; ++ c_byte = SQUASHFS_COMPRESSED_SIZE_BLOCK(c_byte); ++ ++ TRACE("Block @ 0x%llx, %scompressed size %d\n", index, compressed ++ ? "" : "un", (unsigned int) c_byte); ++ ++ if (!(bh[0] = sb_getblk(s, cur_index))) ++ goto block_release; ++ ++ for (b = 1; bytes < c_byte; b++) { ++ if (!(bh[b] = sb_getblk(s, ++cur_index))) ++ goto block_release; ++ bytes += msblk->devblksize; ++ } ++ ll_rw_block(READ, b, bh); ++ } else { ++ if (!(bh[0] = get_block_length(s, &cur_index, &offset, ++ &c_byte))) ++ goto read_failure; ++ ++ bytes = msblk->devblksize - offset; ++ compressed = SQUASHFS_COMPRESSED(c_byte); ++ c_buffer = compressed ? msblk->read_data : buffer; ++ c_byte = SQUASHFS_COMPRESSED_SIZE(c_byte); ++ ++ TRACE("Block @ 0x%llx, %scompressed size %d\n", index, compressed ++ ? "" : "un", (unsigned int) c_byte); ++ ++ for (b = 1; bytes < c_byte; b++) { ++ if (!(bh[b] = sb_getblk(s, ++cur_index))) ++ goto block_release; ++ bytes += msblk->devblksize; ++ } ++ ll_rw_block(READ, b - 1, bh + 1); ++ } ++ ++ if (compressed) ++ down(&msblk->read_data_mutex); ++ ++ for (bytes = 0, k = 0; k < b; k++) { ++ avail_bytes = (c_byte - bytes) > (msblk->devblksize - offset) ? ++ msblk->devblksize - offset : ++ c_byte - bytes; ++ wait_on_buffer(bh[k]); ++ if (!buffer_uptodate(bh[k])) ++ goto block_release; ++ memcpy(c_buffer + bytes, bh[k]->b_data + offset, avail_bytes); ++ bytes += avail_bytes; ++ offset = 0; ++ brelse(bh[k]); ++ } ++ ++ /* ++ * uncompress block ++ */ ++ if (compressed) { ++ int zlib_err; ++ ++ stream.next_in = c_buffer; ++ stream.avail_in = c_byte; ++ stream.next_out = buffer; ++ stream.avail_out = msblk->read_size; ++ ++ if (((zlib_err = zlib_inflateInit(&stream)) != Z_OK) || ++ ((zlib_err = zlib_inflate(&stream, Z_FINISH)) ++ != Z_STREAM_END) || ((zlib_err = ++ zlib_inflateEnd(&stream)) != Z_OK)) { ++ ERROR("zlib_fs returned unexpected result 0x%x\n", ++ zlib_err); ++ bytes = 0; ++ } else ++ bytes = stream.total_out; ++ ++ up(&msblk->read_data_mutex); ++ } ++ ++ if (next_index) ++ *next_index = index + c_byte + (length ? 0 : ++ (SQUASHFS_CHECK_DATA(msblk->sblk.flags) ++ ? 3 : 2)); ++ return bytes; ++ ++block_release: ++ while (--b >= 0) ++ brelse(bh[b]); ++ ++read_failure: ++ ERROR("sb_bread failed reading block 0x%x\n", cur_index); ++ return 0; ++} ++ ++ ++SQSH_EXTERN int squashfs_get_cached_block(struct super_block *s, char *buffer, ++ long long block, unsigned int offset, ++ int length, long long *next_block, ++ unsigned int *next_offset) ++{ ++ struct squashfs_sb_info *msblk = s->s_fs_info; ++ int n, i, bytes, return_length = length; ++ long long next_index; ++ ++ TRACE("Entered squashfs_get_cached_block [%llx:%x]\n", block, offset); ++ ++ while ( 1 ) { ++ for (i = 0; i < SQUASHFS_CACHED_BLKS; i++) ++ if (msblk->block_cache[i].block == block) ++ break; ++ ++ down(&msblk->block_cache_mutex); ++ ++ if (i == SQUASHFS_CACHED_BLKS) { ++ /* read inode header block */ ++ for (i = msblk->next_cache, n = SQUASHFS_CACHED_BLKS; ++ n ; n --, i = (i + 1) % ++ SQUASHFS_CACHED_BLKS) ++ if (msblk->block_cache[i].block != ++ SQUASHFS_USED_BLK) ++ break; ++ ++ if (n == 0) { ++ wait_queue_t wait; ++ ++ init_waitqueue_entry(&wait, current); ++ add_wait_queue(&msblk->waitq, &wait); ++ set_current_state(TASK_UNINTERRUPTIBLE); ++ up(&msblk->block_cache_mutex); ++ schedule(); ++ set_current_state(TASK_RUNNING); ++ remove_wait_queue(&msblk->waitq, &wait); ++ continue; ++ } ++ msblk->next_cache = (i + 1) % SQUASHFS_CACHED_BLKS; ++ ++ if (msblk->block_cache[i].block == ++ SQUASHFS_INVALID_BLK) { ++ if (!(msblk->block_cache[i].data = ++ kmalloc(SQUASHFS_METADATA_SIZE, ++ GFP_KERNEL))) { ++ ERROR("Failed to allocate cache" ++ "block\n"); ++ up(&msblk->block_cache_mutex); ++ goto out; ++ } ++ } ++ ++ msblk->block_cache[i].block = SQUASHFS_USED_BLK; ++ up(&msblk->block_cache_mutex); ++ ++ if (!(msblk->block_cache[i].length = ++ squashfs_read_data(s, ++ msblk->block_cache[i].data, ++ block, 0, &next_index))) { ++ ERROR("Unable to read cache block [%llx:%x]\n", ++ block, offset); ++ goto out; ++ } ++ ++ down(&msblk->block_cache_mutex); ++ wake_up(&msblk->waitq); ++ msblk->block_cache[i].block = block; ++ msblk->block_cache[i].next_index = next_index; ++ TRACE("Read cache block [%llx:%x]\n", block, offset); ++ } ++ ++ if (msblk->block_cache[i].block != block) { ++ up(&msblk->block_cache_mutex); ++ continue; ++ } ++ ++ if ((bytes = msblk->block_cache[i].length - offset) >= length) { ++ if (buffer) ++ memcpy(buffer, msblk->block_cache[i].data + ++ offset, length); ++ if (msblk->block_cache[i].length - offset == length) { ++ *next_block = msblk->block_cache[i].next_index; ++ *next_offset = 0; ++ } else { ++ *next_block = block; ++ *next_offset = offset + length; ++ } ++ up(&msblk->block_cache_mutex); ++ goto finish; ++ } else { ++ if (buffer) { ++ memcpy(buffer, msblk->block_cache[i].data + ++ offset, bytes); ++ buffer += bytes; ++ } ++ block = msblk->block_cache[i].next_index; ++ up(&msblk->block_cache_mutex); ++ length -= bytes; ++ offset = 0; ++ } ++ } ++ ++finish: ++ return return_length; ++out: ++ return 0; ++} ++ ++ ++static int get_fragment_location(struct super_block *s, unsigned int fragment, ++ long long *fragment_start_block, ++ unsigned int *fragment_size) ++{ ++ struct squashfs_sb_info *msblk = s->s_fs_info; ++ long long start_block = ++ msblk->fragment_index[SQUASHFS_FRAGMENT_INDEX(fragment)]; ++ int offset = SQUASHFS_FRAGMENT_INDEX_OFFSET(fragment); ++ struct squashfs_fragment_entry fragment_entry; ++ ++ if (msblk->swap) { ++ struct squashfs_fragment_entry sfragment_entry; ++ ++ if (!squashfs_get_cached_block(s, (char *) &sfragment_entry, ++ start_block, offset, ++ sizeof(sfragment_entry), &start_block, ++ &offset)) ++ goto out; ++ SQUASHFS_SWAP_FRAGMENT_ENTRY(&fragment_entry, &sfragment_entry); ++ } else ++ if (!squashfs_get_cached_block(s, (char *) &fragment_entry, ++ start_block, offset, ++ sizeof(fragment_entry), &start_block, ++ &offset)) ++ goto out; ++ ++ *fragment_start_block = fragment_entry.start_block; ++ *fragment_size = fragment_entry.size; ++ ++ return 1; ++ ++out: ++ return 0; ++} ++ ++ ++SQSH_EXTERN void release_cached_fragment(struct squashfs_sb_info *msblk, struct ++ squashfs_fragment_cache *fragment) ++{ ++ down(&msblk->fragment_mutex); ++ fragment->locked --; ++ wake_up(&msblk->fragment_wait_queue); ++ up(&msblk->fragment_mutex); ++} ++ ++ ++SQSH_EXTERN struct squashfs_fragment_cache *get_cached_fragment(struct super_block ++ *s, long long start_block, ++ int length) ++{ ++ int i, n, nf; ++ struct squashfs_sb_info *msblk = s->s_fs_info; ++ ++ while ( 1 ) { ++ down(&msblk->fragment_mutex); ++ ++ for (i = 0; i < SQUASHFS_CACHED_FRAGMENTS && ++ msblk->fragment[i].block != start_block; i++); ++ ++ if (i == SQUASHFS_CACHED_FRAGMENTS) { ++ nf = (msblk->next_fragment + 1) % ++ SQUASHFS_CACHED_FRAGMENTS; ++ for (i = msblk->next_fragment, n = ++ SQUASHFS_CACHED_FRAGMENTS; n && ++ msblk->fragment[i].locked; n--, i = (i + 1) % ++ SQUASHFS_CACHED_FRAGMENTS); ++ ++ if (n == 0) { ++ wait_queue_t wait; ++ ++ init_waitqueue_entry(&wait, current); ++ add_wait_queue(&msblk->fragment_wait_queue, ++ &wait); ++ set_current_state(TASK_UNINTERRUPTIBLE); ++ up(&msblk->fragment_mutex); ++ schedule(); ++ set_current_state(TASK_RUNNING); ++ remove_wait_queue(&msblk->fragment_wait_queue, ++ &wait); ++ continue; ++ } ++ msblk->next_fragment = nf; ++ ++ if (msblk->fragment[i].data == NULL) ++ if (!(msblk->fragment[i].data = SQUASHFS_ALLOC ++ (SQUASHFS_FILE_MAX_SIZE))) { ++ ERROR("Failed to allocate fragment " ++ "cache block\n"); ++ up(&msblk->fragment_mutex); ++ goto out; ++ } ++ ++ msblk->fragment[i].block = SQUASHFS_INVALID_BLK; ++ msblk->fragment[i].locked = 1; ++ up(&msblk->fragment_mutex); ++ ++ if (!(msblk->fragment[i].length = squashfs_read_data(s, ++ msblk->fragment[i].data, ++ start_block, length, NULL))) { ++ ERROR("Unable to read fragment cache block " ++ "[%llx]\n", start_block); ++ msblk->fragment[i].locked = 0; ++ goto out; ++ } ++ ++ msblk->fragment[i].block = start_block; ++ TRACE("New fragment %d, start block %lld, locked %d\n", ++ i, msblk->fragment[i].block, ++ msblk->fragment[i].locked); ++ break; ++ } ++ ++ msblk->fragment[i].locked++; ++ up(&msblk->fragment_mutex); ++ TRACE("Got fragment %d, start block %lld, locked %d\n", i, ++ msblk->fragment[i].block, ++ msblk->fragment[i].locked); ++ break; ++ } ++ ++ return &msblk->fragment[i]; ++ ++out: ++ return NULL; ++} ++ ++ ++static struct inode *squashfs_new_inode(struct super_block *s, ++ struct squashfs_base_inode_header *inodeb) ++{ ++ struct squashfs_sb_info *msblk = s->s_fs_info; ++ struct inode *i = new_inode(s); ++ ++ if (i) { ++ i->i_ino = inodeb->inode_number; ++ i->i_mtime.tv_sec = inodeb->mtime; ++ i->i_atime.tv_sec = inodeb->mtime; ++ i->i_ctime.tv_sec = inodeb->mtime; ++ i->i_uid = msblk->uid[inodeb->uid]; ++ i->i_mode = inodeb->mode; ++ i->i_size = 0; ++ if (inodeb->guid == SQUASHFS_GUIDS) ++ i->i_gid = i->i_uid; ++ else ++ i->i_gid = msblk->guid[inodeb->guid]; ++ } ++ ++ return i; ++} ++ ++ ++static struct inode *squashfs_iget(struct super_block *s, squashfs_inode_t inode) ++{ ++ struct inode *i; ++ struct squashfs_sb_info *msblk = s->s_fs_info; ++ struct squashfs_super_block *sblk = &msblk->sblk; ++ long long block = SQUASHFS_INODE_BLK(inode) + ++ sblk->inode_table_start; ++ unsigned int offset = SQUASHFS_INODE_OFFSET(inode); ++ long long next_block; ++ unsigned int next_offset; ++ union squashfs_inode_header id, sid; ++ struct squashfs_base_inode_header *inodeb = &id.base, ++ *sinodeb = &sid.base; ++ ++ TRACE("Entered squashfs_iget\n"); ++ ++ if (msblk->swap) { ++ if (!squashfs_get_cached_block(s, (char *) sinodeb, block, ++ offset, sizeof(*sinodeb), &next_block, ++ &next_offset)) ++ goto failed_read; ++ SQUASHFS_SWAP_BASE_INODE_HEADER(inodeb, sinodeb, ++ sizeof(*sinodeb)); ++ } else ++ if (!squashfs_get_cached_block(s, (char *) inodeb, block, ++ offset, sizeof(*inodeb), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ switch(inodeb->inode_type) { ++ case SQUASHFS_FILE_TYPE: { ++ unsigned int frag_size; ++ long long frag_blk; ++ struct squashfs_reg_inode_header *inodep = &id.reg; ++ struct squashfs_reg_inode_header *sinodep = &sid.reg; ++ ++ if (msblk->swap) { ++ if (!squashfs_get_cached_block(s, (char *) ++ sinodep, block, offset, ++ sizeof(*sinodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ SQUASHFS_SWAP_REG_INODE_HEADER(inodep, sinodep); ++ } else ++ if (!squashfs_get_cached_block(s, (char *) ++ inodep, block, offset, ++ sizeof(*inodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ frag_blk = SQUASHFS_INVALID_BLK; ++ if (inodep->fragment != SQUASHFS_INVALID_FRAG && ++ !get_fragment_location(s, ++ inodep->fragment, &frag_blk, &frag_size)) ++ goto failed_read; ++ ++ if((i = squashfs_new_inode(s, inodeb)) == NULL) ++ goto failed_read1; ++ ++ i->i_nlink = 1; ++ i->i_size = inodep->file_size; ++ i->i_fop = &generic_ro_fops; ++ i->i_mode |= S_IFREG; ++ i->i_blocks = ((i->i_size - 1) >> 9) + 1; ++ SQUASHFS_I(i)->u.s1.fragment_start_block = frag_blk; ++ SQUASHFS_I(i)->u.s1.fragment_size = frag_size; ++ SQUASHFS_I(i)->u.s1.fragment_offset = inodep->offset; ++ SQUASHFS_I(i)->start_block = inodep->start_block; ++ SQUASHFS_I(i)->u.s1.block_list_start = next_block; ++ SQUASHFS_I(i)->offset = next_offset; ++ if (sblk->block_size > 4096) ++ i->i_data.a_ops = &squashfs_aops; ++ else ++ i->i_data.a_ops = &squashfs_aops_4K; ++ ++ TRACE("File inode %x:%x, start_block %llx, " ++ "block_list_start %llx, offset %x\n", ++ SQUASHFS_INODE_BLK(inode), offset, ++ inodep->start_block, next_block, ++ next_offset); ++ break; ++ } ++ case SQUASHFS_LREG_TYPE: { ++ unsigned int frag_size; ++ long long frag_blk; ++ struct squashfs_lreg_inode_header *inodep = &id.lreg; ++ struct squashfs_lreg_inode_header *sinodep = &sid.lreg; ++ ++ if (msblk->swap) { ++ if (!squashfs_get_cached_block(s, (char *) ++ sinodep, block, offset, ++ sizeof(*sinodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ SQUASHFS_SWAP_LREG_INODE_HEADER(inodep, sinodep); ++ } else ++ if (!squashfs_get_cached_block(s, (char *) ++ inodep, block, offset, ++ sizeof(*inodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ frag_blk = SQUASHFS_INVALID_BLK; ++ if (inodep->fragment != SQUASHFS_INVALID_FRAG && ++ !get_fragment_location(s, ++ inodep->fragment, &frag_blk, &frag_size)) ++ goto failed_read; ++ ++ if((i = squashfs_new_inode(s, inodeb)) == NULL) ++ goto failed_read1; ++ ++ i->i_nlink = inodep->nlink; ++ i->i_size = inodep->file_size; ++ i->i_fop = &generic_ro_fops; ++ i->i_mode |= S_IFREG; ++ i->i_blocks = ((i->i_size - 1) >> 9) + 1; ++ SQUASHFS_I(i)->u.s1.fragment_start_block = frag_blk; ++ SQUASHFS_I(i)->u.s1.fragment_size = frag_size; ++ SQUASHFS_I(i)->u.s1.fragment_offset = inodep->offset; ++ SQUASHFS_I(i)->start_block = inodep->start_block; ++ SQUASHFS_I(i)->u.s1.block_list_start = next_block; ++ SQUASHFS_I(i)->offset = next_offset; ++ if (sblk->block_size > 4096) ++ i->i_data.a_ops = &squashfs_aops; ++ else ++ i->i_data.a_ops = &squashfs_aops_4K; ++ ++ TRACE("File inode %x:%x, start_block %llx, " ++ "block_list_start %llx, offset %x\n", ++ SQUASHFS_INODE_BLK(inode), offset, ++ inodep->start_block, next_block, ++ next_offset); ++ break; ++ } ++ case SQUASHFS_DIR_TYPE: { ++ struct squashfs_dir_inode_header *inodep = &id.dir; ++ struct squashfs_dir_inode_header *sinodep = &sid.dir; ++ ++ if (msblk->swap) { ++ if (!squashfs_get_cached_block(s, (char *) ++ sinodep, block, offset, ++ sizeof(*sinodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ SQUASHFS_SWAP_DIR_INODE_HEADER(inodep, sinodep); ++ } else ++ if (!squashfs_get_cached_block(s, (char *) ++ inodep, block, offset, ++ sizeof(*inodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ if((i = squashfs_new_inode(s, inodeb)) == NULL) ++ goto failed_read1; ++ ++ i->i_nlink = inodep->nlink; ++ i->i_size = inodep->file_size; ++ i->i_op = &squashfs_dir_inode_ops; ++ i->i_fop = &squashfs_dir_ops; ++ i->i_mode |= S_IFDIR; ++ SQUASHFS_I(i)->start_block = inodep->start_block; ++ SQUASHFS_I(i)->offset = inodep->offset; ++ SQUASHFS_I(i)->u.s2.directory_index_count = 0; ++ SQUASHFS_I(i)->u.s2.parent_inode = inodep->parent_inode; ++ ++ TRACE("Directory inode %x:%x, start_block %x, offset " ++ "%x\n", SQUASHFS_INODE_BLK(inode), ++ offset, inodep->start_block, ++ inodep->offset); ++ break; ++ } ++ case SQUASHFS_LDIR_TYPE: { ++ struct squashfs_ldir_inode_header *inodep = &id.ldir; ++ struct squashfs_ldir_inode_header *sinodep = &sid.ldir; ++ ++ if (msblk->swap) { ++ if (!squashfs_get_cached_block(s, (char *) ++ sinodep, block, offset, ++ sizeof(*sinodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ SQUASHFS_SWAP_LDIR_INODE_HEADER(inodep, ++ sinodep); ++ } else ++ if (!squashfs_get_cached_block(s, (char *) ++ inodep, block, offset, ++ sizeof(*inodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ if((i = squashfs_new_inode(s, inodeb)) == NULL) ++ goto failed_read1; ++ ++ i->i_nlink = inodep->nlink; ++ i->i_size = inodep->file_size; ++ i->i_op = &squashfs_dir_inode_ops; ++ i->i_fop = &squashfs_dir_ops; ++ i->i_mode |= S_IFDIR; ++ SQUASHFS_I(i)->start_block = inodep->start_block; ++ SQUASHFS_I(i)->offset = inodep->offset; ++ SQUASHFS_I(i)->u.s2.directory_index_start = next_block; ++ SQUASHFS_I(i)->u.s2.directory_index_offset = ++ next_offset; ++ SQUASHFS_I(i)->u.s2.directory_index_count = ++ inodep->i_count; ++ SQUASHFS_I(i)->u.s2.parent_inode = inodep->parent_inode; ++ ++ TRACE("Long directory inode %x:%x, start_block %x, " ++ "offset %x\n", ++ SQUASHFS_INODE_BLK(inode), offset, ++ inodep->start_block, inodep->offset); ++ break; ++ } ++ case SQUASHFS_SYMLINK_TYPE: { ++ struct squashfs_symlink_inode_header *inodep = ++ &id.symlink; ++ struct squashfs_symlink_inode_header *sinodep = ++ &sid.symlink; ++ ++ if (msblk->swap) { ++ if (!squashfs_get_cached_block(s, (char *) ++ sinodep, block, offset, ++ sizeof(*sinodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ SQUASHFS_SWAP_SYMLINK_INODE_HEADER(inodep, ++ sinodep); ++ } else ++ if (!squashfs_get_cached_block(s, (char *) ++ inodep, block, offset, ++ sizeof(*inodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ if((i = squashfs_new_inode(s, inodeb)) == NULL) ++ goto failed_read1; ++ ++ i->i_nlink = inodep->nlink; ++ i->i_size = inodep->symlink_size; ++ i->i_op = &page_symlink_inode_operations; ++ i->i_data.a_ops = &squashfs_symlink_aops; ++ i->i_mode |= S_IFLNK; ++ SQUASHFS_I(i)->start_block = next_block; ++ SQUASHFS_I(i)->offset = next_offset; ++ ++ TRACE("Symbolic link inode %x:%x, start_block %llx, " ++ "offset %x\n", ++ SQUASHFS_INODE_BLK(inode), offset, ++ next_block, next_offset); ++ break; ++ } ++ case SQUASHFS_BLKDEV_TYPE: ++ case SQUASHFS_CHRDEV_TYPE: { ++ struct squashfs_dev_inode_header *inodep = &id.dev; ++ struct squashfs_dev_inode_header *sinodep = &sid.dev; ++ ++ if (msblk->swap) { ++ if (!squashfs_get_cached_block(s, (char *) ++ sinodep, block, offset, ++ sizeof(*sinodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ SQUASHFS_SWAP_DEV_INODE_HEADER(inodep, sinodep); ++ } else ++ if (!squashfs_get_cached_block(s, (char *) ++ inodep, block, offset, ++ sizeof(*inodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ if ((i = squashfs_new_inode(s, inodeb)) == NULL) ++ goto failed_read1; ++ ++ i->i_nlink = inodep->nlink; ++ i->i_mode |= (inodeb->inode_type == ++ SQUASHFS_CHRDEV_TYPE) ? S_IFCHR : ++ S_IFBLK; ++ init_special_inode(i, i->i_mode, ++ old_decode_dev(inodep->rdev)); ++ ++ TRACE("Device inode %x:%x, rdev %x\n", ++ SQUASHFS_INODE_BLK(inode), offset, ++ inodep->rdev); ++ break; ++ } ++ case SQUASHFS_FIFO_TYPE: ++ case SQUASHFS_SOCKET_TYPE: { ++ struct squashfs_ipc_inode_header *inodep = &id.ipc; ++ struct squashfs_ipc_inode_header *sinodep = &sid.ipc; ++ ++ if (msblk->swap) { ++ if (!squashfs_get_cached_block(s, (char *) ++ sinodep, block, offset, ++ sizeof(*sinodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ SQUASHFS_SWAP_IPC_INODE_HEADER(inodep, sinodep); ++ } else ++ if (!squashfs_get_cached_block(s, (char *) ++ inodep, block, offset, ++ sizeof(*inodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ if ((i = squashfs_new_inode(s, inodeb)) == NULL) ++ goto failed_read1; ++ ++ i->i_nlink = inodep->nlink; ++ i->i_mode |= (inodeb->inode_type == SQUASHFS_FIFO_TYPE) ++ ? S_IFIFO : S_IFSOCK; ++ init_special_inode(i, i->i_mode, 0); ++ break; ++ } ++ default: ++ ERROR("Unknown inode type %d in squashfs_iget!\n", ++ inodeb->inode_type); ++ goto failed_read1; ++ } ++ ++ insert_inode_hash(i); ++ return i; ++ ++failed_read: ++ ERROR("Unable to read inode [%llx:%x]\n", block, offset); ++ ++failed_read1: ++ return NULL; ++} ++ ++ ++static int read_fragment_index_table(struct super_block *s) ++{ ++ struct squashfs_sb_info *msblk = s->s_fs_info; ++ struct squashfs_super_block *sblk = &msblk->sblk; ++ ++ /* Allocate fragment index table */ ++ if (!(msblk->fragment_index = kmalloc(SQUASHFS_FRAGMENT_INDEX_BYTES ++ (sblk->fragments), GFP_KERNEL))) { ++ ERROR("Failed to allocate uid/gid table\n"); ++ return 0; ++ } ++ ++ if (SQUASHFS_FRAGMENT_INDEX_BYTES(sblk->fragments) && ++ !squashfs_read_data(s, (char *) ++ msblk->fragment_index, ++ sblk->fragment_table_start, ++ SQUASHFS_FRAGMENT_INDEX_BYTES ++ (sblk->fragments) | ++ SQUASHFS_COMPRESSED_BIT_BLOCK, NULL)) { ++ ERROR("unable to read fragment index table\n"); ++ return 0; ++ } ++ ++ if (msblk->swap) { ++ int i; ++ long long fragment; ++ ++ for (i = 0; i < SQUASHFS_FRAGMENT_INDEXES(sblk->fragments); ++ i++) { ++ SQUASHFS_SWAP_FRAGMENT_INDEXES((&fragment), ++ &msblk->fragment_index[i], 1); ++ msblk->fragment_index[i] = fragment; ++ } ++ } ++ ++ return 1; ++} ++ ++ ++static int supported_squashfs_filesystem(struct squashfs_sb_info *msblk, int silent) ++{ ++ struct squashfs_super_block *sblk = &msblk->sblk; ++ ++ msblk->iget = squashfs_iget; ++ msblk->read_blocklist = read_blocklist; ++ msblk->read_fragment_index_table = read_fragment_index_table; ++ ++ if (sblk->s_major == 1) { ++ if (!squashfs_1_0_supported(msblk)) { ++ SERROR("Major/Minor mismatch, Squashfs 1.0 filesystems " ++ "are unsupported\n"); ++ SERROR("Please recompile with " ++ "Squashfs 1.0 support enabled\n"); ++ return 0; ++ } ++ } else if (sblk->s_major == 2) { ++ if (!squashfs_2_0_supported(msblk)) { ++ SERROR("Major/Minor mismatch, Squashfs 2.0 filesystems " ++ "are unsupported\n"); ++ SERROR("Please recompile with " ++ "Squashfs 2.0 support enabled\n"); ++ return 0; ++ } ++ } else if(sblk->s_major != SQUASHFS_MAJOR || sblk->s_minor > ++ SQUASHFS_MINOR) { ++ SERROR("Major/Minor mismatch, trying to mount newer %d.%d " ++ "filesystem\n", sblk->s_major, sblk->s_minor); ++ SERROR("Please update your kernel\n"); ++ return 0; ++ } ++ ++ return 1; ++} ++ ++ ++static int squashfs_fill_super(struct super_block *s, void *data, int silent) ++{ ++ struct squashfs_sb_info *msblk; ++ struct squashfs_super_block *sblk; ++ int i; ++ char b[BDEVNAME_SIZE]; ++ struct inode *root; ++ ++ TRACE("Entered squashfs_read_superblock\n"); ++ ++ if (!(s->s_fs_info = kmalloc(sizeof(struct squashfs_sb_info), ++ GFP_KERNEL))) { ++ ERROR("Failed to allocate superblock\n"); ++ goto failure; ++ } ++ memset(s->s_fs_info, 0, sizeof(struct squashfs_sb_info)); ++ msblk = s->s_fs_info; ++ sblk = &msblk->sblk; ++ ++ msblk->devblksize = sb_min_blocksize(s, BLOCK_SIZE); ++ msblk->devblksize_log2 = ffz(~msblk->devblksize); ++ ++ init_MUTEX(&msblk->read_data_mutex); ++ init_MUTEX(&msblk->read_page_mutex); ++ init_MUTEX(&msblk->block_cache_mutex); ++ init_MUTEX(&msblk->fragment_mutex); ++ init_MUTEX(&msblk->meta_index_mutex); ++ ++ init_waitqueue_head(&msblk->waitq); ++ init_waitqueue_head(&msblk->fragment_wait_queue); ++ ++ if (!squashfs_read_data(s, (char *) sblk, SQUASHFS_START, ++ sizeof(struct squashfs_super_block) | ++ SQUASHFS_COMPRESSED_BIT_BLOCK, NULL)) { ++ SERROR("unable to read superblock\n"); ++ goto failed_mount; ++ } ++ ++ /* Check it is a SQUASHFS superblock */ ++ msblk->swap = 0; ++ if ((s->s_magic = sblk->s_magic) != SQUASHFS_MAGIC) { ++ if (sblk->s_magic == SQUASHFS_MAGIC_SWAP) { ++ struct squashfs_super_block ssblk; ++ ++ WARNING("Mounting a different endian SQUASHFS " ++ "filesystem on %s\n", bdevname(s->s_bdev, b)); ++ ++ SQUASHFS_SWAP_SUPER_BLOCK(&ssblk, sblk); ++ memcpy(sblk, &ssblk, sizeof(struct squashfs_super_block)); ++ msblk->swap = 1; ++ } else { ++ SERROR("Can't find a SQUASHFS superblock on %s\n", ++ bdevname(s->s_bdev, b)); ++ goto failed_mount; ++ } ++ } ++ ++ /* Check the MAJOR & MINOR versions */ ++ if(!supported_squashfs_filesystem(msblk, silent)) ++ goto failed_mount; ++ ++ TRACE("Found valid superblock on %s\n", bdevname(s->s_bdev, b)); ++ TRACE("Inodes are %scompressed\n", ++ SQUASHFS_UNCOMPRESSED_INODES ++ (sblk->flags) ? "un" : ""); ++ TRACE("Data is %scompressed\n", ++ SQUASHFS_UNCOMPRESSED_DATA(sblk->flags) ++ ? "un" : ""); ++ TRACE("Check data is %s present in the filesystem\n", ++ SQUASHFS_CHECK_DATA(sblk->flags) ? ++ "" : "not"); ++ TRACE("Filesystem size %lld bytes\n", sblk->bytes_used); ++ TRACE("Block size %d\n", sblk->block_size); ++ TRACE("Number of inodes %d\n", sblk->inodes); ++ if (sblk->s_major > 1) ++ TRACE("Number of fragments %d\n", sblk->fragments); ++ TRACE("Number of uids %d\n", sblk->no_uids); ++ TRACE("Number of gids %d\n", sblk->no_guids); ++ TRACE("sblk->inode_table_start %llx\n", sblk->inode_table_start); ++ TRACE("sblk->directory_table_start %llx\n", sblk->directory_table_start); ++ if (sblk->s_major > 1) ++ TRACE("sblk->fragment_table_start %llx\n", ++ sblk->fragment_table_start); ++ TRACE("sblk->uid_start %llx\n", sblk->uid_start); ++ ++ s->s_flags |= MS_RDONLY; ++ s->s_op = &squashfs_ops; ++ ++ /* Init inode_table block pointer array */ ++ if (!(msblk->block_cache = kmalloc(sizeof(struct squashfs_cache) * ++ SQUASHFS_CACHED_BLKS, GFP_KERNEL))) { ++ ERROR("Failed to allocate block cache\n"); ++ goto failed_mount; ++ } ++ ++ for (i = 0; i < SQUASHFS_CACHED_BLKS; i++) ++ msblk->block_cache[i].block = SQUASHFS_INVALID_BLK; ++ ++ msblk->next_cache = 0; ++ ++ /* Allocate read_data block */ ++ msblk->read_size = (sblk->block_size < SQUASHFS_METADATA_SIZE) ? ++ SQUASHFS_METADATA_SIZE : ++ sblk->block_size; ++ ++ if (!(msblk->read_data = kmalloc(msblk->read_size, GFP_KERNEL))) { ++ ERROR("Failed to allocate read_data block\n"); ++ goto failed_mount; ++ } ++ ++ /* Allocate read_page block */ ++ if (!(msblk->read_page = kmalloc(sblk->block_size, GFP_KERNEL))) { ++ ERROR("Failed to allocate read_page block\n"); ++ goto failed_mount; ++ } ++ ++ /* Allocate uid and gid tables */ ++ if (!(msblk->uid = kmalloc((sblk->no_uids + sblk->no_guids) * ++ sizeof(unsigned int), GFP_KERNEL))) { ++ ERROR("Failed to allocate uid/gid table\n"); ++ goto failed_mount; ++ } ++ msblk->guid = msblk->uid + sblk->no_uids; ++ ++ if (msblk->swap) { ++ unsigned int suid[sblk->no_uids + sblk->no_guids]; ++ ++ if (!squashfs_read_data(s, (char *) &suid, sblk->uid_start, ++ ((sblk->no_uids + sblk->no_guids) * ++ sizeof(unsigned int)) | ++ SQUASHFS_COMPRESSED_BIT_BLOCK, NULL)) { ++ ERROR("unable to read uid/gid table\n"); ++ goto failed_mount; ++ } ++ ++ SQUASHFS_SWAP_DATA(msblk->uid, suid, (sblk->no_uids + ++ sblk->no_guids), (sizeof(unsigned int) * 8)); ++ } else ++ if (!squashfs_read_data(s, (char *) msblk->uid, sblk->uid_start, ++ ((sblk->no_uids + sblk->no_guids) * ++ sizeof(unsigned int)) | ++ SQUASHFS_COMPRESSED_BIT_BLOCK, NULL)) { ++ ERROR("unable to read uid/gid table\n"); ++ goto failed_mount; ++ } ++ ++ ++ if (sblk->s_major == 1 && squashfs_1_0_supported(msblk)) ++ goto allocate_root; ++ ++ if (!(msblk->fragment = kmalloc(sizeof(struct squashfs_fragment_cache) * ++ SQUASHFS_CACHED_FRAGMENTS, GFP_KERNEL))) { ++ ERROR("Failed to allocate fragment block cache\n"); ++ goto failed_mount; ++ } ++ ++ for (i = 0; i < SQUASHFS_CACHED_FRAGMENTS; i++) { ++ msblk->fragment[i].locked = 0; ++ msblk->fragment[i].block = SQUASHFS_INVALID_BLK; ++ msblk->fragment[i].data = NULL; ++ } ++ ++ msblk->next_fragment = 0; ++ ++ /* Allocate fragment index table */ ++ if (msblk->read_fragment_index_table(s) == 0) ++ goto failed_mount; ++ ++allocate_root: ++ if ((root = (msblk->iget)(s, sblk->root_inode)) == NULL) ++ goto failed_mount; ++ ++ if ((s->s_root = d_alloc_root(root)) == NULL) { ++ ERROR("Root inode create failed\n"); ++ iput(root); ++ goto failed_mount; ++ } ++ ++ TRACE("Leaving squashfs_read_super\n"); ++ return 0; ++ ++failed_mount: ++ kfree(msblk->fragment_index); ++ kfree(msblk->fragment); ++ kfree(msblk->uid); ++ kfree(msblk->read_page); ++ kfree(msblk->read_data); ++ kfree(msblk->block_cache); ++ kfree(msblk->fragment_index_2); ++ kfree(s->s_fs_info); ++ s->s_fs_info = NULL; ++ return -EINVAL; ++ ++failure: ++ return -ENOMEM; ++} ++ ++ ++static int squashfs_statfs(struct dentry *dentry, struct kstatfs *buf) ++{ ++ struct super_block *s = dentry->d_sb; ++ struct squashfs_sb_info *msblk = s->s_fs_info; ++ struct squashfs_super_block *sblk = &msblk->sblk; ++ ++ TRACE("Entered squashfs_statfs\n"); ++ ++ buf->f_type = SQUASHFS_MAGIC; ++ buf->f_bsize = sblk->block_size; ++ buf->f_blocks = ((sblk->bytes_used - 1) >> sblk->block_log) + 1; ++ buf->f_bfree = buf->f_bavail = 0; ++ buf->f_files = sblk->inodes; ++ buf->f_ffree = 0; ++ buf->f_namelen = SQUASHFS_NAME_LEN; ++ ++ return 0; ++} ++ ++ ++static int squashfs_symlink_readpage(struct file *file, struct page *page) ++{ ++ struct inode *inode = page->mapping->host; ++ int index = page->index << PAGE_CACHE_SHIFT, length, bytes; ++ long long block = SQUASHFS_I(inode)->start_block; ++ int offset = SQUASHFS_I(inode)->offset; ++ void *pageaddr = kmap(page); ++ ++ TRACE("Entered squashfs_symlink_readpage, page index %ld, start block " ++ "%llx, offset %x\n", page->index, ++ SQUASHFS_I(inode)->start_block, ++ SQUASHFS_I(inode)->offset); ++ ++ for (length = 0; length < index; length += bytes) { ++ if (!(bytes = squashfs_get_cached_block(inode->i_sb, NULL, ++ block, offset, PAGE_CACHE_SIZE, &block, ++ &offset))) { ++ ERROR("Unable to read symbolic link [%llx:%x]\n", block, ++ offset); ++ goto skip_read; ++ } ++ } ++ ++ if (length != index) { ++ ERROR("(squashfs_symlink_readpage) length != index\n"); ++ bytes = 0; ++ goto skip_read; ++ } ++ ++ bytes = (i_size_read(inode) - length) > PAGE_CACHE_SIZE ? PAGE_CACHE_SIZE : ++ i_size_read(inode) - length; ++ ++ if (!(bytes = squashfs_get_cached_block(inode->i_sb, pageaddr, block, ++ offset, bytes, &block, &offset))) ++ ERROR("Unable to read symbolic link [%llx:%x]\n", block, offset); ++ ++skip_read: ++ memset(pageaddr + bytes, 0, PAGE_CACHE_SIZE - bytes); ++ kunmap(page); ++ SetPageUptodate(page); ++ unlock_page(page); ++ ++ return 0; ++} ++ ++ ++struct meta_index *locate_meta_index(struct inode *inode, int index, int offset) ++{ ++ struct meta_index *meta = NULL; ++ struct squashfs_sb_info *msblk = inode->i_sb->s_fs_info; ++ int i; ++ ++ down(&msblk->meta_index_mutex); ++ ++ TRACE("locate_meta_index: index %d, offset %d\n", index, offset); ++ ++ if(msblk->meta_index == NULL) ++ goto not_allocated; ++ ++ for (i = 0; i < SQUASHFS_META_NUMBER; i ++) ++ if (msblk->meta_index[i].inode_number == inode->i_ino && ++ msblk->meta_index[i].offset >= offset && ++ msblk->meta_index[i].offset <= index && ++ msblk->meta_index[i].locked == 0) { ++ TRACE("locate_meta_index: entry %d, offset %d\n", i, ++ msblk->meta_index[i].offset); ++ meta = &msblk->meta_index[i]; ++ offset = meta->offset; ++ } ++ ++ if (meta) ++ meta->locked = 1; ++ ++not_allocated: ++ up(&msblk->meta_index_mutex); ++ ++ return meta; ++} ++ ++ ++struct meta_index *empty_meta_index(struct inode *inode, int offset, int skip) ++{ ++ struct squashfs_sb_info *msblk = inode->i_sb->s_fs_info; ++ struct meta_index *meta = NULL; ++ int i; ++ ++ down(&msblk->meta_index_mutex); ++ ++ TRACE("empty_meta_index: offset %d, skip %d\n", offset, skip); ++ ++ if(msblk->meta_index == NULL) { ++ if (!(msblk->meta_index = kmalloc(sizeof(struct meta_index) * ++ SQUASHFS_META_NUMBER, GFP_KERNEL))) { ++ ERROR("Failed to allocate meta_index\n"); ++ goto failed; ++ } ++ for(i = 0; i < SQUASHFS_META_NUMBER; i++) { ++ msblk->meta_index[i].inode_number = 0; ++ msblk->meta_index[i].locked = 0; ++ } ++ msblk->next_meta_index = 0; ++ } ++ ++ for(i = SQUASHFS_META_NUMBER; i && ++ msblk->meta_index[msblk->next_meta_index].locked; i --) ++ msblk->next_meta_index = (msblk->next_meta_index + 1) % ++ SQUASHFS_META_NUMBER; ++ ++ if(i == 0) { ++ TRACE("empty_meta_index: failed!\n"); ++ goto failed; ++ } ++ ++ TRACE("empty_meta_index: returned meta entry %d, %p\n", ++ msblk->next_meta_index, ++ &msblk->meta_index[msblk->next_meta_index]); ++ ++ meta = &msblk->meta_index[msblk->next_meta_index]; ++ msblk->next_meta_index = (msblk->next_meta_index + 1) % ++ SQUASHFS_META_NUMBER; ++ ++ meta->inode_number = inode->i_ino; ++ meta->offset = offset; ++ meta->skip = skip; ++ meta->entries = 0; ++ meta->locked = 1; ++ ++failed: ++ up(&msblk->meta_index_mutex); ++ return meta; ++} ++ ++ ++void release_meta_index(struct inode *inode, struct meta_index *meta) ++{ ++ meta->locked = 0; ++} ++ ++ ++static int read_block_index(struct super_block *s, int blocks, char *block_list, ++ long long *start_block, int *offset) ++{ ++ struct squashfs_sb_info *msblk = s->s_fs_info; ++ unsigned int *block_listp; ++ int block = 0; ++ ++ if (msblk->swap) { ++ char sblock_list[blocks << 2]; ++ ++ if (!squashfs_get_cached_block(s, sblock_list, *start_block, ++ *offset, blocks << 2, start_block, offset)) { ++ ERROR("Unable to read block list [%llx:%x]\n", ++ *start_block, *offset); ++ goto failure; ++ } ++ SQUASHFS_SWAP_INTS(((unsigned int *)block_list), ++ ((unsigned int *)sblock_list), blocks); ++ } else ++ if (!squashfs_get_cached_block(s, block_list, *start_block, ++ *offset, blocks << 2, start_block, offset)) { ++ ERROR("Unable to read block list [%llx:%x]\n", ++ *start_block, *offset); ++ goto failure; ++ } ++ ++ for (block_listp = (unsigned int *) block_list; blocks; ++ block_listp++, blocks --) ++ block += SQUASHFS_COMPRESSED_SIZE_BLOCK(*block_listp); ++ ++ return block; ++ ++failure: ++ return -1; ++} ++ ++ ++#define SIZE 256 ++ ++static inline int calculate_skip(int blocks) { ++ int skip = (blocks - 1) / ((SQUASHFS_SLOTS * SQUASHFS_META_ENTRIES + 1) * SQUASHFS_META_INDEXES); ++ return skip >= 7 ? 7 : skip + 1; ++} ++ ++ ++static int get_meta_index(struct inode *inode, int index, ++ long long *index_block, int *index_offset, ++ long long *data_block, char *block_list) ++{ ++ struct squashfs_sb_info *msblk = inode->i_sb->s_fs_info; ++ struct squashfs_super_block *sblk = &msblk->sblk; ++ int skip = calculate_skip(i_size_read(inode) >> sblk->block_log); ++ int offset = 0; ++ struct meta_index *meta; ++ struct meta_entry *meta_entry; ++ long long cur_index_block = SQUASHFS_I(inode)->u.s1.block_list_start; ++ int cur_offset = SQUASHFS_I(inode)->offset; ++ long long cur_data_block = SQUASHFS_I(inode)->start_block; ++ int i; ++ ++ index /= SQUASHFS_META_INDEXES * skip; ++ ++ while ( offset < index ) { ++ meta = locate_meta_index(inode, index, offset + 1); ++ ++ if (meta == NULL) { ++ if ((meta = empty_meta_index(inode, offset + 1, ++ skip)) == NULL) ++ goto all_done; ++ } else { ++ offset = index < meta->offset + meta->entries ? index : ++ meta->offset + meta->entries - 1; ++ meta_entry = &meta->meta_entry[offset - meta->offset]; ++ cur_index_block = meta_entry->index_block + sblk->inode_table_start; ++ cur_offset = meta_entry->offset; ++ cur_data_block = meta_entry->data_block; ++ TRACE("get_meta_index: offset %d, meta->offset %d, " ++ "meta->entries %d\n", offset, meta->offset, ++ meta->entries); ++ TRACE("get_meta_index: index_block 0x%llx, offset 0x%x" ++ " data_block 0x%llx\n", cur_index_block, ++ cur_offset, cur_data_block); ++ } ++ ++ for (i = meta->offset + meta->entries; i <= index && ++ i < meta->offset + SQUASHFS_META_ENTRIES; i++) { ++ int blocks = skip * SQUASHFS_META_INDEXES; ++ ++ while (blocks) { ++ int block = blocks > (SIZE >> 2) ? (SIZE >> 2) : ++ blocks; ++ int res = read_block_index(inode->i_sb, block, ++ block_list, &cur_index_block, ++ &cur_offset); ++ ++ if (res == -1) ++ goto failed; ++ ++ cur_data_block += res; ++ blocks -= block; ++ } ++ ++ meta_entry = &meta->meta_entry[i - meta->offset]; ++ meta_entry->index_block = cur_index_block - sblk->inode_table_start; ++ meta_entry->offset = cur_offset; ++ meta_entry->data_block = cur_data_block; ++ meta->entries ++; ++ offset ++; ++ } ++ ++ TRACE("get_meta_index: meta->offset %d, meta->entries %d\n", ++ meta->offset, meta->entries); ++ ++ release_meta_index(inode, meta); ++ } ++ ++all_done: ++ *index_block = cur_index_block; ++ *index_offset = cur_offset; ++ *data_block = cur_data_block; ++ ++ return offset * SQUASHFS_META_INDEXES * skip; ++ ++failed: ++ release_meta_index(inode, meta); ++ return -1; ++} ++ ++ ++static long long read_blocklist(struct inode *inode, int index, ++ int readahead_blks, char *block_list, ++ unsigned short **block_p, unsigned int *bsize) ++{ ++ long long block_ptr; ++ int offset; ++ long long block; ++ int res = get_meta_index(inode, index, &block_ptr, &offset, &block, ++ block_list); ++ ++ TRACE("read_blocklist: res %d, index %d, block_ptr 0x%llx, offset" ++ " 0x%x, block 0x%llx\n", res, index, block_ptr, offset, ++ block); ++ ++ if(res == -1) ++ goto failure; ++ ++ index -= res; ++ ++ while ( index ) { ++ int blocks = index > (SIZE >> 2) ? (SIZE >> 2) : index; ++ int res = read_block_index(inode->i_sb, blocks, block_list, ++ &block_ptr, &offset); ++ if (res == -1) ++ goto failure; ++ block += res; ++ index -= blocks; ++ } ++ ++ if (read_block_index(inode->i_sb, 1, block_list, ++ &block_ptr, &offset) == -1) ++ goto failure; ++ *bsize = *((unsigned int *) block_list); ++ ++ return block; ++ ++failure: ++ return 0; ++} ++ ++ ++static int squashfs_readpage(struct file *file, struct page *page) ++{ ++ struct inode *inode = page->mapping->host; ++ struct squashfs_sb_info *msblk = inode->i_sb->s_fs_info; ++ struct squashfs_super_block *sblk = &msblk->sblk; ++ unsigned char block_list[SIZE]; ++ long long block; ++ unsigned int bsize, i = 0, bytes = 0, byte_offset = 0; ++ int index = page->index >> (sblk->block_log - PAGE_CACHE_SHIFT); ++ void *pageaddr; ++ struct squashfs_fragment_cache *fragment = NULL; ++ char *data_ptr = msblk->read_page; ++ ++ int mask = (1 << (sblk->block_log - PAGE_CACHE_SHIFT)) - 1; ++ int start_index = page->index & ~mask; ++ int end_index = start_index | mask; ++ ++ TRACE("Entered squashfs_readpage, page index %lx, start block %llx\n", ++ page->index, ++ SQUASHFS_I(inode)->start_block); ++ ++ if (page->index >= ((i_size_read(inode) + PAGE_CACHE_SIZE - 1) >> ++ PAGE_CACHE_SHIFT)) ++ goto skip_read; ++ ++ if (SQUASHFS_I(inode)->u.s1.fragment_start_block == SQUASHFS_INVALID_BLK ++ || index < (i_size_read(inode) >> ++ sblk->block_log)) { ++ if ((block = (msblk->read_blocklist)(inode, index, 1, ++ block_list, NULL, &bsize)) == 0) ++ goto skip_read; ++ ++ down(&msblk->read_page_mutex); ++ ++ if (!(bytes = squashfs_read_data(inode->i_sb, msblk->read_page, ++ block, bsize, NULL))) { ++ ERROR("Unable to read page, block %llx, size %x\n", block, ++ bsize); ++ up(&msblk->read_page_mutex); ++ goto skip_read; ++ } ++ } else { ++ if ((fragment = get_cached_fragment(inode->i_sb, ++ SQUASHFS_I(inode)-> ++ u.s1.fragment_start_block, ++ SQUASHFS_I(inode)->u.s1.fragment_size)) ++ == NULL) { ++ ERROR("Unable to read page, block %llx, size %x\n", ++ SQUASHFS_I(inode)-> ++ u.s1.fragment_start_block, ++ (int) SQUASHFS_I(inode)-> ++ u.s1.fragment_size); ++ goto skip_read; ++ } ++ bytes = SQUASHFS_I(inode)->u.s1.fragment_offset + ++ (i_size_read(inode) & (sblk->block_size ++ - 1)); ++ byte_offset = SQUASHFS_I(inode)->u.s1.fragment_offset; ++ data_ptr = fragment->data; ++ } ++ ++ for (i = start_index; i <= end_index && byte_offset < bytes; ++ i++, byte_offset += PAGE_CACHE_SIZE) { ++ struct page *push_page; ++ int available_bytes = (bytes - byte_offset) > PAGE_CACHE_SIZE ? ++ PAGE_CACHE_SIZE : bytes - byte_offset; ++ ++ TRACE("bytes %d, i %d, byte_offset %d, available_bytes %d\n", ++ bytes, i, byte_offset, available_bytes); ++ ++ if (i == page->index) { ++ pageaddr = kmap_atomic(page, KM_USER0); ++ memcpy(pageaddr, data_ptr + byte_offset, ++ available_bytes); ++ memset(pageaddr + available_bytes, 0, ++ PAGE_CACHE_SIZE - available_bytes); ++ kunmap_atomic(pageaddr, KM_USER0); ++ flush_dcache_page(page); ++ SetPageUptodate(page); ++ unlock_page(page); ++ } else if ((push_page = ++ grab_cache_page_nowait(page->mapping, i))) { ++ pageaddr = kmap_atomic(push_page, KM_USER0); ++ ++ memcpy(pageaddr, data_ptr + byte_offset, ++ available_bytes); ++ memset(pageaddr + available_bytes, 0, ++ PAGE_CACHE_SIZE - available_bytes); ++ kunmap_atomic(pageaddr, KM_USER0); ++ flush_dcache_page(push_page); ++ SetPageUptodate(push_page); ++ unlock_page(push_page); ++ page_cache_release(push_page); ++ } ++ } ++ ++ if (SQUASHFS_I(inode)->u.s1.fragment_start_block == SQUASHFS_INVALID_BLK ++ || index < (i_size_read(inode) >> ++ sblk->block_log)) ++ up(&msblk->read_page_mutex); ++ else ++ release_cached_fragment(msblk, fragment); ++ ++ return 0; ++ ++skip_read: ++ pageaddr = kmap_atomic(page, KM_USER0); ++ memset(pageaddr + bytes, 0, PAGE_CACHE_SIZE - bytes); ++ kunmap_atomic(pageaddr, KM_USER0); ++ flush_dcache_page(page); ++ SetPageUptodate(page); ++ unlock_page(page); ++ ++ return 0; ++} ++ ++ ++static int squashfs_readpage4K(struct file *file, struct page *page) ++{ ++ struct inode *inode = page->mapping->host; ++ struct squashfs_sb_info *msblk = inode->i_sb->s_fs_info; ++ struct squashfs_super_block *sblk = &msblk->sblk; ++ unsigned char block_list[SIZE]; ++ long long block; ++ unsigned int bsize, bytes = 0; ++ void *pageaddr; ++ ++ TRACE("Entered squashfs_readpage4K, page index %lx, start block %llx\n", ++ page->index, ++ SQUASHFS_I(inode)->start_block); ++ ++ if (page->index >= ((i_size_read(inode) + PAGE_CACHE_SIZE - 1) >> ++ PAGE_CACHE_SHIFT)) { ++ pageaddr = kmap_atomic(page, KM_USER0); ++ goto skip_read; ++ } ++ ++ if (SQUASHFS_I(inode)->u.s1.fragment_start_block == SQUASHFS_INVALID_BLK ++ || page->index < (i_size_read(inode) >> ++ sblk->block_log)) { ++ block = (msblk->read_blocklist)(inode, page->index, 1, ++ block_list, NULL, &bsize); ++ ++ down(&msblk->read_page_mutex); ++ bytes = squashfs_read_data(inode->i_sb, msblk->read_page, block, ++ bsize, NULL); ++ pageaddr = kmap_atomic(page, KM_USER0); ++ if (bytes) ++ memcpy(pageaddr, msblk->read_page, bytes); ++ else ++ ERROR("Unable to read page, block %llx, size %x\n", ++ block, bsize); ++ up(&msblk->read_page_mutex); ++ } else { ++ struct squashfs_fragment_cache *fragment = ++ get_cached_fragment(inode->i_sb, ++ SQUASHFS_I(inode)-> ++ u.s1.fragment_start_block, ++ SQUASHFS_I(inode)-> u.s1.fragment_size); ++ pageaddr = kmap_atomic(page, KM_USER0); ++ if (fragment) { ++ bytes = i_size_read(inode) & (sblk->block_size - 1); ++ memcpy(pageaddr, fragment->data + SQUASHFS_I(inode)-> ++ u.s1.fragment_offset, bytes); ++ release_cached_fragment(msblk, fragment); ++ } else ++ ERROR("Unable to read page, block %llx, size %x\n", ++ SQUASHFS_I(inode)-> ++ u.s1.fragment_start_block, (int) ++ SQUASHFS_I(inode)-> u.s1.fragment_size); ++ } ++ ++skip_read: ++ memset(pageaddr + bytes, 0, PAGE_CACHE_SIZE - bytes); ++ kunmap_atomic(pageaddr, KM_USER0); ++ flush_dcache_page(page); ++ SetPageUptodate(page); ++ unlock_page(page); ++ ++ return 0; ++} ++ ++ ++static int get_dir_index_using_offset(struct super_block *s, long long ++ *next_block, unsigned int *next_offset, ++ long long index_start, ++ unsigned int index_offset, int i_count, ++ long long f_pos) ++{ ++ struct squashfs_sb_info *msblk = s->s_fs_info; ++ struct squashfs_super_block *sblk = &msblk->sblk; ++ int i, length = 0; ++ struct squashfs_dir_index index; ++ ++ TRACE("Entered get_dir_index_using_offset, i_count %d, f_pos %d\n", ++ i_count, (unsigned int) f_pos); ++ ++ f_pos =- 3; ++ if (f_pos == 0) ++ goto finish; ++ ++ for (i = 0; i < i_count; i++) { ++ if (msblk->swap) { ++ struct squashfs_dir_index sindex; ++ squashfs_get_cached_block(s, (char *) &sindex, ++ index_start, index_offset, ++ sizeof(sindex), &index_start, ++ &index_offset); ++ SQUASHFS_SWAP_DIR_INDEX(&index, &sindex); ++ } else ++ squashfs_get_cached_block(s, (char *) &index, ++ index_start, index_offset, ++ sizeof(index), &index_start, ++ &index_offset); ++ ++ if (index.index > f_pos) ++ break; ++ ++ squashfs_get_cached_block(s, NULL, index_start, index_offset, ++ index.size + 1, &index_start, ++ &index_offset); ++ ++ length = index.index; ++ *next_block = index.start_block + sblk->directory_table_start; ++ } ++ ++ *next_offset = (length + *next_offset) % SQUASHFS_METADATA_SIZE; ++ ++finish: ++ return length + 3; ++} ++ ++ ++static int get_dir_index_using_name(struct super_block *s, long long ++ *next_block, unsigned int *next_offset, ++ long long index_start, ++ unsigned int index_offset, int i_count, ++ const char *name, int size) ++{ ++ struct squashfs_sb_info *msblk = s->s_fs_info; ++ struct squashfs_super_block *sblk = &msblk->sblk; ++ int i, length = 0; ++ char buffer[sizeof(struct squashfs_dir_index) + SQUASHFS_NAME_LEN + 1]; ++ struct squashfs_dir_index *index = (struct squashfs_dir_index *) buffer; ++ char str[SQUASHFS_NAME_LEN + 1]; ++ ++ TRACE("Entered get_dir_index_using_name, i_count %d\n", i_count); ++ ++ strncpy(str, name, size); ++ str[size] = '\0'; ++ ++ for (i = 0; i < i_count; i++) { ++ if (msblk->swap) { ++ struct squashfs_dir_index sindex; ++ squashfs_get_cached_block(s, (char *) &sindex, ++ index_start, index_offset, ++ sizeof(sindex), &index_start, ++ &index_offset); ++ SQUASHFS_SWAP_DIR_INDEX(index, &sindex); ++ } else ++ squashfs_get_cached_block(s, (char *) index, ++ index_start, index_offset, ++ sizeof(struct squashfs_dir_index), ++ &index_start, &index_offset); ++ ++ squashfs_get_cached_block(s, index->name, index_start, ++ index_offset, index->size + 1, ++ &index_start, &index_offset); ++ ++ index->name[index->size + 1] = '\0'; ++ ++ if (strcmp(index->name, str) > 0) ++ break; ++ ++ length = index->index; ++ *next_block = index->start_block + sblk->directory_table_start; ++ } ++ ++ *next_offset = (length + *next_offset) % SQUASHFS_METADATA_SIZE; ++ return length + 3; ++} ++ ++ ++static int squashfs_readdir(struct file *file, void *dirent, filldir_t filldir) ++{ ++ struct inode *i = file->f_dentry->d_inode; ++ struct squashfs_sb_info *msblk = i->i_sb->s_fs_info; ++ struct squashfs_super_block *sblk = &msblk->sblk; ++ long long next_block = SQUASHFS_I(i)->start_block + ++ sblk->directory_table_start; ++ int next_offset = SQUASHFS_I(i)->offset, length = 0, dirs_read = 0, ++ dir_count; ++ struct squashfs_dir_header dirh; ++ char buffer[sizeof(struct squashfs_dir_entry) + SQUASHFS_NAME_LEN + 1]; ++ struct squashfs_dir_entry *dire = (struct squashfs_dir_entry *) buffer; ++ ++ TRACE("Entered squashfs_readdir [%llx:%x]\n", next_block, next_offset); ++ ++ while(file->f_pos < 3) { ++ char *name; ++ int size, i_ino; ++ ++ if(file->f_pos == 0) { ++ name = "."; ++ size = 1; ++ i_ino = i->i_ino; ++ } else { ++ name = ".."; ++ size = 2; ++ i_ino = SQUASHFS_I(i)->u.s2.parent_inode; ++ } ++ TRACE("Calling filldir(%x, %s, %d, %d, %d, %d)\n", ++ (unsigned int) dirent, name, size, (int) ++ file->f_pos, i_ino, ++ squashfs_filetype_table[1]); ++ ++ if (filldir(dirent, name, size, ++ file->f_pos, i_ino, ++ squashfs_filetype_table[1]) < 0) { ++ TRACE("Filldir returned less than 0\n"); ++ goto finish; ++ } ++ file->f_pos += size; ++ dirs_read++; ++ } ++ ++ length = get_dir_index_using_offset(i->i_sb, &next_block, &next_offset, ++ SQUASHFS_I(i)->u.s2.directory_index_start, ++ SQUASHFS_I(i)->u.s2.directory_index_offset, ++ SQUASHFS_I(i)->u.s2.directory_index_count, ++ file->f_pos); ++ ++ while (length < i_size_read(i)) { ++ /* read directory header */ ++ if (msblk->swap) { ++ struct squashfs_dir_header sdirh; ++ ++ if (!squashfs_get_cached_block(i->i_sb, (char *) &sdirh, ++ next_block, next_offset, sizeof(sdirh), ++ &next_block, &next_offset)) ++ goto failed_read; ++ ++ length += sizeof(sdirh); ++ SQUASHFS_SWAP_DIR_HEADER(&dirh, &sdirh); ++ } else { ++ if (!squashfs_get_cached_block(i->i_sb, (char *) &dirh, ++ next_block, next_offset, sizeof(dirh), ++ &next_block, &next_offset)) ++ goto failed_read; ++ ++ length += sizeof(dirh); ++ } ++ ++ dir_count = dirh.count + 1; ++ while (dir_count--) { ++ if (msblk->swap) { ++ struct squashfs_dir_entry sdire; ++ if (!squashfs_get_cached_block(i->i_sb, (char *) ++ &sdire, next_block, next_offset, ++ sizeof(sdire), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ length += sizeof(sdire); ++ SQUASHFS_SWAP_DIR_ENTRY(dire, &sdire); ++ } else { ++ if (!squashfs_get_cached_block(i->i_sb, (char *) ++ dire, next_block, next_offset, ++ sizeof(*dire), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ length += sizeof(*dire); ++ } ++ ++ if (!squashfs_get_cached_block(i->i_sb, dire->name, ++ next_block, next_offset, ++ dire->size + 1, &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ length += dire->size + 1; ++ ++ if (file->f_pos >= length) ++ continue; ++ ++ dire->name[dire->size + 1] = '\0'; ++ ++ TRACE("Calling filldir(%x, %s, %d, %d, %x:%x, %d, %d)\n", ++ (unsigned int) dirent, dire->name, ++ dire->size + 1, (int) file->f_pos, ++ dirh.start_block, dire->offset, ++ dirh.inode_number + dire->inode_number, ++ squashfs_filetype_table[dire->type]); ++ ++ if (filldir(dirent, dire->name, dire->size + 1, ++ file->f_pos, ++ dirh.inode_number + dire->inode_number, ++ squashfs_filetype_table[dire->type]) ++ < 0) { ++ TRACE("Filldir returned less than 0\n"); ++ goto finish; ++ } ++ file->f_pos = length; ++ dirs_read++; ++ } ++ } ++ ++finish: ++ return dirs_read; ++ ++failed_read: ++ ERROR("Unable to read directory block [%llx:%x]\n", next_block, ++ next_offset); ++ return 0; ++} ++ ++ ++static struct dentry *squashfs_lookup(struct inode *i, struct dentry *dentry, ++ struct nameidata *nd) ++{ ++ const unsigned char *name = dentry->d_name.name; ++ int len = dentry->d_name.len; ++ struct inode *inode = NULL; ++ struct squashfs_sb_info *msblk = i->i_sb->s_fs_info; ++ struct squashfs_super_block *sblk = &msblk->sblk; ++ long long next_block = SQUASHFS_I(i)->start_block + ++ sblk->directory_table_start; ++ int next_offset = SQUASHFS_I(i)->offset, length = 0, ++ dir_count; ++ struct squashfs_dir_header dirh; ++ char buffer[sizeof(struct squashfs_dir_entry) + SQUASHFS_NAME_LEN]; ++ struct squashfs_dir_entry *dire = (struct squashfs_dir_entry *) buffer; ++ ++ TRACE("Entered squashfs_lookup [%llx:%x]\n", next_block, next_offset); ++ ++ if (len > SQUASHFS_NAME_LEN) ++ goto exit_loop; ++ ++ length = get_dir_index_using_name(i->i_sb, &next_block, &next_offset, ++ SQUASHFS_I(i)->u.s2.directory_index_start, ++ SQUASHFS_I(i)->u.s2.directory_index_offset, ++ SQUASHFS_I(i)->u.s2.directory_index_count, name, ++ len); ++ ++ while (length < i_size_read(i)) { ++ /* read directory header */ ++ if (msblk->swap) { ++ struct squashfs_dir_header sdirh; ++ if (!squashfs_get_cached_block(i->i_sb, (char *) &sdirh, ++ next_block, next_offset, sizeof(sdirh), ++ &next_block, &next_offset)) ++ goto failed_read; ++ ++ length += sizeof(sdirh); ++ SQUASHFS_SWAP_DIR_HEADER(&dirh, &sdirh); ++ } else { ++ if (!squashfs_get_cached_block(i->i_sb, (char *) &dirh, ++ next_block, next_offset, sizeof(dirh), ++ &next_block, &next_offset)) ++ goto failed_read; ++ ++ length += sizeof(dirh); ++ } ++ ++ dir_count = dirh.count + 1; ++ while (dir_count--) { ++ if (msblk->swap) { ++ struct squashfs_dir_entry sdire; ++ if (!squashfs_get_cached_block(i->i_sb, (char *) ++ &sdire, next_block,next_offset, ++ sizeof(sdire), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ length += sizeof(sdire); ++ SQUASHFS_SWAP_DIR_ENTRY(dire, &sdire); ++ } else { ++ if (!squashfs_get_cached_block(i->i_sb, (char *) ++ dire, next_block,next_offset, ++ sizeof(*dire), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ length += sizeof(*dire); ++ } ++ ++ if (!squashfs_get_cached_block(i->i_sb, dire->name, ++ next_block, next_offset, dire->size + 1, ++ &next_block, &next_offset)) ++ goto failed_read; ++ ++ length += dire->size + 1; ++ ++ if (name[0] < dire->name[0]) ++ goto exit_loop; ++ ++ if ((len == dire->size + 1) && !strncmp(name, ++ dire->name, len)) { ++ squashfs_inode_t ino = ++ SQUASHFS_MKINODE(dirh.start_block, ++ dire->offset); ++ ++ TRACE("calling squashfs_iget for directory " ++ "entry %s, inode %x:%x, %d\n", name, ++ dirh.start_block, dire->offset, ++ dirh.inode_number + dire->inode_number); ++ ++ inode = (msblk->iget)(i->i_sb, ino); ++ ++ goto exit_loop; ++ } ++ } ++ } ++ ++exit_loop: ++ d_add(dentry, inode); ++ return ERR_PTR(0); ++ ++failed_read: ++ ERROR("Unable to read directory block [%llx:%x]\n", next_block, ++ next_offset); ++ goto exit_loop; ++} ++ ++ ++static void squashfs_put_super(struct super_block *s) ++{ ++ int i; ++ ++ if (s->s_fs_info) { ++ struct squashfs_sb_info *sbi = s->s_fs_info; ++ if (sbi->block_cache) ++ for (i = 0; i < SQUASHFS_CACHED_BLKS; i++) ++ if (sbi->block_cache[i].block != ++ SQUASHFS_INVALID_BLK) ++ kfree(sbi->block_cache[i].data); ++ if (sbi->fragment) ++ for (i = 0; i < SQUASHFS_CACHED_FRAGMENTS; i++) ++ SQUASHFS_FREE(sbi->fragment[i].data); ++ kfree(sbi->fragment); ++ kfree(sbi->block_cache); ++ kfree(sbi->read_data); ++ kfree(sbi->read_page); ++ kfree(sbi->uid); ++ kfree(sbi->fragment_index); ++ kfree(sbi->fragment_index_2); ++ kfree(sbi->meta_index); ++ kfree(s->s_fs_info); ++ s->s_fs_info = NULL; ++ } ++} ++ ++ ++static int squashfs_get_sb(struct file_system_type *fs_type, ++ int flags, const char *dev_name, void *data, struct vfsmount *mnt) ++{ ++ return get_sb_bdev(fs_type, flags, dev_name, data, squashfs_fill_super, mnt); ++} ++ ++ ++static int __init init_squashfs_fs(void) ++{ ++ int err = init_inodecache(); ++ if (err) ++ goto out; ++ ++ printk(KERN_INFO "squashfs: version 3.0 (2006/03/15) " ++ "Phillip Lougher\n"); ++ ++ if (!(stream.workspace = vmalloc(zlib_inflate_workspacesize()))) { ++ ERROR("Failed to allocate zlib workspace\n"); ++ destroy_inodecache(); ++ err = -ENOMEM; ++ goto out; ++ } ++ ++ if ((err = register_filesystem(&squashfs_fs_type))) { ++ vfree(stream.workspace); ++ destroy_inodecache(); ++ } ++ ++out: ++ return err; ++} ++ ++ ++static void __exit exit_squashfs_fs(void) ++{ ++ vfree(stream.workspace); ++ unregister_filesystem(&squashfs_fs_type); ++ destroy_inodecache(); ++} ++ ++ ++static kmem_cache_t * squashfs_inode_cachep; ++ ++ ++static struct inode *squashfs_alloc_inode(struct super_block *sb) ++{ ++ struct squashfs_inode_info *ei; ++ ei = kmem_cache_alloc(squashfs_inode_cachep, GFP_KERNEL); ++ if (!ei) ++ return NULL; ++ return &ei->vfs_inode; ++} ++ ++ ++static void squashfs_destroy_inode(struct inode *inode) ++{ ++ kmem_cache_free(squashfs_inode_cachep, SQUASHFS_I(inode)); ++} ++ ++ ++static void init_once(void * foo, kmem_cache_t * cachep, unsigned long flags) ++{ ++ struct squashfs_inode_info *ei = foo; ++ ++ if ((flags & (SLAB_CTOR_VERIFY|SLAB_CTOR_CONSTRUCTOR)) == ++ SLAB_CTOR_CONSTRUCTOR) ++ inode_init_once(&ei->vfs_inode); ++} ++ ++ ++static int __init init_inodecache(void) ++{ ++ squashfs_inode_cachep = kmem_cache_create("squashfs_inode_cache", ++ sizeof(struct squashfs_inode_info), ++ 0, SLAB_HWCACHE_ALIGN|SLAB_RECLAIM_ACCOUNT, ++ init_once, NULL); ++ if (squashfs_inode_cachep == NULL) ++ return -ENOMEM; ++ return 0; ++} ++ ++ ++static void destroy_inodecache(void) ++{ ++ kmem_cache_destroy(squashfs_inode_cachep); ++} ++ ++ ++module_init(init_squashfs_fs); ++module_exit(exit_squashfs_fs); ++MODULE_DESCRIPTION("squashfs, a compressed read-only filesystem"); ++MODULE_AUTHOR("Phillip Lougher <phillip@lougher.org.uk>"); +Index: git/fs/squashfs/Makefile +=================================================================== +--- /dev/null 1970-01-01 00:00:00.000000000 +0000 ++++ git/fs/squashfs/Makefile 2006-10-31 21:26:45.000000000 +0000 +@@ -0,0 +1,7 @@ ++# ++# Makefile for the linux squashfs routines. ++# ++ ++obj-$(CONFIG_SQUASHFS) += squashfs.o ++squashfs-y += inode.o ++squashfs-y += squashfs2_0.o +Index: git/fs/squashfs/squashfs2_0.c +=================================================================== +--- /dev/null 1970-01-01 00:00:00.000000000 +0000 ++++ git/fs/squashfs/squashfs2_0.c 2006-11-01 18:01:49.000000000 +0000 +@@ -0,0 +1,757 @@ ++/* ++ * Squashfs - a compressed read only filesystem for Linux ++ * ++ * Copyright (c) 2002, 2003, 2004, 2005, 2006 ++ * Phillip Lougher <phillip@lougher.org.uk> ++ * ++ * 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, ++ * or (at your option) any later version. ++ * ++ * This program is distributed in the hope that it will be useful, ++ * but WITHOUT ANY WARRANTY; without even the implied warranty of ++ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ++ * GNU General Public License for more details. ++ * ++ * You should have received a copy of the GNU General Public License ++ * along with this program; if not, write to the Free Software ++ * Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. ++ * ++ * squashfs2_0.c ++ */ ++ ++#include <linux/types.h> ++#include <linux/squashfs_fs.h> ++#include <linux/module.h> ++#include <linux/errno.h> ++#include <linux/slab.h> ++#include <linux/fs.h> ++#include <linux/smp_lock.h> ++#include <linux/slab.h> ++#include <linux/squashfs_fs_sb.h> ++#include <linux/squashfs_fs_i.h> ++#include <linux/buffer_head.h> ++#include <linux/vfs.h> ++#include <linux/init.h> ++#include <linux/dcache.h> ++#include <linux/wait.h> ++#include <linux/zlib.h> ++#include <linux/blkdev.h> ++#include <linux/vmalloc.h> ++#include <asm/uaccess.h> ++#include <asm/semaphore.h> ++ ++#include "squashfs.h" ++static int squashfs_readdir_2(struct file *file, void *dirent, filldir_t filldir); ++static struct dentry *squashfs_lookup_2(struct inode *, struct dentry *, ++ struct nameidata *); ++ ++static struct file_operations squashfs_dir_ops_2 = { ++ .read = generic_read_dir, ++ .readdir = squashfs_readdir_2 ++}; ++ ++static struct inode_operations squashfs_dir_inode_ops_2 = { ++ .lookup = squashfs_lookup_2 ++}; ++ ++static unsigned char squashfs_filetype_table[] = { ++ DT_UNKNOWN, DT_DIR, DT_REG, DT_LNK, DT_BLK, DT_CHR, DT_FIFO, DT_SOCK ++}; ++ ++static int read_fragment_index_table_2(struct super_block *s) ++{ ++ struct squashfs_sb_info *msblk = s->s_fs_info; ++ struct squashfs_super_block *sblk = &msblk->sblk; ++ ++ if (!(msblk->fragment_index_2 = kmalloc(SQUASHFS_FRAGMENT_INDEX_BYTES_2 ++ (sblk->fragments), GFP_KERNEL))) { ++ ERROR("Failed to allocate uid/gid table\n"); ++ return 0; ++ } ++ ++ if (SQUASHFS_FRAGMENT_INDEX_BYTES_2(sblk->fragments) && ++ !squashfs_read_data(s, (char *) ++ msblk->fragment_index_2, ++ sblk->fragment_table_start, ++ SQUASHFS_FRAGMENT_INDEX_BYTES_2 ++ (sblk->fragments) | ++ SQUASHFS_COMPRESSED_BIT_BLOCK, NULL)) { ++ ERROR("unable to read fragment index table\n"); ++ return 0; ++ } ++ ++ if (msblk->swap) { ++ int i; ++ unsigned int fragment; ++ ++ for (i = 0; i < SQUASHFS_FRAGMENT_INDEXES_2(sblk->fragments); ++ i++) { ++ SQUASHFS_SWAP_FRAGMENT_INDEXES_2((&fragment), ++ &msblk->fragment_index_2[i], 1); ++ msblk->fragment_index_2[i] = fragment; ++ } ++ } ++ ++ return 1; ++} ++ ++ ++static int get_fragment_location_2(struct super_block *s, unsigned int fragment, ++ long long *fragment_start_block, ++ unsigned int *fragment_size) ++{ ++ struct squashfs_sb_info *msblk = s->s_fs_info; ++ long long start_block = ++ msblk->fragment_index_2[SQUASHFS_FRAGMENT_INDEX_2(fragment)]; ++ int offset = SQUASHFS_FRAGMENT_INDEX_OFFSET_2(fragment); ++ struct squashfs_fragment_entry_2 fragment_entry; ++ ++ if (msblk->swap) { ++ struct squashfs_fragment_entry_2 sfragment_entry; ++ ++ if (!squashfs_get_cached_block(s, (char *) &sfragment_entry, ++ start_block, offset, ++ sizeof(sfragment_entry), &start_block, ++ &offset)) ++ goto out; ++ SQUASHFS_SWAP_FRAGMENT_ENTRY_2(&fragment_entry, &sfragment_entry); ++ } else ++ if (!squashfs_get_cached_block(s, (char *) &fragment_entry, ++ start_block, offset, ++ sizeof(fragment_entry), &start_block, ++ &offset)) ++ goto out; ++ ++ *fragment_start_block = fragment_entry.start_block; ++ *fragment_size = fragment_entry.size; ++ ++ return 1; ++ ++out: ++ return 0; ++} ++ ++ ++static struct inode *squashfs_new_inode(struct super_block *s, ++ struct squashfs_base_inode_header_2 *inodeb, unsigned int ino) ++{ ++ struct squashfs_sb_info *msblk = s->s_fs_info; ++ struct squashfs_super_block *sblk = &msblk->sblk; ++ struct inode *i = new_inode(s); ++ ++ if (i) { ++ i->i_ino = ino; ++ i->i_mtime.tv_sec = sblk->mkfs_time; ++ i->i_atime.tv_sec = sblk->mkfs_time; ++ i->i_ctime.tv_sec = sblk->mkfs_time; ++ i->i_uid = msblk->uid[inodeb->uid]; ++ i->i_mode = inodeb->mode; ++ i->i_nlink = 1; ++ i->i_size = 0; ++ if (inodeb->guid == SQUASHFS_GUIDS) ++ i->i_gid = i->i_uid; ++ else ++ i->i_gid = msblk->guid[inodeb->guid]; ++ } ++ ++ return i; ++} ++ ++ ++static struct inode *squashfs_iget_2(struct super_block *s, squashfs_inode_t inode) ++{ ++ struct inode *i; ++ struct squashfs_sb_info *msblk = s->s_fs_info; ++ struct squashfs_super_block *sblk = &msblk->sblk; ++ unsigned int block = SQUASHFS_INODE_BLK(inode) + ++ sblk->inode_table_start; ++ unsigned int offset = SQUASHFS_INODE_OFFSET(inode); ++ unsigned int ino = SQUASHFS_MK_VFS_INODE(block ++ - sblk->inode_table_start, offset); ++ long long next_block; ++ unsigned int next_offset; ++ union squashfs_inode_header_2 id, sid; ++ struct squashfs_base_inode_header_2 *inodeb = &id.base, ++ *sinodeb = &sid.base; ++ ++ TRACE("Entered squashfs_iget\n"); ++ ++ if (msblk->swap) { ++ if (!squashfs_get_cached_block(s, (char *) sinodeb, block, ++ offset, sizeof(*sinodeb), &next_block, ++ &next_offset)) ++ goto failed_read; ++ SQUASHFS_SWAP_BASE_INODE_HEADER_2(inodeb, sinodeb, ++ sizeof(*sinodeb)); ++ } else ++ if (!squashfs_get_cached_block(s, (char *) inodeb, block, ++ offset, sizeof(*inodeb), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ switch(inodeb->inode_type) { ++ case SQUASHFS_FILE_TYPE: { ++ struct squashfs_reg_inode_header_2 *inodep = &id.reg; ++ struct squashfs_reg_inode_header_2 *sinodep = &sid.reg; ++ long long frag_blk; ++ unsigned int frag_size; ++ ++ if (msblk->swap) { ++ if (!squashfs_get_cached_block(s, (char *) ++ sinodep, block, offset, ++ sizeof(*sinodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ SQUASHFS_SWAP_REG_INODE_HEADER_2(inodep, sinodep); ++ } else ++ if (!squashfs_get_cached_block(s, (char *) ++ inodep, block, offset, ++ sizeof(*inodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ frag_blk = SQUASHFS_INVALID_BLK; ++ if (inodep->fragment != SQUASHFS_INVALID_FRAG && ++ !get_fragment_location_2(s, ++ inodep->fragment, &frag_blk, &frag_size)) ++ goto failed_read; ++ ++ if((i = squashfs_new_inode(s, inodeb, ino)) == NULL) ++ goto failed_read1; ++ ++ i->i_size = inodep->file_size; ++ i->i_fop = &generic_ro_fops; ++ i->i_mode |= S_IFREG; ++ i->i_mtime.tv_sec = inodep->mtime; ++ i->i_atime.tv_sec = inodep->mtime; ++ i->i_ctime.tv_sec = inodep->mtime; ++ i->i_blocks = ((i->i_size - 1) >> 9) + 1; ++ SQUASHFS_I(i)->u.s1.fragment_start_block = frag_blk; ++ SQUASHFS_I(i)->u.s1.fragment_size = frag_size; ++ SQUASHFS_I(i)->u.s1.fragment_offset = inodep->offset; ++ SQUASHFS_I(i)->start_block = inodep->start_block; ++ SQUASHFS_I(i)->u.s1.block_list_start = next_block; ++ SQUASHFS_I(i)->offset = next_offset; ++ if (sblk->block_size > 4096) ++ i->i_data.a_ops = &squashfs_aops; ++ else ++ i->i_data.a_ops = &squashfs_aops_4K; ++ ++ TRACE("File inode %x:%x, start_block %x, " ++ "block_list_start %llx, offset %x\n", ++ SQUASHFS_INODE_BLK(inode), offset, ++ inodep->start_block, next_block, ++ next_offset); ++ break; ++ } ++ case SQUASHFS_DIR_TYPE: { ++ struct squashfs_dir_inode_header_2 *inodep = &id.dir; ++ struct squashfs_dir_inode_header_2 *sinodep = &sid.dir; ++ ++ if (msblk->swap) { ++ if (!squashfs_get_cached_block(s, (char *) ++ sinodep, block, offset, ++ sizeof(*sinodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ SQUASHFS_SWAP_DIR_INODE_HEADER_2(inodep, sinodep); ++ } else ++ if (!squashfs_get_cached_block(s, (char *) ++ inodep, block, offset, ++ sizeof(*inodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ if((i = squashfs_new_inode(s, inodeb, ino)) == NULL) ++ goto failed_read1; ++ ++ i->i_size = inodep->file_size; ++ i->i_op = &squashfs_dir_inode_ops_2; ++ i->i_fop = &squashfs_dir_ops_2; ++ i->i_mode |= S_IFDIR; ++ i->i_mtime.tv_sec = inodep->mtime; ++ i->i_atime.tv_sec = inodep->mtime; ++ i->i_ctime.tv_sec = inodep->mtime; ++ SQUASHFS_I(i)->start_block = inodep->start_block; ++ SQUASHFS_I(i)->offset = inodep->offset; ++ SQUASHFS_I(i)->u.s2.directory_index_count = 0; ++ SQUASHFS_I(i)->u.s2.parent_inode = 0; ++ ++ TRACE("Directory inode %x:%x, start_block %x, offset " ++ "%x\n", SQUASHFS_INODE_BLK(inode), ++ offset, inodep->start_block, ++ inodep->offset); ++ break; ++ } ++ case SQUASHFS_LDIR_TYPE: { ++ struct squashfs_ldir_inode_header_2 *inodep = &id.ldir; ++ struct squashfs_ldir_inode_header_2 *sinodep = &sid.ldir; ++ ++ if (msblk->swap) { ++ if (!squashfs_get_cached_block(s, (char *) ++ sinodep, block, offset, ++ sizeof(*sinodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ SQUASHFS_SWAP_LDIR_INODE_HEADER_2(inodep, ++ sinodep); ++ } else ++ if (!squashfs_get_cached_block(s, (char *) ++ inodep, block, offset, ++ sizeof(*inodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ if((i = squashfs_new_inode(s, inodeb, ino)) == NULL) ++ goto failed_read1; ++ ++ i->i_size = inodep->file_size; ++ i->i_op = &squashfs_dir_inode_ops_2; ++ i->i_fop = &squashfs_dir_ops_2; ++ i->i_mode |= S_IFDIR; ++ i->i_mtime.tv_sec = inodep->mtime; ++ i->i_atime.tv_sec = inodep->mtime; ++ i->i_ctime.tv_sec = inodep->mtime; ++ SQUASHFS_I(i)->start_block = inodep->start_block; ++ SQUASHFS_I(i)->offset = inodep->offset; ++ SQUASHFS_I(i)->u.s2.directory_index_start = next_block; ++ SQUASHFS_I(i)->u.s2.directory_index_offset = ++ next_offset; ++ SQUASHFS_I(i)->u.s2.directory_index_count = ++ inodep->i_count; ++ SQUASHFS_I(i)->u.s2.parent_inode = 0; ++ ++ TRACE("Long directory inode %x:%x, start_block %x, " ++ "offset %x\n", ++ SQUASHFS_INODE_BLK(inode), offset, ++ inodep->start_block, inodep->offset); ++ break; ++ } ++ case SQUASHFS_SYMLINK_TYPE: { ++ struct squashfs_symlink_inode_header_2 *inodep = ++ &id.symlink; ++ struct squashfs_symlink_inode_header_2 *sinodep = ++ &sid.symlink; ++ ++ if (msblk->swap) { ++ if (!squashfs_get_cached_block(s, (char *) ++ sinodep, block, offset, ++ sizeof(*sinodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ SQUASHFS_SWAP_SYMLINK_INODE_HEADER_2(inodep, ++ sinodep); ++ } else ++ if (!squashfs_get_cached_block(s, (char *) ++ inodep, block, offset, ++ sizeof(*inodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ if((i = squashfs_new_inode(s, inodeb, ino)) == NULL) ++ goto failed_read1; ++ ++ i->i_size = inodep->symlink_size; ++ i->i_op = &page_symlink_inode_operations; ++ i->i_data.a_ops = &squashfs_symlink_aops; ++ i->i_mode |= S_IFLNK; ++ SQUASHFS_I(i)->start_block = next_block; ++ SQUASHFS_I(i)->offset = next_offset; ++ ++ TRACE("Symbolic link inode %x:%x, start_block %llx, " ++ "offset %x\n", ++ SQUASHFS_INODE_BLK(inode), offset, ++ next_block, next_offset); ++ break; ++ } ++ case SQUASHFS_BLKDEV_TYPE: ++ case SQUASHFS_CHRDEV_TYPE: { ++ struct squashfs_dev_inode_header_2 *inodep = &id.dev; ++ struct squashfs_dev_inode_header_2 *sinodep = &sid.dev; ++ ++ if (msblk->swap) { ++ if (!squashfs_get_cached_block(s, (char *) ++ sinodep, block, offset, ++ sizeof(*sinodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ SQUASHFS_SWAP_DEV_INODE_HEADER_2(inodep, sinodep); ++ } else ++ if (!squashfs_get_cached_block(s, (char *) ++ inodep, block, offset, ++ sizeof(*inodep), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ if ((i = squashfs_new_inode(s, inodeb, ino)) == NULL) ++ goto failed_read1; ++ ++ i->i_mode |= (inodeb->inode_type == ++ SQUASHFS_CHRDEV_TYPE) ? S_IFCHR : ++ S_IFBLK; ++ init_special_inode(i, i->i_mode, ++ old_decode_dev(inodep->rdev)); ++ ++ TRACE("Device inode %x:%x, rdev %x\n", ++ SQUASHFS_INODE_BLK(inode), offset, ++ inodep->rdev); ++ break; ++ } ++ case SQUASHFS_FIFO_TYPE: ++ case SQUASHFS_SOCKET_TYPE: { ++ if ((i = squashfs_new_inode(s, inodeb, ino)) == NULL) ++ goto failed_read1; ++ ++ i->i_mode |= (inodeb->inode_type == SQUASHFS_FIFO_TYPE) ++ ? S_IFIFO : S_IFSOCK; ++ init_special_inode(i, i->i_mode, 0); ++ break; ++ } ++ default: ++ ERROR("Unknown inode type %d in squashfs_iget!\n", ++ inodeb->inode_type); ++ goto failed_read1; ++ } ++ ++ insert_inode_hash(i); ++ return i; ++ ++failed_read: ++ ERROR("Unable to read inode [%x:%x]\n", block, offset); ++ ++failed_read1: ++ return NULL; ++} ++ ++ ++static int get_dir_index_using_offset(struct super_block *s, long long ++ *next_block, unsigned int *next_offset, ++ long long index_start, ++ unsigned int index_offset, int i_count, ++ long long f_pos) ++{ ++ struct squashfs_sb_info *msblk = s->s_fs_info; ++ struct squashfs_super_block *sblk = &msblk->sblk; ++ int i, length = 0; ++ struct squashfs_dir_index_2 index; ++ ++ TRACE("Entered get_dir_index_using_offset, i_count %d, f_pos %d\n", ++ i_count, (unsigned int) f_pos); ++ ++ if (f_pos == 0) ++ goto finish; ++ ++ for (i = 0; i < i_count; i++) { ++ if (msblk->swap) { ++ struct squashfs_dir_index_2 sindex; ++ squashfs_get_cached_block(s, (char *) &sindex, ++ index_start, index_offset, ++ sizeof(sindex), &index_start, ++ &index_offset); ++ SQUASHFS_SWAP_DIR_INDEX_2(&index, &sindex); ++ } else ++ squashfs_get_cached_block(s, (char *) &index, ++ index_start, index_offset, ++ sizeof(index), &index_start, ++ &index_offset); ++ ++ if (index.index > f_pos) ++ break; ++ ++ squashfs_get_cached_block(s, NULL, index_start, index_offset, ++ index.size + 1, &index_start, ++ &index_offset); ++ ++ length = index.index; ++ *next_block = index.start_block + sblk->directory_table_start; ++ } ++ ++ *next_offset = (length + *next_offset) % SQUASHFS_METADATA_SIZE; ++ ++finish: ++ return length; ++} ++ ++ ++static int get_dir_index_using_name(struct super_block *s, long long ++ *next_block, unsigned int *next_offset, ++ long long index_start, ++ unsigned int index_offset, int i_count, ++ const char *name, int size) ++{ ++ struct squashfs_sb_info *msblk = s->s_fs_info; ++ struct squashfs_super_block *sblk = &msblk->sblk; ++ int i, length = 0; ++ char buffer[sizeof(struct squashfs_dir_index_2) + SQUASHFS_NAME_LEN + 1]; ++ struct squashfs_dir_index_2 *index = (struct squashfs_dir_index_2 *) buffer; ++ char str[SQUASHFS_NAME_LEN + 1]; ++ ++ TRACE("Entered get_dir_index_using_name, i_count %d\n", i_count); ++ ++ strncpy(str, name, size); ++ str[size] = '\0'; ++ ++ for (i = 0; i < i_count; i++) { ++ if (msblk->swap) { ++ struct squashfs_dir_index_2 sindex; ++ squashfs_get_cached_block(s, (char *) &sindex, ++ index_start, index_offset, ++ sizeof(sindex), &index_start, ++ &index_offset); ++ SQUASHFS_SWAP_DIR_INDEX_2(index, &sindex); ++ } else ++ squashfs_get_cached_block(s, (char *) index, ++ index_start, index_offset, ++ sizeof(struct squashfs_dir_index_2), ++ &index_start, &index_offset); ++ ++ squashfs_get_cached_block(s, index->name, index_start, ++ index_offset, index->size + 1, ++ &index_start, &index_offset); ++ ++ index->name[index->size + 1] = '\0'; ++ ++ if (strcmp(index->name, str) > 0) ++ break; ++ ++ length = index->index; ++ *next_block = index->start_block + sblk->directory_table_start; ++ } ++ ++ *next_offset = (length + *next_offset) % SQUASHFS_METADATA_SIZE; ++ return length; ++} ++ ++ ++static int squashfs_readdir_2(struct file *file, void *dirent, filldir_t filldir) ++{ ++ struct inode *i = file->f_dentry->d_inode; ++ struct squashfs_sb_info *msblk = i->i_sb->s_fs_info; ++ struct squashfs_super_block *sblk = &msblk->sblk; ++ long long next_block = SQUASHFS_I(i)->start_block + ++ sblk->directory_table_start; ++ int next_offset = SQUASHFS_I(i)->offset, length = 0, dirs_read = 0, ++ dir_count; ++ struct squashfs_dir_header_2 dirh; ++ char buffer[sizeof(struct squashfs_dir_entry_2) + SQUASHFS_NAME_LEN + 1]; ++ struct squashfs_dir_entry_2 *dire = (struct squashfs_dir_entry_2 *) buffer; ++ ++ TRACE("Entered squashfs_readdir_2 [%llx:%x]\n", next_block, next_offset); ++ ++ length = get_dir_index_using_offset(i->i_sb, &next_block, &next_offset, ++ SQUASHFS_I(i)->u.s2.directory_index_start, ++ SQUASHFS_I(i)->u.s2.directory_index_offset, ++ SQUASHFS_I(i)->u.s2.directory_index_count, ++ file->f_pos); ++ ++ while (length < i_size_read(i)) { ++ /* read directory header */ ++ if (msblk->swap) { ++ struct squashfs_dir_header_2 sdirh; ++ ++ if (!squashfs_get_cached_block(i->i_sb, (char *) &sdirh, ++ next_block, next_offset, sizeof(sdirh), ++ &next_block, &next_offset)) ++ goto failed_read; ++ ++ length += sizeof(sdirh); ++ SQUASHFS_SWAP_DIR_HEADER_2(&dirh, &sdirh); ++ } else { ++ if (!squashfs_get_cached_block(i->i_sb, (char *) &dirh, ++ next_block, next_offset, sizeof(dirh), ++ &next_block, &next_offset)) ++ goto failed_read; ++ ++ length += sizeof(dirh); ++ } ++ ++ dir_count = dirh.count + 1; ++ while (dir_count--) { ++ if (msblk->swap) { ++ struct squashfs_dir_entry_2 sdire; ++ if (!squashfs_get_cached_block(i->i_sb, (char *) ++ &sdire, next_block, next_offset, ++ sizeof(sdire), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ length += sizeof(sdire); ++ SQUASHFS_SWAP_DIR_ENTRY_2(dire, &sdire); ++ } else { ++ if (!squashfs_get_cached_block(i->i_sb, (char *) ++ dire, next_block, next_offset, ++ sizeof(*dire), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ length += sizeof(*dire); ++ } ++ ++ if (!squashfs_get_cached_block(i->i_sb, dire->name, ++ next_block, next_offset, ++ dire->size + 1, &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ length += dire->size + 1; ++ ++ if (file->f_pos >= length) ++ continue; ++ ++ dire->name[dire->size + 1] = '\0'; ++ ++ TRACE("Calling filldir(%x, %s, %d, %d, %x:%x, %d)\n", ++ (unsigned int) dirent, dire->name, ++ dire->size + 1, (int) file->f_pos, ++ dirh.start_block, dire->offset, ++ squashfs_filetype_table[dire->type]); ++ ++ if (filldir(dirent, dire->name, dire->size + 1, ++ file->f_pos, SQUASHFS_MK_VFS_INODE( ++ dirh.start_block, dire->offset), ++ squashfs_filetype_table[dire->type]) ++ < 0) { ++ TRACE("Filldir returned less than 0\n"); ++ goto finish; ++ } ++ file->f_pos = length; ++ dirs_read++; ++ } ++ } ++ ++finish: ++ return dirs_read; ++ ++failed_read: ++ ERROR("Unable to read directory block [%llx:%x]\n", next_block, ++ next_offset); ++ return 0; ++} ++ ++ ++static struct dentry *squashfs_lookup_2(struct inode *i, struct dentry *dentry, ++ struct nameidata *nd) ++{ ++ const unsigned char *name = dentry->d_name.name; ++ int len = dentry->d_name.len; ++ struct inode *inode = NULL; ++ struct squashfs_sb_info *msblk = i->i_sb->s_fs_info; ++ struct squashfs_super_block *sblk = &msblk->sblk; ++ long long next_block = SQUASHFS_I(i)->start_block + ++ sblk->directory_table_start; ++ int next_offset = SQUASHFS_I(i)->offset, length = 0, ++ dir_count; ++ struct squashfs_dir_header_2 dirh; ++ char buffer[sizeof(struct squashfs_dir_entry_2) + SQUASHFS_NAME_LEN]; ++ struct squashfs_dir_entry_2 *dire = (struct squashfs_dir_entry_2 *) buffer; ++ int sorted = sblk->s_major == 2 && sblk->s_minor >= 1; ++ ++ TRACE("Entered squashfs_lookup [%llx:%x]\n", next_block, next_offset); ++ ++ if (len > SQUASHFS_NAME_LEN) ++ goto exit_loop; ++ ++ length = get_dir_index_using_name(i->i_sb, &next_block, &next_offset, ++ SQUASHFS_I(i)->u.s2.directory_index_start, ++ SQUASHFS_I(i)->u.s2.directory_index_offset, ++ SQUASHFS_I(i)->u.s2.directory_index_count, name, ++ len); ++ ++ while (length < i_size_read(i)) { ++ /* read directory header */ ++ if (msblk->swap) { ++ struct squashfs_dir_header_2 sdirh; ++ if (!squashfs_get_cached_block(i->i_sb, (char *) &sdirh, ++ next_block, next_offset, sizeof(sdirh), ++ &next_block, &next_offset)) ++ goto failed_read; ++ ++ length += sizeof(sdirh); ++ SQUASHFS_SWAP_DIR_HEADER_2(&dirh, &sdirh); ++ } else { ++ if (!squashfs_get_cached_block(i->i_sb, (char *) &dirh, ++ next_block, next_offset, sizeof(dirh), ++ &next_block, &next_offset)) ++ goto failed_read; ++ ++ length += sizeof(dirh); ++ } ++ ++ dir_count = dirh.count + 1; ++ while (dir_count--) { ++ if (msblk->swap) { ++ struct squashfs_dir_entry_2 sdire; ++ if (!squashfs_get_cached_block(i->i_sb, (char *) ++ &sdire, next_block,next_offset, ++ sizeof(sdire), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ length += sizeof(sdire); ++ SQUASHFS_SWAP_DIR_ENTRY_2(dire, &sdire); ++ } else { ++ if (!squashfs_get_cached_block(i->i_sb, (char *) ++ dire, next_block,next_offset, ++ sizeof(*dire), &next_block, ++ &next_offset)) ++ goto failed_read; ++ ++ length += sizeof(*dire); ++ } ++ ++ if (!squashfs_get_cached_block(i->i_sb, dire->name, ++ next_block, next_offset, dire->size + 1, ++ &next_block, &next_offset)) ++ goto failed_read; ++ ++ length += dire->size + 1; ++ ++ if (sorted && name[0] < dire->name[0]) ++ goto exit_loop; ++ ++ if ((len == dire->size + 1) && !strncmp(name, ++ dire->name, len)) { ++ squashfs_inode_t ino = ++ SQUASHFS_MKINODE(dirh.start_block, ++ dire->offset); ++ ++ TRACE("calling squashfs_iget for directory " ++ "entry %s, inode %x:%x, %lld\n", name, ++ dirh.start_block, dire->offset, ino); ++ ++ inode = (msblk->iget)(i->i_sb, ino); ++ ++ goto exit_loop; ++ } ++ } ++ } ++ ++exit_loop: ++ d_add(dentry, inode); ++ return ERR_PTR(0); ++ ++failed_read: ++ ERROR("Unable to read directory block [%llx:%x]\n", next_block, ++ next_offset); ++ goto exit_loop; ++} ++ ++ ++int squashfs_2_0_supported(struct squashfs_sb_info *msblk) ++{ ++ struct squashfs_super_block *sblk = &msblk->sblk; ++ ++ msblk->iget = squashfs_iget_2; ++ msblk->read_fragment_index_table = read_fragment_index_table_2; ++ ++ sblk->bytes_used = sblk->bytes_used_2; ++ sblk->uid_start = sblk->uid_start_2; ++ sblk->guid_start = sblk->guid_start_2; ++ sblk->inode_table_start = sblk->inode_table_start_2; ++ sblk->directory_table_start = sblk->directory_table_start_2; ++ sblk->fragment_table_start = sblk->fragment_table_start_2; ++ ++ return 1; ++} +Index: git/fs/squashfs/squashfs.h +=================================================================== +--- /dev/null 1970-01-01 00:00:00.000000000 +0000 ++++ git/fs/squashfs/squashfs.h 2006-10-31 21:26:45.000000000 +0000 +@@ -0,0 +1,86 @@ ++/* ++ * Squashfs - a compressed read only filesystem for Linux ++ * ++ * Copyright (c) 2002, 2003, 2004, 2005, 2006 ++ * Phillip Lougher <phillip@lougher.org.uk> ++ * ++ * 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, ++ * or (at your option) any later version. ++ * ++ * This program is distributed in the hope that it will be useful, ++ * but WITHOUT ANY WARRANTY; without even the implied warranty of ++ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ++ * GNU General Public License for more details. ++ * ++ * You should have received a copy of the GNU General Public License ++ * along with this program; if not, write to the Free Software ++ * Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. ++ * ++ * squashfs.h ++ */ ++ ++#ifdef CONFIG_SQUASHFS_1_0_COMPATIBILITY ++#undef CONFIG_SQUASHFS_1_0_COMPATIBILITY ++#endif ++ ++#ifdef SQUASHFS_TRACE ++#define TRACE(s, args...) printk(KERN_NOTICE "SQUASHFS: "s, ## args) ++#else ++#define TRACE(s, args...) {} ++#endif ++ ++#define ERROR(s, args...) printk(KERN_ERR "SQUASHFS error: "s, ## args) ++ ++#define SERROR(s, args...) do { \ ++ if (!silent) \ ++ printk(KERN_ERR "SQUASHFS error: "s, ## args);\ ++ } while(0) ++ ++#define WARNING(s, args...) printk(KERN_WARNING "SQUASHFS: "s, ## args) ++ ++static inline struct squashfs_inode_info *SQUASHFS_I(struct inode *inode) ++{ ++ return list_entry(inode, struct squashfs_inode_info, vfs_inode); ++} ++ ++#if defined(CONFIG_SQUASHFS_1_0_COMPATIBILITY ) || defined(CONFIG_SQUASHFS_2_0_COMPATIBILITY) ++#define SQSH_EXTERN ++extern unsigned int squashfs_read_data(struct super_block *s, char *buffer, ++ long long index, unsigned int length, ++ long long *next_index); ++extern int squashfs_get_cached_block(struct super_block *s, char *buffer, ++ long long block, unsigned int offset, ++ int length, long long *next_block, ++ unsigned int *next_offset); ++extern void release_cached_fragment(struct squashfs_sb_info *msblk, struct ++ squashfs_fragment_cache *fragment); ++extern struct squashfs_fragment_cache *get_cached_fragment(struct super_block ++ *s, long long start_block, ++ int length); ++extern struct address_space_operations squashfs_symlink_aops; ++extern struct address_space_operations squashfs_aops; ++extern struct address_space_operations squashfs_aops_4K; ++extern struct inode_operations squashfs_dir_inode_ops; ++#else ++#define SQSH_EXTERN static ++#endif ++ ++#ifdef CONFIG_SQUASHFS_1_0_COMPATIBILITY ++extern int squashfs_1_0_supported(struct squashfs_sb_info *msblk); ++#else ++static inline int squashfs_1_0_supported(struct squashfs_sb_info *msblk) ++{ ++ return 0; ++} ++#endif ++ ++#ifdef CONFIG_SQUASHFS_2_0_COMPATIBILITY ++extern int squashfs_2_0_supported(struct squashfs_sb_info *msblk); ++#else ++static inline int squashfs_2_0_supported(struct squashfs_sb_info *msblk) ++{ ++ return 0; ++} ++#endif +Index: git/include/linux/squashfs_fs.h +=================================================================== +--- /dev/null 1970-01-01 00:00:00.000000000 +0000 ++++ git/include/linux/squashfs_fs.h 2006-10-31 21:26:45.000000000 +0000 +@@ -0,0 +1,911 @@ ++#ifndef SQUASHFS_FS ++#define SQUASHFS_FS ++ ++/* ++ * Squashfs ++ * ++ * Copyright (c) 2002, 2003, 2004, 2005, 2006 ++ * Phillip Lougher <phillip@lougher.org.uk> ++ * ++ * 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, ++ * or (at your option) any later version. ++ * ++ * This program is distributed in the hope that it will be useful, ++ * but WITHOUT ANY WARRANTY; without even the implied warranty of ++ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ++ * GNU General Public License for more details. ++ * ++ * You should have received a copy of the GNU General Public License ++ * along with this program; if not, write to the Free Software ++ * Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. ++ * ++ * squashfs_fs.h ++ */ ++ ++#ifndef CONFIG_SQUASHFS_2_0_COMPATIBILITY ++#define CONFIG_SQUASHFS_2_0_COMPATIBILITY ++#endif ++ ++#ifdef CONFIG_SQUASHFS_VMALLOC ++#define SQUASHFS_ALLOC(a) vmalloc(a) ++#define SQUASHFS_FREE(a) vfree(a) ++#else ++#define SQUASHFS_ALLOC(a) kmalloc(a, GFP_KERNEL) ++#define SQUASHFS_FREE(a) kfree(a) ++#endif ++#define SQUASHFS_CACHED_FRAGMENTS CONFIG_SQUASHFS_FRAGMENT_CACHE_SIZE ++#define SQUASHFS_MAJOR 3 ++#define SQUASHFS_MINOR 0 ++#define SQUASHFS_MAGIC 0x73717368 ++#define SQUASHFS_MAGIC_SWAP 0x68737173 ++#define SQUASHFS_START 0 ++ ++/* size of metadata (inode and directory) blocks */ ++#define SQUASHFS_METADATA_SIZE 8192 ++#define SQUASHFS_METADATA_LOG 13 ++ ++/* default size of data blocks */ ++#define SQUASHFS_FILE_SIZE 65536 ++#define SQUASHFS_FILE_LOG 16 ++ ++#define SQUASHFS_FILE_MAX_SIZE 65536 ++ ++/* Max number of uids and gids */ ++#define SQUASHFS_UIDS 256 ++#define SQUASHFS_GUIDS 255 ++ ++/* Max length of filename (not 255) */ ++#define SQUASHFS_NAME_LEN 256 ++ ++#define SQUASHFS_INVALID ((long long) 0xffffffffffff) ++#define SQUASHFS_INVALID_FRAG ((unsigned int) 0xffffffff) ++#define SQUASHFS_INVALID_BLK ((long long) -1) ++#define SQUASHFS_USED_BLK ((long long) -2) ++ ++/* Filesystem flags */ ++#define SQUASHFS_NOI 0 ++#define SQUASHFS_NOD 1 ++#define SQUASHFS_CHECK 2 ++#define SQUASHFS_NOF 3 ++#define SQUASHFS_NO_FRAG 4 ++#define SQUASHFS_ALWAYS_FRAG 5 ++#define SQUASHFS_DUPLICATE 6 ++ ++#define SQUASHFS_BIT(flag, bit) ((flag >> bit) & 1) ++ ++#define SQUASHFS_UNCOMPRESSED_INODES(flags) SQUASHFS_BIT(flags, \ ++ SQUASHFS_NOI) ++ ++#define SQUASHFS_UNCOMPRESSED_DATA(flags) SQUASHFS_BIT(flags, \ ++ SQUASHFS_NOD) ++ ++#define SQUASHFS_UNCOMPRESSED_FRAGMENTS(flags) SQUASHFS_BIT(flags, \ ++ SQUASHFS_NOF) ++ ++#define SQUASHFS_NO_FRAGMENTS(flags) SQUASHFS_BIT(flags, \ ++ SQUASHFS_NO_FRAG) ++ ++#define SQUASHFS_ALWAYS_FRAGMENTS(flags) SQUASHFS_BIT(flags, \ ++ SQUASHFS_ALWAYS_FRAG) ++ ++#define SQUASHFS_DUPLICATES(flags) SQUASHFS_BIT(flags, \ ++ SQUASHFS_DUPLICATE) ++ ++#define SQUASHFS_CHECK_DATA(flags) SQUASHFS_BIT(flags, \ ++ SQUASHFS_CHECK) ++ ++#define SQUASHFS_MKFLAGS(noi, nod, check_data, nof, no_frag, always_frag, \ ++ duplicate_checking) (noi | (nod << 1) | (check_data << 2) \ ++ | (nof << 3) | (no_frag << 4) | (always_frag << 5) | \ ++ (duplicate_checking << 6)) ++ ++/* Max number of types and file types */ ++#define SQUASHFS_DIR_TYPE 1 ++#define SQUASHFS_FILE_TYPE 2 ++#define SQUASHFS_SYMLINK_TYPE 3 ++#define SQUASHFS_BLKDEV_TYPE 4 ++#define SQUASHFS_CHRDEV_TYPE 5 ++#define SQUASHFS_FIFO_TYPE 6 ++#define SQUASHFS_SOCKET_TYPE 7 ++#define SQUASHFS_LDIR_TYPE 8 ++#define SQUASHFS_LREG_TYPE 9 ++ ++/* 1.0 filesystem type definitions */ ++#define SQUASHFS_TYPES 5 ++#define SQUASHFS_IPC_TYPE 0 ++ ++/* Flag whether block is compressed or uncompressed, bit is set if block is ++ * uncompressed */ ++#define SQUASHFS_COMPRESSED_BIT (1 << 15) ++ ++#define SQUASHFS_COMPRESSED_SIZE(B) (((B) & ~SQUASHFS_COMPRESSED_BIT) ? \ ++ (B) & ~SQUASHFS_COMPRESSED_BIT : SQUASHFS_COMPRESSED_BIT) ++ ++#define SQUASHFS_COMPRESSED(B) (!((B) & SQUASHFS_COMPRESSED_BIT)) ++ ++#define SQUASHFS_COMPRESSED_BIT_BLOCK (1 << 24) ++ ++#define SQUASHFS_COMPRESSED_SIZE_BLOCK(B) (((B) & \ ++ ~SQUASHFS_COMPRESSED_BIT_BLOCK) ? (B) & \ ++ ~SQUASHFS_COMPRESSED_BIT_BLOCK : SQUASHFS_COMPRESSED_BIT_BLOCK) ++ ++#define SQUASHFS_COMPRESSED_BLOCK(B) (!((B) & SQUASHFS_COMPRESSED_BIT_BLOCK)) ++ ++/* ++ * Inode number ops. Inodes consist of a compressed block number, and an ++ * uncompressed offset within that block ++ */ ++#define SQUASHFS_INODE_BLK(a) ((unsigned int) ((a) >> 16)) ++ ++#define SQUASHFS_INODE_OFFSET(a) ((unsigned int) ((a) & 0xffff)) ++ ++#define SQUASHFS_MKINODE(A, B) ((squashfs_inode_t)(((squashfs_inode_t) (A)\ ++ << 16) + (B))) ++ ++/* Compute 32 bit VFS inode number from squashfs inode number */ ++#define SQUASHFS_MK_VFS_INODE(a, b) ((unsigned int) (((a) << 8) + \ ++ ((b) >> 2) + 1)) ++/* XXX */ ++ ++/* Translate between VFS mode and squashfs mode */ ++#define SQUASHFS_MODE(a) ((a) & 0xfff) ++ ++/* fragment and fragment table defines */ ++#define SQUASHFS_FRAGMENT_BYTES(A) (A * sizeof(struct squashfs_fragment_entry)) ++ ++#define SQUASHFS_FRAGMENT_INDEX(A) (SQUASHFS_FRAGMENT_BYTES(A) / \ ++ SQUASHFS_METADATA_SIZE) ++ ++#define SQUASHFS_FRAGMENT_INDEX_OFFSET(A) (SQUASHFS_FRAGMENT_BYTES(A) % \ ++ SQUASHFS_METADATA_SIZE) ++ ++#define SQUASHFS_FRAGMENT_INDEXES(A) ((SQUASHFS_FRAGMENT_BYTES(A) + \ ++ SQUASHFS_METADATA_SIZE - 1) / \ ++ SQUASHFS_METADATA_SIZE) ++ ++#define SQUASHFS_FRAGMENT_INDEX_BYTES(A) (SQUASHFS_FRAGMENT_INDEXES(A) *\ ++ sizeof(long long)) ++ ++/* cached data constants for filesystem */ ++#define SQUASHFS_CACHED_BLKS 8 ++ ++#define SQUASHFS_MAX_FILE_SIZE_LOG 64 ++ ++#define SQUASHFS_MAX_FILE_SIZE ((long long) 1 << \ ++ (SQUASHFS_MAX_FILE_SIZE_LOG - 2)) ++ ++#define SQUASHFS_MARKER_BYTE 0xff ++ ++/* meta index cache */ ++#define SQUASHFS_META_INDEXES (SQUASHFS_METADATA_SIZE / sizeof(unsigned int)) ++#define SQUASHFS_META_ENTRIES 31 ++#define SQUASHFS_META_NUMBER 8 ++#define SQUASHFS_SLOTS 4 ++ ++struct meta_entry { ++ long long data_block; ++ unsigned int index_block; ++ unsigned short offset; ++ unsigned short pad; ++}; ++ ++struct meta_index { ++ unsigned int inode_number; ++ unsigned int offset; ++ unsigned short entries; ++ unsigned short skip; ++ unsigned short locked; ++ unsigned short pad; ++ struct meta_entry meta_entry[SQUASHFS_META_ENTRIES]; ++}; ++ ++ ++/* ++ * definitions for structures on disk ++ */ ++ ++typedef long long squashfs_block_t; ++typedef long long squashfs_inode_t; ++ ++struct squashfs_super_block { ++ unsigned int s_magic; ++ unsigned int inodes; ++ unsigned int bytes_used_2; ++ unsigned int uid_start_2; ++ unsigned int guid_start_2; ++ unsigned int inode_table_start_2; ++ unsigned int directory_table_start_2; ++ unsigned int s_major:16; ++ unsigned int s_minor:16; ++ unsigned int block_size_1:16; ++ unsigned int block_log:16; ++ unsigned int flags:8; ++ unsigned int no_uids:8; ++ unsigned int no_guids:8; ++ unsigned int mkfs_time /* time of filesystem creation */; ++ squashfs_inode_t root_inode; ++ unsigned int block_size; ++ unsigned int fragments; ++ unsigned int fragment_table_start_2; ++ long long bytes_used; ++ long long uid_start; ++ long long guid_start; ++ long long inode_table_start; ++ long long directory_table_start; ++ long long fragment_table_start; ++ long long unused; ++} __attribute__ ((packed)); ++ ++struct squashfs_dir_index { ++ unsigned int index; ++ unsigned int start_block; ++ unsigned char size; ++ unsigned char name[0]; ++} __attribute__ ((packed)); ++ ++#define SQUASHFS_BASE_INODE_HEADER \ ++ unsigned int inode_type:4; \ ++ unsigned int mode:12; \ ++ unsigned int uid:8; \ ++ unsigned int guid:8; \ ++ unsigned int mtime; \ ++ unsigned int inode_number; ++ ++struct squashfs_base_inode_header { ++ SQUASHFS_BASE_INODE_HEADER; ++} __attribute__ ((packed)); ++ ++struct squashfs_ipc_inode_header { ++ SQUASHFS_BASE_INODE_HEADER; ++ unsigned int nlink; ++} __attribute__ ((packed)); ++ ++struct squashfs_dev_inode_header { ++ SQUASHFS_BASE_INODE_HEADER; ++ unsigned int nlink; ++ unsigned short rdev; ++} __attribute__ ((packed)); ++ ++struct squashfs_symlink_inode_header { ++ SQUASHFS_BASE_INODE_HEADER; ++ unsigned int nlink; ++ unsigned short symlink_size; ++ char symlink[0]; ++} __attribute__ ((packed)); ++ ++struct squashfs_reg_inode_header { ++ SQUASHFS_BASE_INODE_HEADER; ++ squashfs_block_t start_block; ++ unsigned int fragment; ++ unsigned int offset; ++ unsigned int file_size; ++ unsigned short block_list[0]; ++} __attribute__ ((packed)); ++ ++struct squashfs_lreg_inode_header { ++ SQUASHFS_BASE_INODE_HEADER; ++ unsigned int nlink; ++ squashfs_block_t start_block; ++ unsigned int fragment; ++ unsigned int offset; ++ long long file_size; ++ unsigned short block_list[0]; ++} __attribute__ ((packed)); ++ ++struct squashfs_dir_inode_header { ++ SQUASHFS_BASE_INODE_HEADER; ++ unsigned int nlink; ++ unsigned int file_size:19; ++ unsigned int offset:13; ++ unsigned int start_block; ++ unsigned int parent_inode; ++} __attribute__ ((packed)); ++ ++struct squashfs_ldir_inode_header { ++ SQUASHFS_BASE_INODE_HEADER; ++ unsigned int nlink; ++ unsigned int file_size:27; ++ unsigned int offset:13; ++ unsigned int start_block; ++ unsigned int i_count:16; ++ unsigned int parent_inode; ++ struct squashfs_dir_index index[0]; ++} __attribute__ ((packed)); ++ ++union squashfs_inode_header { ++ struct squashfs_base_inode_header base; ++ struct squashfs_dev_inode_header dev; ++ struct squashfs_symlink_inode_header symlink; ++ struct squashfs_reg_inode_header reg; ++ struct squashfs_lreg_inode_header lreg; ++ struct squashfs_dir_inode_header dir; ++ struct squashfs_ldir_inode_header ldir; ++ struct squashfs_ipc_inode_header ipc; ++}; ++ ++struct squashfs_dir_entry { ++ unsigned int offset:13; ++ unsigned int type:3; ++ unsigned int size:8; ++ int inode_number:16; ++ char name[0]; ++} __attribute__ ((packed)); ++ ++struct squashfs_dir_header { ++ unsigned int count:8; ++ unsigned int start_block; ++ unsigned int inode_number; ++} __attribute__ ((packed)); ++ ++struct squashfs_fragment_entry { ++ long long start_block; ++ unsigned int size; ++ unsigned int unused; ++} __attribute__ ((packed)); ++ ++extern int squashfs_uncompress_block(void *d, int dstlen, void *s, int srclen); ++extern int squashfs_uncompress_init(void); ++extern int squashfs_uncompress_exit(void); ++ ++/* ++ * macros to convert each packed bitfield structure from little endian to big ++ * endian and vice versa. These are needed when creating or using a filesystem ++ * on a machine with different byte ordering to the target architecture. ++ * ++ */ ++ ++#define SQUASHFS_SWAP_START \ ++ int bits;\ ++ int b_pos;\ ++ unsigned long long val;\ ++ unsigned char *s;\ ++ unsigned char *d; ++ ++#define SQUASHFS_SWAP_SUPER_BLOCK(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_MEMSET(s, d, sizeof(struct squashfs_super_block));\ ++ SQUASHFS_SWAP((s)->s_magic, d, 0, 32);\ ++ SQUASHFS_SWAP((s)->inodes, d, 32, 32);\ ++ SQUASHFS_SWAP((s)->bytes_used_2, d, 64, 32);\ ++ SQUASHFS_SWAP((s)->uid_start_2, d, 96, 32);\ ++ SQUASHFS_SWAP((s)->guid_start_2, d, 128, 32);\ ++ SQUASHFS_SWAP((s)->inode_table_start_2, d, 160, 32);\ ++ SQUASHFS_SWAP((s)->directory_table_start_2, d, 192, 32);\ ++ SQUASHFS_SWAP((s)->s_major, d, 224, 16);\ ++ SQUASHFS_SWAP((s)->s_minor, d, 240, 16);\ ++ SQUASHFS_SWAP((s)->block_size_1, d, 256, 16);\ ++ SQUASHFS_SWAP((s)->block_log, d, 272, 16);\ ++ SQUASHFS_SWAP((s)->flags, d, 288, 8);\ ++ SQUASHFS_SWAP((s)->no_uids, d, 296, 8);\ ++ SQUASHFS_SWAP((s)->no_guids, d, 304, 8);\ ++ SQUASHFS_SWAP((s)->mkfs_time, d, 312, 32);\ ++ SQUASHFS_SWAP((s)->root_inode, d, 344, 64);\ ++ SQUASHFS_SWAP((s)->block_size, d, 408, 32);\ ++ SQUASHFS_SWAP((s)->fragments, d, 440, 32);\ ++ SQUASHFS_SWAP((s)->fragment_table_start_2, d, 472, 32);\ ++ SQUASHFS_SWAP((s)->bytes_used, d, 504, 64);\ ++ SQUASHFS_SWAP((s)->uid_start, d, 568, 64);\ ++ SQUASHFS_SWAP((s)->guid_start, d, 632, 64);\ ++ SQUASHFS_SWAP((s)->inode_table_start, d, 696, 64);\ ++ SQUASHFS_SWAP((s)->directory_table_start, d, 760, 64);\ ++ SQUASHFS_SWAP((s)->fragment_table_start, d, 824, 64);\ ++ SQUASHFS_SWAP((s)->unused, d, 888, 64);\ ++} ++ ++#define SQUASHFS_SWAP_BASE_INODE_CORE(s, d, n)\ ++ SQUASHFS_MEMSET(s, d, n);\ ++ SQUASHFS_SWAP((s)->inode_type, d, 0, 4);\ ++ SQUASHFS_SWAP((s)->mode, d, 4, 12);\ ++ SQUASHFS_SWAP((s)->uid, d, 16, 8);\ ++ SQUASHFS_SWAP((s)->guid, d, 24, 8);\ ++ SQUASHFS_SWAP((s)->mtime, d, 32, 32);\ ++ SQUASHFS_SWAP((s)->inode_number, d, 64, 32); ++ ++#define SQUASHFS_SWAP_BASE_INODE_HEADER(s, d, n) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE(s, d, n)\ ++} ++ ++#define SQUASHFS_SWAP_IPC_INODE_HEADER(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE(s, d, \ ++ sizeof(struct squashfs_ipc_inode_header))\ ++ SQUASHFS_SWAP((s)->nlink, d, 96, 32);\ ++} ++ ++#define SQUASHFS_SWAP_DEV_INODE_HEADER(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE(s, d, \ ++ sizeof(struct squashfs_dev_inode_header)); \ ++ SQUASHFS_SWAP((s)->nlink, d, 96, 32);\ ++ SQUASHFS_SWAP((s)->rdev, d, 128, 16);\ ++} ++ ++#define SQUASHFS_SWAP_SYMLINK_INODE_HEADER(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE(s, d, \ ++ sizeof(struct squashfs_symlink_inode_header));\ ++ SQUASHFS_SWAP((s)->nlink, d, 96, 32);\ ++ SQUASHFS_SWAP((s)->symlink_size, d, 128, 16);\ ++} ++ ++#define SQUASHFS_SWAP_REG_INODE_HEADER(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE(s, d, \ ++ sizeof(struct squashfs_reg_inode_header));\ ++ SQUASHFS_SWAP((s)->start_block, d, 96, 64);\ ++ SQUASHFS_SWAP((s)->fragment, d, 160, 32);\ ++ SQUASHFS_SWAP((s)->offset, d, 192, 32);\ ++ SQUASHFS_SWAP((s)->file_size, d, 224, 32);\ ++} ++ ++#define SQUASHFS_SWAP_LREG_INODE_HEADER(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE(s, d, \ ++ sizeof(struct squashfs_lreg_inode_header));\ ++ SQUASHFS_SWAP((s)->nlink, d, 96, 32);\ ++ SQUASHFS_SWAP((s)->start_block, d, 128, 64);\ ++ SQUASHFS_SWAP((s)->fragment, d, 192, 32);\ ++ SQUASHFS_SWAP((s)->offset, d, 224, 32);\ ++ SQUASHFS_SWAP((s)->file_size, d, 256, 64);\ ++} ++ ++#define SQUASHFS_SWAP_DIR_INODE_HEADER(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE(s, d, \ ++ sizeof(struct squashfs_dir_inode_header));\ ++ SQUASHFS_SWAP((s)->nlink, d, 96, 32);\ ++ SQUASHFS_SWAP((s)->file_size, d, 128, 19);\ ++ SQUASHFS_SWAP((s)->offset, d, 147, 13);\ ++ SQUASHFS_SWAP((s)->start_block, d, 160, 32);\ ++ SQUASHFS_SWAP((s)->parent_inode, d, 192, 32);\ ++} ++ ++#define SQUASHFS_SWAP_LDIR_INODE_HEADER(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE(s, d, \ ++ sizeof(struct squashfs_ldir_inode_header));\ ++ SQUASHFS_SWAP((s)->nlink, d, 96, 32);\ ++ SQUASHFS_SWAP((s)->file_size, d, 128, 27);\ ++ SQUASHFS_SWAP((s)->offset, d, 155, 13);\ ++ SQUASHFS_SWAP((s)->start_block, d, 168, 32);\ ++ SQUASHFS_SWAP((s)->i_count, d, 200, 16);\ ++ SQUASHFS_SWAP((s)->parent_inode, d, 216, 32);\ ++} ++ ++#define SQUASHFS_SWAP_DIR_INDEX(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_MEMSET(s, d, sizeof(struct squashfs_dir_index));\ ++ SQUASHFS_SWAP((s)->index, d, 0, 32);\ ++ SQUASHFS_SWAP((s)->start_block, d, 32, 32);\ ++ SQUASHFS_SWAP((s)->size, d, 64, 8);\ ++} ++ ++#define SQUASHFS_SWAP_DIR_HEADER(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_MEMSET(s, d, sizeof(struct squashfs_dir_header));\ ++ SQUASHFS_SWAP((s)->count, d, 0, 8);\ ++ SQUASHFS_SWAP((s)->start_block, d, 8, 32);\ ++ SQUASHFS_SWAP((s)->inode_number, d, 40, 32);\ ++} ++ ++#define SQUASHFS_SWAP_DIR_ENTRY(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_MEMSET(s, d, sizeof(struct squashfs_dir_entry));\ ++ SQUASHFS_SWAP((s)->offset, d, 0, 13);\ ++ SQUASHFS_SWAP((s)->type, d, 13, 3);\ ++ SQUASHFS_SWAP((s)->size, d, 16, 8);\ ++ SQUASHFS_SWAP((s)->inode_number, d, 24, 16);\ ++} ++ ++#define SQUASHFS_SWAP_FRAGMENT_ENTRY(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_MEMSET(s, d, sizeof(struct squashfs_fragment_entry));\ ++ SQUASHFS_SWAP((s)->start_block, d, 0, 64);\ ++ SQUASHFS_SWAP((s)->size, d, 64, 32);\ ++} ++ ++#define SQUASHFS_SWAP_SHORTS(s, d, n) {\ ++ int entry;\ ++ int bit_position;\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_MEMSET(s, d, n * 2);\ ++ for(entry = 0, bit_position = 0; entry < n; entry++, bit_position += \ ++ 16)\ ++ SQUASHFS_SWAP(s[entry], d, bit_position, 16);\ ++} ++ ++#define SQUASHFS_SWAP_INTS(s, d, n) {\ ++ int entry;\ ++ int bit_position;\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_MEMSET(s, d, n * 4);\ ++ for(entry = 0, bit_position = 0; entry < n; entry++, bit_position += \ ++ 32)\ ++ SQUASHFS_SWAP(s[entry], d, bit_position, 32);\ ++} ++ ++#define SQUASHFS_SWAP_LONG_LONGS(s, d, n) {\ ++ int entry;\ ++ int bit_position;\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_MEMSET(s, d, n * 8);\ ++ for(entry = 0, bit_position = 0; entry < n; entry++, bit_position += \ ++ 64)\ ++ SQUASHFS_SWAP(s[entry], d, bit_position, 64);\ ++} ++ ++#define SQUASHFS_SWAP_DATA(s, d, n, bits) {\ ++ int entry;\ ++ int bit_position;\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_MEMSET(s, d, n * bits / 8);\ ++ for(entry = 0, bit_position = 0; entry < n; entry++, bit_position += \ ++ bits)\ ++ SQUASHFS_SWAP(s[entry], d, bit_position, bits);\ ++} ++ ++#define SQUASHFS_SWAP_FRAGMENT_INDEXES(s, d, n) SQUASHFS_SWAP_LONG_LONGS(s, d, n) ++ ++#ifdef CONFIG_SQUASHFS_1_0_COMPATIBILITY ++ ++struct squashfs_base_inode_header_1 { ++ unsigned int inode_type:4; ++ unsigned int mode:12; /* protection */ ++ unsigned int uid:4; /* index into uid table */ ++ unsigned int guid:4; /* index into guid table */ ++} __attribute__ ((packed)); ++ ++struct squashfs_ipc_inode_header_1 { ++ unsigned int inode_type:4; ++ unsigned int mode:12; /* protection */ ++ unsigned int uid:4; /* index into uid table */ ++ unsigned int guid:4; /* index into guid table */ ++ unsigned int type:4; ++ unsigned int offset:4; ++} __attribute__ ((packed)); ++ ++struct squashfs_dev_inode_header_1 { ++ unsigned int inode_type:4; ++ unsigned int mode:12; /* protection */ ++ unsigned int uid:4; /* index into uid table */ ++ unsigned int guid:4; /* index into guid table */ ++ unsigned short rdev; ++} __attribute__ ((packed)); ++ ++struct squashfs_symlink_inode_header_1 { ++ unsigned int inode_type:4; ++ unsigned int mode:12; /* protection */ ++ unsigned int uid:4; /* index into uid table */ ++ unsigned int guid:4; /* index into guid table */ ++ unsigned short symlink_size; ++ char symlink[0]; ++} __attribute__ ((packed)); ++ ++struct squashfs_reg_inode_header_1 { ++ unsigned int inode_type:4; ++ unsigned int mode:12; /* protection */ ++ unsigned int uid:4; /* index into uid table */ ++ unsigned int guid:4; /* index into guid table */ ++ unsigned int mtime; ++ unsigned int start_block; ++ unsigned int file_size:32; ++ unsigned short block_list[0]; ++} __attribute__ ((packed)); ++ ++struct squashfs_dir_inode_header_1 { ++ unsigned int inode_type:4; ++ unsigned int mode:12; /* protection */ ++ unsigned int uid:4; /* index into uid table */ ++ unsigned int guid:4; /* index into guid table */ ++ unsigned int file_size:19; ++ unsigned int offset:13; ++ unsigned int mtime; ++ unsigned int start_block:24; ++} __attribute__ ((packed)); ++ ++#define SQUASHFS_SWAP_BASE_INODE_CORE_1(s, d, n) \ ++ SQUASHFS_MEMSET(s, d, n);\ ++ SQUASHFS_SWAP((s)->inode_type, d, 0, 4);\ ++ SQUASHFS_SWAP((s)->mode, d, 4, 12);\ ++ SQUASHFS_SWAP((s)->uid, d, 16, 4);\ ++ SQUASHFS_SWAP((s)->guid, d, 20, 4); ++ ++#define SQUASHFS_SWAP_BASE_INODE_HEADER_1(s, d, n) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE_1(s, d, n)\ ++} ++ ++#define SQUASHFS_SWAP_IPC_INODE_HEADER_1(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE_1(s, d, \ ++ sizeof(struct squashfs_ipc_inode_header_1));\ ++ SQUASHFS_SWAP((s)->type, d, 24, 4);\ ++ SQUASHFS_SWAP((s)->offset, d, 28, 4);\ ++} ++ ++#define SQUASHFS_SWAP_DEV_INODE_HEADER_1(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE_1(s, d, \ ++ sizeof(struct squashfs_dev_inode_header_1));\ ++ SQUASHFS_SWAP((s)->rdev, d, 24, 16);\ ++} ++ ++#define SQUASHFS_SWAP_SYMLINK_INODE_HEADER_1(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE_1(s, d, \ ++ sizeof(struct squashfs_symlink_inode_header_1));\ ++ SQUASHFS_SWAP((s)->symlink_size, d, 24, 16);\ ++} ++ ++#define SQUASHFS_SWAP_REG_INODE_HEADER_1(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE_1(s, d, \ ++ sizeof(struct squashfs_reg_inode_header_1));\ ++ SQUASHFS_SWAP((s)->mtime, d, 24, 32);\ ++ SQUASHFS_SWAP((s)->start_block, d, 56, 32);\ ++ SQUASHFS_SWAP((s)->file_size, d, 88, 32);\ ++} ++ ++#define SQUASHFS_SWAP_DIR_INODE_HEADER_1(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE_1(s, d, \ ++ sizeof(struct squashfs_dir_inode_header_1));\ ++ SQUASHFS_SWAP((s)->file_size, d, 24, 19);\ ++ SQUASHFS_SWAP((s)->offset, d, 43, 13);\ ++ SQUASHFS_SWAP((s)->mtime, d, 56, 32);\ ++ SQUASHFS_SWAP((s)->start_block, d, 88, 24);\ ++} ++ ++#endif ++ ++#ifdef CONFIG_SQUASHFS_2_0_COMPATIBILITY ++ ++struct squashfs_dir_index_2 { ++ unsigned int index:27; ++ unsigned int start_block:29; ++ unsigned char size; ++ unsigned char name[0]; ++} __attribute__ ((packed)); ++ ++struct squashfs_base_inode_header_2 { ++ unsigned int inode_type:4; ++ unsigned int mode:12; /* protection */ ++ unsigned int uid:8; /* index into uid table */ ++ unsigned int guid:8; /* index into guid table */ ++} __attribute__ ((packed)); ++ ++struct squashfs_ipc_inode_header_2 { ++ unsigned int inode_type:4; ++ unsigned int mode:12; /* protection */ ++ unsigned int uid:8; /* index into uid table */ ++ unsigned int guid:8; /* index into guid table */ ++} __attribute__ ((packed)); ++ ++struct squashfs_dev_inode_header_2 { ++ unsigned int inode_type:4; ++ unsigned int mode:12; /* protection */ ++ unsigned int uid:8; /* index into uid table */ ++ unsigned int guid:8; /* index into guid table */ ++ unsigned short rdev; ++} __attribute__ ((packed)); ++ ++struct squashfs_symlink_inode_header_2 { ++ unsigned int inode_type:4; ++ unsigned int mode:12; /* protection */ ++ unsigned int uid:8; /* index into uid table */ ++ unsigned int guid:8; /* index into guid table */ ++ unsigned short symlink_size; ++ char symlink[0]; ++} __attribute__ ((packed)); ++ ++struct squashfs_reg_inode_header_2 { ++ unsigned int inode_type:4; ++ unsigned int mode:12; /* protection */ ++ unsigned int uid:8; /* index into uid table */ ++ unsigned int guid:8; /* index into guid table */ ++ unsigned int mtime; ++ unsigned int start_block; ++ unsigned int fragment; ++ unsigned int offset; ++ unsigned int file_size:32; ++ unsigned short block_list[0]; ++} __attribute__ ((packed)); ++ ++struct squashfs_dir_inode_header_2 { ++ unsigned int inode_type:4; ++ unsigned int mode:12; /* protection */ ++ unsigned int uid:8; /* index into uid table */ ++ unsigned int guid:8; /* index into guid table */ ++ unsigned int file_size:19; ++ unsigned int offset:13; ++ unsigned int mtime; ++ unsigned int start_block:24; ++} __attribute__ ((packed)); ++ ++struct squashfs_ldir_inode_header_2 { ++ unsigned int inode_type:4; ++ unsigned int mode:12; /* protection */ ++ unsigned int uid:8; /* index into uid table */ ++ unsigned int guid:8; /* index into guid table */ ++ unsigned int file_size:27; ++ unsigned int offset:13; ++ unsigned int mtime; ++ unsigned int start_block:24; ++ unsigned int i_count:16; ++ struct squashfs_dir_index_2 index[0]; ++} __attribute__ ((packed)); ++ ++union squashfs_inode_header_2 { ++ struct squashfs_base_inode_header_2 base; ++ struct squashfs_dev_inode_header_2 dev; ++ struct squashfs_symlink_inode_header_2 symlink; ++ struct squashfs_reg_inode_header_2 reg; ++ struct squashfs_dir_inode_header_2 dir; ++ struct squashfs_ldir_inode_header_2 ldir; ++ struct squashfs_ipc_inode_header_2 ipc; ++}; ++ ++struct squashfs_dir_header_2 { ++ unsigned int count:8; ++ unsigned int start_block:24; ++} __attribute__ ((packed)); ++ ++struct squashfs_dir_entry_2 { ++ unsigned int offset:13; ++ unsigned int type:3; ++ unsigned int size:8; ++ char name[0]; ++} __attribute__ ((packed)); ++ ++struct squashfs_fragment_entry_2 { ++ unsigned int start_block; ++ unsigned int size; ++} __attribute__ ((packed)); ++ ++#define SQUASHFS_SWAP_BASE_INODE_CORE_2(s, d, n)\ ++ SQUASHFS_MEMSET(s, d, n);\ ++ SQUASHFS_SWAP((s)->inode_type, d, 0, 4);\ ++ SQUASHFS_SWAP((s)->mode, d, 4, 12);\ ++ SQUASHFS_SWAP((s)->uid, d, 16, 8);\ ++ SQUASHFS_SWAP((s)->guid, d, 24, 8);\ ++ ++#define SQUASHFS_SWAP_BASE_INODE_HEADER_2(s, d, n) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE_2(s, d, n)\ ++} ++ ++#define SQUASHFS_SWAP_IPC_INODE_HEADER_2(s, d) \ ++ SQUASHFS_SWAP_BASE_INODE_HEADER_2(s, d, sizeof(struct squashfs_ipc_inode_header_2)) ++ ++#define SQUASHFS_SWAP_DEV_INODE_HEADER_2(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE_2(s, d, \ ++ sizeof(struct squashfs_dev_inode_header_2)); \ ++ SQUASHFS_SWAP((s)->rdev, d, 32, 16);\ ++} ++ ++#define SQUASHFS_SWAP_SYMLINK_INODE_HEADER_2(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE_2(s, d, \ ++ sizeof(struct squashfs_symlink_inode_header_2));\ ++ SQUASHFS_SWAP((s)->symlink_size, d, 32, 16);\ ++} ++ ++#define SQUASHFS_SWAP_REG_INODE_HEADER_2(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE_2(s, d, \ ++ sizeof(struct squashfs_reg_inode_header_2));\ ++ SQUASHFS_SWAP((s)->mtime, d, 32, 32);\ ++ SQUASHFS_SWAP((s)->start_block, d, 64, 32);\ ++ SQUASHFS_SWAP((s)->fragment, d, 96, 32);\ ++ SQUASHFS_SWAP((s)->offset, d, 128, 32);\ ++ SQUASHFS_SWAP((s)->file_size, d, 160, 32);\ ++} ++ ++#define SQUASHFS_SWAP_DIR_INODE_HEADER_2(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE_2(s, d, \ ++ sizeof(struct squashfs_dir_inode_header_2));\ ++ SQUASHFS_SWAP((s)->file_size, d, 32, 19);\ ++ SQUASHFS_SWAP((s)->offset, d, 51, 13);\ ++ SQUASHFS_SWAP((s)->mtime, d, 64, 32);\ ++ SQUASHFS_SWAP((s)->start_block, d, 96, 24);\ ++} ++ ++#define SQUASHFS_SWAP_LDIR_INODE_HEADER_2(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_SWAP_BASE_INODE_CORE_2(s, d, \ ++ sizeof(struct squashfs_ldir_inode_header_2));\ ++ SQUASHFS_SWAP((s)->file_size, d, 32, 27);\ ++ SQUASHFS_SWAP((s)->offset, d, 59, 13);\ ++ SQUASHFS_SWAP((s)->mtime, d, 72, 32);\ ++ SQUASHFS_SWAP((s)->start_block, d, 104, 24);\ ++ SQUASHFS_SWAP((s)->i_count, d, 128, 16);\ ++} ++ ++#define SQUASHFS_SWAP_DIR_INDEX_2(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_MEMSET(s, d, sizeof(struct squashfs_dir_index_2));\ ++ SQUASHFS_SWAP((s)->index, d, 0, 27);\ ++ SQUASHFS_SWAP((s)->start_block, d, 27, 29);\ ++ SQUASHFS_SWAP((s)->size, d, 56, 8);\ ++} ++#define SQUASHFS_SWAP_DIR_HEADER_2(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_MEMSET(s, d, sizeof(struct squashfs_dir_header_2));\ ++ SQUASHFS_SWAP((s)->count, d, 0, 8);\ ++ SQUASHFS_SWAP((s)->start_block, d, 8, 24);\ ++} ++ ++#define SQUASHFS_SWAP_DIR_ENTRY_2(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_MEMSET(s, d, sizeof(struct squashfs_dir_entry_2));\ ++ SQUASHFS_SWAP((s)->offset, d, 0, 13);\ ++ SQUASHFS_SWAP((s)->type, d, 13, 3);\ ++ SQUASHFS_SWAP((s)->size, d, 16, 8);\ ++} ++ ++#define SQUASHFS_SWAP_FRAGMENT_ENTRY_2(s, d) {\ ++ SQUASHFS_SWAP_START\ ++ SQUASHFS_MEMSET(s, d, sizeof(struct squashfs_fragment_entry_2));\ ++ SQUASHFS_SWAP((s)->start_block, d, 0, 32);\ ++ SQUASHFS_SWAP((s)->size, d, 32, 32);\ ++} ++ ++#define SQUASHFS_SWAP_FRAGMENT_INDEXES_2(s, d, n) SQUASHFS_SWAP_INTS(s, d, n) ++ ++/* fragment and fragment table defines */ ++#define SQUASHFS_FRAGMENT_BYTES_2(A) (A * sizeof(struct squashfs_fragment_entry_2)) ++ ++#define SQUASHFS_FRAGMENT_INDEX_2(A) (SQUASHFS_FRAGMENT_BYTES_2(A) / \ ++ SQUASHFS_METADATA_SIZE) ++ ++#define SQUASHFS_FRAGMENT_INDEX_OFFSET_2(A) (SQUASHFS_FRAGMENT_BYTES_2(A) % \ ++ SQUASHFS_METADATA_SIZE) ++ ++#define SQUASHFS_FRAGMENT_INDEXES_2(A) ((SQUASHFS_FRAGMENT_BYTES_2(A) + \ ++ SQUASHFS_METADATA_SIZE - 1) / \ ++ SQUASHFS_METADATA_SIZE) ++ ++#define SQUASHFS_FRAGMENT_INDEX_BYTES_2(A) (SQUASHFS_FRAGMENT_INDEXES_2(A) *\ ++ sizeof(int)) ++ ++#endif ++ ++#ifdef __KERNEL__ ++ ++/* ++ * macros used to swap each structure entry, taking into account ++ * bitfields and different bitfield placing conventions on differing ++ * architectures ++ */ ++ ++#include <asm/byteorder.h> ++ ++#ifdef __BIG_ENDIAN ++ /* convert from little endian to big endian */ ++#define SQUASHFS_SWAP(value, p, pos, tbits) _SQUASHFS_SWAP(value, p, pos, \ ++ tbits, b_pos) ++#else ++ /* convert from big endian to little endian */ ++#define SQUASHFS_SWAP(value, p, pos, tbits) _SQUASHFS_SWAP(value, p, pos, \ ++ tbits, 64 - tbits - b_pos) ++#endif ++ ++#define _SQUASHFS_SWAP(value, p, pos, tbits, SHIFT) {\ ++ b_pos = pos % 8;\ ++ val = 0;\ ++ s = (unsigned char *)p + (pos / 8);\ ++ d = ((unsigned char *) &val) + 7;\ ++ for(bits = 0; bits < (tbits + b_pos); bits += 8) \ ++ *d-- = *s++;\ ++ value = (val >> (SHIFT))/* & ((1 << tbits) - 1)*/;\ ++} ++ ++#define SQUASHFS_MEMSET(s, d, n) memset(s, 0, n); ++ ++#endif ++#endif +Index: git/include/linux/squashfs_fs_i.h +=================================================================== +--- /dev/null 1970-01-01 00:00:00.000000000 +0000 ++++ git/include/linux/squashfs_fs_i.h 2006-10-31 21:26:45.000000000 +0000 +@@ -0,0 +1,45 @@ ++#ifndef SQUASHFS_FS_I ++#define SQUASHFS_FS_I ++/* ++ * Squashfs ++ * ++ * Copyright (c) 2002, 2003, 2004, 2005, 2006 ++ * Phillip Lougher <phillip@lougher.org.uk> ++ * ++ * 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, ++ * or (at your option) any later version. ++ * ++ * This program is distributed in the hope that it will be useful, ++ * but WITHOUT ANY WARRANTY; without even the implied warranty of ++ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ++ * GNU General Public License for more details. ++ * ++ * You should have received a copy of the GNU General Public License ++ * along with this program; if not, write to the Free Software ++ * Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. ++ * ++ * squashfs_fs_i.h ++ */ ++ ++struct squashfs_inode_info { ++ long long start_block; ++ unsigned int offset; ++ union { ++ struct { ++ long long fragment_start_block; ++ unsigned int fragment_size; ++ unsigned int fragment_offset; ++ long long block_list_start; ++ } s1; ++ struct { ++ long long directory_index_start; ++ unsigned int directory_index_offset; ++ unsigned int directory_index_count; ++ unsigned int parent_inode; ++ } s2; ++ } u; ++ struct inode vfs_inode; ++}; ++#endif +Index: git/include/linux/squashfs_fs_sb.h +=================================================================== +--- /dev/null 1970-01-01 00:00:00.000000000 +0000 ++++ git/include/linux/squashfs_fs_sb.h 2006-10-31 21:26:45.000000000 +0000 +@@ -0,0 +1,74 @@ ++#ifndef SQUASHFS_FS_SB ++#define SQUASHFS_FS_SB ++/* ++ * Squashfs ++ * ++ * Copyright (c) 2002, 2003, 2004, 2005, 2006 ++ * Phillip Lougher <phillip@lougher.org.uk> ++ * ++ * 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, ++ * or (at your option) any later version. ++ * ++ * This program is distributed in the hope that it will be useful, ++ * but WITHOUT ANY WARRANTY; without even the implied warranty of ++ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ++ * GNU General Public License for more details. ++ * ++ * You should have received a copy of the GNU General Public License ++ * along with this program; if not, write to the Free Software ++ * Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. ++ * ++ * squashfs_fs_sb.h ++ */ ++ ++#include <linux/squashfs_fs.h> ++ ++struct squashfs_cache { ++ long long block; ++ int length; ++ long long next_index; ++ char *data; ++}; ++ ++struct squashfs_fragment_cache { ++ long long block; ++ int length; ++ unsigned int locked; ++ char *data; ++}; ++ ++struct squashfs_sb_info { ++ struct squashfs_super_block sblk; ++ int devblksize; ++ int devblksize_log2; ++ int swap; ++ struct squashfs_cache *block_cache; ++ struct squashfs_fragment_cache *fragment; ++ int next_cache; ++ int next_fragment; ++ int next_meta_index; ++ unsigned int *uid; ++ unsigned int *guid; ++ long long *fragment_index; ++ unsigned int *fragment_index_2; ++ unsigned int read_size; ++ char *read_data; ++ char *read_page; ++ struct semaphore read_data_mutex; ++ struct semaphore read_page_mutex; ++ struct semaphore block_cache_mutex; ++ struct semaphore fragment_mutex; ++ struct semaphore meta_index_mutex; ++ wait_queue_head_t waitq; ++ wait_queue_head_t fragment_wait_queue; ++ struct meta_index *meta_index; ++ struct inode *(*iget)(struct super_block *s, squashfs_inode_t \ ++ inode); ++ long long (*read_blocklist)(struct inode *inode, int \ ++ index, int readahead_blks, char *block_list, \ ++ unsigned short **block_p, unsigned int *bsize); ++ int (*read_fragment_index_table)(struct super_block *s); ++}; ++#endif +Index: git/init/do_mounts_rd.c +=================================================================== +--- git.orig/init/do_mounts_rd.c 2006-10-31 16:09:47.000000000 +0000 ++++ git/init/do_mounts_rd.c 2006-10-31 21:26:45.000000000 +0000 +@@ -5,6 +5,7 @@ + #include <linux/ext2_fs.h> + #include <linux/romfs_fs.h> + #include <linux/cramfs_fs.h> ++#include <linux/squashfs_fs.h> + #include <linux/initrd.h> + #include <linux/string.h> + +@@ -39,6 +40,7 @@ static int __init crd_load(int in_fd, in + * numbers could not be found. + * + * We currently check for the following magic numbers: ++ * squashfs + * minix + * ext2 + * romfs +@@ -53,6 +55,7 @@ identify_ramdisk_image(int fd, int start + struct ext2_super_block *ext2sb; + struct romfs_super_block *romfsb; + struct cramfs_super *cramfsb; ++ struct squashfs_super_block *squashfsb; + int nblocks = -1; + unsigned char *buf; + +@@ -64,6 +67,7 @@ identify_ramdisk_image(int fd, int start + ext2sb = (struct ext2_super_block *) buf; + romfsb = (struct romfs_super_block *) buf; + cramfsb = (struct cramfs_super *) buf; ++ squashfsb = (struct squashfs_super_block *) buf; + memset(buf, 0xe5, size); + + /* +@@ -101,6 +105,15 @@ identify_ramdisk_image(int fd, int start + goto done; + } + ++ /* squashfs is at block zero too */ ++ if (squashfsb->s_magic == SQUASHFS_MAGIC) { ++ printk(KERN_NOTICE ++ "RAMDISK: squashfs filesystem found at block %d\n", ++ start_block); ++ nblocks = (squashfsb->bytes_used+BLOCK_SIZE-1)>>BLOCK_SIZE_BITS; ++ goto done; ++ } ++ + /* + * Read block 1 to test for minix and ext2 superblock + */ diff --git a/packages/linux/linux-rp-2.6.20/vesafb-tng-1.0-rc2-2.6.20-rc2.patch b/packages/linux/linux-rp-2.6.20/vesafb-tng-1.0-rc2-2.6.20-rc2.patch new file mode 100644 index 0000000000..b1b0fc3549 --- /dev/null +++ b/packages/linux/linux-rp-2.6.20/vesafb-tng-1.0-rc2-2.6.20-rc2.patch @@ -0,0 +1,3141 @@ +diff --git a/Documentation/fb/vesafb.txt b/Documentation/fb/vesafb.txt +index ee277dd..93d6e6e 100644 +--- a/Documentation/fb/vesafb.txt ++++ b/Documentation/fb/vesafb.txt +@@ -2,16 +2,18 @@ + What is vesafb? + =============== + +-This is a generic driver for a graphic framebuffer on intel boxes. ++Vesafb is a generic framebuffer driver for x86 and x86_64 boxes. + +-The idea is simple: Turn on graphics mode at boot time with the help +-of the BIOS, and use this as framebuffer device /dev/fb0, like the m68k +-(and other) ports do. ++VESA BIOS Extensions Version 2.0 are required, because we need access to ++a linear frame buffer. VBE 3.0 is required if you want to use modes with a ++higher (than the standard 60 Hz) refresh rate. + +-This means we decide at boot time whenever we want to run in text or +-graphics mode. Switching mode later on (in protected mode) is +-impossible; BIOS calls work in real mode only. VESA BIOS Extensions +-Version 2.0 are required, because we need a linear frame buffer. ++The VESA framebuffer driver comes in two flavors - the standard 'vesafb' ++and 'vesafb-tng'. Vesafb-tng is available only on 32-bit x86 due to the ++technology it uses (vm86). Vesafb-tng has more features than vesafb ++(adjusting the refresh rate on VBE 3.0 compliant boards, switching the ++video mode without rebooting, selecting a mode by providing its ++modedb name, and more). + + Advantages: + +@@ -29,26 +31,35 @@ Disadvantages: + How to use it? + ============== + +-Switching modes is done using the vga=... boot parameter. Read +-Documentation/svga.txt for details. ++If you are running a 32-bit x86 system and you decide to use vesafb-tng, ++you can either compile the driver into the kernel or use it as a module. ++The graphics mode you want to use is in both cases specified using the ++standard modedb format. + +-You should compile in both vgacon (for text mode) and vesafb (for +-graphics mode). Which of them takes over the console depends on +-whenever the specified mode is text or graphics. ++If your system doesn't support vm86 calls, things get a little more tricky. ++Since on such systems you can't do BIOS calls from protected mode in which ++kernel runs, you have to decide at boot time whenever you want to run in text ++or in graphics mode. Switching mode later on is impossible. Switching modes ++is done using the vga=... boot parameter. Read Documentation/svga.txt for ++details. Below is a more detailed description of what to do on systems using ++the standard vesafb driver. + +-The graphic modes are NOT in the list which you get if you boot with +-vga=ask and hit return. The mode you wish to use is derived from the +-VESA mode number. Here are those VESA mode numbers: ++You should compile in both vgacon (for text mode) and vesafb (for graphics ++mode). Which of them takes over the console depends on whenever the ++specified mode is text or graphics. ++ ++The graphic modes are NOT in the list which you get if you boot with vga=ask ++and hit return. The mode you wish to use is derived from the VESA mode number. ++Here are those VESA mode numbers: + + | 640x480 800x600 1024x768 1280x1024 + ----+------------------------------------- +-256 | 0x101 0x103 0x105 0x107 +-32k | 0x110 0x113 0x116 0x119 +-64k | 0x111 0x114 0x117 0x11A +-16M | 0x112 0x115 0x118 0x11B ++256 | 0x101 0x103 0x105 0x107 ++32k | 0x110 0x113 0x116 0x119 ++64k | 0x111 0x114 0x117 0x11A ++16M | 0x112 0x115 0x118 0x11B + +-The video mode number of the Linux kernel is the VESA mode number plus +-0x200. ++The video mode number of the Linux kernel is the VESA mode number plus 0x200. + + Linux_kernel_mode_number = VESA_mode_number + 0x200 + +@@ -56,15 +67,15 @@ So the table for the Kernel mode numbers are: + + | 640x480 800x600 1024x768 1280x1024 + ----+------------------------------------- +-256 | 0x301 0x303 0x305 0x307 +-32k | 0x310 0x313 0x316 0x319 +-64k | 0x311 0x314 0x317 0x31A +-16M | 0x312 0x315 0x318 0x31B ++256 | 0x301 0x303 0x305 0x307 ++32k | 0x310 0x313 0x316 0x319 ++64k | 0x311 0x314 0x317 0x31A ++16M | 0x312 0x315 0x318 0x31B + +-To enable one of those modes you have to specify "vga=ask" in the +-lilo.conf file and rerun LILO. Then you can type in the desired +-mode at the "vga=ask" prompt. For example if you like to use +-1024x768x256 colors you have to say "305" at this prompt. ++To enable one of those modes you have to specify "vga=ask" in the lilo.conf ++file and rerun LILO. Then you can type in the desired mode at the "vga=ask" ++prompt. For example if you like to use 1024x768x256 colors you have to say ++"305" at this prompt. + + If this does not work, this might be because your BIOS does not support + linear framebuffers or because it does not support this mode at all. +@@ -72,11 +83,12 @@ Even if your board does, it might be the BIOS which does not. VESA BIOS + Extensions v2.0 are required, 1.2 is NOT sufficient. You will get a + "bad mode number" message if something goes wrong. + +-1. Note: LILO cannot handle hex, for booting directly with ++1. Note: LILO cannot handle hex, for booting directly with + "vga=mode-number" you have to transform the numbers to decimal. + 2. Note: Some newer versions of LILO appear to work with those hex values, + if you set the 0x in front of the numbers. + ++ + X11 + === + +@@ -84,98 +96,164 @@ XF68_FBDev should work just fine, but it is non-accelerated. Running + another (accelerated) X-Server like XF86_SVGA might or might not work. + It depends on X-Server and graphics board. + +-The X-Server must restore the video mode correctly, else you end up ++The X-Server must restore the video mode correctly, or else you end up + with a broken console (and vesafb cannot do anything about this). ++With vesafb-tng chances are that the console will be restored properly ++even if the X server messes up the video mode. + + + Refresh rates + ============= + +-There is no way to change the vesafb video mode and/or timings after +-booting linux. If you are not happy with the 60 Hz refresh rate, you +-have these options: ++With VBE 3.0 compatible BIOSes and vesafb-tng it is possible to change ++the refresh rate either at boot time (by specifying the @<rr> part of ++the mode name) or later, using the fbset utility. ++ ++If you want to use the default BIOS refresh rate while switching modes ++on a running system, set pixclock to 0. + +- * configure and load the DOS-Tools for your the graphics board (if +- available) and boot linux with loadlin. +- * use a native driver (matroxfb/atyfb) instead if vesafb. If none ++With VBE 2.0 there is no way to change the mode timings after booting ++Linux. If you are not happy with the 60 Hz refresh rate, you have ++the following options: ++ ++ * Configure and load the DOS tools for your the graphics board (if ++ available) and boot Linux with loadlin. ++ * Use a native driver (matroxfb/atyfb) instead of vesafb. If none + is available, write a new one! +- * VBE 3.0 might work too. I have neither a gfx board with VBE 3.0 +- support nor the specs, so I have not checked this yet. ++ * Use a BIOS editor to change the default refresh rate (such an ++ editor does exist at least for ATI Radeon BIOSes). ++ * If you're running a non-vm86 and VBE 3.0 compatible system, you can ++ use a kernel patch (vesafb-rrc) to hard-code some mode timings in ++ the kernel and use these while setting the video mode at boot time. ++ ++Note that there are some boards (nVidia 59**, 57** and newer models) ++claiming that their Video BIOS is VBE 3.0 compliant, while ignoring the ++CRTC values provided by software such as vesafb-tng. You'll not be able ++to adjust the refresh rate if you're using one of these boards. + + + Configuration + ============= + +-The VESA BIOS provides protected mode interface for changing +-some parameters. vesafb can use it for palette changes and +-to pan the display. It is turned off by default because it +-seems not to work with some BIOS versions, but there are options +-to turn it on. +- +-You can pass options to vesafb using "video=vesafb:option" on +-the kernel command line. Multiple options should be separated +-by comma, like this: "video=vesafb:ypan,invers" +- +-Accepted options: +- +-invers no comment... +- +-ypan enable display panning using the VESA protected mode +- interface. The visible screen is just a window of the +- video memory, console scrolling is done by changing the +- start of the window. +- pro: * scrolling (fullscreen) is fast, because there is +- no need to copy around data. +- * You'll get scrollback (the Shift-PgUp thing), +- the video memory can be used as scrollback buffer +- kontra: * scrolling only parts of the screen causes some +- ugly flicker effects (boot logo flickers for +- example). +- +-ywrap Same as ypan, but assumes your gfx board can wrap-around +- the video memory (i.e. starts reading from top if it +- reaches the end of video memory). Faster than ypan. +- +-redraw scroll by redrawing the affected part of the screen, this +- is the safe (and slow) default. +- +- +-vgapal Use the standard vga registers for palette changes. +- This is the default. +-pmipal Use the protected mode interface for palette changes. +- +-mtrr:n setup memory type range registers for the vesafb framebuffer +- where n: +- 0 - disabled (equivalent to nomtrr) (default) +- 1 - uncachable +- 2 - write-back +- 3 - write-combining +- 4 - write-through +- +- If you see the following in dmesg, choose the type that matches the +- old one. In this example, use "mtrr:2". ++The VESA BIOS provides protected mode interface for changing some parameters. ++vesafb can use it for palette changes and to pan the display. It is turned ++off by default because it seems not to work with some BIOS versions, but ++there are options to turn it on. ++ ++You can pass options to vesafb using "video=vesafb:option" on the kernel ++command line. Multiple options should be separated by a comma, like this: ++"video=vesafb:ypan,1024x768-32@85" ++ ++Note that vesafb-tng still uses the "video=vesafb:option" format of the ++kernel command line video parameter. "video=vesafb-tng:xxx" is incorrect. ++ ++Accepted options (both vesafb and vesafb-tng): ++ ++ypan Enable display panning using the VESA protected mode interface ++ The visible screen is just a window of the video memory, ++ console scrolling is done by changing the start of the window. ++ pro: * scrolling (fullscreen) is fast, because there is ++ no need to copy around data. ++ * you'll get scrollback (the Shift-PgUp thing), ++ the video memory can be used as scrollback buffer ++ con: * scrolling only parts of the screen causes some ++ ugly flicker effects (boot logo flickers for ++ example). ++ ++ywrap Same as ypan, but assumes your gfx board can wrap-around the video ++ memory (i.e. starts reading from top if it reaches the end of ++ video memory). Faster than ypan. ++ ++redraw Scroll by redrawing the affected part of the screen, this is the ++ safe (and slow) default. ++ ++vgapal Use the standard VGA registers for palette changes. ++ ++pmipal Use the protected mode interface for palette changes. ++ This is the default is the protected mode interface is available. ++ ++mtrr:n Setup memory type range registers for the vesafb framebuffer ++ where n: ++ 0 - disabled (equivalent to nomtrr) (default) ++ 1 - uncachable ++ 2 - write-back ++ 3 - write-combining ++ 4 - write-through ++ ++ If you see the following in dmesg, choose the type that matches ++ the old one. In this example, use "mtrr:2". + ... + mtrr: type mismatch for e0000000,8000000 old: write-back new: write-combining + ... + +-nomtrr disable mtrr ++nomtrr Do not use memory type range registers for vesafb. + + vremap:n + remap 'n' MiB of video RAM. If 0 or not specified, remap memory +- according to video mode. (2.5.66 patch/idea by Antonino Daplas +- reversed to give override possibility (allocate more fb memory +- than the kernel would) to 2.4 by tmb@iki.fi) ++ according to video mode. (2.5.66 patch/idea by Antonino Daplas ++ reversed to give override possibility (allocate more fb memory ++ than the kernel would) to 2.4 by tmb@iki.fi) + + vtotal:n + if the video BIOS of your card incorrectly determines the total + amount of video RAM, use this option to override the BIOS (in MiB). + +-Have fun! ++Options accepted only by vesafb-tng: ++ ++<mode> The mode you want to set, in the standard modedb format. Refer to ++ modedb.txt for a detailed description. If you specify a mode that is ++ not supported by your board's BIOS, vesafb-tng will attempt to set a ++ similar mode. The list of supported modes can be found in ++ /proc/fbx/modes, where x is the framebuffer number (usually 0). ++ When vesafb-tng is compiled as a module, the mode string should be ++ provided as a value of the parameter 'mode'. ++ ++vbemode:x ++ Force the use of VBE mode x. The mode will only be set if it's ++ found in the VBE-provided list of supported modes. ++ NOTE: The mode number 'x' should be specified in VESA mode number ++ notation, not the Linux kernel one (eg. 257 instead of 769). ++ HINT: If you use this option because normal <mode> parameter does ++ not work for you and you use a X server, you'll probably want to ++ set the 'nocrtc' option to ensure that the video mode is properly ++ restored after console <-> X switches. ++ ++nocrtc Do not use CRTC timings while setting the video mode. This option ++ makes sence only with VBE 3.0 compliant systems. Use it if you have ++ problems with modes set in the standard way. Note that using this ++ option means that any refresh rate adjustments will be ignored ++ and the refresh rate will stay at your BIOS default (60 Hz). ++ ++noedid Do not try to fetch and use EDID-provided modes. ++ ++noblank Disable hardware blanking. ++ ++gtf Force the use of VESA's GTF (Generalized Timing Formula). Specifying ++ this will cause vesafb to skip its internal modedb and EDID-modedb ++ and jump straight to the GTF part of the code (normally used only if ++ everything else failed). This can be useful if you want to get as ++ much as possible from your graphics board but your BIOS doesn't ++ support modes with the refresh rates you require. Note that you may ++ need to specify the maxhf, maxvf and maxclk parameters if they are not ++ provided by the EDID block. ++ ++Additionally, the following parameters may be provided. They all override the ++EDID-provided values and BIOS defaults. Refer to your monitor's specs to get ++the correct values for maxhf, maxvf and maxclk for your hardware. ++ ++maxhf:n Maximum horizontal frequency (in kHz). ++maxvf:n Maximum vertical frequency (in Hz). ++maxclk:n Maximum pixel clock (in MHz). + +- Gerd ++Have fun! + + -- ++Original document for the vesafb driver by + Gerd Knorr <kraxel@goldbach.in-berlin.de> + +-Minor (mostly typo) changes +-by Nico Schmoigl <schmoigl@rumms.uni-mannheim.de> ++Minor (mostly typo) changes by ++Nico Schmoigl <schmoigl@rumms.uni-mannheim.de> ++ ++Extended documentation for vm86, VBE 3.0 and vesafb-tng by ++Michal Januszewski <spock@gentoo.org> ++ +diff --git a/arch/i386/boot/video.S b/arch/i386/boot/video.S +index 2c5b5cc..2c2d4b5 100644 +--- a/arch/i386/boot/video.S ++++ b/arch/i386/boot/video.S +@@ -163,10 +163,12 @@ basret: ret + # parameters in the default 80x25 mode -- these are set directly, + # because some very obscure BIOSes supply insane values. + mode_params: ++#ifdef CONFIG_FB_VESA_STD + #ifdef CONFIG_VIDEO_SELECT + cmpb $0, graphic_mode + jnz mopar_gr + #endif ++#endif + movb $0x03, %ah # Read cursor position + xorb %bh, %bh + int $0x10 +@@ -199,6 +201,7 @@ mopar2: movb %al, %fs:(PARAM_VIDEO_LINES) + ret + + #ifdef CONFIG_VIDEO_SELECT ++#ifdef CONFIG_FB_VESA_STD + # Fetching of VESA frame buffer parameters + mopar_gr: + leaw modelist+1024, %di +@@ -281,6 +284,7 @@ dac_done: + movw %es, %fs:(PARAM_VESAPM_SEG) + movw %di, %fs:(PARAM_VESAPM_OFF) + no_pm: ret ++#endif + + # The video mode menu + mode_menu: +@@ -495,10 +499,12 @@ mode_set: + + cmpb $VIDEO_FIRST_V7>>8, %ah + jz setv7 +- ++ ++#ifdef CONFIG_FB_VESA_STD + cmpb $VIDEO_FIRST_VESA>>8, %ah + jnc check_vesa +- ++#endif ++ + orb %ah, %ah + jz setmenu + +@@ -570,6 +576,7 @@ setr1: lodsw + movw -4(%si), %ax # Fetch mode ID + jmp _m_s + ++#ifdef CONFIG_FB_VESA_STD + check_vesa: + leaw modelist+1024, %di + subb $VIDEO_FIRST_VESA>>8, %bh +@@ -603,6 +610,7 @@ check_vesa: + ret + + _setbad: jmp setbad # Ugly... ++#endif + + # Recalculate vertical display end registers -- this fixes various + # inconsistencies of extended modes on many adapters. Called when +diff --git a/drivers/char/sysrq.c b/drivers/char/sysrq.c +index 1393523..8a05f95 100644 +--- a/drivers/char/sysrq.c ++++ b/drivers/char/sysrq.c +@@ -240,7 +240,7 @@ static void send_sig_all(int sig) + struct task_struct *p; + + for_each_process(p) { +- if (p->mm && !is_init(p)) ++ if (p->mm && !is_init(p) && !(p->flags & PF_BORROWED_MM)) + /* Not swapper, init nor kernel thread */ + force_sig(sig, p); + } +diff --git a/drivers/video/Kconfig b/drivers/video/Kconfig +index 4e83f01..ae122fd 100644 +--- a/drivers/video/Kconfig ++++ b/drivers/video/Kconfig +@@ -547,8 +547,22 @@ config FB_TGA + cards. Say Y if you have one of those. + + config FB_VESA +- bool "VESA VGA graphics support" +- depends on (FB = y) && X86 ++ tristate "VESA VGA graphics support" ++ depends on (FB = y) && (X86 || X86_64) ++ help ++ This is the frame buffer device driver for generic VESA 2.0 ++ compliant graphic cards. The older VESA 1.2 cards are not supported. ++ You will get a boot time penguin logo at no additional cost. Please ++ read <file:Documentation/fb/vesafb.txt>. If unsure, say Y. ++ ++choice ++ prompt "VESA driver type" ++ depends on FB_VESA ++ default FB_VESA_STD if X86_64 ++ default FB_VESA_TNG if X86 ++ ++config FB_VESA_STD ++ bool "vesafb" + select FB_CFB_FILLRECT + select FB_CFB_COPYAREA + select FB_CFB_IMAGEBLIT +@@ -557,7 +571,43 @@ config FB_VESA + This is the frame buffer device driver for generic VESA 2.0 + compliant graphic cards. The older VESA 1.2 cards are not supported. + You will get a boot time penguin logo at no additional cost. Please +- read <file:Documentation/fb/vesafb.txt>. If unsure, say Y. ++ read <file:Documentation/fb/vesafb.txt>. Choose this driver if you ++ are experiencing problems with vesafb-tng or if you own a 64-bit system. ++ ++ Note that this driver cannot be compiled as a module. ++ ++config FB_VESA_TNG ++ bool "vesafb-tng" ++ depends on !X86_64 ++ select FB_MODE_HELPERS ++ select FB_CFB_FILLRECT ++ select FB_CFB_COPYAREA ++ select FB_CFB_IMAGEBLIT ++ help ++ This is an enhanced generic frame buffer device driver for ++ VBE 2.0 compliant graphic cards. It can take advantage of VBE 3.0 ++ features (refresh rate adjustment) when these are available. ++ The driver also makes it possible to change the video mode ++ on the fly and to switch back to text mode when it's unloaded. ++ ++ If the driver is compiled as a module, the module will be called ++ vesafb-tng. ++ ++endchoice ++ ++config FB_VESA_DEFAULT_MODE ++ string "VESA default mode" ++ depends on FB_VESA_TNG ++ default "640x480@60" ++ help ++ This option is used to determine the default mode vesafb is ++ supposed to switch to in case no mode is provided as a kernel ++ command line parameter. ++ ++config VIDEO_SELECT ++ bool ++ depends on FB_VESA ++ default y + + config FB_IMAC + bool "Intel-based Macintosh Framebuffer Support" +diff --git a/drivers/video/Makefile b/drivers/video/Makefile +index 309a26d..e57b0e7 100644 +--- a/drivers/video/Makefile ++++ b/drivers/video/Makefile +@@ -102,7 +102,11 @@ obj-$(CONFIG_FB_PNX4008_DUM_RGB) += pnx4008/ + obj-$(CONFIG_FB_IBM_GXT4500) += gxt4500.o + + # Platform or fallback drivers go here +-obj-$(CONFIG_FB_VESA) += vesafb.o ++ifeq ($(CONFIG_FB_VESA_STD),y) ++ obj-y += vesafb.o ++else ++ obj-$(CONFIG_FB_VESA) += vesafb-thread.o vesafb-tng.o ++endif + obj-$(CONFIG_FB_IMAC) += imacfb.o + obj-$(CONFIG_FB_VGA16) += vga16fb.o vgastate.o + obj-$(CONFIG_FB_OF) += offb.o +diff --git a/drivers/video/fbmem.c b/drivers/video/fbmem.c +index 3cfea31..bfb39cc 100644 +--- a/drivers/video/fbmem.c ++++ b/drivers/video/fbmem.c +@@ -1408,6 +1408,7 @@ fbmem_init(void) + printk(KERN_WARNING "Unable to create fb class; errno = %ld\n", PTR_ERR(fb_class)); + fb_class = NULL; + } ++ + return 0; + } + +diff --git a/drivers/video/modedb.c b/drivers/video/modedb.c +index 5df41f6..f838a53 100644 +--- a/drivers/video/modedb.c ++++ b/drivers/video/modedb.c +@@ -674,6 +674,7 @@ void fb_var_to_videomode(struct fb_videomode *mode, + { + u32 pixclock, hfreq, htotal, vtotal; + ++ mode->refresh = 0; + mode->name = NULL; + mode->xres = var->xres; + mode->yres = var->yres; +@@ -1025,3 +1026,4 @@ EXPORT_SYMBOL(fb_find_best_mode); + EXPORT_SYMBOL(fb_find_nearest_mode); + EXPORT_SYMBOL(fb_videomode_to_modelist); + EXPORT_SYMBOL(fb_find_mode); ++EXPORT_SYMBOL(fb_destroy_modelist); +diff --git a/drivers/video/vesafb-thread.c b/drivers/video/vesafb-thread.c +new file mode 100644 +index 0000000..543e202 +--- /dev/null ++++ b/drivers/video/vesafb-thread.c +@@ -0,0 +1,751 @@ ++/* ++ * Framebuffer driver for VBE 2.0+ compliant graphic boards. ++ * Kernel thread and vm86 routines. ++ * ++ * (c) 2004-2006 Michal Januszewski <spock@gentoo.org> ++ * ++ */ ++ ++#include <linux/workqueue.h> ++#include <linux/completion.h> ++#include <linux/module.h> ++#include <linux/kernel.h> ++#include <linux/errno.h> ++#include <linux/mm.h> ++#include <linux/delay.h> ++#include <linux/signal.h> ++#include <linux/freezer.h> ++#include <linux/suspend.h> ++#include <linux/unistd.h> ++#include <video/vesa.h> ++#include <video/edid.h> ++#include <asm/mman.h> ++#include <asm/page.h> ++#include <asm/vm86.h> ++#include <asm/thread_info.h> ++#include <asm/uaccess.h> ++#include <asm/mmu_context.h> ++#include "edid.h" ++ ++static int errno; ++ ++static DECLARE_COMPLETION(vesafb_th_completion); ++static DECLARE_MUTEX(vesafb_task_list_sem); ++static LIST_HEAD(vesafb_task_list); ++static DECLARE_WAIT_QUEUE_HEAD(vesafb_wait); ++ ++static struct vm86_struct vm86; ++static int vesafb_pid = 0; ++ ++#define DEFAULT_VM86_FLAGS (IF_MASK | IOPL_MASK) ++#define VM86_PUSHW(x) \ ++do { \ ++ vm86.regs.esp -= 2; \ ++ *(u16*)(STACK_ADDR + vm86.regs.esp) = x; \ ++} while(0); ++ ++/* Stack, the return code and buffers will be put into ++ * one contiguous memory chunk: ++ * ++ * [ STACK | RET_CODE | BUFFER ] ++ * ++ * Some video BIOSes (sis6326) try to store data somewhere ++ * in 0x7000-0x7fff, so we zeromap more memory to be safe. ++ */ ++#define IVTBDA_SIZE PAGE_SIZE ++#define RET_CODE_SIZE 0x0010 ++#define STACK_SIZE 0x0500 ++#define BUFFER_SIZE 0x10000 ++ ++/* The amount of memory that will be allocated should be a multiple ++ * of PAGE_SIZE. */ ++#define __MEM_SIZE (RET_CODE_SIZE + STACK_SIZE + BUFFER_SIZE) ++#define REAL_MEM_SIZE (((__MEM_SIZE / PAGE_SIZE) + 1) * PAGE_SIZE) ++ ++#define IVTBDA_ADDR 0x00000 ++#define STACK_ADDR (IVTBDA_ADDR + IVTBDA_SIZE) ++#define RET_CODE_ADDR (STACK_ADDR + STACK_SIZE) ++#define BUF_ADDR (RET_CODE_ADDR + RET_CODE_SIZE) ++ ++#define FLAG_D (1 << 10) ++ ++/* Syscalls used by the vesafb thread */ ++static int vm86old(struct vm86_struct __user* v86) ++{ ++ long res; ++ __asm__ volatile ("push %%ebx; movl %2, %%ebx ; int $0x80 ; pop %%ebx" ++ : "=a" (res) ++ : "0" (__NR_vm86old), "ri" ((long)(v86)) : "memory"); ++ ++ if ((unsigned long)(res) >= (unsigned long)(-MAX_ERRNO)) { ++ errno = -res; ++ res = -1; ++ } ++ return (int)res; ++} ++ ++static int ioperm(unsigned long a, unsigned long b, unsigned long c) ++{ ++ long res; ++ __asm__ volatile ("push %%ebx; movl %2, %%ebx ; int $0x80 ; pop %%ebx" ++ : "=a" (res) ++ : "0" (__NR_ioperm), "ri" ((long)(a)), "c" ((long)(b)), ++ "d" ((long)(c)) : "memory"); ++ ++ if ((unsigned long)(res) >= (unsigned long)(-MAX_ERRNO)) { ++ errno = -res; ++ res = -1; ++ } ++ return (int)res; ++} ++ ++/* Segment prefix opcodes */ ++enum { ++ P_CS = 0x2e, ++ P_SS = 0x36, ++ P_DS = 0x3e, ++ P_ES = 0x26, ++ P_FS = 0x64, ++ P_GS = 0x65 ++}; ++ ++/* Emulated vm86 ins instruction */ ++static void vm86_ins(int size) ++{ ++ u32 edx, edi; ++ edx = vm86.regs.edx & 0xffff; ++ edi = (vm86.regs.edi & 0xffff) + (u32)(vm86.regs.es << 4); ++ ++ if (vm86.regs.eflags & FLAG_D) ++ asm volatile ("std\n"); ++ else ++ asm volatile ("cld\n"); ++ ++ switch (size) { ++ case 4: ++ asm volatile ("insl\n" : "=D" (edi) : "d" (edx), "0" (edi)); ++ break; ++ case 2: ++ asm volatile ("insw\n" : "=D" (edi) : "d" (edx), "0" (edi)); ++ break; ++ case 1: ++ asm volatile ("insb\n" : "=D" (edi) : "d" (edx), "0" (edi)); ++ break; ++ } ++ ++ if (vm86.regs.eflags & FLAG_D) ++ asm volatile ("cld\n"); ++ ++ edi -= (u32)(vm86.regs.es << 4); ++ ++ vm86.regs.edi &= 0xffff0000; ++ vm86.regs.edi |= edi & 0xffff; ++} ++ ++static void vm86_rep_ins(int size) ++{ ++ u16 cx = vm86.regs.ecx; ++ while (cx--) ++ vm86_ins(size); ++ ++ vm86.regs.ecx &= 0xffff0000; ++} ++ ++/* Emulated vm86 outs instruction */ ++static void vm86_outs(int size, int segment) ++{ ++ u32 edx, esi, base; ++ ++ edx = vm86.regs.edx & 0xffff; ++ esi = vm86.regs.esi & 0xffff; ++ ++ switch (segment) { ++ case P_CS: base = vm86.regs.cs; break; ++ case P_SS: base = vm86.regs.ss; break; ++ case P_ES: base = vm86.regs.es; break; ++ case P_FS: base = vm86.regs.fs; break; ++ case P_GS: base = vm86.regs.gs; break; ++ default: base = vm86.regs.ds; break; ++ } ++ ++ esi += base << 4; ++ ++ if (vm86.regs.eflags & FLAG_D) ++ asm volatile ("std\n"); ++ else ++ asm volatile ("cld\n"); ++ ++ switch (size) { ++ case 4: ++ asm volatile ("outsl\n" : "=S" (esi) : "d" (edx), "0" (esi)); ++ break; ++ case 2: ++ asm volatile ("outsw\n" : "=S" (esi) : "d" (edx), "0" (esi)); ++ break; ++ case 1: ++ asm volatile ("outsb\n" : "=S" (esi) : "d" (edx), "0" (esi)); ++ break; ++ } ++ ++ if (vm86.regs.eflags & FLAG_D) ++ asm volatile ("cld"); ++ ++ esi -= base << 4; ++ vm86.regs.esi &= 0xffff0000; ++ vm86.regs.esi |= (esi & 0xffff); ++} ++ ++static void vm86_rep_outs(int size, int segment) ++{ ++ u16 cx = vm86.regs.ecx; ++ while (cx--) ++ vm86_outs(size, segment); ++ ++ vm86.regs.ecx &= 0xffff0000; ++} ++ ++static int vm86_do_unknown(void) ++{ ++ u8 data32 = 0, segment = P_DS, rep = 0; ++ u8 *instr; ++ int ret = 0, i = 0; ++ ++ instr = (u8*)((vm86.regs.cs << 4) + vm86.regs.eip); ++ ++ while (1) { ++ switch(instr[i]) { ++ case 0x66: /* operand size prefix */ ++ data32 = 1 - data32; ++ i++; ++ break; ++ case 0xf2: /* repnz */ ++ case 0xf3: /* rep */ ++ rep = 1; ++ i++; ++ break; ++ case P_CS: /* segment prefix */ ++ case P_SS: ++ case P_DS: ++ case P_ES: ++ case P_FS: ++ case P_GS: ++ segment = instr[i]; ++ i++; ++ break; ++ case 0xf0: /* LOCK - ignored */ ++ case 0x67: /* address size prefix - ignored */ ++ i++; ++ break; ++ case 0x6c: /* insb */ ++ if (rep) ++ vm86_rep_ins(1); ++ else ++ vm86_ins(1); ++ i++; ++ goto out; ++ case 0x6d: /* insw / insd */ ++ if (rep) { ++ if (data32) ++ vm86_rep_ins(4); ++ else ++ vm86_rep_ins(2); ++ } else { ++ if (data32) ++ vm86_ins(4); ++ else ++ vm86_ins(2); ++ } ++ i++; ++ goto out; ++ case 0x6e: /* outsb */ ++ if (rep) ++ vm86_rep_outs(1, segment); ++ else ++ vm86_outs(1, segment); ++ i++; ++ goto out; ++ case 0x6f: /* outsw / outsd */ ++ if (rep) { ++ if (data32) ++ vm86_rep_outs(4, segment); ++ else ++ vm86_rep_outs(2, segment); ++ } else { ++ if (data32) ++ vm86_outs(4, segment); ++ else ++ vm86_outs(2, segment); ++ } ++ i++; ++ goto out; ++ case 0xe4: /* inb xx */ ++ asm volatile ( ++ "inb %w1, %b0" ++ : "=a" (vm86.regs.eax) ++ : "d" (instr[i+1]), "0" (vm86.regs.eax)); ++ i += 2; ++ goto out; ++ case 0xe5: /* inw xx / ind xx */ ++ if (data32) { ++ asm volatile ( ++ "inl %w1, %0" ++ : "=a" (vm86.regs.eax) ++ : "d" (instr[i+1]), ++ "0" (vm86.regs.eax)); ++ } else { ++ asm volatile ( ++ "inw %w1, %w0" ++ : "=a" (vm86.regs.eax) ++ : "d" (instr[i+1]), ++ "0" (vm86.regs.eax)); ++ } ++ i += 2; ++ goto out; ++ ++ case 0xec: /* inb dx */ ++ asm volatile ( ++ "inb %w1, %b0" ++ : "=a" (vm86.regs.eax) ++ : "d" (vm86.regs.edx), "0" (vm86.regs.eax)); ++ i++; ++ goto out; ++ case 0xed: /* inw dx / ind dx */ ++ if (data32) { ++ asm volatile ( ++ "inl %w1, %0" ++ : "=a" (vm86.regs.eax) ++ : "d" (vm86.regs.edx)); ++ } else { ++ asm volatile ( ++ "inw %w1, %w0" ++ : "=a" (vm86.regs.eax) ++ : "d" (vm86.regs.edx)); ++ } ++ i++; ++ goto out; ++ case 0xe6: /* outb xx */ ++ asm volatile ( ++ "outb %b0, %w1" ++ : /* no return value */ ++ : "a" (vm86.regs.eax), "d" (instr[i+1])); ++ i += 2; ++ goto out; ++ case 0xe7: /* outw xx / outd xx */ ++ if (data32) { ++ asm volatile ( ++ "outl %0, %w1" ++ : /* no return value */ ++ : "a" (vm86.regs.eax), ++ "d" (instr[i+1])); ++ } else { ++ asm volatile ( ++ "outw %w0, %w1" ++ : /* no return value */ ++ : "a" (vm86.regs.eax), ++ "d" (instr[i+1])); ++ } ++ i += 2; ++ goto out; ++ case 0xee: /* outb dx */ ++ asm volatile ( ++ "outb %b0, %w1" ++ : /* no return value */ ++ : "a" (vm86.regs.eax), "d" (vm86.regs.edx)); ++ i++; ++ goto out; ++ case 0xef: /* outw dx / outd dx */ ++ if (data32) { ++ asm volatile ( ++ "outl %0, %w1" ++ : /* no return value */ ++ : "a" (vm86.regs.eax), ++ "d" (vm86.regs.edx)); ++ } else { ++ asm volatile ( ++ "outw %w0, %w1" ++ : /* no return value */ ++ : "a" (vm86.regs.eax), ++ "d" (vm86.regs.edx)); ++ } ++ i++; ++ goto out; ++ default: ++ printk(KERN_ERR "vesafb: BUG, opcode 0x%x emulation " ++ "not supported (EIP: 0x%lx)\n", ++ instr[i], (u32)(vm86.regs.cs << 4) + ++ vm86.regs.eip); ++ ret = 1; ++ goto out; ++ } ++ } ++out: vm86.regs.eip += i; ++ return ret; ++} ++ ++void vesafb_do_vm86(struct vm86_regs *regs) ++{ ++ unsigned int ret; ++ u8 *retcode = (void*)RET_CODE_ADDR; ++ ++ memset(&vm86,0,sizeof(vm86)); ++ memcpy(&vm86.regs, regs, sizeof(struct vm86_regs)); ++ ++ /* The return code */ ++ retcode[0] = 0xcd; /* int opcode */ ++ retcode[1] = 0xff; /* int number (255) */ ++ ++ /* We use int 0xff to get back to protected mode */ ++ memset(&vm86.int_revectored, 0, sizeof(vm86.int_revectored)); ++ ((unsigned char *)&vm86.int_revectored)[0xff / 8] |= (1 << (0xff % 8)); ++ ++ /* ++ * We want to call int 0x10, so we set: ++ * CS = 0x42 = 0x10 * 4 + 2 ++ * IP = 0x40 = 0x10 * 4 ++ * and SS:ESP. It's up to the caller to set the rest of the registers. ++ */ ++ vm86.regs.eflags = DEFAULT_VM86_FLAGS; ++ vm86.regs.cs = *(unsigned short *)0x42; ++ vm86.regs.eip = *(unsigned short *)0x40; ++ vm86.regs.ss = (STACK_ADDR >> 4); ++ vm86.regs.esp = ((STACK_ADDR & 0x0000f) + STACK_SIZE); ++ ++ /* These will be fetched off the stack when we come to an iret in the ++ * int's 0x10 code. */ ++ VM86_PUSHW(DEFAULT_VM86_FLAGS); ++ VM86_PUSHW((RET_CODE_ADDR >> 4)); /* return code segment */ ++ VM86_PUSHW((RET_CODE_ADDR & 0x0000f)); /* return code offset */ ++ ++ while(1) { ++ ret = vm86old(&vm86); ++ ++ if (VM86_TYPE(ret) == VM86_INTx) { ++ int vint = VM86_ARG(ret); ++ ++ /* If exit from vm86 was caused by int 0xff, then ++ * we're done.. */ ++ if (vint == 0xff) ++ goto out; ++ ++ /* .. otherwise, we have to call the int handler ++ * manually */ ++ VM86_PUSHW(vm86.regs.eflags); ++ VM86_PUSHW(vm86.regs.cs); ++ VM86_PUSHW(vm86.regs.eip); ++ ++ vm86.regs.cs = *(u16 *)((vint << 2) + 2); ++ vm86.regs.eip = *(u16 *)(vint << 2); ++ vm86.regs.eflags &= ~(VIF_MASK | TF_MASK); ++ } else if (VM86_TYPE(ret) == VM86_UNKNOWN) { ++ if (vm86_do_unknown()) ++ goto out; ++ } else { ++ printk(KERN_ERR "vesafb: BUG, returned from " ++ "vm86 with %x (EIP: 0x%lx)\n", ++ ret, (u32)(vm86.regs.cs << 4) + ++ vm86.regs.eip); ++ goto out; ++ } ++ } ++ ++out: /* copy the registers' state back to the caller's struct */ ++ memcpy(regs, &vm86.regs, sizeof(struct vm86_regs)); ++} ++ ++static int vesafb_remap_pfn_range(unsigned long start, unsigned long end, ++ unsigned long pgoff, unsigned long prot, ++ int type) ++{ ++ struct vm_area_struct *vma; ++ struct mm_struct *mm = current->mm; ++ int ret = 0; ++ ++ vma = kmem_cache_alloc(vm_area_cachep, GFP_KERNEL); ++ if (!vma) ++ return -ENOMEM; ++ memset(vma, 0, sizeof(*vma)); ++ down_write(&mm->mmap_sem); ++ vma->vm_mm = mm; ++ vma->vm_start = start; ++ vma->vm_end = end; ++ vma->vm_flags = VM_READ | VM_WRITE | VM_EXEC; ++ vma->vm_flags |= mm->def_flags; ++ vma->vm_page_prot.pgprot = prot; ++ vma->vm_pgoff = pgoff; ++ ++ if ((ret = insert_vm_struct(mm, vma))) { ++ up_write(&mm->mmap_sem); ++ kmem_cache_free(vm_area_cachep, vma); ++ return ret; ++ } ++ ++ if (type) { ++ ret = zeromap_page_range(vma, ++ vma->vm_start, ++ vma->vm_end - vma->vm_start, ++ vma->vm_page_prot); ++ } else { ++ vma->vm_flags |= VM_SHARED; ++ ret = remap_pfn_range(vma, ++ vma->vm_start, ++ vma->vm_pgoff, ++ vma->vm_end - vma->vm_start, ++ vma->vm_page_prot); ++ } ++ up_write(&mm->mmap_sem); ++ return ret; ++} ++ ++static inline int vesafb_init_mem(void) ++{ ++ int ret = 0; ++ ++ /* The memory chunks we're remapping here should be multiples ++ * of PAGE_SIZE. */ ++ ret += vesafb_remap_pfn_range(0x00000, IVTBDA_SIZE, 0, ++ PROT_READ | PROT_EXEC | PROT_WRITE, 0); ++ ret += vesafb_remap_pfn_range(IVTBDA_SIZE, REAL_MEM_SIZE, 0, ++ PROT_READ | PROT_EXEC | PROT_WRITE, 1); ++ ret += vesafb_remap_pfn_range(0x9f000, 0x100000, ++ 0x9f000 >> PAGE_SHIFT, ++ PROT_READ | PROT_EXEC | PROT_WRITE, 0); ++ if (ret) ++ printk(KERN_ERR "vesafb thread: memory remapping failed\n"); ++ ++ return ret; ++} ++ ++#define vesafb_get_string(str) \ ++{ \ ++ /* The address is in the form ssssoooo, where oooo = offset, \ ++ * ssss = segment */ \ ++ addr = ((p_vbe(tsk->buf)->str & 0xffff0000) >> 12) + \ ++ (p_vbe(tsk->buf)->str & 0x0000ffff); \ ++ \ ++ /* The data is in ROM which is shared between processes, so we \ ++ * just translate the real mode address into one visible from \ ++ * kernel space */ \ ++ if (addr >= 0xa0000) { \ ++ p_vbe(tsk->buf)->str = (u32) __va(addr); \ ++ \ ++ /* The data is in the buffer, we just have to convert the \ ++ * address so that it points into the buffer user provided. */ \ ++ } else if (addr > BUF_ADDR && addr < BUF_ADDR + \ ++ sizeof(struct vesafb_vbe_ib)) { \ ++ addr -= BUF_ADDR; \ ++ p_vbe(tsk->buf)->str = (u32) (tsk->buf + addr); \ ++ \ ++ /* This should never happen: someone was insane enough to put \ ++ * the data somewhere in RAM.. */ \ ++ } else { \ ++ p_vbe(tsk->buf)->str = (u32) ""; \ ++ } \ ++} ++ ++void vesafb_handle_getvbeib(struct vesafb_task *tsk) ++{ ++ int addr, res; ++ ++ tsk->regs.es = (BUF_ADDR >> 4); ++ tsk->regs.edi = (BUF_ADDR & 0x000f); ++ strncpy(p_vbe(BUF_ADDR)->vbe_signature, "VBE2", 4); ++ ++ vesafb_do_vm86(&tsk->regs); ++ memcpy(tsk->buf, (void*)(BUF_ADDR), sizeof(struct vesafb_vbe_ib)); ++ ++ /* The OEM fields were not defined prior to VBE 2.0 */ ++ if (p_vbe(tsk->buf)->vbe_version >= 0x200) { ++ vesafb_get_string(oem_string_ptr); ++ vesafb_get_string(oem_vendor_name_ptr); ++ vesafb_get_string(oem_product_name_ptr); ++ vesafb_get_string(oem_product_rev_ptr); ++ } ++ ++ /* This is basically the same as vesafb_get_string() */ ++ addr = ((p_vbe(tsk->buf)->mode_list_ptr & 0xffff0000) >> 12) + ++ (p_vbe(tsk->buf)->mode_list_ptr & 0x0000ffff); ++ ++ if (addr >= 0xa0000) { ++ p_vbe(tsk->buf)->mode_list_ptr = (u32) __va(addr); ++ } else if (addr > BUF_ADDR && addr < BUF_ADDR + ++ sizeof(struct vesafb_vbe_ib)) { ++ addr -= BUF_ADDR; ++ p_vbe(tsk->buf)->mode_list_ptr = (u32) (tsk->buf + addr); ++ } else { ++ res = 0; ++ printk(KERN_WARNING "vesafb: warning, copying modelist " ++ "from somewhere in RAM!\n"); ++ while (*(u16*)(addr+res) != 0xffff && ++ res < (sizeof(p_vbe(tsk->buf)->reserved) - 2)) { ++ *(u16*) ((u32)&(p_vbe(tsk->buf)->reserved) + res) = ++ *(u16*)(addr+res); ++ res += 2; ++ } ++ *(u16*) ((u32)&(p_vbe(tsk->buf)->reserved) + res) = 0xffff; ++ } ++} ++ ++int vesafb_handle_tasks(void) ++{ ++ struct vesafb_task *tsk; ++ struct list_head *curr, *next; ++ int ret = 0; ++ ++ down(&vesafb_task_list_sem); ++ list_for_each_safe(curr, next, &vesafb_task_list) { ++ tsk = list_entry(curr, struct vesafb_task, node); ++ ++ if (tsk->flags & TF_EXIT) { ++ ret = 1; ++ goto task_done; ++ } ++ if (tsk->flags & TF_GETVBEIB) { ++ vesafb_handle_getvbeib(tsk); ++ goto task_done; ++ } ++ /* Do we need to store a pointer to the buffer in ES:EDI? */ ++ if (tsk->flags & TF_BUF_DI) { ++ tsk->regs.es = (BUF_ADDR >> 4); ++ tsk->regs.edi = (BUF_ADDR & 0x000f); ++ } ++ /* Sometimes the pointer has to be in ES:EBX. */ ++ if (tsk->flags & TF_BUF_BX) { ++ tsk->regs.es = (BUF_ADDR >> 4); ++ tsk->regs.ebx = (BUF_ADDR & 0x000f); ++ } ++ if (tsk->flags & (TF_BUF_DI | TF_BUF_BX)) ++ memcpy((void*)BUF_ADDR, tsk->buf, tsk->buf_len); ++ ++ vesafb_do_vm86(&tsk->regs); ++ ++ if (tsk->flags & TF_RETURN_BUF) ++ memcpy(tsk->buf, (void*)BUF_ADDR, tsk->buf_len); ++ ++task_done: list_del(curr); ++ complete(&tsk->done); ++ } ++ ++ /* If we're going to kill this thread, don't allow any elements ++ * to be added to the task list. */ ++ if (!ret) ++ up(&vesafb_task_list_sem); ++ ++ return ret; ++} ++ ++/* ++ * This 'hybrid' thread serves as a backend for vesafb-tng, handling all vm86 ++ * calls. It is started as a kernel thread. It then creates its own mm struct, ++ * thus separating itself from any userspace processes. At this moment, it ++ * stops being a kernel thread (kernel threads have mm = NULL) and becomes ++ * a 'hybrid' thread -- one that has full access to kernel space, yet runs ++ * with its own address space. ++ * ++ * This is necessary because in order to make vm86 calls some parts of the ++ * first 1MB of RAM have to be setup to mimic the real mode. These are: ++ * - interrupt vector table [0x00000-0x003ff] ++ * - BIOS data area [0x00400-0x004ff] ++ * - Extended BIOS data area [0x9fc00-0x9ffff] ++ * - the video RAM [0xa0000-0xbffff] ++ * - video BIOS [0xc0000-0xcffff] ++ * - motherboard BIOS [0xf0000-0xfffff] ++ */ ++int vesafb_thread(void *unused) ++{ ++ int err = 0; ++ ++ set_fs(KERNEL_DS); ++ daemonize("vesafb"); ++ ++ if (set_new_mm()) { ++ err = -ENOMEM; ++ goto thr_end; ++ } ++ if (vesafb_init_mem()) { ++ err = -ENOMEM; ++ goto thr_end; ++ } ++ ++ DPRINTK("started vesafb thread\n"); ++ ++ /* Having an IO bitmap makes things faster as we avoid GPFs ++ * when running vm86 code. We can live if it fails, though, ++ * so don't bother checking for errors. */ ++ ioperm(0,1024,1); ++ set_user_nice(current, -10); ++ ++ complete(&vesafb_th_completion); ++ ++ while (1) { ++ if (vesafb_handle_tasks()) ++ break; ++ wait_event_interruptible(vesafb_wait, ++ !list_empty(&vesafb_task_list)); ++ try_to_freeze(); ++ } ++ ++out: DPRINTK("exiting the vesafb thread\n"); ++ vesafb_pid = -1; ++ ++ /* Now that all callers know this thread is no longer running ++ * (pid < 0), allow them to continue. */ ++ up(&vesafb_task_list_sem); ++ return err; ++thr_end: ++ down(&vesafb_task_list_sem); ++ complete(&vesafb_th_completion); ++ goto out; ++} ++ ++int vesafb_queue_task(struct vesafb_task *tsk) ++{ ++ down(&vesafb_task_list_sem); ++ if (vesafb_pid < 0) ++ return -1; ++ list_add_tail(&tsk->node, &vesafb_task_list); ++ up(&vesafb_task_list_sem); ++ wake_up(&vesafb_wait); ++ return 0; ++} ++ ++int vesafb_wait_for_thread(void) ++{ ++ /* PID 0 means that the thread is still initializing. */ ++ if (vesafb_pid < 0) ++ return -1; ++ wait_for_completion(&vesafb_th_completion); ++ return 0; ++} ++ ++int __init vesafb_init_thread(void) ++{ ++ vesafb_pid = kernel_thread(vesafb_thread,NULL,0); ++ return 0; ++} ++ ++#ifdef MODULE ++void __exit vesafb_kill_thread(void) ++{ ++ struct vesafb_task *tsk; ++ if (vesafb_pid <= 0) ++ return; ++ ++ vesafb_create_task(tsk); ++ if (!tsk) ++ return; ++ tsk->flags |= TF_EXIT; ++ vesafb_queue_task(tsk); ++ vesafb_wait_for_task(tsk); ++ kfree(tsk); ++ return; ++} ++module_exit(vesafb_kill_thread); ++#endif ++module_init(vesafb_init_thread); ++ ++EXPORT_SYMBOL_GPL(vesafb_queue_task); ++EXPORT_SYMBOL_GPL(vesafb_wait_for_thread); ++ ++MODULE_LICENSE("GPL"); ++MODULE_AUTHOR("Michal Januszewski"); ++ +diff --git a/drivers/video/vesafb-tng.c b/drivers/video/vesafb-tng.c +new file mode 100644 +index 0000000..b4d4394 +--- /dev/null ++++ b/drivers/video/vesafb-tng.c +@@ -0,0 +1,1586 @@ ++/* ++ * Framebuffer driver for VBE 2.0+ compliant graphic boards ++ * ++ * (c) 2004-2006 Michal Januszewski <spock@gentoo.org> ++ * Based upon vesafb code by Gerd Knorr <kraxel@goldbach.in-berlin.de> ++ * ++ */ ++ ++#include <linux/module.h> ++#include <linux/kernel.h> ++#include <linux/errno.h> ++#include <linux/string.h> ++#include <linux/mm.h> ++#include <linux/tty.h> ++#include <linux/delay.h> ++#include <linux/fb.h> ++#include <linux/ioport.h> ++#include <linux/init.h> ++#include <linux/proc_fs.h> ++#include <linux/completion.h> ++#include <linux/platform_device.h> ++#include <video/edid.h> ++#include <video/vesa.h> ++#include <video/vga.h> ++#include <asm/io.h> ++#include <asm/mtrr.h> ++#include <asm/page.h> ++#include <asm/pgtable.h> ++#include "edid.h" ++ ++#define dac_reg (0x3c8) ++#define dac_val (0x3c9) ++ ++#define VESAFB_NEED_EXACT_RES 1 ++#define VESAFB_NEED_EXACT_DEPTH 2 ++ ++/* --------------------------------------------------------------------- */ ++ ++static struct fb_var_screeninfo vesafb_defined __initdata = { ++ .activate = FB_ACTIVATE_NOW, ++ .height = 0, ++ .width = 0, ++ .right_margin = 32, ++ .upper_margin = 16, ++ .lower_margin = 4, ++ .vsync_len = 4, ++ .vmode = FB_VMODE_NONINTERLACED, ++}; ++ ++static struct fb_fix_screeninfo vesafb_fix __initdata = { ++ .id = "VESA VGA", ++ .type = FB_TYPE_PACKED_PIXELS, ++ .accel = FB_ACCEL_NONE, ++}; ++ ++static int mtrr = 0; /* disable mtrr by default */ ++static int blank = 1; /* enable blanking by default */ ++static int ypan = 0; /* 0 - nothing, 1 - ypan, 2 - ywrap */ ++static int pmi_setpal = 1; /* pmi for palette changes */ ++static u16 *pmi_base = NULL; /* protected mode interface location */ ++static void (*pmi_start)(void) = NULL; ++static void (*pmi_pal)(void) = NULL; ++static struct vesafb_vbe_ib vbe_ib; ++static struct vesafb_mode_ib *vbe_modes; ++static int vbe_modes_cnt = 0; ++static struct fb_info *vesafb_info = NULL; ++static int nocrtc = 0; /* ignore CRTC settings */ ++static int noedid __initdata = 0; /* don't try DDC transfers */ ++static int vram_remap __initdata = 0; /* set amount of memory to be used */ ++static int vram_total __initdata = 0; /* set total amount of memory */ ++static u16 maxclk __initdata = 0; /* maximum pixel clock */ ++static u16 maxvf __initdata = 0; /* maximum vertical frequency */ ++static u16 maxhf __initdata = 0; /* maximum horizontal frequency */ ++static int gtf __initdata = 0; /* forces use of the GTF */ ++static char *mode_option __initdata = NULL; ++static u16 vbemode __initdata = 0; ++ ++/* --------------------------------------------------------------------- */ ++ ++static int vesafb_find_vbe_mode(int xres, int yres, int depth, ++ unsigned char flags) ++{ ++ int i, match = -1, h = 0, d = 0x7fffffff; ++ ++ for (i = 0; i < vbe_modes_cnt; i++) { ++ h = abs(vbe_modes[i].x_res - xres) + ++ abs(vbe_modes[i].y_res - yres) + ++ abs(depth - vbe_modes[i].depth); ++ if (h == 0) ++ return i; ++ if (h < d || (h == d && vbe_modes[i].depth > depth)) { ++ d = h; ++ match = i; ++ } ++ } ++ i = 1; ++ ++ if (flags & VESAFB_NEED_EXACT_DEPTH && vbe_modes[match].depth != depth) ++ i = 0; ++ if (flags & VESAFB_NEED_EXACT_RES && d > 24) ++ i = 0; ++ if (i != 0) ++ return match; ++ else ++ return -1; ++} ++ ++static int vesafb_pan_display(struct fb_var_screeninfo *var, ++ struct fb_info *info) ++{ ++ int offset; ++ ++ offset = (var->yoffset * info->fix.line_length + var->xoffset) / 4; ++ ++ /* It turns out it's not the best idea to do panning via vm86, ++ * so we only allow it if we have a PMI. */ ++ if (pmi_start) { ++ __asm__ __volatile__( ++ "call *(%%edi)" ++ : /* no return value */ ++ : "a" (0x4f07), /* EAX */ ++ "b" (0), /* EBX */ ++ "c" (offset), /* ECX */ ++ "d" (offset >> 16), /* EDX */ ++ "D" (&pmi_start)); /* EDI */ ++ } ++ return 0; ++} ++ ++static int vesafb_blank(int blank, struct fb_info *info) ++{ ++ struct vesafb_task *tsk; ++ int err = 1; ++ ++ if (vbe_ib.capabilities & VBE_CAP_VGACOMPAT) { ++ int loop = 10000; ++ u8 seq = 0, crtc17 = 0; ++ ++ if (blank == FB_BLANK_POWERDOWN) { ++ seq = 0x20; ++ crtc17 = 0x00; ++ err = 0; ++ } else { ++ seq = 0x00; ++ crtc17 = 0x80; ++ err = (blank == FB_BLANK_UNBLANK) ? 0 : -EINVAL; ++ } ++ ++ vga_wseq(NULL, 0x00, 0x01); ++ seq |= vga_rseq(NULL, 0x01) & ~0x20; ++ vga_wseq(NULL, 0x00, seq); ++ ++ crtc17 |= vga_rcrt(NULL, 0x17) & ~0x80; ++ while (loop--); ++ vga_wcrt(NULL, 0x17, crtc17); ++ vga_wseq(NULL, 0x00, 0x03); ++ } else { ++ vesafb_create_task (tsk); ++ if (!tsk) ++ return -ENOMEM; ++ tsk->regs.eax = 0x4f10; ++ switch (blank) { ++ case FB_BLANK_UNBLANK: ++ tsk->regs.ebx = 0x0001; ++ break; ++ case FB_BLANK_NORMAL: ++ tsk->regs.ebx = 0x0101; /* standby */ ++ break; ++ case FB_BLANK_POWERDOWN: ++ tsk->regs.ebx = 0x0401; /* powerdown */ ++ break; ++ default: ++ goto out; ++ } ++ tsk->flags = TF_CALL; ++ if (!vesafb_queue_task (tsk)) ++ vesafb_wait_for_task(tsk); ++ ++ if ((tsk->regs.eax & 0xffff) == 0x004f) ++ err = 0; ++out: kfree(tsk); ++ } ++ return err; ++} ++ ++static int vesafb_setpalette(struct vesafb_pal_entry *entries, int count, ++ int start, struct fb_info *info) ++{ ++ struct vesafb_task *tsk; ++ int i = ((struct vesafb_par*)info->par)->mode_idx; ++ int ret = 0; ++ ++ /* We support palette modifications for 8 bpp modes only, so ++ * there can never be more than 256 entries. */ ++ if (start + count > 256) ++ return -EINVAL; ++ ++ /* Use VGA registers if mode is VGA-compatible. */ ++ if (i >= 0 && i < vbe_modes_cnt && ++ vbe_modes[i].mode_attr & VBE_MODE_VGACOMPAT) { ++ for (i = 0; i < count; i++) { ++ outb_p(start + i, dac_reg); ++ outb_p(entries[i].red, dac_val); ++ outb_p(entries[i].green, dac_val); ++ outb_p(entries[i].blue, dac_val); ++ } ++ } else if (pmi_setpal) { ++ __asm__ __volatile__( ++ "call *(%%esi)" ++ : /* no return value */ ++ : "a" (0x4f09), /* EAX */ ++ "b" (0), /* EBX */ ++ "c" (count), /* ECX */ ++ "d" (start), /* EDX */ ++ "D" (entries), /* EDI */ ++ "S" (&pmi_pal)); /* ESI */ ++ } else { ++ vesafb_create_task (tsk); ++ if (!tsk) ++ return -ENOMEM; ++ tsk->regs.eax = 0x4f09; ++ tsk->regs.ebx = 0x0; ++ tsk->regs.ecx = count; ++ tsk->regs.edx = start; ++ tsk->buf = entries; ++ tsk->buf_len = sizeof(struct vesafb_pal_entry) * count; ++ tsk->flags = TF_CALL | TF_BUF_DI; ++ ++ if (!vesafb_queue_task (tsk)) ++ vesafb_wait_for_task(tsk); ++ if ((tsk->regs.eax & 0xffff) != 0x004f) ++ ret = 1; ++ kfree(tsk); ++ } ++ return ret; ++} ++ ++static int vesafb_setcolreg(unsigned regno, unsigned red, unsigned green, ++ unsigned blue, unsigned transp, ++ struct fb_info *info) ++{ ++ struct vesafb_pal_entry entry; ++ int shift = 16 - info->var.green.length; ++ int ret = 0; ++ ++ if (regno >= info->cmap.len) ++ return -EINVAL; ++ ++ if (info->var.bits_per_pixel == 8) { ++ entry.red = red >> shift; ++ entry.green = green >> shift; ++ entry.blue = blue >> shift; ++ entry.pad = 0; ++ ++ ret = vesafb_setpalette(&entry, 1, regno, info); ++ } else if (regno < 16) { ++ switch (info->var.bits_per_pixel) { ++ case 16: ++ if (info->var.red.offset == 10) { ++ /* 1:5:5:5 */ ++ ((u32*) (info->pseudo_palette))[regno] = ++ ((red & 0xf800) >> 1) | ++ ((green & 0xf800) >> 6) | ++ ((blue & 0xf800) >> 11); ++ } else { ++ /* 0:5:6:5 */ ++ ((u32*) (info->pseudo_palette))[regno] = ++ ((red & 0xf800) ) | ++ ((green & 0xfc00) >> 5) | ++ ((blue & 0xf800) >> 11); ++ } ++ break; ++ ++ case 24: ++ case 32: ++ red >>= 8; ++ green >>= 8; ++ blue >>= 8; ++ ((u32 *)(info->pseudo_palette))[regno] = ++ (red << info->var.red.offset) | ++ (green << info->var.green.offset) | ++ (blue << info->var.blue.offset); ++ break; ++ } ++ } ++ return ret; ++} ++ ++static int vesafb_setcmap(struct fb_cmap *cmap, struct fb_info *info) ++{ ++ struct vesafb_pal_entry *entries; ++ int shift = 16 - info->var.green.length; ++ int i, ret = 0; ++ ++ if (info->var.bits_per_pixel == 8) { ++ if (cmap->start + cmap->len > info->cmap.start + ++ info->cmap.len || cmap->start < info->cmap.start) ++ return -EINVAL; ++ ++ entries = vmalloc(sizeof(struct vesafb_pal_entry) * cmap->len); ++ if (!entries) ++ return -ENOMEM; ++ for (i = 0; i < cmap->len; i++) { ++ entries[i].red = cmap->red[i] >> shift; ++ entries[i].green = cmap->green[i] >> shift; ++ entries[i].blue = cmap->blue[i] >> shift; ++ entries[i].pad = 0; ++ } ++ ret = vesafb_setpalette(entries, cmap->len, cmap->start, info); ++ vfree(entries); ++ } else { ++ /* For modes with bpp > 8, we only set the pseudo palette in ++ * the fb_info struct. We rely on vesafb_setcolreg to do all ++ * sanity checking. */ ++ for (i = 0; i < cmap->len; i++) { ++ ret += vesafb_setcolreg(cmap->start + i, cmap->red[i], ++ cmap->green[i], cmap->blue[i], ++ 0, info); ++ } ++ } ++ return ret; ++} ++ ++static int vesafb_set_par(struct fb_info *info) ++{ ++ struct vesafb_par *par = (struct vesafb_par *) info->par; ++ struct vesafb_task *tsk; ++ struct vesafb_crtc_ib *crtc = NULL; ++ struct vesafb_mode_ib *mode = NULL; ++ int i, err = 0, depth = info->var.bits_per_pixel; ++ ++ if (depth > 8 && depth != 32) ++ depth = info->var.red.length + info->var.green.length + ++ info->var.blue.length; ++ ++ i = vesafb_find_vbe_mode(info->var.xres, info->var.yres, depth, ++ VESAFB_NEED_EXACT_RES | ++ VESAFB_NEED_EXACT_DEPTH); ++ if (i >= 0) ++ mode = &vbe_modes[i]; ++ else ++ return -EINVAL; ++ ++ vesafb_create_task (tsk); ++ if (!tsk) ++ return -ENOMEM; ++ tsk->regs.eax = 0x4f02; ++ tsk->regs.ebx = mode->mode_id | 0x4000; /* use LFB */ ++ tsk->flags = TF_CALL; ++ ++ if (vbe_ib.vbe_version >= 0x0300 && !nocrtc && ++ info->var.pixclock != 0) { ++ tsk->regs.ebx |= 0x0800; /* use CRTC data */ ++ tsk->flags |= TF_BUF_DI; ++ crtc = kmalloc(sizeof(struct vesafb_crtc_ib), GFP_KERNEL); ++ if (!crtc) { ++ err = -ENOMEM; ++ goto out; ++ } ++ crtc->horiz_start = info->var.xres + info->var.right_margin; ++ crtc->horiz_end = crtc->horiz_start + info->var.hsync_len; ++ crtc->horiz_total = crtc->horiz_end + info->var.left_margin; ++ ++ crtc->vert_start = info->var.yres + info->var.lower_margin; ++ crtc->vert_end = crtc->vert_start + info->var.vsync_len; ++ crtc->vert_total = crtc->vert_end + info->var.upper_margin; ++ ++ crtc->pixel_clock = PICOS2KHZ(info->var.pixclock) * 1000; ++ crtc->refresh_rate = (u16)(100 * (crtc->pixel_clock / ++ (crtc->vert_total * crtc->horiz_total))); ++ crtc->flags = 0; ++ ++ if (info->var.vmode & FB_VMODE_DOUBLE) ++ crtc->flags |= 0x1; ++ if (info->var.vmode & FB_VMODE_INTERLACED) ++ crtc->flags |= 0x2; ++ if (!(info->var.sync & FB_SYNC_HOR_HIGH_ACT)) ++ crtc->flags |= 0x4; ++ if (!(info->var.sync & FB_SYNC_VERT_HIGH_ACT)) ++ crtc->flags |= 0x8; ++ memcpy(&par->crtc, crtc, sizeof(struct vesafb_crtc_ib)); ++ } else ++ memset(&par->crtc, 0, sizeof(struct vesafb_crtc_ib)); ++ ++ tsk->buf = (void*)crtc; ++ tsk->buf_len = sizeof(struct vesafb_crtc_ib); ++ ++ if (vesafb_queue_task (tsk)) { ++ err = -EINVAL; ++ goto out; ++ } ++ vesafb_wait_for_task(tsk); ++ ++ if ((tsk->regs.eax & 0xffff) != 0x004f) { ++ printk(KERN_ERR "vesafb: mode switch failed (eax: 0x%lx)\n", ++ tsk->regs.eax); ++ err = -EINVAL; ++ goto out; ++ } ++ par->mode_idx = i; ++ ++ /* For 8bpp modes, always try to set the DAC to 8 bits. */ ++ if (vbe_ib.capabilities & VBE_CAP_CAN_SWITCH_DAC && ++ mode->bits_per_pixel <= 8) { ++ vesafb_reset_task(tsk); ++ tsk->flags = TF_CALL; ++ tsk->regs.eax = 0x4f08; ++ tsk->regs.ebx = 0x0800; ++ ++ if (!vesafb_queue_task (tsk)) ++ vesafb_wait_for_task(tsk); ++ ++ if ((tsk->regs.eax & 0xffff) != 0x004f || ++ ((tsk->regs.ebx & 0xff00) >> 8) != 8) { ++ /* We've failed to set the DAC palette format - ++ * time to correct var. */ ++ info->var.red.length = 6; ++ info->var.green.length = 6; ++ info->var.blue.length = 6; ++ } ++ } ++ ++ info->fix.visual = (info->var.bits_per_pixel == 8) ? ++ FB_VISUAL_PSEUDOCOLOR : FB_VISUAL_TRUECOLOR; ++ info->fix.line_length = mode->bytes_per_scan_line; ++ ++ DPRINTK("set new mode %dx%d-%d (0x%x)\n", ++ info->var.xres, info->var.yres, info->var.bits_per_pixel, ++ mode->mode_id); ++ ++out: if (crtc != NULL) ++ kfree(crtc); ++ kfree(tsk); ++ ++ return err; ++} ++ ++static void vesafb_setup_var(struct fb_var_screeninfo *var, struct fb_info *info, ++ struct vesafb_mode_ib *mode) ++{ ++ var->xres = mode->x_res; ++ var->yres = mode->y_res; ++ var->xres_virtual = mode->x_res; ++ var->yres_virtual = (ypan) ? ++ info->fix.smem_len / mode->bytes_per_scan_line : ++ mode->y_res; ++ var->xoffset = 0; ++ var->yoffset = 0; ++ var->bits_per_pixel = mode->bits_per_pixel; ++ ++ if (var->bits_per_pixel == 15) ++ var->bits_per_pixel = 16; ++ ++ if (var->bits_per_pixel > 8) { ++ var->red.offset = mode->red_off; ++ var->red.length = mode->red_len; ++ var->green.offset = mode->green_off; ++ var->green.length = mode->green_len; ++ var->blue.offset = mode->blue_off; ++ var->blue.length = mode->blue_len; ++ var->transp.offset = mode->rsvd_off; ++ var->transp.length = mode->rsvd_len; ++ ++ DPRINTK("directcolor: size=%d:%d:%d:%d, shift=%d:%d:%d:%d\n", ++ mode->rsvd_len, ++ mode->red_len, ++ mode->green_len, ++ mode->blue_len, ++ mode->rsvd_off, ++ mode->red_off, ++ mode->green_off, ++ mode->blue_off); ++ } else { ++ var->red.offset = 0; ++ var->green.offset = 0; ++ var->blue.offset = 0; ++ var->transp.offset = 0; ++ ++ /* We're assuming that we can switch the DAC to 8 bits. If ++ * this proves to be incorrect, we'll update the fields ++ * later in set_par(). */ ++ if (vbe_ib.capabilities & VBE_CAP_CAN_SWITCH_DAC) { ++ var->red.length = 8; ++ var->green.length = 8; ++ var->blue.length = 8; ++ var->transp.length = 0; ++ } else { ++ var->red.length = 6; ++ var->green.length = 6; ++ var->blue.length = 6; ++ var->transp.length = 0; ++ } ++ } ++} ++ ++static void inline vesafb_check_limits(struct fb_var_screeninfo *var, ++ struct fb_info *info) ++{ ++ struct fb_videomode *mode; ++ ++ if (!var->pixclock) ++ return; ++ if (vbe_ib.vbe_version < 0x0300) { ++ fb_get_mode(FB_VSYNCTIMINGS | FB_IGNOREMON, 60, var, info); ++ return; ++ } ++ if (!fb_validate_mode(var, info)) ++ return; ++ mode = fb_find_best_mode(var, &info->modelist); ++ if (mode) { ++ DPRINTK("find_best_mode: %d %d @ %d (vmode: %d)\n", ++ mode->xres, mode->yres, mode->refresh, mode->vmode); ++ if (mode->xres == var->xres && mode->yres == var->yres && ++ !(mode->vmode & (FB_VMODE_INTERLACED | FB_VMODE_DOUBLE))) { ++ fb_videomode_to_var(var, mode); ++ return; ++ } ++ } ++ if (info->monspecs.gtf && !fb_get_mode(FB_MAXTIMINGS, 0, var, info)) ++ return; ++ /* Use default refresh rate */ ++ var->pixclock = 0; ++} ++ ++static int vesafb_check_var(struct fb_var_screeninfo *var, ++ struct fb_info *info) ++{ ++ int match = -1; ++ int depth = var->red.length + var->green.length + var->blue.length; ++ ++ /* Various apps will use bits_per_pixel to set the color depth, ++ * which is theoretically incorrect, but which we'll try to handle ++ * here. */ ++ if (depth == 0 || abs(depth - var->bits_per_pixel) >= 8) ++ depth = var->bits_per_pixel; ++ match = vesafb_find_vbe_mode(var->xres, var->yres, depth, ++ VESAFB_NEED_EXACT_RES); ++ ++ if (match == -1) { ++ DPRINTK("vesafb: mode %dx%d-%d not found\n", var->xres, ++ var->yres, depth); ++ return -EINVAL; ++ } ++ ++ vesafb_setup_var(var, info, &vbe_modes[match]); ++ DPRINTK("found mode 0x%x (%dx%d-%dbpp)\n", ++ vbe_modes[match].mode_id, vbe_modes[match].x_res, ++ vbe_modes[match].y_res, vbe_modes[match].depth); ++ ++ /* Check whether we have remapped enough memory for this mode. */ ++ if (var->yres * vbe_modes[match].bytes_per_scan_line > ++ info->fix.smem_len) { ++ return -EINVAL; ++ } ++ ++ if ((var->vmode & FB_VMODE_DOUBLE) && ++ !(vbe_modes[match].mode_attr & 0x100)) ++ var->vmode &= ~FB_VMODE_DOUBLE; ++ if ((var->vmode & FB_VMODE_INTERLACED) && ++ !(vbe_modes[match].mode_attr & 0x200)) ++ var->vmode &= ~FB_VMODE_INTERLACED; ++ vesafb_check_limits(var, info); ++ return 0; ++} ++ ++static int vesafb_open(struct fb_info *info, int user) ++{ ++ struct vesafb_task *tsk = NULL; ++ struct vesafb_par *par = info->par; ++ int cnt = atomic_read(&par->ref_count); ++ ++ if (!cnt) { ++ vesafb_create_task(tsk); ++ if (!tsk) ++ goto out; ++ ++ /* Get the VBE state buffer size. We want all available ++ * hardware state data (CL = 0x0f). */ ++ tsk->regs.eax = 0x4f04; ++ tsk->regs.ecx = 0x000f; ++ tsk->regs.edx = 0x0000; ++ tsk->flags = TF_CALL; ++ ++ if (vesafb_queue_task(tsk)) ++ goto out; ++ ++ vesafb_wait_for_task(tsk); ++ ++ if ((tsk->regs.eax & 0xffff) != 0x004f) { ++ printk(KERN_WARNING "vesafb: VBE state buffer size " ++ "cannot be determined (eax: 0x%lx)\n", ++ tsk->regs.eax); ++ goto out; ++ } ++ ++ par->vbe_state_size = 64 * (tsk->regs.ebx & 0xffff); ++ par->vbe_state = kzalloc(par->vbe_state_size, GFP_KERNEL); ++ if (!par->vbe_state) ++ goto out; ++ ++ vesafb_reset_task(tsk); ++ tsk->regs.eax = 0x4f04; ++ tsk->regs.ecx = 0x000f; ++ tsk->regs.edx = 0x0001; ++ tsk->flags = TF_CALL | TF_BUF_BX | TF_RETURN_BUF; ++ tsk->buf = (void*)(par->vbe_state); ++ tsk->buf_len = par->vbe_state_size; ++ ++ if (vesafb_queue_task(tsk)) ++ goto getstate_failed; ++ vesafb_wait_for_task(tsk); ++ ++ if ((tsk->regs.eax & 0xffff) != 0x004f) { ++ printk(KERN_WARNING "vesafb: VBE get state call " ++ "failed (eax: 0x%lx)\n", tsk->regs.eax); ++ goto getstate_failed; ++ } ++ } ++out: ++ atomic_inc(&par->ref_count); ++ if (tsk) ++ kfree(tsk); ++ return 0; ++ ++getstate_failed: ++ kfree(par->vbe_state); ++ par->vbe_state = NULL; ++ par->vbe_state_size = 0; ++ goto out; ++} ++ ++static int vesafb_release(struct fb_info *info, int user) ++{ ++ struct vesafb_task *tsk = NULL; ++ struct vesafb_par *par = info->par; ++ int cnt = atomic_read(&par->ref_count); ++ ++ if (!cnt) ++ return -EINVAL; ++ ++ if (cnt == 1 && par->vbe_state && par->vbe_state_size) { ++ vesafb_create_task(tsk); ++ if (!tsk) ++ goto out; ++ ++ tsk->regs.eax = 0x0003; ++ tsk->regs.ebx = 0x0000; ++ tsk->flags = TF_CALL; ++ ++ if (vesafb_queue_task(tsk)) ++ goto out; ++ ++ vesafb_wait_for_task(tsk); ++ ++ vesafb_reset_task(tsk); ++ tsk->regs.eax = 0x4f04; ++ tsk->regs.ecx = 0x000f; ++ tsk->regs.edx = 0x0002; ++ tsk->buf = (void*)(par->vbe_state); ++ tsk->buf_len = par->vbe_state_size; ++ tsk->flags = TF_CALL | TF_BUF_BX; ++ ++ if (vesafb_queue_task(tsk)) ++ goto out; ++ ++ vesafb_wait_for_task(tsk); ++ ++ if ((tsk->regs.eax & 0xffff) != 0x004f) ++ printk(KERN_WARNING "vesafb: VBE state restore call " ++ "failed (eax: 0x%lx)\n", ++ tsk->regs.eax); ++ } ++out: ++ atomic_dec(&par->ref_count); ++ if (tsk) ++ kfree(tsk); ++ return 0; ++} ++ ++static int __init vesafb_probe(struct platform_device *device); ++ ++static struct fb_ops vesafb_ops = { ++ .owner = THIS_MODULE, ++ .fb_open = vesafb_open, ++ .fb_release = vesafb_release, ++ .fb_setcolreg = vesafb_setcolreg, ++ .fb_setcmap = vesafb_setcmap, ++ .fb_pan_display = vesafb_pan_display, ++ .fb_blank = vesafb_blank, ++ .fb_fillrect = cfb_fillrect, ++ .fb_copyarea = cfb_copyarea, ++ .fb_imageblit = cfb_imageblit, ++ .fb_check_var = vesafb_check_var, ++ .fb_set_par = vesafb_set_par ++}; ++ ++static struct platform_driver vesafb_driver = { ++ .probe = vesafb_probe, ++ .driver = { ++ .name = "vesafb", ++ }, ++}; ++ ++static struct platform_device *vesafb_device; ++ ++#ifndef MODULE ++int __init vesafb_setup(char *options) ++{ ++ char *this_opt; ++ ++ if (!options || !*options) ++ return 0; ++ ++ DPRINTK("options %s\n",options); ++ ++ while ((this_opt = strsep(&options, ",")) != NULL) { ++ if (!*this_opt) continue; ++ ++ DPRINTK("this_opt: %s\n",this_opt); ++ ++ if (! strcmp(this_opt, "redraw")) ++ ypan=0; ++ else if (! strcmp(this_opt, "ypan")) ++ ypan=1; ++ else if (! strcmp(this_opt, "ywrap")) ++ ypan=2; ++ else if (! strcmp(this_opt, "vgapal")) ++ pmi_setpal=0; ++ else if (! strcmp(this_opt, "pmipal")) ++ pmi_setpal=1; ++ else if (! strncmp(this_opt, "mtrr:", 5)) ++ mtrr = simple_strtoul(this_opt+5, NULL, 0); ++ else if (! strcmp(this_opt, "nomtrr")) ++ mtrr=0; ++ else if (! strcmp(this_opt, "nocrtc")) ++ nocrtc=1; ++ else if (! strcmp(this_opt, "noedid")) ++ noedid=1; ++ else if (! strcmp(this_opt, "noblank")) ++ blank=0; ++ else if (! strcmp(this_opt, "gtf")) ++ gtf=1; ++ else if (! strncmp(this_opt, "vtotal:", 7)) ++ vram_total = simple_strtoul(this_opt + 7, NULL, 0); ++ else if (! strncmp(this_opt, "vremap:", 7)) ++ vram_remap = simple_strtoul(this_opt + 7, NULL, 0); ++ else if (! strncmp(this_opt, "maxhf:", 6)) ++ maxhf = simple_strtoul(this_opt + 6, NULL, 0); ++ else if (! strncmp(this_opt, "maxvf:", 6)) ++ maxvf = simple_strtoul(this_opt + 6, NULL, 0); ++ else if (! strncmp(this_opt, "maxclk:", 7)) ++ maxclk = simple_strtoul(this_opt + 7, NULL, 0); ++ else if (! strncmp(this_opt, "vbemode:", 8)) ++ vbemode = simple_strtoul(this_opt + 8, NULL,0); ++ else if (this_opt[0] >= '0' && this_opt[0] <= '9') { ++ DPRINTK("mode_option: %s\n",this_opt); ++ mode_option = this_opt; ++ } else { ++ printk(KERN_WARNING ++ "vesafb: unrecognized option %s\n", this_opt); ++ } ++ } ++ ++ return 0; ++} ++#endif /* !MODULE */ ++ ++static int vesafb_read_proc_modes(char *buf, char **start, off_t offset, ++ int len, int *eof, void *private) ++{ ++ int clen = 0, i; ++ ++ for (i = 0; i < vbe_modes_cnt; i++) { ++ clen += min(snprintf(buf + clen, len - clen, "%dx%d-%d\n", vbe_modes[i].x_res, ++ vbe_modes[i].y_res, vbe_modes[i].depth), len - clen); ++ } ++ *eof = 1; ++ return clen; ++} ++ ++static int vesafb_read_proc_vbe_info(char *buf, char **start, off_t offset, ++ int len, int *eof, void *private) ++{ ++ int clen = 0; ++ ++ clen += min(snprintf(buf + clen, len, "Version: %d.%d\n", ++ ((vbe_ib.vbe_version & 0xff00) >> 8), ++ vbe_ib.vbe_version & 0xff), len); ++ clen += min(snprintf(buf + clen, len - clen, "Vendor: %s\n", ++ (char*)vbe_ib.oem_vendor_name_ptr), len - clen); ++ clen += min(snprintf(buf + clen, len - clen, "Product: %s\n", ++ (char*)vbe_ib.oem_product_name_ptr), len - clen); ++ clen += min(snprintf(buf + clen, len - clen, "OEM rev: %s\n", ++ (char*)vbe_ib.oem_product_rev_ptr), len - clen); ++ clen += min(snprintf(buf + clen, len - clen, "OEM string: %s\n", ++ (char*)vbe_ib.oem_string_ptr), len - clen); ++ ++ *eof = 1; ++ return clen; ++} ++ ++static int __init inline vesafb_vbe_getinfo(struct vesafb_task *tsk) ++{ ++ tsk->regs.eax = 0x4f00; ++ tsk->flags = TF_CALL | TF_GETVBEIB; ++ tsk->buf = &vbe_ib; ++ tsk->buf_len = sizeof(vbe_ib); ++ if (vesafb_queue_task (tsk)) ++ return -EINVAL; ++ vesafb_wait_for_task(tsk); ++ ++ if (vbe_ib.vbe_version < 0x0200) { ++ printk(KERN_ERR "vesafb: Sorry, pre-VBE 2.0 cards are " ++ "not supported.\n"); ++ return -EINVAL; ++ } ++ ++ if ((tsk->regs.eax & 0xffff) != 0x004f) { ++ printk(KERN_ERR "vesafb: Getting mode info block failed " ++ "(eax=0x%x)\n", (u32)tsk->regs.eax); ++ return -EINVAL; ++ } ++ ++ printk(KERN_INFO "vesafb: %s, %s, %s (OEM: %s)\n", ++ (char*)vbe_ib.oem_vendor_name_ptr, ++ (char*)vbe_ib.oem_product_name_ptr, ++ (char*)vbe_ib.oem_product_rev_ptr, ++ (char*)vbe_ib.oem_string_ptr); ++ ++ printk(KERN_INFO "vesafb: VBE version: %d.%d\n", ++ ((vbe_ib.vbe_version & 0xff00) >> 8), ++ vbe_ib.vbe_version & 0xff); ++ return 0; ++} ++ ++static int __init inline vesafb_vbe_getmodes(struct vesafb_task *tsk) ++{ ++ u16 *mode = 0; ++ int off = 0; ++ ++ /* Count available modes. */ ++ mode = (u16*)vbe_ib.mode_list_ptr; ++ while (*mode != 0xffff) { ++ vbe_modes_cnt++; ++ mode++; ++ } ++ ++ vbe_modes = kmalloc(sizeof(struct vesafb_mode_ib)* ++ vbe_modes_cnt, GFP_KERNEL); ++ if (!vbe_modes) ++ return -ENOMEM; ++ ++ /* Get mode info for all available modes. */ ++ mode = (u16*)vbe_ib.mode_list_ptr; ++ ++ while (*mode != 0xffff) { ++ struct vesafb_mode_ib *mib; ++ ++ vesafb_reset_task(tsk); ++ tsk->regs.eax = 0x4f01; ++ tsk->regs.ecx = (u32) *mode; ++ tsk->flags = TF_CALL | TF_RETURN_BUF | TF_BUF_DI; ++ tsk->buf = vbe_modes+off; ++ tsk->buf_len = sizeof(struct vesafb_mode_ib); ++ if (vesafb_queue_task(tsk)) ++ return -EINVAL; ++ vesafb_wait_for_task(tsk); ++ mib = p_mode(tsk->buf); ++ mib->mode_id = *mode; ++ ++ /* We only want modes that are supported with the currennt ++ * hardware configuration (D0), color (D3), graphics (D4) ++ * and that have support for the LFB (D7). */ ++ if ((mib->mode_attr & 0x99) == 0x99 && ++ mib->bits_per_pixel >= 8) { ++ off++; ++ } else { ++ vbe_modes_cnt--; ++ } ++ mode++; ++ mib->depth = mib->red_len + mib->green_len + mib->blue_len; ++ /* Handle 8bpp modes and modes with broken color component ++ * lengths. */ ++ if (mib->depth == 0 || ++ (mib->depth == 24 && mib->bits_per_pixel == 32)) ++ mib->depth = mib->bits_per_pixel; ++ } ++ ++ return 0; ++} ++ ++static int __init inline vesafb_vbe_getpmi(struct vesafb_task *tsk) ++{ ++ int i; ++ ++ vesafb_reset_task(tsk); ++ tsk->regs.eax = 0x4f0a; ++ tsk->regs.ebx = 0x0; ++ tsk->flags = TF_CALL; ++ if (vesafb_queue_task(tsk)) ++ return -EINVAL; ++ vesafb_wait_for_task(tsk); ++ ++ if ((tsk->regs.eax & 0xffff) != 0x004f || tsk->regs.es < 0xc000) { ++ pmi_setpal = ypan = 0; ++ } else { ++ pmi_base = (u16*)phys_to_virt(((u32)tsk->regs.es << 4) + ++ tsk->regs.edi); ++ pmi_start = (void*)((char*)pmi_base + pmi_base[1]); ++ pmi_pal = (void*)((char*)pmi_base + pmi_base[2]); ++ printk(KERN_INFO "vesafb: protected mode interface info at " ++ "%04x:%04x\n", ++ (u16)tsk->regs.es, (u16)tsk->regs.edi); ++ printk(KERN_INFO "vesafb: pmi: set display start = %p, " ++ "set palette = %p\n", pmi_start, pmi_pal); ++ ++ if (pmi_base[3]) { ++ printk(KERN_INFO "vesafb: pmi: ports = "); ++ for (i = pmi_base[3]/2; pmi_base[i] != 0xffff; i++) ++ printk("%x ",pmi_base[i]); ++ printk("\n"); ++ ++ /* ++ * memory areas not supported (yet?) ++ * ++ * Rules are: we have to set up a descriptor for the ++ * requested memory area and pass it in the ES register ++ * to the BIOS function. ++ */ ++ if (pmi_base[i] != 0xffff) { ++ printk(KERN_INFO "vesafb: can't handle memory " ++ "requests, pmi disabled\n"); ++ ypan = pmi_setpal = 0; ++ } ++ } ++ } ++ return 0; ++} ++ ++static int __init inline vesafb_vbe_getedid(struct vesafb_task *tsk, ++ struct fb_info *info) ++{ ++ int res = 0; ++ ++ if (noedid || vbe_ib.vbe_version < 0x0300) ++ return -EINVAL; ++ ++ vesafb_reset_task(tsk); ++ tsk->regs.eax = 0x4f15; ++ tsk->regs.ebx = 0; ++ tsk->regs.ecx = 0; ++ if (vesafb_queue_task(tsk)) ++ return -EINVAL; ++ vesafb_wait_for_task(tsk); ++ ++ if ((tsk->regs.eax & 0xffff) != 0x004f) ++ return -EINVAL; ++ ++ if ((tsk->regs.ebx & 0x3) == 3) { ++ printk(KERN_INFO "vesafb: VBIOS/hardware supports both " ++ "DDC1 and DDC2 transfers\n"); ++ } else if ((tsk->regs.ebx & 0x3) == 2) { ++ printk(KERN_INFO "vesafb: VBIOS/hardware supports DDC2 " ++ "transfers\n"); ++ } else if ((tsk->regs.ebx & 0x3) == 1) { ++ printk(KERN_INFO "vesafb: VBIOS/hardware supports DDC1 " ++ "transfers\n"); ++ } else { ++ printk(KERN_INFO "vesafb: VBIOS/hardware doesn't support " ++ "DDC transfers\n"); ++ return -EINVAL; ++ } ++ ++ vesafb_reset_task(tsk); ++ tsk->regs.eax = 0x4f15; ++ tsk->regs.ebx = 1; ++ tsk->regs.ecx = tsk->regs.edx = 0; ++ tsk->flags = TF_CALL | TF_RETURN_BUF | TF_BUF_DI; ++ tsk->buf = kmalloc(EDID_LENGTH, GFP_KERNEL); ++ tsk->buf_len = EDID_LENGTH; ++ ++ if (vesafb_queue_task(tsk)) { ++ res = -EINVAL; ++ goto out; ++ } ++ vesafb_wait_for_task(tsk); ++ ++ if ((tsk->regs.eax & 0xffff) == 0x004f) { ++ fb_edid_to_monspecs(tsk->buf, &info->monspecs); ++ fb_videomode_to_modelist(info->monspecs.modedb, ++ info->monspecs.modedb_len, &info->modelist); ++ if (info->monspecs.vfmax && info->monspecs.hfmax) { ++ /* If the maximum pixel clock wasn't specified in ++ * the EDID block, set it to 300 MHz. */ ++ if (info->monspecs.dclkmax == 0) ++ info->monspecs.dclkmax = 300 * 1000000; ++ info->monspecs.gtf = 1; ++ } else { ++ res = -EINVAL; ++ } ++ } ++ ++out: kfree(tsk->buf); ++ return res; ++} ++ ++static void __init inline vesafb_vbe_getmonspecs(struct vesafb_task *tsk, ++ struct fb_info *info) ++{ ++ struct fb_var_screeninfo var; ++ int i; ++ memset(&info->monspecs, 0, sizeof(struct fb_monspecs)); ++ ++ /* If we didn't get all necessary data from the EDID block, ++ * mark it as incompatible with the GTF. */ ++ if (vesafb_vbe_getedid(tsk, info)) ++ info->monspecs.gtf = 0; ++ ++ /* Kernel command line overrides. */ ++ if (maxclk) ++ info->monspecs.dclkmax = maxclk * 1000000; ++ if (maxvf) ++ info->monspecs.vfmax = maxvf; ++ if (maxhf) ++ info->monspecs.hfmax = maxhf * 1000; ++ ++ /* In case DDC transfers are not supported the user can provide ++ * monitor limits manually. Lower limits are set to "safe" values. */ ++ if (info->monspecs.gtf == 0 && maxclk && maxvf && maxhf) { ++ info->monspecs.dclkmin = 0; ++ info->monspecs.vfmin = 60; ++ info->monspecs.hfmin = 29000; ++ info->monspecs.gtf = 1; ++ } ++ ++ if (info->monspecs.gtf) { ++ printk(KERN_INFO ++ "vesafb: monitor limits: vf = %d Hz, hf = %d kHz, " ++ "clk = %d MHz\n", info->monspecs.vfmax, ++ (int)(info->monspecs.hfmax / 1000), ++ (int)(info->monspecs.dclkmax / 1000000)); ++ /* Add valid VESA video modes to our modelist. */ ++ for (i = 0; i < VESA_MODEDB_SIZE; i++) { ++ fb_videomode_to_var(&var, (struct fb_videomode *) ++ &vesa_modes[i]); ++ if (!fb_validate_mode(&var, info)) ++ fb_add_videomode((struct fb_videomode *) ++ &vesa_modes[i], ++ &info->modelist); ++ } ++ } else { ++ /* Add all VESA video modes to our modelist. */ ++ fb_videomode_to_modelist((struct fb_videomode *)vesa_modes, ++ VESA_MODEDB_SIZE, &info->modelist); ++ printk(KERN_INFO "vesafb: no monitor limits have been set\n"); ++ } ++ return; ++} ++ ++static int __init inline vesafb_vbe_init(struct fb_info *info) ++{ ++ struct vesafb_task *tsk; ++ int res = 0; ++ ++ vesafb_create_task(tsk); ++ if (!tsk) ++ return -EINVAL; ++ if ((res = vesafb_vbe_getinfo(tsk)) != 0) ++ goto out; ++ if ((res = vesafb_vbe_getmodes(tsk)) != 0) ++ goto out; ++ if (pmi_setpal || ypan) ++ vesafb_vbe_getpmi(tsk); ++ ++ INIT_LIST_HEAD(&info->modelist); ++ vesafb_vbe_getmonspecs(tsk, info); ++ ++out: kfree(tsk); ++ return res; ++} ++ ++static int __init decode_mode(u32 *xres, u32 *yres, u32 *bpp, u32 *refresh) ++{ ++ int len = strlen(mode_option), i, err = 0; ++ u8 res_specified = 0, bpp_specified = 0, refresh_specified = 0, ++ yres_specified = 0; ++ ++ for (i = len-1; i >= 0; i--) { ++ switch (mode_option[i]) { ++ case '@': ++ len = i; ++ if (!refresh_specified && !bpp_specified && ++ !yres_specified) { ++ *refresh = simple_strtoul(&mode_option[i+1], ++ NULL, 0); ++ refresh_specified = 1; ++ } else ++ goto out; ++ break; ++ case '-': ++ len = i; ++ if (!bpp_specified && !yres_specified) { ++ *bpp = simple_strtoul(&mode_option[i+1], ++ NULL, 0); ++ bpp_specified = 1; ++ } else ++ goto out; ++ break; ++ case 'x': ++ if (!yres_specified) { ++ *yres = simple_strtoul(&mode_option[i+1], ++ NULL, 0); ++ yres_specified = 1; ++ } else ++ goto out; ++ break; ++ case '0'...'9': ++ break; ++ default: ++ goto out; ++ } ++ } ++ ++ if (i < 0 && yres_specified) { ++ *xres = simple_strtoul(mode_option, NULL, 0); ++ res_specified = 1; ++ } ++ ++out: if (!res_specified || !yres_specified) { ++ printk(KERN_ERR "vesafb: invalid resolution, " ++ "%s not specified\n", ++ (!res_specified) ? "width" : "height"); ++ err = -EINVAL; ++ } ++ ++ return err; ++} ++ ++static int __init vesafb_init_set_mode(struct fb_info *info) ++{ ++ struct fb_videomode *fbmode; ++ struct fb_videomode mode; ++ int i, modeid, refresh = 0; ++ u8 refresh_specified = 0; ++ ++ if (!mode_option) ++ mode_option = CONFIG_FB_VESA_DEFAULT_MODE; ++ ++ if (vbemode > 0) { ++ for (i = 0; i < vbe_modes_cnt; i++) { ++ if (vbe_modes[i].mode_id == vbemode) { ++ info->var.vmode = FB_VMODE_NONINTERLACED; ++ info->var.sync = FB_SYNC_VERT_HIGH_ACT; ++ vesafb_setup_var(&info->var, info, ++ &vbe_modes[i]); ++ fb_get_mode(FB_VSYNCTIMINGS | FB_IGNOREMON, ++ 60, &info->var, info); ++ /* With pixclock set to 0, the default BIOS ++ * timings will be used in set_par(). */ ++ info->var.pixclock = 0; ++ modeid = i; ++ goto out; ++ } ++ } ++ printk(KERN_INFO "specified VBE mode %d not found\n", ++ vbemode); ++ vbemode = 0; ++ } ++ ++ /* Decode the mode specified on the kernel command line. We save ++ * the depth into bits_per_pixel, which is wrong, but will work ++ * anyway. */ ++ if (decode_mode(&info->var.xres, &info->var.yres, ++ &info->var.bits_per_pixel, &refresh)) ++ return -EINVAL; ++ if (refresh) ++ refresh_specified = 1; ++ else ++ refresh = 60; ++ ++ /* Look for a matching VBE mode. We can live if an exact match ++ * cannot be found. */ ++ modeid = vesafb_find_vbe_mode(info->var.xres, info->var.yres, ++ info->var.bits_per_pixel, 0); ++ ++ if (modeid == -1) { ++ return -EINVAL; ++ } else { ++ info->var.vmode = FB_VMODE_NONINTERLACED; ++ info->var.sync = FB_SYNC_VERT_HIGH_ACT; ++ vesafb_setup_var(&info->var, info, &vbe_modes[modeid]); ++ } ++ if (vbe_ib.vbe_version < 0x0300) { ++ fb_get_mode(FB_VSYNCTIMINGS | FB_IGNOREMON, 60, ++ &info->var, info); ++ goto out; ++ } ++ if (!gtf) { ++ struct fb_videomode tmode; ++ ++ if (refresh_specified) { ++ fb_var_to_videomode(&tmode, &info->var); ++ tmode.refresh = refresh; ++ fbmode = fb_find_nearest_mode(&tmode, ++ &info->modelist); ++ } else ++ fbmode = fb_find_best_mode(&info->var, ++ &info->modelist); ++ ++ if (fbmode->xres == info->var.xres && ++ fbmode->yres == info->var.yres && ++ !(fbmode->vmode & (FB_VMODE_INTERLACED | FB_VMODE_DOUBLE)) ++ && (!refresh_specified || ++ abs(refresh - fbmode->refresh) <= 5)) { ++ fb_videomode_to_var(&info->var, fbmode); ++ return modeid; ++ } ++ } ++ i = FB_MAXTIMINGS; ++ if (!info->monspecs.gtf) ++ i = FB_IGNOREMON | FB_VSYNCTIMINGS; ++ else if (refresh_specified) ++ i = FB_VSYNCTIMINGS; ++ if (!fb_get_mode(i, refresh, &info->var, info)) ++ goto out; ++ if (info->monspecs.gtf && ++ !fb_get_mode(FB_MAXTIMINGS, 0, &info->var, info)) ++ goto out; ++ /* Use default refresh rate */ ++ printk(KERN_WARNING "vesafb: using default BIOS refresh rate\n"); ++ info->var.pixclock = 0; ++ ++out: ++ fb_var_to_videomode(&mode, &info->var); ++ fb_add_videomode(&mode, &info->modelist); ++ return modeid; ++} ++ ++static int __init vesafb_probe(struct platform_device *dev) ++{ ++ char entry[16]; ++ struct fb_info *info; ++ struct vesafb_mode_ib *mode = NULL; ++ int err = 0, i, h; ++ unsigned int size_vmode; ++ unsigned int size_remap; ++ unsigned int size_total; ++ ++ vesafb_info = info = framebuffer_alloc(sizeof(struct vesafb_par) + ++ sizeof(u32) * 256, &dev->dev); ++ if (!info) ++ return -ENOMEM; ++ ++ if (vesafb_wait_for_thread()) { ++ printk(KERN_ERR "vesafb: vesafb thread not running\n"); ++ framebuffer_release(info); ++ return -EINVAL; ++ } ++ ++ if (vesafb_vbe_init(info)) { ++ printk(KERN_ERR "vesafb: vbe_init failed\n"); ++ err = -EINVAL; ++ goto out; ++ } ++ ++ vesafb_fix.ypanstep = ypan ? 1 : 0; ++ vesafb_fix.ywrapstep = (ypan>1) ? 1 : 0; ++ ++ info->pseudo_palette = ((u8*)info->par + sizeof(struct vesafb_par)); ++ info->fbops = &vesafb_ops; ++ info->var = vesafb_defined; ++ info->fix = vesafb_fix; ++ ++ if (fb_alloc_cmap(&info->cmap, 256, 0) < 0) { ++ err = -ENXIO; ++ goto out; ++ } ++ ++ i = vesafb_init_set_mode(info); ++ if (i < 0) { ++ err = -EINVAL; ++ goto out_cmap; ++ } else ++ mode = &vbe_modes[i]; ++ ++ /* Disable blanking if the user requested so. */ ++ if (!blank) { ++ info->fbops->fb_blank = NULL; ++ } ++ ++ /* Find out how much IO memory is required for the mode with ++ * the highest resolution. */ ++ size_remap = 0; ++ for (i = 0; i < vbe_modes_cnt; i++) { ++ h = vbe_modes[i].bytes_per_scan_line * vbe_modes[i].y_res; ++ if (h > size_remap) ++ size_remap = h; ++ } ++ size_remap *= 2; ++ ++ /* size_vmode -- that is the amount of memory needed for the ++ * used video mode, i.e. the minimum amount of ++ * memory we need. */ ++ if (mode != NULL) { ++ size_vmode = info->var.yres * mode->bytes_per_scan_line; ++ } else { ++ size_vmode = info->var.yres * info->var.xres * ++ ((info->var.bits_per_pixel + 7) >> 3); ++ } ++ ++ /* size_total -- all video memory we have. Used for mtrr ++ * entries, ressource allocation and bounds ++ * checking. */ ++ size_total = vbe_ib.total_memory * 65536; ++ if (vram_total) ++ size_total = vram_total * 1024 * 1024; ++ if (size_total < size_vmode) ++ size_total = size_vmode; ++ ((struct vesafb_par*)(info->par))->mem_total = size_total; ++ ++ /* size_remap -- the amount of video memory we are going to ++ * use for vesafb. With modern cards it is no ++ * option to simply use size_total as th ++ * wastes plenty of kernel address space. */ ++ if (vram_remap) ++ size_remap = vram_remap * 1024 * 1024; ++ if (size_remap < size_vmode) ++ size_remap = size_vmode; ++ if (size_remap > size_total) ++ size_remap = size_total; ++ ++ info->fix.smem_len = size_remap; ++ info->fix.smem_start = mode->phys_base_ptr; ++ ++ /* We have to set it here, because when setup_var() was called, ++ * smem_len wasn't defined yet. */ ++ info->var.yres_virtual = info->fix.smem_len / ++ mode->bytes_per_scan_line; ++ ++ if (ypan && info->var.yres_virtual > info->var.yres) { ++ printk(KERN_INFO "vesafb: scrolling: %s " ++ "using protected mode interface, " ++ "yres_virtual=%d\n", ++ (ypan > 1) ? "ywrap" : "ypan",info->var.yres_virtual); ++ } else { ++ printk(KERN_INFO "vesafb: scrolling: redraw\n"); ++ info->var.yres_virtual = info->var.yres; ++ ypan = 0; ++ } ++ ++ info->flags = FBINFO_FLAG_DEFAULT | ++ (ypan) ? FBINFO_HWACCEL_YPAN : 0; ++ ++ if (!ypan) ++ info->fbops->fb_pan_display = NULL; ++ ++ if (!request_mem_region(info->fix.smem_start, size_total, "vesafb")) { ++ printk(KERN_WARNING "vesafb: cannot reserve video memory at " ++ "0x%lx\n", info->fix.smem_start); ++ /* We cannot make this fatal. Sometimes this comes from magic ++ spaces our resource handlers simply don't know about. */ ++ } ++ ++ info->screen_base = ioremap(info->fix.smem_start, info->fix.smem_len); ++ ++ if (!info->screen_base) { ++ printk(KERN_ERR ++ "vesafb: abort, cannot ioremap video memory " ++ "0x%x @ 0x%lx\n", ++ info->fix.smem_len, info->fix.smem_start); ++ err = -EIO; ++ goto out_mem; ++ } ++ ++ /* Request failure does not faze us, as vgacon probably has this ++ region already (FIXME) */ ++ request_region(0x3c0, 32, "vesafb"); ++ ++#ifdef CONFIG_MTRR ++ if (mtrr && !(info->fix.smem_start & (PAGE_SIZE - 1))) { ++ int temp_size = size_total; ++ unsigned int type = 0; ++ ++ switch (mtrr) { ++ case 1: ++ type = MTRR_TYPE_UNCACHABLE; ++ break; ++ case 2: ++ type = MTRR_TYPE_WRBACK; ++ break; ++ case 3: ++ type = MTRR_TYPE_WRCOMB; ++ break; ++ case 4: ++ type = MTRR_TYPE_WRTHROUGH; ++ break; ++ default: ++ type = 0; ++ break; ++ } ++ ++ if (type) { ++ int rc; ++ ++ /* Find the largest power-of-two */ ++ while (temp_size & (temp_size - 1)) ++ temp_size &= (temp_size - 1); ++ ++ /* Try and find a power of two to add */ ++ do { ++ rc = mtrr_add(info->fix.smem_start, ++ temp_size, type, 1); ++ temp_size >>= 1; ++ } while (temp_size >= PAGE_SIZE && rc == -EINVAL); ++ } ++ } ++#endif /* CONFIG_MTRR */ ++ ++ if (register_framebuffer(info) < 0) { ++ printk(KERN_ERR ++ "vesafb: failed to register framebuffer device\n"); ++ err = -EINVAL; ++ goto out_mem; ++ } ++ ++ printk(KERN_INFO "vesafb: framebuffer at 0x%lx, mapped to 0x%p, " ++ "using %dk, total %dk\n", info->fix.smem_start, ++ info->screen_base, size_remap/1024, size_total/1024); ++ printk(KERN_INFO "fb%d: %s frame buffer device\n", info->node, ++ info->fix.id); ++ ++ sprintf(entry, "fb%d", info->node); ++ proc_mkdir(entry, 0); ++ ++ sprintf(entry, "fb%d/modes", info->node); ++ create_proc_read_entry(entry, 0, 0, vesafb_read_proc_modes, NULL); ++ ++ sprintf(entry, "fb%d/vbe_info", info->node); ++ create_proc_read_entry(entry, 0, 0, vesafb_read_proc_vbe_info, NULL); ++ return 0; ++ ++out_mem: ++ release_mem_region(info->fix.smem_start, size_total); ++ if (!list_empty(&info->modelist)) ++ fb_destroy_modelist(&info->modelist); ++ fb_destroy_modedb(info->monspecs.modedb); ++out_cmap: ++ fb_dealloc_cmap(&info->cmap); ++out: ++ framebuffer_release(info); ++ vesafb_info = NULL; ++ kfree(vbe_modes); ++ vbe_modes = NULL; ++ return err; ++} ++ ++int __init vesafb_init(void) ++{ ++ int ret; ++#ifndef MODULE ++ char *option = NULL; ++ ++ if (fb_get_options("vesafb", &option)) ++ return -ENODEV; ++ vesafb_setup(option); ++#endif ++ ret = platform_driver_register(&vesafb_driver); ++ ++ if (!ret) { ++ vesafb_device = platform_device_alloc("vesafb", 0); ++ ++ if (vesafb_device) ++ ret = platform_device_add(vesafb_device); ++ else ++ ret = -ENOMEM; ++ ++ if (ret) { ++ platform_device_put(vesafb_device); ++ platform_driver_unregister(&vesafb_driver); ++ } ++ } ++ return ret; ++} ++ ++module_init(vesafb_init); ++ ++#ifdef MODULE ++void __exit vesafb_exit(void) ++{ ++ char entry[16]; ++ ++ if (vesafb_info) ++ unregister_framebuffer(vesafb_info); ++ ++ platform_device_unregister(vesafb_device); ++ platform_driver_unregister(&vesafb_driver); ++ ++ if (vesafb_info) { ++ struct vesafb_par *par = (struct vesafb_par*)vesafb_info->par; ++ ++ sprintf(entry, "fb%d/modes", vesafb_info->node); ++ remove_proc_entry(entry, NULL); ++ ++ sprintf(entry, "fb%d/vbe_info", vesafb_info->node); ++ remove_proc_entry(entry, NULL); ++ ++ sprintf(entry, "fb%d", vesafb_info->node); ++ remove_proc_entry(entry, NULL); ++ ++ iounmap(vesafb_info->screen_base); ++ release_mem_region(vesafb_info->fix.smem_start, ++ par->mem_total); ++ fb_dealloc_cmap(&vesafb_info->cmap); ++ if (!list_empty(&vesafb_info->modelist)) ++ fb_destroy_modelist(&vesafb_info->modelist); ++ fb_destroy_modedb(vesafb_info->monspecs.modedb); ++ framebuffer_release(vesafb_info); ++ } ++ ++ if (vbe_modes != NULL) ++ kfree(vbe_modes); ++} ++ ++module_exit(vesafb_exit); ++ ++static inline int param_get_scroll(char *buffer, struct kernel_param *kp) ++{ ++ return 0; ++} ++static inline int param_set_scroll(const char *val, struct kernel_param *kp) ++{ ++ ypan = 0; ++ ++ if (! strcmp(val, "redraw")) ++ ypan = 0; ++ else if (! strcmp(val, "ypan")) ++ ypan = 1; ++ else if (! strcmp(val, "ywrap")) ++ ypan = 2; ++ ++ return 0; ++} ++ ++#define param_check_scroll(name, p) __param_check(name, p, void); ++ ++module_param_named(scroll, ypan, scroll, 0); ++MODULE_PARM_DESC(scroll,"Scrolling mode, set to 'redraw', 'ypan' or 'ywrap'"); ++module_param_named(vgapal, pmi_setpal, invbool, 0); ++MODULE_PARM_DESC(vgapal,"bool: set palette using VGA registers"); ++module_param_named(pmipal, pmi_setpal, bool, 0); ++MODULE_PARM_DESC(pmipal,"bool: set palette using PMI calls"); ++module_param(mtrr, uint, 0); ++MODULE_PARM_DESC(mtrr,"Memory Type Range Registers setting. Use 0 to disable."); ++module_param(blank, bool, 1); ++MODULE_PARM_DESC(blank,"bool: enable hardware blanking"); ++module_param(nocrtc, bool, 0); ++MODULE_PARM_DESC(nocrtc,"bool: ignore CRTC timings when setting modes"); ++module_param(noedid, bool, 0); ++MODULE_PARM_DESC(noedid,"bool: ignore EDID-provided monitor limits " ++ "when setting modes"); ++module_param(gtf, bool, 0); ++MODULE_PARM_DESC(gtf,"bool: force use of VESA GTF to calculate mode timings"); ++module_param(vram_remap, uint, 0); ++MODULE_PARM_DESC(vram_remap,"Set amount of video memory to be used [MiB]"); ++module_param(vram_total, uint, 0); ++MODULE_PARM_DESC(vram_total,"Set total amount of video memoery [MiB]"); ++module_param(maxclk, ushort, 0); ++MODULE_PARM_DESC(maxclk,"Maximum pixelclock [MHz], overrides EDID data"); ++module_param(maxhf, ushort, 0); ++MODULE_PARM_DESC(maxhf,"Maximum horizontal frequency [kHz], " ++ "overrides EDID data"); ++module_param(maxvf, ushort, 0); ++MODULE_PARM_DESC(maxvf,"Maximum vertical frequency [Hz], " ++ "overrides EDID data"); ++module_param_named(mode, mode_option, charp, 0); ++MODULE_PARM_DESC(mode, "Specify resolution as " ++ "\"<xres>x<yres>[-<bpp>][@<refresh>]\""); ++module_param(vbemode, ushort, 0); ++MODULE_PARM_DESC(vbemode,"VBE mode number to set, overrides 'mode' setting"); ++ ++#endif /* MODULE */ ++ ++MODULE_LICENSE("GPL"); ++MODULE_AUTHOR("Michal Januszewski"); ++MODULE_DESCRIPTION("Framebuffer driver for VBE2.0+ compliant graphics boards"); ++ +diff --git a/include/linux/sched.h b/include/linux/sched.h +index 4463735..7283e48 100644 +--- a/include/linux/sched.h ++++ b/include/linux/sched.h +@@ -1390,6 +1390,8 @@ extern void mmput(struct mm_struct *); + extern struct mm_struct *get_task_mm(struct task_struct *task); + /* Remove the current tasks stale references to the old mm_struct */ + extern void mm_release(struct task_struct *, struct mm_struct *); ++/* Create a new mm for a kernel thread */ ++extern int set_new_mm(void); + + extern int copy_thread(int, unsigned long, unsigned long, unsigned long, struct task_struct *, struct pt_regs *); + extern void flush_thread(void); +diff --git a/include/video/vesa.h b/include/video/vesa.h +new file mode 100644 +index 0000000..bb5abcf +--- /dev/null ++++ b/include/video/vesa.h +@@ -0,0 +1,150 @@ ++#if 0 ++#define DPRINTK(fmt, args...) printk(KERN_DEBUG "%s: " fmt, __FUNCTION__ , \ ++ ## args) ++#else ++#define DPRINTK(fmt, args...) ++#endif ++ ++#define p_crtc(arg) ((struct vesafb_crtc_ib*)(arg)) ++#define p_vbe(arg) ((struct vesafb_vbe_ib*)(arg)) ++#define p_mode(arg) ((struct vesafb_mode_ib*)(arg)) ++ ++struct vesafb_task { ++ u8 flags; ++ void *buf; ++ int buf_len; ++ struct vm86_regs regs; ++ struct list_head node; ++ struct completion done; ++}; ++ ++/* Vesafb task flags and masks */ ++#define TF_CALL 0x00 ++#define TF_EXIT 0x01 ++#define TF_GETVBEIB 0x02 ++#define TF_BUF_DI 0x04 ++#define TF_BUF_BX 0x08 ++#define TF_RETURN_BUF 0x10 ++ ++/* Macros and functions for manipulating vesafb tasks */ ++#define vesafb_create_task(task) \ ++do { \ ++ task = kmalloc(sizeof(struct vesafb_task), GFP_ATOMIC); \ ++ if (task) \ ++ memset(task, 0, sizeof(struct vesafb_task)); \ ++ init_completion(&task->done); \ ++} while (0) ++ ++#define vesafb_wait_for_task(task) wait_for_completion(&task->done); ++#define vesafb_reset_task(task) init_completion(&task->done); ++int vesafb_queue_task(struct vesafb_task *task); ++ ++/* Functions for controlling the vesafb thread */ ++int vesafb_wait_for_thread(void); ++ ++#define VBE_CAP_CAN_SWITCH_DAC 0x01 ++#define VBE_CAP_VGACOMPAT 0x02 ++ ++/* This struct is 512 bytes long */ ++struct vesafb_vbe_ib { ++ char vbe_signature[4]; ++ u16 vbe_version; ++ u32 oem_string_ptr; ++ u32 capabilities; ++ u32 mode_list_ptr; ++ u16 total_memory; ++ u16 oem_software_rev; ++ u32 oem_vendor_name_ptr; ++ u32 oem_product_name_ptr; ++ u32 oem_product_rev_ptr; ++ u8 reserved[222]; ++ char oem_data[256]; ++} __attribute__ ((packed)); ++ ++struct vesafb_crtc_ib { ++ u16 horiz_total; ++ u16 horiz_start; ++ u16 horiz_end; ++ u16 vert_total; ++ u16 vert_start; ++ u16 vert_end; ++ u8 flags; ++ u32 pixel_clock; ++ u16 refresh_rate; ++ u8 reserved[40]; ++} __attribute__ ((packed)); ++ ++#define VBE_MODE_VGACOMPAT 0x20 ++ ++struct vesafb_mode_ib { ++ /* for all VBE revisions */ ++ u16 mode_attr; ++ u8 winA_attr; ++ u8 winB_attr; ++ u16 win_granularity; ++ u16 win_size; ++ u16 winA_seg; ++ u16 winB_seg; ++ u32 win_func_ptr; ++ u16 bytes_per_scan_line; ++ ++ /* for VBE 1.2+ */ ++ u16 x_res; ++ u16 y_res; ++ u8 x_char_size; ++ u8 y_char_size; ++ u8 planes; ++ u8 bits_per_pixel; ++ u8 banks; ++ u8 memory_model; ++ u8 bank_size; ++ u8 image_pages; ++ u8 reserved1; ++ ++ /* Direct color fields for direct/6 and YUV/7 memory models. */ ++ /* Offsets are bit positions of lsb in the mask. */ ++ u8 red_len; ++ u8 red_off; ++ u8 green_len; ++ u8 green_off; ++ u8 blue_len; ++ u8 blue_off; ++ u8 rsvd_len; ++ u8 rsvd_off; ++ u8 direct_color_info; /* direct color mode attributes */ ++ ++ /* for VBE 2.0+ */ ++ u32 phys_base_ptr; ++ u8 reserved2[6]; ++ ++ /* for VBE 3.0+ */ ++ u16 lin_bytes_per_scan_line; ++ u8 bnk_image_pages; ++ u8 lin_image_pages; ++ u8 lin_red_len; ++ u8 lin_red_off; ++ u8 lin_green_len; ++ u8 lin_green_off; ++ u8 lin_blue_len; ++ u8 lin_blue_off; ++ u8 lin_rsvd_len; ++ u8 lin_rsvd_off; ++ u32 max_pixel_clock; ++ u16 mode_id; ++ u8 depth; ++} __attribute__ ((packed)); ++ ++struct vesafb_pal_entry { ++ u_char blue, green, red, pad; ++} __attribute__ ((packed)); ++ ++struct vesafb_par { ++ u8 *vbe_state; ++ int vbe_state_size; ++ atomic_t ref_count; ++ ++ u32 mem_total; ++ int mode_idx; ++ struct vesafb_crtc_ib crtc; ++}; ++ +diff --git a/kernel/fork.c b/kernel/fork.c +index fc723e5..dc8f93b 100644 +--- a/kernel/fork.c ++++ b/kernel/fork.c +@@ -100,6 +100,7 @@ struct kmem_cache *fs_cachep; + + /* SLAB cache for vm_area_struct structures */ + struct kmem_cache *vm_area_cachep; ++EXPORT_SYMBOL_GPL(vm_area_cachep); + + /* SLAB cache for mm_struct structures (tsk->mm) */ + static struct kmem_cache *mm_cachep; +@@ -399,6 +400,40 @@ void mmput(struct mm_struct *mm) + EXPORT_SYMBOL_GPL(mmput); + + /** ++ * set_new_mm - allocate, init and activate a new mm for a kernel thread ++ */ ++int set_new_mm(void) ++{ ++ struct mm_struct *mm; ++ struct task_struct *tsk = current; ++ struct mm_struct *active_mm; ++ ++ mm = mm_alloc(); ++ if (!mm) ++ goto fail_nomem; ++ if (init_new_context(current,mm)) ++ goto fail_nocontext; ++ ++ task_lock(tsk); ++ tsk->flags |= PF_BORROWED_MM; ++ active_mm = tsk->active_mm; ++ current->mm = mm; ++ current->active_mm = mm; ++ activate_mm(active_mm, mm); ++ task_unlock(current); ++ ++ /* Drop the previous active_mm */ ++ mmdrop(active_mm); ++ return 0; ++ ++fail_nocontext: ++ mmdrop(mm); ++fail_nomem: ++ return -EINVAL; ++} ++EXPORT_SYMBOL_GPL(set_new_mm); ++ ++/** + * get_task_mm - acquire a reference to the task's mm + * + * Returns %NULL if the task has no mm. Checks PF_BORROWED_MM (meaning +diff --git a/mm/memory.c b/mm/memory.c +index 563792f..a9519ea 100644 +--- a/mm/memory.c ++++ b/mm/memory.c +@@ -1193,6 +1193,7 @@ int zeromap_page_range(struct vm_area_struct *vma, + } while (pgd++, addr = next, addr != end); + return err; + } ++EXPORT_SYMBOL_GPL(zeromap_page_range); + + pte_t * fastcall get_locked_pte(struct mm_struct *mm, unsigned long addr, spinlock_t **ptl) + { +diff --git a/mm/mmap.c b/mm/mmap.c +index 9717337..6fa5b1c 100644 +--- a/mm/mmap.c ++++ b/mm/mmap.c +@@ -2024,6 +2024,7 @@ int insert_vm_struct(struct mm_struct * mm, struct vm_area_struct * vma) + vma_link(mm, vma, prev, rb_link, rb_parent); + return 0; + } ++EXPORT_SYMBOL_GPL(insert_vm_struct); + + /* + * Copy the vma structure to a new location in the same mm, diff --git a/packages/linux/linux-rp-2.6.20/wm97xx-lcdnoise-r0.patch b/packages/linux/linux-rp-2.6.20/wm97xx-lcdnoise-r0.patch new file mode 100644 index 0000000000..191de3af22 --- /dev/null +++ b/packages/linux/linux-rp-2.6.20/wm97xx-lcdnoise-r0.patch @@ -0,0 +1,208 @@ +Index: linux-tosa/drivers/input/touchscreen/wm9712.c +=================================================================== +--- linux-tosa.orig/drivers/input/touchscreen/wm9712.c 2006-08-29 16:52:36.008543280 +0100 ++++ linux-tosa/drivers/input/touchscreen/wm9712.c 2006-08-29 16:52:50.923275896 +0100 +@@ -1,7 +1,7 @@ + /* + * wm9712.c -- Codec driver for Wolfson WM9712 AC97 Codecs. + * +- * Copyright 2003, 2004, 2005 Wolfson Microelectronics PLC. ++ * Copyright 2003, 2004, 2005, 2006 Wolfson Microelectronics PLC. + * Author: Liam Girdwood + * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com + * Parts Copyright : Ian Molton <spyro@f2s.com> +@@ -13,6 +13,12 @@ + * Free Software Foundation; either version 2 of the License, or (at your + * option) any later version. + * ++ * Revision history ++ * 4th Jul 2005 Initial version. ++ * 29th Aug 2006 Mike Arthur <mike@mikearthur.co.uk> ++ * Added fixes for Sharp SL-6000 (Tosa) LCD noise causing ++ * touchscreen interference. ++ * + */ + + #include <linux/module.h> +@@ -28,6 +34,10 @@ + #define WM9705_VERSION "0.60" + #define DEFAULT_PRESSURE 0xb0c0 + ++#define CCNT(a) asm volatile ("mrc p14, 0, %0, C1, C1, 0" : "=r"(a)) ++#define CCNT_ON() asm("mcr p14, 0, %0, C0, C0, 0" : : "r"(1)) ++#define CCNT_OFF() asm("mcr p14, 0, %0, C0, C0, 0" : : "r"(1)) ++ + /* + * Debug + */ +@@ -243,6 +253,36 @@ + return wm->dig[2] & WM9712_PDEN; + } + ++ ++#ifdef CONFIG_MACH_TOSA ++/* On the Sharp SL-6000 (Tosa), due to a noisy LCD, we need to perform a wait ++ * before sampling the Y axis of the touchscreen */ ++static inline void wm9712_lcd_sync_on(struct wm97xx* wm, int adcsel) { ++ unsigned long timer1 = 0, timer2 = 0, wait_time = 0; ++ if (adcsel == WM97XX_ADCSEL_Y) { ++ wait_time = wm97xx_calc_lcd_waittime(wm); ++ ++ CCNT_ON(); ++ ++ if (wait_time) { ++ /* wait for LCD rising edge */ ++ wm_machinfo->wait_hsync(); ++ /* get clock */ ++ CCNT(timer1); ++ CCNT(timer2); ++ ++ while ((timer2 - timer1) < wait_time) { ++ CCNT(timer2); ++ } ++ } ++ } ++} ++ ++static inline void wm9712_lcd_sync_off(void) { ++ CCNT_OFF(); ++} ++#endif ++ + /* + * Read a sample from the WM9712 adc in polling mode. + */ +@@ -260,6 +300,9 @@ + /* set up digitiser */ + if (adcsel & 0x8000) + adcsel = ((adcsel & 0x7fff) + 3) << 12; ++ #ifdef CONFIG_MACH_TOSA ++ wm9712_lcd_sync_on(wm, adcsel); ++ #endif + wm97xx_reg_write(wm, AC97_WM97XX_DIGITISER1, adcsel | WM97XX_POLL | WM97XX_DELAY(delay)); + + /* wait 3 AC97 time slots + delay for conversion */ +@@ -282,6 +325,10 @@ + + *sample = wm97xx_reg_read(wm, AC97_WM97XX_DIGITISER_RD); + ++ #ifdef CONFIG_MACH_TOSA ++ wm9712_lcd_sync_off(); ++ #endif ++ + /* check we have correct sample */ + if ((*sample & WM97XX_ADCSEL_MASK) != adcsel) { + dbg ("adc wrong sample, read %x got %x", adcsel, +@@ -303,11 +350,12 @@ + static int wm9712_poll_touch(struct wm97xx* wm, struct wm97xx_data *data) + { + int rc; +- + if ((rc = wm9712_poll_sample(wm, WM97XX_ADCSEL_X, &data->x)) != RC_VALID) + return rc; ++ + if ((rc = wm9712_poll_sample(wm, WM97XX_ADCSEL_Y, &data->y)) != RC_VALID) + return rc; ++ + if (pil && !five_wire) { + if ((rc = wm9712_poll_sample(wm, WM97XX_ADCSEL_PRES, &data->p)) != RC_VALID) + return rc; +Index: linux-tosa/drivers/input/touchscreen/wm97xx-core.c +=================================================================== +--- linux-tosa.orig/drivers/input/touchscreen/wm97xx-core.c 2006-08-29 16:52:36.008543280 +0100 ++++ linux-tosa/drivers/input/touchscreen/wm97xx-core.c 2006-08-29 16:52:50.924275744 +0100 +@@ -2,7 +2,7 @@ + * wm97xx-core.c -- Touch screen driver core for Wolfson WM9705, WM9712 + * and WM9713 AC97 Codecs. + * +- * Copyright 2003, 2004, 2005 Wolfson Microelectronics PLC. ++ * Copyright 2003, 2004, 2005, 2006 Wolfson Microelectronics PLC. + * Author: Liam Girdwood + * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com + * Parts Copyright : Ian Molton <spyro@f2s.com> +@@ -67,6 +67,9 @@ + * GPIOs) and 2.6 power management. + * 29th Nov 2004 Added WM9713 support. + * 4th Jul 2005 Moved codec specific code out to seperate files. ++ * 29th Aug 2006 Mike Arthur <mike@mikearthur.co.uk> ++ * Added fixes for Sharp SL-6000 (Tosa) LCD noise causing ++ * touchscreen interference. + */ + + #include <linux/module.h> +@@ -94,6 +97,7 @@ + static DECLARE_MUTEX(gpio_sem); + static LIST_HEAD(wm97xx_misc_list); + static struct wm97xx* wm_codec = NULL; ++struct wm97xx_machinfo *wm_machinfo; + + /* + * WM97xx - enable/disable AUX ADC sysfs +@@ -832,6 +836,23 @@ + mdev->remove(wm_codec); + } + ++#ifdef CONFIG_MACH_TOSA ++/* On the Sharp SL-6000 (Tosa), due to a noisy LCD, we need to perform a wait ++ * before sampling the Y axis of the touchscreen */ ++unsigned long wm97xx_calc_lcd_waittime(struct wm97xx *wm) { ++ unsigned long hsync_time = wm_machinfo->get_hsync_time(); ++ return hsync_time; ++} ++ ++void wm97xx_set_machinfo(struct wm97xx_machinfo *machinfo) { ++ wm_machinfo = machinfo; ++} ++ ++void wm97xx_unset_machinfo() { ++ wm_machinfo = NULL; ++} ++#endif ++ + static struct device_driver wm97xx_driver = { + .name = "ac97", + .bus = &ac97_bus_type, +@@ -861,6 +882,9 @@ + EXPORT_SYMBOL_GPL(wm97xx_reg_write); + EXPORT_SYMBOL_GPL(wm97xx_register_misc_dev); + EXPORT_SYMBOL_GPL(wm97xx_unregister_misc_dev); ++EXPORT_SYMBOL_GPL(wm97xx_calc_lcd_waittime); ++EXPORT_SYMBOL_GPL(wm97xx_set_machinfo); ++EXPORT_SYMBOL_GPL(wm97xx_unset_machinfo); + + module_init(wm97xx_init); + module_exit(wm97xx_exit); +Index: linux-tosa/include/linux/wm97xx.h +=================================================================== +--- linux-tosa.orig/include/linux/wm97xx.h 2006-08-29 16:52:36.008543280 +0100 ++++ linux-tosa/include/linux/wm97xx.h 2006-08-29 16:52:50.924275744 +0100 +@@ -207,6 +207,7 @@ + + struct wm97xx; + extern struct wm97xx_codec_drv wm97xx_codec; ++extern struct wm97xx_machinfo *wm_machinfo; + + /* + * Codec driver interface - allows mapping to WM9705/12/13 and newer codecs +@@ -253,6 +254,11 @@ + struct list_head list; + }; + ++struct wm97xx_machinfo { ++ unsigned long (*get_hsync_time)(void); ++ void (*wait_hsync)(void); ++}; ++ + int wm97xx_register_misc_dev(struct wm97xx_misc_dev* mdev); + void wm97xx_unregister_misc_dev(struct wm97xx_misc_dev* mdev); + +@@ -281,4 +287,9 @@ + int wm97xx_acc_startup(struct wm97xx* wm); + void wm97xx_acc_shutdown(struct wm97xx* wm); + ++ ++unsigned long wm97xx_calc_lcd_waittime(struct wm97xx *wm); ++void wm97xx_set_machinfo(struct wm97xx_machinfo *machinfo); ++void wm97xx_unset_machinfo(void); ++ + #endif diff --git a/packages/linux/linux-rp_2.6.18.bb b/packages/linux/linux-rp_2.6.18.bb index 7150881cad..2f3d63dee2 100644 --- a/packages/linux/linux-rp_2.6.18.bb +++ b/packages/linux/linux-rp_2.6.18.bb @@ -1,6 +1,6 @@ require linux-rp.inc -PR = "r7" +PR = "r9" # Handy URLs # git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux-2.6.git;protocol=git;tag=ef7d1b244fa6c94fb76d5f787b8629df64ea4046 @@ -49,6 +49,7 @@ SRC_URI = "${KERNELORG_MIRROR}/pub/linux/kernel/v2.6/linux-2.6.18.tar.bz2 \ file://connectplus-remove-ide-HACK.patch;patch=1;status=hack \ file://orinoco-remove-all-which-are-in-hostap-HACK.patch;patch=1;status=unmergable-hack \ file://squashfs3.0-2.6.15.patch;patch=1;status=external \ + http://oe.reversefold.com/sharpsl-rc/sharpsl-rc-2.6.18-r0.patch;patch=1 \ file://defconfig-c7x0 \ file://defconfig-hx2000 \ file://defconfig-collie \ diff --git a/packages/linux/linux-rp_2.6.20.bb b/packages/linux/linux-rp_2.6.20.bb new file mode 100644 index 0000000000..f0f62c36b2 --- /dev/null +++ b/packages/linux/linux-rp_2.6.20.bb @@ -0,0 +1,103 @@ +require linux-rp.inc + +PR = "r1" + +# Handy URLs +# git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux-2.6.git;protocol=git;tag=ef7d1b244fa6c94fb76d5f787b8629df64ea4046 +# http://www.kernel.org/pub/linux/kernel/v2.6/linux-2.6.18.tar.bz2 +# http://www.kernel.org/pub/linux/kernel/v2.6/testing/linux-2.6.20-rc4.tar.bz2 +# http://www.kernel.org/pub/linux/kernel/v2.6/testing/patch-2.6.18-rc6.bz2;patch=1 +# http://www.kernel.org/pub/linux/kernel/v2.6/snapshots/patch-2.6.18-rc2-git1.bz2;patch=1 +# http://www.kernel.org/pub/linux/kernel/people/alan/linux-2.6/2.6.10/patch-2.6.10-ac8.gz;patch=1 +# http://www.kernel.org/pub/linux/kernel/people/akpm/patches/2.6/2.6.14-rc2/2.6.14-rc2-mm1/2.6.14-rc2-mm1.bz2;patch=1 + +# Patches submitted upstream are towards top of this list +# Hacks should clearly named and at the bottom +SRC_URI = "http://www.kernel.org/pub/linux/kernel/v2.6/linux-2.6.20.tar.bz2 \ + http://opensource.wolfsonmicro.com/~lg/asoc/asoc-v0.13.patch;patch=1;status=merged \ + ${RPSRC}/hx2750_base-r28.patch;patch=1 \ + ${RPSRC}/hx2750_bl-r8.patch;patch=1 \ + ${RPSRC}/hx2750_pcmcia-r2.patch;patch=1 \ + ${RPSRC}/pxa_keys-r7.patch;patch=1 \ + ${RPSRC}/tsc2101-r16.patch;patch=1 \ + ${RPSRC}/hx2750_test1-r6.patch;patch=1 \ + ${RPSRC}/pxa_timerfix-r0.patch;patch=1 \ + ${RPSRC}/input_power-r6.patch;patch=1 \ + ${RPSRC}/pxa25x_cpufreq-r2.patch;patch=1 \ + ${RPSRC}/sharpsl_pm_fixes1-r0.patch;patch=1 \ + ${RPSRC}/pm_changes-r1.patch;patch=1 \ + ${RPSRC}/usb_add_epalloc-r3.patch;patch=1 \ + ${RPSRC}/usb_pxa27x_udc-r3.patch;patch=1 \ + ${RPSRC}/kexec-arm-r4.patch;patch=1 \ + ${RPSRC}/locomo_kbd_tweak-r1.patch;patch=1 \ + ${RPSRC}/poodle_pm-r3.patch;patch=1 \ + ${RPSRC}/pxa27x_overlay-r5.patch;patch=1 \ + ${RPSRC}/w100_extaccel-r0.patch;patch=1 \ + file://serial-add-support-for-non-standard-xtals-to-16c950-driver.patch;patch=1 \ + ${RPSRC}/logo_oh-r0.patch.bz2;patch=1;status=unmergable \ + ${RPSRC}/logo_oz-r2.patch.bz2;patch=1;status=unmergable \ + ${RPSRC}/pxa-linking-bug.patch;patch=1;status=unmergable \ + ${RPSRC}/mmcsd_large_cards-r0.patch;patch=1;status=hack \ + ${RPSRC}/mmcsd_no_scr_check-r0.patch;patch=1;status=hack \ + ${RPSRC}/integrator_rgb-r1.patch;patch=1;status=hack \ + ${RPSRC}/pxa_cf_initorder_hack-r1.patch;patch=1;status=hack \ + file://pxa-serial-hack.patch;patch=1;status=hack \ + file://connectplus-remove-ide-HACK.patch;patch=1;status=hack \ + file://squashfs3.0-2.6.15.patch;patch=1;status=external \ + file://vesafb-tng-1.0-rc2-2.6.20-rc2.patch;patch=1;status=external \ + file://defconfig-c7x0 \ + file://defconfig-hx2000 \ + file://defconfig-collie \ + file://defconfig-poodle \ + file://defconfig-akita \ + file://defconfig-spitz \ + file://defconfig-qemuarm \ + file://defconfig-qemux86 \ + file://defconfig-tosa " + +# Add this to enable pm debug code (useful with a serial lead) +# ${RPSRC}/sharpsl_pm_debug-r0.patch;patch=1 + +# Disabled until I find the reason this gives issues with cdc_subset +# ${RPSRC}/usb_rndis_tweaks-r0.patch;patch=1 \ + +# Is anything out of this still needed? Parts were commited to mainline by rmk (drivers/mfd/) +# (Pavel Machek's git tree has updated versions of this?) +# ${JLSRC}/zaurus-lcd-2.6.11.diff.gz;patch=1 + +# These patches are extracted from Pavel Machek's git tree +# (diff against vanilla kernel) +SRC_URI_append_collie = "\ + ${DOSRC}/collie/mtd-sharp-flash-hack-r0.patch;patch=1 \ + ${DOSRC}/collie/collie-r0.patch;patch=1 \ + ${DOSRC}/collie/locomolcd-backlight-r0.patch;patch=1 \ + ${DOSRC}/collie/ucb1x00-touch-audio-r0.patch;patch=1 \ + file://collie-mcp-r1.patch;patch=1 \ + ${DOSRC}/collie/sa1100-udc-r0.patch;patch=1 \ +# ${DOSRC}/collie/collie-pm-r1.patch;patch=1 \ +" + +SRC_URI_append_tosa = "\ + ${CHSRC}/usb-ohci-hooks-r1.patch;patch=1 \ + ${CHSRC}/tmio-core-r4.patch;patch=1 \ + file://tmio-tc6393-r8.patch;patch=1 \ + file://tmio-nand-r7.patch;patch=1 \ + file://tmio-ohci-r6.patch;patch=1 \ + ${CHSRC}/tmio-fb-r6.patch;patch=1 \ + file://tosa-keyboard-r18.patch;patch=1 \ + ${DOSRC}/tosa-pxaac97-r6.patch;patch=1 \ + ${DOSRC}/tosa-tmio-r6.patch;patch=1 \ + ${DOSRC}/tosa-power-r17.patch;patch=1 \ + file://tosa-tmio-lcd-r10.patch;patch=1 \ + ${DOSRC}/tosa-bluetooth-r8.patch;patch=1 \ + ${DOSRC}/wm97xx-lg7-r0.patch;patch=1 \ + file://wm9712-suspend-cold-res-r2.patch;patch=1 \ + file://sharpsl-pm-postresume-r1.patch;patch=1 \ + ${DOSRC}/wm97xx-dig-restore-r0.patch;patch=1 \ + ${DOSRC}/wm97xx-miscdevs-resume-r0.patch;patch=1 \ + file://wm9712-reset-loop-r2.patch;patch=1 \ + file://tosa-lcdnoise-r1.patch;patch=1 \ + file://wm97xx-lcdnoise-r0.patch;patch=1 " +# ${DOSRC}/tosa-asoc-r1.patch;patch=1 " + +S = "${WORKDIR}/linux-2.6.20" diff --git a/packages/linux/linux/progear/defconfig b/packages/linux/linux/progear/defconfig index 1dccddab5e..a6257b00f3 100644 --- a/packages/linux/linux/progear/defconfig +++ b/packages/linux/linux/progear/defconfig @@ -1,7 +1,7 @@ # # Automatically generated make config: don't edit -# Linux kernel version: 2.6.19 -# Sat Jan 6 17:55:01 2007 +# Linux kernel version: 2.6.20 +# Tue Feb 6 12:23:13 2007 # CONFIG_X86_32=y CONFIG_GENERIC_TIME=y @@ -12,6 +12,7 @@ CONFIG_X86=y CONFIG_MMU=y CONFIG_GENERIC_ISA_DMA=y CONFIG_GENERIC_IOMAP=y +CONFIG_GENERIC_BUG=y CONFIG_GENERIC_HWEIGHT=y CONFIG_ARCH_MAY_HAVE_PC_FDC=y CONFIG_DMI=y @@ -41,6 +42,7 @@ CONFIG_BSD_PROCESS_ACCT=y CONFIG_AUDIT=y # CONFIG_AUDITSYSCALL is not set # CONFIG_IKCONFIG is not set +CONFIG_SYSFS_DEPRECATED=y # CONFIG_RELAY is not set CONFIG_INITRAMFS_SOURCE="" CONFIG_CC_OPTIMIZE_FOR_SIZE=y @@ -109,6 +111,7 @@ CONFIG_X86_PC=y # CONFIG_X86_VISWS is not set # CONFIG_X86_GENERICARCH is not set # CONFIG_X86_ES7000 is not set +# CONFIG_PARAVIRT is not set # CONFIG_M386 is not set # CONFIG_M486 is not set # CONFIG_M586 is not set @@ -118,6 +121,7 @@ CONFIG_X86_PC=y # CONFIG_MPENTIUMII is not set # CONFIG_MPENTIUMIII is not set # CONFIG_MPENTIUMM is not set +# CONFIG_MCORE2 is not set # CONFIG_MPENTIUM4 is not set # CONFIG_MK6 is not set # CONFIG_MK7 is not set @@ -136,6 +140,8 @@ CONFIG_X86_CMPXCHG=y CONFIG_X86_XADD=y CONFIG_X86_L1_CACHE_SHIFT=5 CONFIG_RWSEM_XCHGADD_ALGORITHM=y +# CONFIG_ARCH_HAS_ILOG2_U32 is not set +# CONFIG_ARCH_HAS_ILOG2_U64 is not set CONFIG_GENERIC_CALIBRATE_DELAY=y CONFIG_X86_WP_WORKS_OK=y CONFIG_X86_INVLPG=y @@ -194,14 +200,16 @@ CONFIG_RESOURCES_64BIT=y # CONFIG_MATH_EMULATION is not set CONFIG_MTRR=y # CONFIG_EFI is not set -CONFIG_REGPARM=y CONFIG_SECCOMP=y # CONFIG_HZ_100 is not set # CONFIG_HZ_250 is not set +# CONFIG_HZ_300 is not set CONFIG_HZ_1000=y CONFIG_HZ=1000 CONFIG_KEXEC=y CONFIG_PHYSICAL_START=0x100000 +# CONFIG_RELOCATABLE is not set +CONFIG_PHYSICAL_ALIGN=0x100000 # CONFIG_COMPAT_VDSO is not set # @@ -375,6 +383,7 @@ CONFIG_INET_XFRM_MODE_BEET=y # CONFIG_TCP_CONG_ADVANCED is not set CONFIG_TCP_CONG_CUBIC=y CONFIG_DEFAULT_TCP_CONG="cubic" +# CONFIG_TCP_MD5SIG is not set # CONFIG_IPV6 is not set # CONFIG_INET6_XFRM_TUNNEL is not set # CONFIG_INET6_TUNNEL is not set @@ -641,6 +650,7 @@ CONFIG_IDEDMA_AUTO=y # # CONFIG_RAID_ATTRS is not set CONFIG_SCSI=y +# CONFIG_SCSI_TGT is not set # CONFIG_SCSI_NETLINK is not set # CONFIG_SCSI_PROC_FS is not set @@ -661,6 +671,7 @@ CONFIG_CHR_DEV_SG=y CONFIG_SCSI_MULTI_LUN=y # CONFIG_SCSI_CONSTANTS is not set # CONFIG_SCSI_LOGGING is not set +# CONFIG_SCSI_SCAN_ASYNC is not set # # SCSI Transports @@ -708,6 +719,7 @@ CONFIG_SCSI_MULTI_LUN=y # CONFIG_SCSI_DC390T is not set # CONFIG_SCSI_NSP32 is not set # CONFIG_SCSI_DEBUG is not set +# CONFIG_SCSI_SRP is not set # # PCMCIA SCSI adapter support @@ -809,6 +821,7 @@ CONFIG_R8169_NAPI=y # CONFIG_IXGB is not set # CONFIG_S2IO is not set # CONFIG_MYRI10GE is not set +# CONFIG_NETXEN_NIC is not set # # Token Ring devices @@ -947,6 +960,7 @@ CONFIG_TOUCHSCREEN_MTOUCH=m # CONFIG_TOUCHSCREEN_PENMOUNT is not set # CONFIG_TOUCHSCREEN_TOUCHRIGHT is not set # CONFIG_TOUCHSCREEN_TOUCHWIN is not set +# CONFIG_TOUCHSCREEN_UCB1400 is not set CONFIG_INPUT_MISC=y CONFIG_INPUT_PCSPKR=y # CONFIG_INPUT_WISTRON_BTNS is not set @@ -976,11 +990,18 @@ CONFIG_HW_CONSOLE=y # # Serial drivers # -# CONFIG_SERIAL_8250 is not set +CONFIG_SERIAL_8250=m +CONFIG_SERIAL_8250_PCI=m +CONFIG_SERIAL_8250_PNP=m +# CONFIG_SERIAL_8250_CS is not set +CONFIG_SERIAL_8250_NR_UARTS=2 +CONFIG_SERIAL_8250_RUNTIME_UARTS=2 +# CONFIG_SERIAL_8250_EXTENDED is not set # # Non-8250 serial port support # +CONFIG_SERIAL_CORE=m # CONFIG_SERIAL_JSM is not set CONFIG_UNIX98_PTYS=y # CONFIG_LEGACY_PTYS is not set @@ -1001,11 +1022,6 @@ CONFIG_RTC=y # CONFIG_R3964 is not set # CONFIG_APPLICOM is not set # CONFIG_SONYPI is not set - -# -# Ftape, the floppy tape device driver -# -# CONFIG_FTAPE is not set CONFIG_AGP=m CONFIG_AGP_ALI=m # CONFIG_AGP_ATI is not set @@ -1146,6 +1162,7 @@ CONFIG_HWMON=m # CONFIG_SENSORS_LM92 is not set # CONFIG_SENSORS_MAX1619 is not set # CONFIG_SENSORS_PC87360 is not set +# CONFIG_SENSORS_PC87427 is not set # CONFIG_SENSORS_SIS5595 is not set # CONFIG_SENSORS_SMSC47M1 is not set # CONFIG_SENSORS_SMSC47M192 is not set @@ -1156,6 +1173,7 @@ CONFIG_HWMON=m # CONFIG_SENSORS_W83781D is not set # CONFIG_SENSORS_W83791D is not set # CONFIG_SENSORS_W83792D is not set +# CONFIG_SENSORS_W83793 is not set # CONFIG_SENSORS_W83L785TS is not set # CONFIG_SENSORS_W83627HF is not set # CONFIG_SENSORS_W83627EHF is not set @@ -1194,10 +1212,6 @@ CONFIG_FB_TILEBLITTING=y # CONFIG_FB_IMSTT is not set # CONFIG_FB_VGA16 is not set CONFIG_FB_VESA=y -# CONFIG_FB_VESA_STD is not set -CONFIG_FB_VESA_TNG=y -CONFIG_FB_VESA_DEFAULT_MODE="1024x768@60" -CONFIG_VIDEO_SELECT=y # CONFIG_FB_HGA is not set # CONFIG_FB_S1D13XXX is not set # CONFIG_FB_NVIDIA is not set @@ -1224,6 +1238,7 @@ CONFIG_VIDEO_SELECT=y # CONFIG_VGA_CONSOLE=y # CONFIG_VGACON_SOFT_SCROLLBACK is not set +CONFIG_VIDEO_SELECT=y CONFIG_DUMMY_CONSOLE=y # CONFIG_FRAMEBUFFER_CONSOLE is not set @@ -1267,7 +1282,6 @@ CONFIG_SND_RTCTIMER=m # CONFIG_SND_MPU401_UART=m CONFIG_SND_AC97_CODEC=m -CONFIG_SND_AC97_BUS=m # CONFIG_SND_DUMMY is not set # CONFIG_SND_MTPAV is not set # CONFIG_SND_SERIAL_U16550 is not set @@ -1351,6 +1365,12 @@ CONFIG_SND_ALI5451=m # Open Sound System # # CONFIG_SOUND_PRIME is not set +CONFIG_AC97_BUS=m + +# +# HID Devices +# +CONFIG_HID=y # # USB support @@ -1412,7 +1432,6 @@ CONFIG_USB_STORAGE=y # USB Input Devices # CONFIG_USB_HID=y -CONFIG_USB_HIDINPUT=y # CONFIG_USB_HIDINPUT_POWERBOOK is not set # CONFIG_HID_FF is not set CONFIG_USB_HIDDEV=y @@ -1574,6 +1593,11 @@ CONFIG_RTC_INTF_DEV=y # # +# Virtualization +# +# CONFIG_KVM is not set + +# # File systems # CONFIG_EXT2_FS=m @@ -1735,6 +1759,11 @@ CONFIG_NLS_ISO8859_2=m CONFIG_NLS_UTF8=y # +# Distributed Lock Manager +# +# CONFIG_DLM is not set + +# # Instrumentation Support # # CONFIG_PROFILING is not set @@ -1748,12 +1777,11 @@ CONFIG_TRACE_IRQFLAGS_SUPPORT=y CONFIG_ENABLE_MUST_CHECK=y CONFIG_MAGIC_SYSRQ=y # CONFIG_UNUSED_SYMBOLS is not set +# CONFIG_DEBUG_FS is not set +# CONFIG_HEADERS_CHECK is not set # CONFIG_DEBUG_KERNEL is not set CONFIG_LOG_BUF_SHIFT=14 # CONFIG_DEBUG_BUGVERBOSE is not set -# CONFIG_DEBUG_FS is not set -# CONFIG_UNWIND_INFO is not set -# CONFIG_HEADERS_CHECK is not set CONFIG_EARLY_PRINTK=y CONFIG_X86_FIND_SMP_CONFIG=y CONFIG_X86_MPPARSE=y @@ -1773,6 +1801,7 @@ CONFIG_CRYPTO_ALGAPI=y CONFIG_CRYPTO_BLKCIPHER=m CONFIG_CRYPTO_MANAGER=m # CONFIG_CRYPTO_HMAC is not set +# CONFIG_CRYPTO_XCBC is not set CONFIG_CRYPTO_NULL=m CONFIG_CRYPTO_MD4=m CONFIG_CRYPTO_MD5=y @@ -1781,8 +1810,10 @@ CONFIG_CRYPTO_SHA256=m CONFIG_CRYPTO_SHA512=m CONFIG_CRYPTO_WP512=m CONFIG_CRYPTO_TGR192=m +# CONFIG_CRYPTO_GF128MUL is not set CONFIG_CRYPTO_ECB=m CONFIG_CRYPTO_CBC=m +# CONFIG_CRYPTO_LRW is not set CONFIG_CRYPTO_DES=m CONFIG_CRYPTO_BLOWFISH=m CONFIG_CRYPTO_TWOFISH=m @@ -1798,7 +1829,6 @@ CONFIG_CRYPTO_ARC4=m CONFIG_CRYPTO_KHAZAD=m CONFIG_CRYPTO_ANUBIS=m CONFIG_CRYPTO_DEFLATE=m -# CONFIG_CRYPTO_LZF is not set CONFIG_CRYPTO_MICHAEL_MIC=m CONFIG_CRYPTO_CRC32C=m CONFIG_CRYPTO_TEST=m @@ -1807,19 +1837,21 @@ CONFIG_CRYPTO_TEST=m # Hardware crypto devices # # CONFIG_CRYPTO_DEV_PADLOCK is not set +CONFIG_CRYPTO_DEV_GEODE=m # # Library routines # +CONFIG_BITREVERSE=y CONFIG_CRC_CCITT=m # CONFIG_CRC16 is not set CONFIG_CRC32=y CONFIG_LIBCRC32C=m CONFIG_AUDIT_GENERIC=y -CONFIG_DYN_PAGEFLAGS=y CONFIG_ZLIB_INFLATE=m CONFIG_ZLIB_DEFLATE=m CONFIG_PLIST=y +CONFIG_IOMAP_COPY=y CONFIG_GENERIC_HARDIRQS=y CONFIG_GENERIC_IRQ_PROBE=y CONFIG_X86_BIOS_REBOOT=y diff --git a/packages/linux/linux/progear/progear_bl-r6.patch b/packages/linux/linux/progear/progear_bl-r6.patch new file mode 100644 index 0000000000..5ef7469ff3 --- /dev/null +++ b/packages/linux/linux/progear/progear_bl-r6.patch @@ -0,0 +1,204 @@ +From: Marcin Juszkiewicz <openembedded@hrw.one.pl> + +Add control of LCD backlight for Frontpath ProGear HX1050+. +Patch is based on http://downloads.sf.net/progear/progear-lcd-0.2.tar.gz +driver by M Schacht. + +Signed-Off-By: Marcin Juszkiewicz <openembedded@hrw.one.pl> + +--- +Patch follow kernel version 2.6.19-rc6 + + Kconfig | 8 +++ + Makefile | 1 + progear_bl.c | 155 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ + 3 files changed, 164 insertions(+) + + +Index: linux-2.6.20/drivers/video/backlight/Kconfig +=================================================================== +--- linux-2.6.20.orig/drivers/video/backlight/Kconfig 2007-02-04 19:44:54.000000000 +0100 ++++ linux-2.6.20/drivers/video/backlight/Kconfig 2007-02-05 16:13:13.000000000 +0100 +@@ -66,3 +66,11 @@ + If you have a HP Jornada 680, say y to enable the + backlight driver. + ++config BACKLIGHT_PROGEAR ++ tristate "Frontpath ProGear Backlight Driver" ++ depends on BACKLIGHT_DEVICE && PCI ++ default y ++ help ++ If you have a Frontpath ProGear say Y to enable the ++ backlight driver. ++ +Index: linux-2.6.20/drivers/video/backlight/progear_bl.c +=================================================================== +--- /dev/null 1970-01-01 00:00:00.000000000 +0000 ++++ linux-2.6.20/drivers/video/backlight/progear_bl.c 2007-02-05 16:29:14.000000000 +0100 +@@ -0,0 +1,157 @@ ++/* ++ * Backlight Driver for Frontpath ProGear HX1050+ ++ * ++ * Copyright (c) 2006 Marcin Juszkiewicz ++ * ++ * Based on Progear LCD driver by M Schacht ++ * <mschacht at alumni dot washington dot edu> ++ * ++ * Based on Sharp's Corgi Backlight Driver ++ * Based on Backlight Driver for HP Jornada 680 ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ * ++ */ ++ ++#include <linux/module.h> ++#include <linux/kernel.h> ++#include <linux/init.h> ++#include <linux/platform_device.h> ++#include <linux/mutex.h> ++#include <linux/fb.h> ++#include <linux/backlight.h> ++#include <linux/pci.h> ++#include <asm/uaccess.h> ++ ++#define PMU_LPCR 0xB0 ++#define SB_MPS1 0x61 ++#define HW_LEVEL_MAX 0x77 ++#define HW_LEVEL_MIN 0x4f ++ ++static int progearbl_intensity; ++static struct backlight_properties progearbl_data; ++static struct backlight_device *progear_backlight_device; ++ ++static struct pci_dev *pmu_dev = NULL; ++static struct pci_dev *sb_dev = NULL; ++ ++static int progearbl_send_intensity(struct backlight_device *bd) ++{ ++ int intensity = bd->props->brightness; ++ ++ if (bd->props->power != FB_BLANK_UNBLANK) ++ intensity = 0; ++ if (bd->props->fb_blank != FB_BLANK_UNBLANK) ++ intensity = 0; ++ ++ pci_write_config_byte(pmu_dev, PMU_LPCR, intensity + HW_LEVEL_MIN); ++ ++ progearbl_intensity = intensity; ++ ++ return 0; ++} ++ ++static int progearbl_get_intensity(struct backlight_device *bd) ++{ ++ return progearbl_intensity; ++} ++ ++static int progearbl_set_intensity(struct backlight_device *bd) ++{ ++ progearbl_send_intensity(progear_backlight_device); ++ ++ return 0; ++} ++ ++static struct backlight_properties progearbl_data = { ++ .owner = THIS_MODULE, ++ .get_brightness = progearbl_get_intensity, ++ .update_status = progearbl_set_intensity, ++}; ++ ++static int progearbl_probe(struct platform_device *pdev) ++{ ++ u8 temp; ++ ++ pmu_dev = pci_get_device(PCI_VENDOR_ID_AL, PCI_DEVICE_ID_AL_M7101, 0); ++ if (!pmu_dev) { ++ printk("ALI M7101 PMU not found.\n"); ++ return -ENODEV; ++ } ++ ++ sb_dev = pci_get_device(PCI_VENDOR_ID_AL, PCI_DEVICE_ID_AL_M1533, 0); ++ if (!sb_dev) { ++ printk("ALI 1533 SB not found.\n"); ++ pci_dev_put(pmu_dev); ++ return -ENODEV; ++ } ++ ++ /* Set SB_MPS1 to enable brightness control. */ ++ pci_read_config_byte(sb_dev, SB_MPS1, &temp); ++ pci_write_config_byte(sb_dev, SB_MPS1, temp | 0x20); ++ ++ progear_backlight_device = backlight_device_register("progear-bl", ++ &pdev->dev, NULL, &progearbl_data); ++ if (IS_ERR(progear_backlight_device)) ++ return PTR_ERR(progear_backlight_device); ++ ++ progearbl_data.power = FB_BLANK_UNBLANK; ++ progearbl_data.brightness = HW_LEVEL_MAX - HW_LEVEL_MIN; ++ progearbl_data.max_brightness = HW_LEVEL_MAX - HW_LEVEL_MIN; ++ progearbl_send_intensity(progear_backlight_device); ++ ++ return 0; ++} ++ ++static int progearbl_remove(struct platform_device *dev) ++{ ++ backlight_device_unregister(progear_backlight_device); ++ ++ return 0; ++} ++ ++static struct platform_driver progearbl_driver = { ++ .probe = progearbl_probe, ++ .remove = progearbl_remove, ++ .driver = { ++ .name = "progear-bl", ++ }, ++}; ++ ++static struct platform_device *progearbl_device; ++ ++static int __init progearbl_init(void) ++{ ++ int ret = platform_driver_register(&progearbl_driver); ++ if (!ret) { ++ progearbl_device = platform_device_alloc("progear-bl", -1); ++ if (!progearbl_device) ++ return -ENOMEM; ++ ++ ret = platform_device_add(progearbl_device); ++ ++ if (ret) { ++ platform_device_put(progearbl_device); ++ platform_driver_unregister(&progearbl_driver); ++ } ++ } ++ return ret; ++} ++ ++static void __exit progearbl_exit(void) ++{ ++ pci_dev_put(pmu_dev); ++ pci_dev_put(sb_dev); ++ ++ platform_device_unregister(progearbl_device); ++ platform_driver_unregister(&progearbl_driver); ++} ++ ++module_init(progearbl_init); ++module_exit(progearbl_exit); ++ ++MODULE_AUTHOR("Marcin Juszkiewicz <linux@hrw.one.pl>"); ++MODULE_DESCRIPTION("ProGear Backlight Driver"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6.20/drivers/video/backlight/Makefile +=================================================================== +--- linux-2.6.20.orig/drivers/video/backlight/Makefile 2007-02-04 19:44:54.000000000 +0100 ++++ linux-2.6.20/drivers/video/backlight/Makefile 2007-02-05 16:13:13.000000000 +0100 +@@ -5,3 +5,4 @@ + obj-$(CONFIG_BACKLIGHT_CORGI) += corgi_bl.o + obj-$(CONFIG_BACKLIGHT_HP680) += hp680_bl.o + obj-$(CONFIG_BACKLIGHT_LOCOMO) += locomolcd.o ++obj-$(CONFIG_BACKLIGHT_PROGEAR) += progear_bl.o diff --git a/packages/linux/linux_2.6.19.bb b/packages/linux/linux_2.6.20.bb index 8e9346c473..ebf520d3d6 100644 --- a/packages/linux/linux_2.6.19.bb +++ b/packages/linux/linux_2.6.20.bb @@ -2,15 +2,13 @@ DESCRIPTION = "Linux Kernel" SECTION = "kernel" LICENSE = "GPL" -SRC_URI = "${KERNELORG_MIRROR}/pub/linux/kernel/v2.6/linux-2.6.19.tar.bz2 \ +SRC_URI = "${KERNELORG_MIRROR}/pub/linux/kernel/v2.6/linux-${PV}.tar.bz2 \ file://defconfig" SRC_URI_append_progear = " \ - file://progear_bl-r5.patch;patch=1 \ + file://progear_bl-r6.patch;patch=1 \ " -S = "${WORKDIR}/linux-2.6.19" - inherit kernel KERNEL_IMAGETYPE = "bzImage" diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/.mtn2git_empty b/packages/linux/logicpd-pxa270-2.6.19.2/.mtn2git_empty new file mode 100644 index 0000000000..e69de29bb2 --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/.mtn2git_empty diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/asoc-v0.12.4.patch b/packages/linux/logicpd-pxa270-2.6.19.2/asoc-v0.12.4.patch new file mode 100644 index 0000000000..420f4aa2e2 --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/asoc-v0.12.4.patch @@ -0,0 +1,31712 @@ +Index: linux-2.6-pxa-new/Documentation/sound/alsa/soc/DAI.txt +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/Documentation/sound/alsa/soc/DAI.txt +@@ -0,0 +1,546 @@ ++ASoC currently supports the three main Digital Audio Interfaces (DAI) found on ++SoC controllers and portable audio CODECS today, namely AC97, I2S and PCM. ++ ++ ++AC97 ++==== ++ ++ AC97 is a five wire interface commonly found on many PC sound cards. It is ++now also popular in many portable devices. This DAI has a reset line and time ++multiplexes its data on its SDATA_OUT (playback) and SDATA_IN (capture) lines. ++The bit clock (BCLK) is always driven by the CODEC (usually 12.288MHz) and the ++frame (FRAME) (usually 48kHz) is always driven by the controller. Each AC97 ++frame is 21uS long and is divided into 13 time slots. ++ ++The AC97 specification can be found at :- ++http://www.intel.com/design/chipsets/audio/ac97_r23.pdf ++ ++ ++I2S ++=== ++ ++ I2S is a common 4 wire DAI used in HiFi, STB and portable devices. The Tx and ++Rx lines are used for audio transmision, whilst the bit clock (BCLK) and ++left/right clock (LRC) synchronise the link. I2S is flexible in that either the ++controller or CODEC can drive (master) the BCLK and LRC clock lines. Bit clock ++usually varies depending on the sample rate and the master system clock ++(SYSCLK). LRCLK is the same as the sample rate. A few devices support separate ++ADC and DAC LRCLK's, this allows for similtanious capture and playback at ++different sample rates. ++ ++I2S has several different operating modes:- ++ ++ o I2S - MSB is transmitted on the falling edge of the first BCLK after LRC ++ transition. ++ ++ o Left Justified - MSB is transmitted on transition of LRC. ++ ++ o Right Justified - MSB is transmitted sample size BCLK's before LRC ++ transition. ++ ++PCM ++=== ++ ++PCM is another 4 wire interface, very similar to I2S, that can support a more ++flexible protocol. It has bit clock (BCLK) and sync (SYNC) lines that are used ++to synchronise the link whilst the Tx and Rx lines are used to transmit and ++receive the audio data. Bit clock usually varies depending on sample rate ++whilst sync runs at the sample rate. PCM also supports Time Division ++Multiplexing (TDM) in that several devices can use the bus similtaniuosly (This ++is sometimes referred to as network mode). ++ ++Common PCM operating modes:- ++ ++ o Mode A - MSB is transmitted on falling edge of first BCLK after FRAME/SYNC. ++ ++ o Mode B - MSB is transmitted on rising edge of FRAME/SYNC. ++ ++ ++ASoC DAI Configuration ++====================== ++ ++Every CODEC DAI and SoC DAI must have their capabilities defined in order to ++be configured together at runtime when the audio and clocking parameters are ++known. This is achieved by creating an array of struct snd_soc_hw_mode in the ++the CODEC and SoC interface drivers. Each element in the array describes a DAI ++mode and each mode is usually based upon the DAI system clock to sample rate ++ratio (FS). ++ ++i.e. 48k sample rate @ 256 FS = sytem clock of 12.288 MHz ++ 48000 * 256 = 12288000 ++ ++The CPU and Codec DAI modes are then ANDed together at runtime to determine the ++rutime DAI configuration for both the Codec and CPU. ++ ++When creating a new codec or SoC DAI it's probably best to start of with a few ++sample rates first and then test your interface. ++ ++struct snd_soc_dai_mode is defined (in soc.h) as:- ++ ++/* SoC DAI mode */ ++struct snd_soc_dai_mode { ++ u16 fmt; /* SND_SOC_DAIFMT_* */ ++ u16 tdm; /* SND_SOC_HWTDM_* */ ++ u64 pcmfmt; /* SNDRV_PCM_FMTBIT_* */ ++ u16 pcmrate; /* SND_SOC_HWRATE_* */ ++ u16 pcmdir:2; /* SND_SOC_HWDIR_* */ ++ u16 flags:8; /* hw flags */ ++ u16 fs; /* mclk to rate divider */ ++ u64 bfs; /* mclk to bclk dividers */ ++ unsigned long priv; /* private mode data */ ++}; ++ ++fmt: ++---- ++This field defines the DAI mode hardware format (e.g. I2S settings) and ++supports the following settings:- ++ ++ 1) hardware DAI formats ++ ++#define SND_SOC_DAIFMT_I2S (1 << 0) /* I2S mode */ ++#define SND_SOC_DAIFMT_RIGHT_J (1 << 1) /* Right justified mode */ ++#define SND_SOC_DAIFMT_LEFT_J (1 << 2) /* Left Justified mode */ ++#define SND_SOC_DAIFMT_DSP_A (1 << 3) /* L data msb after FRM */ ++#define SND_SOC_DAIFMT_DSP_B (1 << 4) /* L data msb during FRM */ ++#define SND_SOC_DAIFMT_AC97 (1 << 5) /* AC97 */ ++ ++ 2) hw DAI signal inversions ++ ++#define SND_SOC_DAIFMT_NB_NF (1 << 8) /* normal bit clock + frame */ ++#define SND_SOC_DAIFMT_NB_IF (1 << 9) /* normal bclk + inv frm */ ++#define SND_SOC_DAIFMT_IB_NF (1 << 10) /* invert bclk + nor frm */ ++#define SND_SOC_DAIFMT_IB_IF (1 << 11) /* invert bclk + frm */ ++ ++ 3) hw clock masters ++ This is wrt the codec, the inverse is true for the interface ++ i.e. if the codec is clk and frm master then the interface is ++ clk and frame slave. ++ ++#define SND_SOC_DAIFMT_CBM_CFM (1 << 12) /* codec clk & frm master */ ++#define SND_SOC_DAIFMT_CBS_CFM (1 << 13) /* codec clk slave & frm master */ ++#define SND_SOC_DAIFMT_CBM_CFS (1 << 14) /* codec clk master & frame slave */ ++#define SND_SOC_DAIFMT_CBS_CFS (1 << 15) /* codec clk & frm slave */ ++ ++At least one option from each section must be selected. Multiple selections are ++also supported e.g. ++ ++ .fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_LEFT_J | SND_SOC_DAIFMT_RIGHT_J | \ ++ SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_NB_IF | SND_SOC_DAIFMT_IB_NF | \ ++ SND_SOC_DAIFMT_IB_IF ++ ++ ++tdm: ++------ ++This field defines the Time Division Multiplexing left and right word ++positions for the DAI mode if applicable. Set to SND_SOC_DAITDM_LRDW(0,0) for ++no TDM. ++ ++ ++pcmfmt: ++--------- ++The hardware PCM format. This describes the PCM formats supported by the DAI ++mode e.g. ++ ++ .pcmfmt = SNDRV_PCM_FORMAT_S16_LE | SNDRV_PCM_FORMAT_S20_3LE | \ ++ SNDRV_PCM_FORMAT_S24_3LE ++ ++pcmrate: ++---------- ++The PCM sample rates supported by the DAI mode. e.g. ++ ++ .pcmrate = SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000 ++ ++ ++pcmdir: ++--------- ++The stream directions supported by this mode. e.g. playback and capture ++ ++ ++flags: ++-------- ++The DAI hardware flags supported by the mode. ++ ++/* use bfs mclk divider mode (BCLK = MCLK / x) */ ++#define SND_SOC_DAI_BFS_DIV 0x1 ++/* use bfs rate mulitplier (BCLK = RATE * x)*/ ++#define SND_SOC_DAI_BFS_RATE 0x2 ++/* use bfs rcw multiplier (BCLK = RATE * CHN * WORD SIZE) */ ++#define SND_SOC_DAI_BFS_RCW 0x4 ++/* capture and playback can use different clocks */ ++#define SND_SOC_DAI_ASYNC 0x8 ++ ++NOTE: Bitclock division and mulitiplication modes can be safely matched by the ++core logic. ++ ++ ++fs: ++----- ++The FS supported by this DAI mode FS is the ratio between the system clock and ++the sample rate. See above ++ ++bfs: ++------ ++BFS is the ratio of BCLK to MCLK or the ratio of BCLK to sample rate (this ++depends on the codec or CPU DAI). ++ ++The BFS supported by the DAI mode. This can either be the ratio between the ++bitclock (BCLK) and the sample rate OR the ratio between the system clock and ++the sample rate. Depends on the flags above. ++ ++priv: ++----- ++private codec mode data. ++ ++ ++ ++Examples ++======== ++ ++Note that Codec DAI and CPU DAI examples are interchangeable in these examples ++as long as the bus master is reversed. i.e. ++ ++ SND_SOC_DAIFMT_CBM_CFM would become SND_SOC_DAIFMT_CBS_CFS ++ and vice versa. ++ ++This applies to all SND_SOC_DAIFMT_CB*_CF*. ++ ++Example 1 ++--------- ++ ++Simple codec that only runs at 8k & 48k @ 256FS in master mode, can generate a ++BCLK of either MCLK/2 or MCLK/4. ++ ++ /* codec master */ ++ { ++ .fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FORMAT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_48000, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = SND_SOC_FSBD(2) | SND_SOC_FSBD(4), ++ } ++ ++ ++Example 2 ++--------- ++Simple codec that only runs at 8k & 48k @ 256FS in master mode, can generate a ++BCLK of either Rate * 32 or Rate * 64. ++ ++ /* codec master */ ++ { ++ .fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FORMAT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_48000, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 256, ++ .bfs = 32, ++ }, ++ { ++ .fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FORMAT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_48000, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 256, ++ .bfs = 64, ++ }, ++ ++ ++Example 3 ++--------- ++Codec that runs at 8k & 48k @ 256FS in master mode, can generate a BCLK that ++is a multiple of Rate * channels * word size. (RCW) i.e. ++ ++ BCLK = 8000 * 2 * 16 (8k, stereo, 16bit) ++ = 256kHz ++ ++This codecs supports a RCW multiple of 1,2 ++ ++ { ++ .fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FORMAT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_48000, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE, ++ .flags = SND_SOC_DAI_BFS_RCW, ++ .fs = 256, ++ .bfs = SND_SOC_FSBW(1) | SND_SOC_FSBW(2), ++ } ++ ++ ++Example 4 ++--------- ++Codec that only runs at 8k & 48k @ 256FS in master mode, can generate a ++BCLK of either Rate * 32 or Rate * 64. Codec can also run in slave mode as long ++as BCLK is rate * 32 or rate * 64. ++ ++ /* codec master */ ++ { ++ .fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FORMAT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_48000, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 256, ++ .bfs = 32, ++ }, ++ { ++ .fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FORMAT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_48000, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 256, ++ .bfs = 64, ++ }, ++ ++ /* codec slave */ ++ { ++ .fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = SNDRV_PCM_FORMAT_S16_LE, ++ .pcmdir = SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_48000, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = 32, ++ }, ++ { ++ .fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = SNDRV_PCM_FORMAT_S16_LE, ++ .pcmdir = SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_48000, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = 64, ++ }, ++ ++ ++Example 5 ++--------- ++Codec that only runs at 8k, 16k, 32k, 48k, 96k @ 128FS, 192FS & 256FS in master ++mode and can generate a BCLK of MCLK / (1,2,4,8,16). Codec can also run in slave ++mode as and does not care about FS or BCLK (as long as there is enough bandwidth). ++ ++ #define CODEC_FSB \ ++ (SND_SOC_FSBD(1) | SND_SOC_FSBD(2) | SND_SOC_FSBD(4) | \ ++ SND_SOC_FSBD(8) | SND_SOC_FSBD(16)) ++ ++ #define CODEC_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_16000 | SNDRV_PCM_RATE_32000 |\ ++ SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_96000) ++ ++ /* codec master @ 128, 192 & 256 FS */ ++ { ++ .fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FORMAT_S16_LE, ++ .pcmrate = CODEC_RATES, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 128, ++ .bfs = CODEC_FSB, ++ }, ++ ++ { ++ .fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FORMAT_S16_LE, ++ .pcmrate = CODEC_RATES, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 192, ++ .bfs = CODEC_FSB ++ }, ++ ++ { ++ .fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FORMAT_S16_LE, ++ .pcmrate = CODEC_RATES, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = CODEC_FSB, ++ }, ++ ++ /* codec slave */ ++ { ++ .fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = SNDRV_PCM_FORMAT_S16_LE, ++ .pcmrate = CODEC_RATES, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = SND_SOC_FSB_ALL, ++ }, ++ ++ ++Example 6 ++--------- ++Codec that only runs at 8k, 44.1k, 48k @ different FS in master mode (for use ++with a fixed MCLK) and can generate a BCLK of MCLK / (1,2,4,8,16). ++Codec can also run in slave mode as and does not care about FS or BCLK (as long ++as there is enough bandwidth). Codec can support 16, 24 and 32 bit PCM sample ++sizes. ++ ++ #define CODEC_FSB \ ++ (SND_SOC_FSBD(1) | SND_SOC_FSBD(2) | SND_SOC_FSBD(4) | \ ++ SND_SOC_FSBD(8) | SND_SOC_FSBD(16)) ++ ++ #define CODEC_PCM_FORMATS \ ++ (SNDRV_PCM_FORMAT_S16_LE | SNDRV_PCM_FORMAT_S20_3LE | \ ++ SNDRV_PCM_FORMAT_S24_3LE | SNDRV_PCM_FORMAT_S24_LE | SNDRV_PCM_FORMAT_S32_LE) ++ ++ /* codec master */ ++ { ++ .fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FORMAT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1536, ++ .bfs = CODEC_FSB, ++ }, ++ ++ { ++ .fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FORMAT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_44100, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 272, ++ .bfs = CODEC_FSB, ++ }, ++ ++ { ++ .fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FORMAT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_48000, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = CODEC_FSB, ++ }, ++ ++ /* codec slave */ ++ { ++ .fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = SNDRV_PCM_FORMAT_S16_LE, ++ .pcmrate = CODEC_RATES, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = SND_SOC_FSB_ALL, ++ }, ++ ++ ++Example 7 ++--------- ++AC97 Codec that does not support VRA (i.e only runs at 48k). ++ ++ #define AC97_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++ #define AC97_PCM_FORMATS \ ++ (SNDRV_PCM_FORMAT_S16_LE | SNDRV_PCM_FORMAT_S18_3LE | \ ++ SNDRV_PCM_FORMAT_S20_3LE) ++ ++ /* AC97 with no VRA */ ++ { ++ .pcmfmt = AC97_PCM_FORMATS, ++ .pcmrate = SNDRV_PCM_RATE_48000, ++ } ++ ++ ++Example 8 ++--------- ++ ++CPU DAI that supports 8k - 48k @ 256FS and BCLK = MCLK / 4 in master mode. ++Slave mode (CPU DAI is FRAME master) supports 8k - 96k at any FS as long as ++BCLK = 64 * rate. (Intel XScale I2S controller). ++ ++ #define PXA_I2S_DAIFMT \ ++ (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_LEFT_J | SND_SOC_DAIFMT_NB_NF) ++ ++ #define PXA_I2S_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++ #define PXA_I2S_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000) ++ ++ /* priv is divider */ ++ static struct snd_soc_dai_mode pxa2xx_i2s_modes[] = { ++ /* pxa2xx I2S frame and clock master modes */ ++ { ++ .fmt = PXA_I2S_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = PXA_I2S_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = SND_SOC_FSBD(4), ++ .priv = 0x48, ++ }, ++ { ++ .fmt = PXA_I2S_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_11025, ++ .pcmdir = PXA_I2S_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = SND_SOC_FSBD(4), ++ .priv = 0x34, ++ }, ++ { ++ .fmt = PXA_I2S_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_16000, ++ .pcmdir = PXA_I2S_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = SND_SOC_FSBD(4), ++ .priv = 0x24, ++ }, ++ { ++ .fmt = PXA_I2S_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_22050, ++ .pcmdir = PXA_I2S_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = SND_SOC_FSBD(4), ++ .priv = 0x1a, ++ }, ++ { ++ .fmt = PXA_I2S_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_44100, ++ .pcmdir = PXA_I2S_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = SND_SOC_FSBD(4), ++ .priv = 0xd, ++ }, ++ { ++ .fmt = PXA_I2S_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_48000, ++ .pcmdir = PXA_I2S_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = SND_SOC_FSBD(4), ++ .priv = 0xc, ++ }, ++ ++ /* pxa2xx I2S frame master and clock slave mode */ ++ { ++ .fmt = PXA_I2S_DAIFMT | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = PXA_I2S_RATES, ++ .pcmdir = PXA_I2S_DIR, ++ .fs = SND_SOC_FS_ALL, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .bfs = 64, ++ .priv = 0x48, ++ }, ++}; +Index: linux-2.6-pxa-new/Documentation/sound/alsa/soc/clocking.txt +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/Documentation/sound/alsa/soc/clocking.txt +@@ -0,0 +1,314 @@ ++Audio Clocking ++============== ++ ++This text describes the audio clocking terms in ASoC and digital audio in ++general. Note: Audio clocking can be complex ! ++ ++ ++Master Clock ++------------ ++ ++Every audio subsystem is driven by a master clock (sometimes refered to as MCLK ++or SYSCLK). This audio master clock can be derived from a number of sources ++(e.g. crystal, PLL, CPU clock) and is responsible for producing the correct ++audio playback and capture sample rates. ++ ++Some master clocks (e.g. PLL's and CPU based clocks) are configuarble in that ++their speed can be altered by software (depending on the system use and to save ++power). Other master clocks are fixed at at set frequency (i.e. crystals). ++ ++ ++DAI Clocks ++---------- ++The Digital Audio Interface is usually driven by a Bit Clock (often referred to ++as BCLK). This clock is used to drive the digital audio data across the link ++between the codec and CPU. ++ ++The DAI also has a frame clock to signal the start of each audio frame. This ++clock is sometimes referred to as LRC (left right clock) or FRAME. This clock ++runs at exactly the sample rate (LRC = Rate). ++ ++Bit Clock can be generated as follows:- ++ ++BCLK = MCLK / x ++ ++ or ++ ++BCLK = LRC * x ++ ++ or ++ ++BCLK = LRC * Channels * Word Size ++ ++This relationship depends on the codec or SoC CPU in particular. ASoC can quite ++easily match BCLK generated by division (SND_SOC_DAI_BFS_DIV) with BCLK by ++multiplication (SND_SOC_DAI_BFS_RATE) or BCLK generated by ++Rate * Channels * Word size (RCW or SND_SOC_DAI_BFS_RCW). ++ ++ ++ASoC Clocking ++------------- ++ ++The ASoC core determines the clocking for each particular configuration at ++runtime. This is to allow for dynamic audio clocking wereby the audio clock is ++variable and depends on the system state or device usage scenario. i.e. a voice ++call requires slower clocks (and hence less power) than MP3 playback. ++ ++ASoC will call the config_sysclock() function for the target machine during the ++audio parameters configuration. The function is responsible for then clocking ++the machine audio subsytem and returning the audio clock speed to the core. ++This function should also call the codec and cpu DAI clock_config() functions ++to configure their respective internal clocking if required. ++ ++ ++ASoC Clocking Control Flow ++-------------------------- ++ ++The ASoC core will call the machine drivers config_sysclock() when most of the ++DAI capabilities are known. The machine driver is then responsible for calling ++the codec and/or CPU DAI drivers with the selected capabilities and the current ++MCLK. Note that the machine driver is also resonsible for setting the MCLK (and ++enabling it). ++ ++ (1) Match Codec and CPU DAI capabilities. At this point we have ++ matched the majority of the DAI fields and now need to make sure this ++ mode is currently clockable. ++ ++ (2) machine->config_sysclk() is now called with the matched DAI FS, sample ++ rate and BCLK master. This function then gets/sets the current audio ++ clock (depening on usage) and calls the codec and CPUI DAI drivers with ++ the FS, rate, BCLK master and MCLK. ++ ++ (3) Codec/CPU DAI config_sysclock(). This function checks that the FS, rate, ++ BCLK master and MCLK are acceptable for the codec or CPU DAI. It also ++ sets the DAI internal state to work with said clocks. ++ ++The config_sysclk() functions for CPU, codec and machine should return the MCLK ++on success and 0 on failure. ++ ++ ++Examples (b = BCLK, l = LRC) ++============================ ++ ++Example 1 ++--------- ++ ++Simple codec that only runs at 48k @ 256FS in master mode. ++ ++CPU only runs as slave DAI, however it generates a variable MCLK. ++ ++ -------- --------- ++ | | <----mclk--- | | ++ | Codec |b -----------> | CPU | ++ | |l -----------> | | ++ | | | | ++ -------- --------- ++ ++The codec driver has the following config_sysclock() ++ ++ static unsigned int config_sysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++ { ++ /* make sure clock is 256 * rate */ ++ if(info->rate << 8 == clk) { ++ dai->mclk = clk; ++ return clk; ++ } ++ ++ return 0; ++ } ++ ++The CPU I2S DAI driver has the following config_sysclk() ++ ++ static unsigned int config_sysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++ { ++ /* can we support this clk */ ++ if(set_audio_clk(clk) < 0) ++ return -EINVAL; ++ ++ dai->mclk = clk; ++ return dai->clk; ++ } ++ ++The machine driver config_sysclk() in this example is as follows:- ++ ++ unsigned int machine_config_sysclk(struct snd_soc_pcm_runtime *rtd, ++ struct snd_soc_clock_info *info) ++ { ++ int clk = info->rate * info->fs; ++ ++ /* check that CPU can deliver clock */ ++ if(rtd->cpu_dai->config_sysclk(rtd->cpu_dai, info, clk) < 0) ++ return -EINVAL; ++ ++ /* can codec work with this clock */ ++ return rtd->codec_dai->config_sysclk(rtd->codec_dai, info, clk); ++ } ++ ++ ++Example 2 ++--------- ++ ++Codec that can master at 8k and 48k at various FS (and hence supports a fixed ++set of input MCLK's) and can also be slave at various FS . ++ ++The CPU can master at 8k and 48k @256 FS and can be slave at any FS. ++ ++MCLK is a 12.288MHz crystal on this machine. ++ ++ -------- --------- ++ | | <---xtal---> | | ++ | Codec |b <----------> | CPU | ++ | |l <----------> | | ++ | | | | ++ -------- --------- ++ ++ ++The codec driver has the following config_sysclock() ++ ++ /* supported input clocks */ ++ const static int hifi_clks[] = {11289600, 12000000, 12288000, ++ 16934400, 18432000}; ++ ++ static unsigned int config_hsysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++ { ++ int i; ++ ++ /* is clk supported */ ++ for(i = 0; i < ARRAY_SIZE(hifi_clks); i++) { ++ if(clk == hifi_clks[i]) { ++ dai->mclk = clk; ++ return clk; ++ } ++ } ++ ++ /* this clk is not supported */ ++ return 0; ++ } ++ ++The CPU I2S DAI driver has the following config_sysclk() ++ ++ static unsigned int config_sysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++ { ++ /* are we master or slave */ ++ if (info->bclk_master & ++ (SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS)) { ++ ++ /* we can only master @ 256FS */ ++ if(info->rate << 8 == clk) { ++ dai->mclk = clk; ++ return dai->mclk; ++ } ++ } else { ++ /* slave we can run at any FS */ ++ dai->mclk = clk; ++ return dai->mclk; ++ } ++ ++ /* not supported */ ++ return dai->clk; ++ } ++ ++The machine driver config_sysclk() in this example is as follows:- ++ ++ unsigned int machine_config_sysclk(struct snd_soc_pcm_runtime *rtd, ++ struct snd_soc_clock_info *info) ++ { ++ int clk = 12288000; /* 12.288MHz */ ++ ++ /* who's driving the link */ ++ if (info->bclk_master & ++ (SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS)) { ++ /* codec master */ ++ ++ /* check that CPU can work with clock */ ++ if(rtd->cpu_dai->config_sysclk(rtd->cpu_dai, info, clk) < 0) ++ return -EINVAL; ++ ++ /* can codec work with this clock */ ++ return rtd->codec_dai->config_sysclk(rtd->codec_dai, info, clk); ++ } else { ++ /* cpu master */ ++ ++ /* check that codec can work with clock */ ++ if(rtd->codec_dai->config_sysclk(rtd->codec_dai, info, clk) < 0) ++ return -EINVAL; ++ ++ /* can CPU work with this clock */ ++ return rtd->cpu_dai->config_sysclk(rtd->cpu_dai, info, clk); ++ } ++ } ++ ++ ++ ++Example 3 ++--------- ++ ++Codec that masters at 8k ... 48k @256 FS. Codec can also be slave and ++doesn't care about FS. The codec has an internal PLL and dividers to generate ++the necessary internal clocks (for 256FS). ++ ++CPU can only be slave and doesn't care about FS. ++ ++MCLK is a non controllable 13MHz clock from the CPU. ++ ++ ++ -------- --------- ++ | | <----mclk--- | | ++ | Codec |b <----------> | CPU | ++ | |l <----------> | | ++ | | | | ++ -------- --------- ++ ++The codec driver has the following config_sysclock() ++ ++ /* valid PCM clock dividers * 2 */ ++ static int pcm_divs[] = {2, 6, 11, 4, 8, 12, 16}; ++ ++ static unsigned int config_vsysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++ { ++ int i, j, best_clk = info->fs * info->rate; ++ ++ /* can we run at this clk without the PLL ? */ ++ for (i = 0; i < ARRAY_SIZE(pcm_divs); i++) { ++ if ((best_clk >> 1) * pcm_divs[i] == clk) { ++ dai->pll_in = 0; ++ dai->clk_div = pcm_divs[i]; ++ dai->mclk = best_clk; ++ return dai->mclk; ++ } ++ } ++ ++ /* now check for PLL support */ ++ for (i = 0; i < ARRAY_SIZE(pll_div); i++) { ++ if (pll_div[i].pll_in == clk) { ++ for (j = 0; j < ARRAY_SIZE(pcm_divs); j++) { ++ if (pll_div[i].pll_out == pcm_divs[j] * (best_clk >> 1)) { ++ dai->pll_in = clk; ++ dai->pll_out = pll_div[i].pll_out; ++ dai->clk_div = pcm_divs[j]; ++ dai->mclk = best_clk; ++ return dai->mclk; ++ } ++ } ++ } ++ } ++ ++ /* this clk is not supported */ ++ return 0; ++ } ++ ++ ++The CPU I2S DAI driver has the does not need a config_sysclk() as it can slave ++at any FS. ++ ++ unsigned int config_sysclk(struct snd_soc_pcm_runtime *rtd, ++ struct snd_soc_clock_info *info) ++ { ++ /* codec has pll that generates mclk from 13MHz xtal */ ++ return rtd->codec_dai->config_sysclk(rtd->codec_dai, info, 13000000); ++ } +Index: linux-2.6-pxa-new/Documentation/sound/alsa/soc/codec.txt +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/Documentation/sound/alsa/soc/codec.txt +@@ -0,0 +1,232 @@ ++ASoC Codec Driver ++================= ++ ++The codec driver is generic and hardware independent code that configures the ++codec to provide audio capture and playback. It should contain no code that is ++specific to the target platform or machine. All platform and machine specific ++code should be added to the platform and machine drivers respectively. ++ ++Each codec driver must provide the following features:- ++ ++ 1) Digital audio interface (DAI) description ++ 2) Digital audio interface configuration ++ 3) PCM's description ++ 4) Codec control IO - using I2C, 3 Wire(SPI) or both API's ++ 5) Mixers and audio controls ++ 6) Sysclk configuration ++ 7) Codec audio operations ++ ++Optionally, codec drivers can also provide:- ++ ++ 8) DAPM description. ++ 9) DAPM event handler. ++10) DAC Digital mute control. ++ ++It's probably best to use this guide in conjuction with the existing codec ++driver code in sound/soc/codecs/ ++ ++ASoC Codec driver breakdown ++=========================== ++ ++1 - Digital Audio Interface (DAI) description ++--------------------------------------------- ++The DAI is a digital audio data transfer link between the codec and host SoC ++CPU. It typically has data transfer capabilities in both directions ++(playback and capture) and can run at a variety of different speeds. ++Supported interfaces currently include AC97, I2S and generic PCM style links. ++Please read DAI.txt for implementation information. ++ ++ ++2 - Digital Audio Interface (DAI) configuration ++----------------------------------------------- ++DAI configuration is handled by the codec_pcm_prepare function and is ++responsible for configuring and starting the DAI on the codec. This can be ++called multiple times and is atomic. It can access the runtime parameters. ++ ++This usually consists of a large function with numerous switch statements to ++set up each configuration option. These options are set by the core at runtime. ++ ++ ++3 - Codec PCM's ++--------------- ++Each codec must have it's PCM's defined. This defines the number of channels, ++stream names, callbacks and codec name. It is also used to register the DAI ++with the ASoC core. The PCM structure also associates the DAI capabilities with ++the ALSA PCM. ++ ++e.g. ++ ++static struct snd_soc_pcm_codec wm8731_pcm_client = { ++ .name = "WM8731", ++ .playback = { ++ .stream_name = "Playback", ++ .channels_min = 1, ++ .channels_max = 2, ++ }, ++ .capture = { ++ .stream_name = "Capture", ++ .channels_min = 1, ++ .channels_max = 2, ++ }, ++ .config_sysclk = wm8731_config_sysclk, ++ .ops = { ++ .prepare = wm8731_pcm_prepare, ++ }, ++ .caps = { ++ .num_modes = ARRAY_SIZE(wm8731_hwfmt), ++ .modes = &wm8731_hwfmt[0], ++ }, ++}; ++ ++ ++4 - Codec control IO ++-------------------- ++The codec can ususally be controlled via an I2C or SPI style interface (AC97 ++combines control with data in the DAI). The codec drivers will have to provide ++functions to read and write the codec registers along with supplying a register ++cache:- ++ ++ /* IO control data and register cache */ ++ void *control_data; /* codec control (i2c/3wire) data */ ++ void *reg_cache; ++ ++Codec read/write should do any data formatting and call the hardware read write ++below to perform the IO. These functions are called by the core and alsa when ++performing DAPM or changing the mixer:- ++ ++ unsigned int (*read)(struct snd_soc_codec *, unsigned int); ++ int (*write)(struct snd_soc_codec *, unsigned int, unsigned int); ++ ++Codec hardware IO functions - usually points to either the I2C, SPI or AC97 ++read/write:- ++ ++ hw_write_t hw_write; ++ hw_read_t hw_read; ++ ++ ++5 - Mixers and audio controls ++----------------------------- ++All the codec mixers and audio controls can be defined using the convenience ++macros defined in soc.h. ++ ++ #define SOC_SINGLE(xname, reg, shift, mask, invert) ++ ++Defines a single control as follows:- ++ ++ xname = Control name e.g. "Playback Volume" ++ reg = codec register ++ shift = control bit(s) offset in register ++ mask = control bit size(s) e.g. mask of 7 = 3 bits ++ invert = the control is inverted ++ ++Other macros include:- ++ ++ #define SOC_DOUBLE(xname, reg, shift_left, shift_right, mask, invert) ++ ++A stereo control ++ ++ #define SOC_DOUBLE_R(xname, reg_left, reg_right, shift, mask, invert) ++ ++A stereo control spanning 2 registers ++ ++ #define SOC_ENUM_SINGLE(xreg, xshift, xmask, xtexts) ++ ++Defines an single enumerated control as follows:- ++ ++ xreg = register ++ xshift = control bit(s) offset in register ++ xmask = control bit(s) size ++ xtexts = pointer to array of strings that describe each setting ++ ++ #define SOC_ENUM_DOUBLE(xreg, xshift_l, xshift_r, xmask, xtexts) ++ ++Defines a stereo enumerated control ++ ++ ++6 - System clock configuration. ++------------------------------- ++The system clock that drives the audio subsystem can change depending on sample ++rate and the system power state. i.e. ++ ++o Higher sample rates sometimes need a higher system clock. ++o Low system power states can sometimes limit the available clocks. ++ ++This function is a callback that the machine driver can call to set and ++determine if the clock and sample rate combination is supported by the codec at ++the present time (and system state). ++ ++NOTE: If the codec has a PLL then it has a lot more flexability wrt clock and ++sample rate combinations. ++ ++Your config_sysclock function should return the MCLK if it's a valid ++combination for your codec else 0; ++ ++Please read clocking.txt now. ++ ++ ++7 - Codec Audio Operations ++-------------------------- ++The codec driver also supports the following alsa operations:- ++ ++/* SoC audio ops */ ++struct snd_soc_ops { ++ int (*startup)(snd_pcm_substream_t *); ++ void (*shutdown)(snd_pcm_substream_t *); ++ int (*hw_params)(snd_pcm_substream_t *, snd_pcm_hw_params_t *); ++ int (*hw_free)(snd_pcm_substream_t *); ++ int (*prepare)(snd_pcm_substream_t *); ++}; ++ ++Please refer to the alsa driver PCM documentation for details. ++http://www.alsa-project.org/~iwai/writing-an-alsa-driver/c436.htm ++ ++ ++8 - DAPM description. ++--------------------- ++The Dynamic Audio Power Management description describes the codec's power ++components, their relationships and registers to the ASoC core. Please read ++dapm.txt for details of building the description. ++ ++Please also see the examples in other codec drivers. ++ ++ ++9 - DAPM event handler ++---------------------- ++This function is a callback that handles codec domain PM calls and system ++domain PM calls (e.g. suspend and resume). It's used to put the codec to sleep ++when not in use. ++ ++Power states:- ++ ++ SNDRV_CTL_POWER_D0: /* full On */ ++ /* vref/mid, clk and osc on, active */ ++ ++ SNDRV_CTL_POWER_D1: /* partial On */ ++ SNDRV_CTL_POWER_D2: /* partial On */ ++ ++ SNDRV_CTL_POWER_D3hot: /* Off, with power */ ++ /* everything off except vref/vmid, inactive */ ++ ++ SNDRV_CTL_POWER_D3cold: /* Everything Off, without power */ ++ ++ ++10 - Codec DAC digital mute control. ++------------------------------------ ++Most codecs have a digital mute before the DAC's that can be used to minimise ++any system noise. The mute stops any digital data from entering the DAC. ++ ++A callback can be created that is called by the core for each codec DAI when the ++mute is applied or freed. ++ ++i.e. ++ ++static int wm8974_mute(struct snd_soc_codec *codec, ++ struct snd_soc_codec_dai *dai, int mute) ++{ ++ u16 mute_reg = wm8974_read_reg_cache(codec, WM8974_DAC) & 0xffbf; ++ if(mute) ++ wm8974_write(codec, WM8974_DAC, mute_reg | 0x40); ++ else ++ wm8974_write(codec, WM8974_DAC, mute_reg); ++ return 0; ++} +Index: linux-2.6-pxa-new/Documentation/sound/alsa/soc/dapm.txt +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/Documentation/sound/alsa/soc/dapm.txt +@@ -0,0 +1,297 @@ ++Dynamic Audio Power Management for Portable Devices ++=================================================== ++ ++1. Description ++============== ++ ++Dynamic Audio Power Management (DAPM) is designed to allow portable Linux devices ++to use the minimum amount of power within the audio subsystem at all times. It ++is independent of other kernel PM and as such, can easily co-exist with the ++other PM systems. ++ ++DAPM is also completely transparent to all user space applications as all power ++switching is done within the ASoC core. No code changes or recompiling are ++required for user space applications. DAPM makes power switching descisions based ++upon any audio stream (capture/playback) activity and audio mixer settings ++within the device. ++ ++DAPM spans the whole machine. It covers power control within the entire audio ++subsystem, this includes internal codec power blocks and machine level power ++systems. ++ ++There are 4 power domains within DAPM ++ ++ 1. Codec domain - VREF, VMID (core codec and audio power) ++ Usually controlled at codec probe/remove and suspend/resume, although ++ can be set at stream time if power is not needed for sidetone, etc. ++ ++ 2. Platform/Machine domain - physically connected inputs and outputs ++ Is platform/machine and user action specific, is configured by the ++ machine driver and responds to asynchronous events e.g when HP ++ are inserted ++ ++ 3. Path domain - audio susbsystem signal paths ++ Automatically set when mixer and mux settings are changed by the user. ++ e.g. alsamixer, amixer. ++ ++ 4. Stream domain - DAC's and ADC's. ++ Enabled and disabled when stream playback/capture is started and ++ stopped respectively. e.g. aplay, arecord. ++ ++All DAPM power switching descisons are made automatically by consulting an audio ++routing map of the whole machine. This map is specific to each machine and ++consists of the interconnections between every audio component (including ++internal codec components). All audio components that effect power are called ++widgets hereafter. ++ ++ ++2. DAPM Widgets ++=============== ++ ++Audio DAPM widgets fall into a number of types:- ++ ++ o Mixer - Mixes several analog signals into a single analog signal. ++ o Mux - An analog switch that outputs only 1 of it's inputs. ++ o PGA - A programmable gain amplifier or attenuation widget. ++ o ADC - Analog to Digital Converter ++ o DAC - Digital to Analog Converter ++ o Switch - An analog switch ++ o Input - A codec input pin ++ o Output - A codec output pin ++ o Headphone - Headphone (and optional Jack) ++ o Mic - Mic (and optional Jack) ++ o Line - Line Input/Output (and optional Jack) ++ o Speaker - Speaker ++ o Pre - Special PRE widget (exec before all others) ++ o Post - Special POST widget (exec after all others) ++ ++(Widgets are defined in include/sound/soc-dapm.h) ++ ++Widgets are usually added in the codec driver and the machine driver. There are ++convience macros defined in soc-dapm.h that can be used to quickly build a ++list of widgets of the codecs and machines DAPM widgets. ++ ++Most widgets have a name, register, shift and invert. Some widgets have extra ++parameters for stream name and kcontrols. ++ ++ ++2.1 Stream Domain Widgets ++------------------------- ++ ++Stream Widgets relate to the stream power domain and only consist of ADC's ++(analog to digital converters) and DAC's (digital to analog converters). ++ ++Stream widgets have the following format:- ++ ++SND_SOC_DAPM_DAC(name, stream name, reg, shift, invert), ++ ++NOTE: the stream name must match the corresponding stream name in your codecs ++snd_soc_codec_dai. ++ ++e.g. stream widgets for HiFi playback and capture ++ ++SND_SOC_DAPM_DAC("HiFi DAC", "HiFi Playback", REG, 3, 1), ++SND_SOC_DAPM_ADC("HiFi ADC", "HiFi Capture", REG, 2, 1), ++ ++ ++2.2 Path Domain Widgets ++----------------------- ++ ++Path domain widgets have a ability to control or effect the audio signal or ++audio paths within the audio subsystem. They have the following form:- ++ ++SND_SOC_DAPM_PGA(name, reg, shift, invert, controls, num_controls) ++ ++Any widget kcontrols can be set using the controls and num_controls members. ++ ++e.g. Mixer widget (the kcontrols are declared first) ++ ++/* Output Mixer */ ++static const snd_kcontrol_new_t wm8731_output_mixer_controls[] = { ++SOC_DAPM_SINGLE("Line Bypass Switch", WM8731_APANA, 3, 1, 0), ++SOC_DAPM_SINGLE("Mic Sidetone Switch", WM8731_APANA, 5, 1, 0), ++SOC_DAPM_SINGLE("HiFi Playback Switch", WM8731_APANA, 4, 1, 0), ++}; ++ ++SND_SOC_DAPM_MIXER("Output Mixer", WM8731_PWR, 4, 1, wm8731_output_mixer_controls, ++ ARRAY_SIZE(wm8731_output_mixer_controls)), ++ ++ ++2.3 Platform/Machine domain Widgets ++----------------------------------- ++ ++Machine widgets are different from codec widgets in that they don't have a ++codec register bit associated with them. A machine widget is assigned to each ++machine audio component (non codec) that can be independently powered. e.g. ++ ++ o Speaker Amp ++ o Microphone Bias ++ o Jack connectors ++ ++A machine widget can have an optional call back. ++ ++e.g. Jack connector widget for an external Mic that enables Mic Bias ++when the Mic is inserted:- ++ ++static int spitz_mic_bias(struct snd_soc_dapm_widget* w, int event) ++{ ++ if(SND_SOC_DAPM_EVENT_ON(event)) ++ set_scoop_gpio(&spitzscoop2_device.dev, SPITZ_SCP2_MIC_BIAS); ++ else ++ reset_scoop_gpio(&spitzscoop2_device.dev, SPITZ_SCP2_MIC_BIAS); ++ ++ return 0; ++} ++ ++SND_SOC_DAPM_MIC("Mic Jack", spitz_mic_bias), ++ ++ ++2.4 Codec Domain ++---------------- ++ ++The Codec power domain has no widgets and is handled by the codecs DAPM event ++handler. This handler is called when the codec powerstate is changed wrt to any ++stream event or by kernel PM events. ++ ++ ++2.5 Virtual Widgets ++------------------- ++ ++Sometimes widgets exist in the codec or machine audio map that don't have any ++corresponding register bit for power control. In this case it's necessary to ++create a virtual widget - a widget with no control bits e.g. ++ ++SND_SOC_DAPM_MIXER("AC97 Mixer", SND_SOC_DAPM_NOPM, 0, 0, NULL, 0), ++ ++This can be used to merge to signal paths together in software. ++ ++After all the widgets have been defined, they can then be added to the DAPM ++subsystem individually with a call to snd_soc_dapm_new_control(). ++ ++ ++3. Codec Widget Interconnections ++================================ ++ ++Widgets are connected to each other within the codec and machine by audio ++paths (called interconnections). Each interconnection must be defined in order ++to create a map of all audio paths between widgets. ++This is easiest with a diagram of the codec (and schematic of the machine audio ++system), as it requires joining widgets together via their audio signal paths. ++ ++i.e. from the WM8731 codec's output mixer (wm8731.c) ++ ++The WM8731 output mixer has 3 inputs (sources) ++ ++ 1. Line Bypass Input ++ 2. DAC (HiFi playback) ++ 3. Mic Sidetone Input ++ ++Each input in this example has a kcontrol associated with it (defined in example ++above) and is connected to the output mixer via it's kcontrol name. We can now ++connect the destination widget (wrt audio signal) with it's source widgets. ++ ++ /* output mixer */ ++ {"Output Mixer", "Line Bypass Switch", "Line Input"}, ++ {"Output Mixer", "HiFi Playback Switch", "DAC"}, ++ {"Output Mixer", "Mic Sidetone Switch", "Mic Bias"}, ++ ++So we have :- ++ ++ Destination Widget <=== Path Name <=== Source Widget ++ ++Or:- ++ ++ Sink, Path, Source ++ ++Or :- ++ ++ "Output Mixer" is connected to the "DAC" via the "HiFi Playback Switch". ++ ++When there is no path name connecting widgets (e.g. a direct connection) we ++pass NULL for the path name. ++ ++Interconnections are created with a call to:- ++ ++snd_soc_dapm_connect_input(codec, sink, path, source); ++ ++Finally, snd_soc_dapm_new_widgets(codec) must be called after all widgets and ++interconnections have been registered with the core. This causes the core to ++scan the codec and machine so that the internal DAPM state matches the ++physical state of the machine. ++ ++ ++3.1 Machine Widget Interconnections ++----------------------------------- ++Machine widget interconnections are created in the same way as codec ones and ++directly connect the codec pins to machine level widgets. ++ ++e.g. connects the speaker out codec pins to the internal speaker. ++ ++ /* ext speaker connected to codec pins LOUT2, ROUT2 */ ++ {"Ext Spk", NULL , "ROUT2"}, ++ {"Ext Spk", NULL , "LOUT2"}, ++ ++This allows the DAPM to power on and off pins that are connected (and in use) ++and pins that are NC respectively. ++ ++ ++4 Endpoint Widgets ++=================== ++An endpoint is a start or end point (widget) of an audio signal within the ++machine and includes the codec. e.g. ++ ++ o Headphone Jack ++ o Internal Speaker ++ o Internal Mic ++ o Mic Jack ++ o Codec Pins ++ ++When a codec pin is NC it can be marked as not used with a call to ++ ++snd_soc_dapm_set_endpoint(codec, "Widget Name", 0); ++ ++The last argument is 0 for inactive and 1 for active. This way the pin and its ++input widget will never be powered up and consume power. ++ ++This also applies to machine widgets. e.g. if a headphone is connected to a ++jack then the jack can be marked active. If the headphone is removed, then ++the headphone jack can be marked inactive. ++ ++ ++5 DAPM Widget Events ++==================== ++ ++Some widgets can register their interest with the DAPM core in PM events. ++e.g. A Speaker with an amplifier registers a widget so the amplifier can be ++powered only when the spk is in use. ++ ++/* turn speaker amplifier on/off depending on use */ ++static int corgi_amp_event(struct snd_soc_dapm_widget *w, int event) ++{ ++ if (SND_SOC_DAPM_EVENT_ON(event)) ++ set_scoop_gpio(&corgiscoop_device.dev, CORGI_SCP_APM_ON); ++ else ++ reset_scoop_gpio(&corgiscoop_device.dev, CORGI_SCP_APM_ON); ++ ++ return 0; ++} ++ ++/* corgi machine dapm widgets */ ++static const struct snd_soc_dapm_widget wm8731_dapm_widgets = ++ SND_SOC_DAPM_SPK("Ext Spk", corgi_amp_event); ++ ++Please see soc-dapm.h for all other widgets that support events. ++ ++ ++5.1 Event types ++--------------- ++ ++The following event types are supported by event widgets. ++ ++/* dapm event types */ ++#define SND_SOC_DAPM_PRE_PMU 0x1 /* before widget power up */ ++#define SND_SOC_DAPM_POST_PMU 0x2 /* after widget power up */ ++#define SND_SOC_DAPM_PRE_PMD 0x4 /* before widget power down */ ++#define SND_SOC_DAPM_POST_PMD 0x8 /* after widget power down */ ++#define SND_SOC_DAPM_PRE_REG 0x10 /* before audio path setup */ ++#define SND_SOC_DAPM_POST_REG 0x20 /* after audio path setup */ +Index: linux-2.6-pxa-new/Documentation/sound/alsa/soc/machine.txt +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/Documentation/sound/alsa/soc/machine.txt +@@ -0,0 +1,114 @@ ++ASoC Machine Driver ++=================== ++ ++The ASoC machine (or board) driver is the code that glues together the platform ++and codec drivers. ++ ++The machine driver can contain codec and platform specific code. It registers ++the audio subsystem with the kernel as a platform device and is represented by ++the following struct:- ++ ++/* SoC machine */ ++struct snd_soc_machine { ++ char *name; ++ ++ int (*probe)(struct platform_device *pdev); ++ int (*remove)(struct platform_device *pdev); ++ ++ /* the pre and post PM functions are used to do any PM work before and ++ * after the codec and DAI's do any PM work. */ ++ int (*suspend_pre)(struct platform_device *pdev, pm_message_t state); ++ int (*suspend_post)(struct platform_device *pdev, pm_message_t state); ++ int (*resume_pre)(struct platform_device *pdev); ++ int (*resume_post)(struct platform_device *pdev); ++ ++ /* machine stream operations */ ++ struct snd_soc_ops *ops; ++ ++ /* CPU <--> Codec DAI links */ ++ struct snd_soc_dai_link *dai_link; ++ int num_links; ++}; ++ ++probe()/remove() ++---------------- ++probe/remove are optional. Do any machine specific probe here. ++ ++ ++suspend()/resume() ++------------------ ++The machine driver has pre and post versions of suspend and resume to take care ++of any machine audio tasks that have to be done before or after the codec, DAI's ++and DMA is suspended and resumed. Optional. ++ ++ ++Machine operations ++------------------ ++The machine specific audio operations can be set here. Again this is optional. ++ ++ ++Machine DAI Configuration ++------------------------- ++The machine DAI configuration glues all the codec and CPU DAI's together. It can ++also be used to set up the DAI system clock and for any machine related DAI ++initialisation e.g. the machine audio map can be connected to the codec audio ++map, unconnnected codec pins can be set as such. Please see corgi.c, spitz.c ++for examples. ++ ++struct snd_soc_dai_link is used to set up each DAI in your machine. e.g. ++ ++/* corgi digital audio interface glue - connects codec <--> CPU */ ++static struct snd_soc_dai_link corgi_dai = { ++ .name = "WM8731", ++ .stream_name = "WM8731", ++ .cpu_dai = &pxa_i2s_dai, ++ .codec_dai = &wm8731_dai, ++ .init = corgi_wm8731_init, ++ .config_sysclk = corgi_config_sysclk, ++}; ++ ++struct snd_soc_machine then sets up the machine with it's DAI's. e.g. ++ ++/* corgi audio machine driver */ ++static struct snd_soc_machine snd_soc_machine_corgi = { ++ .name = "Corgi", ++ .dai_link = &corgi_dai, ++ .num_links = 1, ++ .ops = &corgi_ops, ++}; ++ ++ ++Machine Audio Subsystem ++----------------------- ++ ++The machine soc device glues the platform, machine and codec driver together. ++Private data can also be set here. e.g. ++ ++/* corgi audio private data */ ++static struct wm8731_setup_data corgi_wm8731_setup = { ++ .i2c_address = 0x1b, ++}; ++ ++/* corgi audio subsystem */ ++static struct snd_soc_device corgi_snd_devdata = { ++ .machine = &snd_soc_machine_corgi, ++ .platform = &pxa2xx_soc_platform, ++ .codec_dev = &soc_codec_dev_wm8731, ++ .codec_data = &corgi_wm8731_setup, ++}; ++ ++ ++Machine Power Map ++----------------- ++ ++The machine driver can optionally extend the codec power map and to become an ++audio power map of the audio subsystem. This allows for automatic power up/down ++of speaker/HP amplifiers, etc. Codec pins can be connected to the machines jack ++sockets in the machine init function. See soc/pxa/spitz.c and dapm.txt for ++details. ++ ++ ++Machine Controls ++---------------- ++ ++Machine specific audio mixer controls can be added in the dai init function. +\ No newline at end of file +Index: linux-2.6-pxa-new/Documentation/sound/alsa/soc/overview.txt +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/Documentation/sound/alsa/soc/overview.txt +@@ -0,0 +1,83 @@ ++ALSA SoC Layer ++============== ++ ++The overall project goal of the ALSA System on Chip (ASoC) layer is to provide ++better ALSA support for embedded system on chip procesors (e.g. pxa2xx, au1x00, ++iMX, etc) and portable audio codecs. Currently there is some support in the ++kernel for SoC audio, however it has some limitations:- ++ ++ * Currently, codec drivers are often tightly coupled to the underlying SoC ++ cpu. This is not ideal and leads to code duplication i.e. Linux now has 4 ++ different wm8731 drivers for 4 different SoC platforms. ++ ++ * There is no standard method to signal user initiated audio events. ++ e.g. Headphone/Mic insertion, Headphone/Mic detection after an insertion ++ event. These are quite common events on portable devices and ofter require ++ machine specific code to re route audio, enable amps etc after such an event. ++ ++ * Current drivers tend to power up the entire codec when playing ++ (or recording) audio. This is fine for a PC, but tends to waste a lot of ++ power on portable devices. There is also no support for saving power via ++ changing codec oversampling rates, bias currents, etc. ++ ++ ++ASoC Design ++=========== ++ ++The ASoC layer is designed to address these issues and provide the following ++features :- ++ ++ * Codec independence. Allows reuse of codec drivers on other platforms ++ and machines. ++ ++ * Easy I2S/PCM audio interface setup between codec and SoC. Each SoC interface ++ and codec registers it's audio interface capabilities with the core and are ++ subsequently matched and configured when the application hw params are known. ++ ++ * Dynamic Audio Power Management (DAPM). DAPM automatically sets the codec to ++ it's minimum power state at all times. This includes powering up/down ++ internal power blocks depending on the internal codec audio routing and any ++ active streams. ++ ++ * Pop and click reduction. Pops and clicks can be reduced by powering the ++ codec up/down in the correct sequence (including using digital mute). ASoC ++ signals the codec when to change power states. ++ ++ * Machine specific controls: Allow machines to add controls to the sound card ++ e.g. volume control for speaker amp. ++ ++To achieve all this, ASoC basically splits an embedded audio system into 3 ++components :- ++ ++ * Codec driver: The codec driver is platform independent and contains audio ++ controls, audio interface capabilities, codec dapm definition and codec IO ++ functions. ++ ++ * Platform driver: The platform driver contains the audio dma engine and audio ++ interface drivers (e.g. I2S, AC97, PCM) for that platform. ++ ++ * Machine driver: The machine driver handles any machine specific controls and ++ audio events. i.e. turing on an amp at start of playback. ++ ++ ++Documentation ++============= ++ ++The documentation is spilt into the following sections:- ++ ++overview.txt: This file. ++ ++codec.txt: Codec driver internals. ++ ++DAI.txt: Description of Digital Audio Interface standards and how to configure ++a DAI within your codec and CPU DAI drivers. ++ ++dapm.txt: Dynamic Audio Power Management ++ ++platform.txt: Platform audio DMA and DAI. ++ ++machine.txt: Machine driver internals. ++ ++pop_clicks.txt: How to minimise audio artifacts. ++ ++clocking.txt: ASoC clocking for best power performance. +\ No newline at end of file +Index: linux-2.6-pxa-new/Documentation/sound/alsa/soc/platform.txt +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/Documentation/sound/alsa/soc/platform.txt +@@ -0,0 +1,58 @@ ++ASoC Platform Driver ++==================== ++ ++An ASoC platform driver can be divided into audio DMA and SoC DAI configuration ++and control. The platform drivers only target the SoC CPU and must have no board ++specific code. ++ ++Audio DMA ++========= ++ ++The platform DMA driver optionally supports the following alsa operations:- ++ ++/* SoC audio ops */ ++struct snd_soc_ops { ++ int (*startup)(snd_pcm_substream_t *); ++ void (*shutdown)(snd_pcm_substream_t *); ++ int (*hw_params)(snd_pcm_substream_t *, snd_pcm_hw_params_t *); ++ int (*hw_free)(snd_pcm_substream_t *); ++ int (*prepare)(snd_pcm_substream_t *); ++ int (*trigger)(snd_pcm_substream_t *, int); ++}; ++ ++The platform driver exports it's DMA functionailty via struct snd_soc_platform:- ++ ++struct snd_soc_platform { ++ char *name; ++ ++ int (*probe)(struct platform_device *pdev); ++ int (*remove)(struct platform_device *pdev); ++ int (*suspend)(struct platform_device *pdev, struct snd_soc_cpu_dai *cpu_dai); ++ int (*resume)(struct platform_device *pdev, struct snd_soc_cpu_dai *cpu_dai); ++ ++ /* pcm creation and destruction */ ++ int (*pcm_new)(snd_card_t *, struct snd_soc_codec_dai *, snd_pcm_t *); ++ void (*pcm_free)(snd_pcm_t *); ++ ++ /* platform stream ops */ ++ snd_pcm_ops_t *pcm_ops; ++}; ++ ++Please refer to the alsa driver documentation for details of audio DMA. ++http://www.alsa-project.org/~iwai/writing-an-alsa-driver/c436.htm ++ ++An example DMA driver is soc/pxa/pxa2xx-pcm.c ++ ++ ++SoC DAI Drivers ++=============== ++ ++Each SoC DAI driver must provide the following features:- ++ ++ 1) Digital audio interface (DAI) description ++ 2) Digital audio interface configuration ++ 3) PCM's description ++ 4) Sysclk configuration ++ 5) Suspend and resume (optional) ++ ++Please see codec.txt for a description of items 1 - 4. +Index: linux-2.6-pxa-new/Documentation/sound/alsa/soc/pops_clicks.txt +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/Documentation/sound/alsa/soc/pops_clicks.txt +@@ -0,0 +1,52 @@ ++Audio Pops and Clicks ++===================== ++ ++Pops and clicks are unwanted audio artifacts caused by the powering up and down ++of components within the audio subsystem. This is noticable on PC's when an audio ++module is either loaded or unloaded (at module load time the sound card is ++powered up and causes a popping noise on the speakers). ++ ++Pops and clicks can be more frequent on portable systems with DAPM. This is because ++the components within the subsystem are being dynamically powered depending on ++the audio usage and this can subsequently cause a small pop or click every time a ++component power state is changed. ++ ++ ++Minimising Playback Pops and Clicks ++=================================== ++ ++Playback pops in portable audio subsystems cannot be completely eliminated atm, ++however future audio codec hardware will have better pop and click supression. ++Pops can be reduced within playback by powering the audio components in a ++specific order. This order is different for startup and shutdown and follows ++some basic rules:- ++ ++ Startup Order :- DAC --> Mixers --> Output PGA --> Digital Unmute ++ ++ Shutdown Order :- Digital Mute --> Output PGA --> Mixers --> DAC ++ ++This assumes that the codec PCM output path from the DAC is via a mixer and then ++a PGA (programmable gain amplifier) before being output to the speakers. ++ ++ ++Minimising Capture Pops and Clicks ++================================== ++ ++Capture artifacts are somewhat easier to get rid as we can delay activating the ++ADC until all the pops have occured. This follows similar power rules to ++playback in that components are powered in a sequence depending upon stream ++startup or shutdown. ++ ++ Startup Order - Input PGA --> Mixers --> ADC ++ ++ Shutdown Order - ADC --> Mixers --> Input PGA ++ ++ ++Zipper Noise ++============ ++An unwanted zipper noise can occur within the audio playback or capture stream ++when a volume control is changed near its maximum gain value. The zipper noise ++is heard when the gain increase or decrease changes the mean audio signal ++amplitude too quickly. It can be minimised by enabling the zero cross setting ++for each volume control. The ZC forces the gain change to occur when the signal ++crosses the zero amplitude line. +Index: linux-2.6-pxa-new/include/sound/ac97_codec.h +=================================================================== +--- linux-2.6-pxa-new.orig/include/sound/ac97_codec.h ++++ linux-2.6-pxa-new/include/sound/ac97_codec.h +@@ -425,6 +425,7 @@ struct snd_ac97_build_ops { + + struct snd_ac97_bus_ops { + void (*reset) (struct snd_ac97 *ac97); ++ void (*warm_reset)(struct snd_ac97 *ac97); + void (*write) (struct snd_ac97 *ac97, unsigned short reg, unsigned short val); + unsigned short (*read) (struct snd_ac97 *ac97, unsigned short reg); + void (*wait) (struct snd_ac97 *ac97); +Index: linux-2.6-pxa-new/include/sound/soc-dapm.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/include/sound/soc-dapm.h +@@ -0,0 +1,286 @@ ++/* ++ * linux/sound/soc-dapm.h -- ALSA SoC Dynamic Audio Power Management ++ * ++ * Author: Liam Girdwood ++ * Created: Aug 11th 2005 ++ * Copyright: Wolfson Microelectronics. PLC. ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#ifndef __LINUX_SND_SOC_DAPM_H ++#define __LINUX_SND_SOC_DAPM_H ++ ++#include <linux/device.h> ++#include <linux/types.h> ++#include <sound/control.h> ++#include <sound/soc.h> ++ ++/* widget has no PM register bit */ ++#define SND_SOC_NOPM -1 ++ ++/* ++ * SoC dynamic audio power managment ++ * ++ * We can have upto 4 power domains ++ * 1. Codec domain - VREF, VMID ++ * Usually controlled at codec probe/remove, although can be set ++ * at stream time if power is not needed for sidetone, etc. ++ * 2. Platform/Machine domain - physically connected inputs and outputs ++ * Is platform/machine and user action specific, is set in the machine ++ * driver and by userspace e.g when HP are inserted ++ * 3. Path domain - Internal codec path mixers ++ * Are automatically set when mixer and mux settings are ++ * changed by the user. ++ * 4. Stream domain - DAC's and ADC's. ++ * Enabled when stream playback/capture is started. ++ */ ++ ++/* codec domain */ ++#define SND_SOC_DAPM_VMID(wname) \ ++{ .id = snd_soc_dapm_vmid, .name = wname, .kcontrols = NULL, \ ++ .num_kcontrols = 0} ++ ++/* platform domain */ ++#define SND_SOC_DAPM_INPUT(wname) \ ++{ .id = snd_soc_dapm_input, .name = wname, .kcontrols = NULL, \ ++ .num_kcontrols = 0} ++#define SND_SOC_DAPM_OUTPUT(wname) \ ++{ .id = snd_soc_dapm_output, .name = wname, .kcontrols = NULL, \ ++ .num_kcontrols = 0} ++#define SND_SOC_DAPM_MIC(wname, wevent) \ ++{ .id = snd_soc_dapm_mic, .name = wname, .kcontrols = NULL, \ ++ .num_kcontrols = 0, .event = wevent, \ ++ .event_flags = SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD} ++#define SND_SOC_DAPM_HP(wname, wevent) \ ++{ .id = snd_soc_dapm_hp, .name = wname, .kcontrols = NULL, \ ++ .num_kcontrols = 0, .event = wevent, \ ++ .event_flags = SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD} ++#define SND_SOC_DAPM_SPK(wname, wevent) \ ++{ .id = snd_soc_dapm_spk, .name = wname, .kcontrols = NULL, \ ++ .num_kcontrols = 0, .event = wevent, \ ++ .event_flags = SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD} ++#define SND_SOC_DAPM_LINE(wname, wevent) \ ++{ .id = snd_soc_dapm_line, .name = wname, .kcontrols = NULL, \ ++ .num_kcontrols = 0, .event = wevent, \ ++ .event_flags = SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD} ++ ++/* path domain */ ++#define SND_SOC_DAPM_PGA(wname, wreg, wshift, winvert,\ ++ wcontrols, wncontrols) \ ++{ .id = snd_soc_dapm_pga, .name = wname, .reg = wreg, .shift = wshift, \ ++ .invert = winvert, .kcontrols = wcontrols, .num_kcontrols = wncontrols} ++#define SND_SOC_DAPM_MIXER(wname, wreg, wshift, winvert, \ ++ wcontrols, wncontrols)\ ++{ .id = snd_soc_dapm_mixer, .name = wname, .reg = wreg, .shift = wshift, \ ++ .invert = winvert, .kcontrols = wcontrols, .num_kcontrols = wncontrols} ++#define SND_SOC_DAPM_MICBIAS(wname, wreg, wshift, winvert) \ ++{ .id = snd_soc_dapm_micbias, .name = wname, .reg = wreg, .shift = wshift, \ ++ .invert = winvert, .kcontrols = NULL, .num_kcontrols = 0} ++#define SND_SOC_DAPM_SWITCH(wname, wreg, wshift, winvert, wcontrols) \ ++{ .id = snd_soc_dapm_switch, .name = wname, .reg = wreg, .shift = wshift, \ ++ .invert = winvert, .kcontrols = wcontrols, .num_kcontrols = 1} ++#define SND_SOC_DAPM_MUX(wname, wreg, wshift, winvert, wcontrols) \ ++{ .id = snd_soc_dapm_mux, .name = wname, .reg = wreg, .shift = wshift, \ ++ .invert = winvert, .kcontrols = wcontrols, .num_kcontrols = 1} ++ ++/* path domain with event - event handler must return 0 for success */ ++#define SND_SOC_DAPM_PGA_E(wname, wreg, wshift, winvert, wcontrols, \ ++ wncontrols, wevent, wflags) \ ++{ .id = snd_soc_dapm_pga, .name = wname, .reg = wreg, .shift = wshift, \ ++ .invert = winvert, .kcontrols = wcontrols, .num_kcontrols = wncontrols, \ ++ .event = wevent, .event_flags = wflags} ++#define SND_SOC_DAPM_MIXER_E(wname, wreg, wshift, winvert, wcontrols, \ ++ wncontrols, wevent, wflags) \ ++{ .id = snd_soc_dapm_mixer, .name = wname, .reg = wreg, .shift = wshift, \ ++ .invert = winvert, .kcontrols = wcontrols, .num_kcontrols = wncontrols, \ ++ .event = wevent, .event_flags = wflags} ++#define SND_SOC_DAPM_MICBIAS_E(wname, wreg, wshift, winvert, wevent, wflags) \ ++{ .id = snd_soc_dapm_micbias, .name = wname, .reg = wreg, .shift = wshift, \ ++ .invert = winvert, .kcontrols = NULL, .num_kcontrols = 0, \ ++ .event = wevent, .event_flags = wflags} ++#define SND_SOC_DAPM_SWITCH_E(wname, wreg, wshift, winvert, wcontrols, \ ++ wevent, wflags) \ ++{ .id = snd_soc_dapm_switch, .name = wname, .reg = wreg, .shift = wshift, \ ++ .invert = winvert, .kcontrols = wcontrols, .num_kcontrols = 1 \ ++ .event = wevent, .event_flags = wflags} ++#define SND_SOC_DAPM_MUX_E(wname, wreg, wshift, winvert, wcontrols, \ ++ wevent, wflags) \ ++{ .id = snd_soc_dapm_mux, .name = wname, .reg = wreg, .shift = wshift, \ ++ .invert = winvert, .kcontrols = wcontrols, .num_kcontrols = 1, \ ++ .event = wevent, .event_flags = wflags} ++ ++/* events that are pre and post DAPM */ ++#define SND_SOC_DAPM_PRE(wname, wevent) \ ++{ .id = snd_soc_dapm_pre, .name = wname, .kcontrols = NULL, \ ++ .num_kcontrols = 0, .event = wevent, \ ++ .event_flags = SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_PRE_PMD} ++#define SND_SOC_DAPM_POST(wname, wevent) \ ++{ .id = snd_soc_dapm_post, .name = wname, .kcontrols = NULL, \ ++ .num_kcontrols = 0, .event = wevent, \ ++ .event_flags = SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD} ++ ++/* stream domain */ ++#define SND_SOC_DAPM_DAC(wname, stname, wreg, wshift, winvert) \ ++{ .id = snd_soc_dapm_dac, .name = wname, .sname = stname, .reg = wreg, \ ++ .shift = wshift, .invert = winvert} ++#define SND_SOC_DAPM_ADC(wname, stname, wreg, wshift, winvert) \ ++{ .id = snd_soc_dapm_adc, .name = wname, .sname = stname, .reg = wreg, \ ++ .shift = wshift, .invert = winvert} ++ ++/* dapm kcontrol types */ ++#define SOC_DAPM_SINGLE(xname, reg, shift, mask, invert) \ ++{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, \ ++ .info = snd_soc_info_volsw, \ ++ .get = snd_soc_dapm_get_volsw, .put = snd_soc_dapm_put_volsw, \ ++ .private_value = SOC_SINGLE_VALUE(reg, shift, mask, invert) } ++#define SOC_DAPM_DOUBLE(xname, reg, shift_left, shift_right, mask, invert, \ ++ power) \ ++{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = (xname), \ ++ .info = snd_soc_info_volsw, \ ++ .get = snd_soc_dapm_get_volsw, .put = snd_soc_dapm_put_volsw, \ ++ .private_value = (reg) | ((shift_left) << 8) | ((shift_right) << 12) |\ ++ ((mask) << 16) | ((invert) << 24) } ++#define SOC_DAPM_ENUM(xname, xenum) \ ++{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, \ ++ .info = snd_soc_info_enum_double, \ ++ .get = snd_soc_dapm_get_enum_double, \ ++ .put = snd_soc_dapm_put_enum_double, \ ++ .private_value = (unsigned long)&xenum } ++ ++/* dapm stream operations */ ++#define SND_SOC_DAPM_STREAM_NOP 0x0 ++#define SND_SOC_DAPM_STREAM_START 0x1 ++#define SND_SOC_DAPM_STREAM_STOP 0x2 ++#define SND_SOC_DAPM_STREAM_SUSPEND 0x4 ++#define SND_SOC_DAPM_STREAM_RESUME 0x8 ++#define SND_SOC_DAPM_STREAM_PAUSE_PUSH 0x10 ++#define SND_SOC_DAPM_STREAM_PAUSE_RELEASE 0x20 ++ ++/* dapm event types */ ++#define SND_SOC_DAPM_PRE_PMU 0x1 /* before widget power up */ ++#define SND_SOC_DAPM_POST_PMU 0x2 /* after widget power up */ ++#define SND_SOC_DAPM_PRE_PMD 0x4 /* before widget power down */ ++#define SND_SOC_DAPM_POST_PMD 0x8 /* after widget power down */ ++#define SND_SOC_DAPM_PRE_REG 0x10 /* before audio path setup */ ++#define SND_SOC_DAPM_POST_REG 0x20 /* after audio path setup */ ++ ++/* convenience event type detection */ ++#define SND_SOC_DAPM_EVENT_ON(e) \ ++ (e & (SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU)) ++#define SND_SOC_DAPM_EVENT_OFF(e) \ ++ (e & (SND_SOC_DAPM_PRE_PMD | SND_SOC_DAPM_POST_PMD)) ++ ++struct snd_soc_dapm_widget; ++enum snd_soc_dapm_type; ++struct snd_soc_dapm_path; ++struct snd_soc_dapm_pin; ++ ++/* dapm controls */ ++int snd_soc_dapm_put_volsw(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol); ++int snd_soc_dapm_get_volsw(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol); ++int snd_soc_dapm_get_enum_double(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol); ++int snd_soc_dapm_put_enum_double(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol); ++int snd_soc_dapm_new_control(struct snd_soc_codec *codec, ++ const struct snd_soc_dapm_widget *widget); ++ ++/* dapm path setup */ ++int snd_soc_dapm_connect_input(struct snd_soc_codec *codec, ++ const char *sink_name, const char *control_name, const char *src_name); ++int snd_soc_dapm_new_widgets(struct snd_soc_codec *codec); ++void snd_soc_dapm_free(struct snd_soc_device *socdev); ++ ++/* dapm events */ ++int snd_soc_dapm_stream_event(struct snd_soc_codec *codec, char *stream, ++ int event); ++ ++/* dapm sys fs - used by the core */ ++int snd_soc_dapm_sys_add(struct device *dev); ++ ++/* dapm audio endpoint control */ ++int snd_soc_dapm_set_endpoint(struct snd_soc_codec *codec, ++ char *pin, int status); ++int snd_soc_dapm_sync_endpoints(struct snd_soc_codec *codec); ++ ++/* dapm widget types */ ++enum snd_soc_dapm_type { ++ snd_soc_dapm_input = 0, /* input pin */ ++ snd_soc_dapm_output, /* output pin */ ++ snd_soc_dapm_mux, /* selects 1 analog signal from many inputs */ ++ snd_soc_dapm_mixer, /* mixes several analog signals together */ ++ snd_soc_dapm_pga, /* programmable gain/attenuation (volume) */ ++ snd_soc_dapm_adc, /* analog to digital converter */ ++ snd_soc_dapm_dac, /* digital to analog converter */ ++ snd_soc_dapm_micbias, /* microphone bias (power) */ ++ snd_soc_dapm_mic, /* microphone */ ++ snd_soc_dapm_hp, /* headphones */ ++ snd_soc_dapm_spk, /* speaker */ ++ snd_soc_dapm_line, /* line input/output */ ++ snd_soc_dapm_switch, /* analog switch */ ++ snd_soc_dapm_vmid, /* codec bias/vmid - to minimise pops */ ++ snd_soc_dapm_pre, /* machine specific pre widget - exec first */ ++ snd_soc_dapm_post, /* machine specific post widget - exec last */ ++}; ++ ++/* dapm audio path between two widgets */ ++struct snd_soc_dapm_path { ++ char *name; ++ char *long_name; ++ ++ /* source (input) and sink (output) widgets */ ++ struct snd_soc_dapm_widget *source; ++ struct snd_soc_dapm_widget *sink; ++ struct snd_kcontrol *kcontrol; ++ ++ /* status */ ++ u32 connect:1; /* source and sink widgets are connected */ ++ u32 walked:1; /* path has been walked */ ++ ++ struct list_head list_source; ++ struct list_head list_sink; ++ struct list_head list; ++}; ++ ++/* dapm widget */ ++struct snd_soc_dapm_widget { ++ enum snd_soc_dapm_type id; ++ char *name; /* widget name */ ++ char *sname; /* stream name */ ++ struct snd_soc_codec *codec; ++ struct list_head list; ++ ++ /* dapm control */ ++ short reg; /* negative reg = no direct dapm */ ++ unsigned char shift; /* bits to shift */ ++ unsigned int saved_value; /* widget saved value */ ++ unsigned int value; /* widget current value */ ++ unsigned char power:1; /* block power status */ ++ unsigned char invert:1; /* invert the power bit */ ++ unsigned char active:1; /* active stream on DAC, ADC's */ ++ unsigned char connected:1; /* connected codec pin */ ++ unsigned char new:1; /* cnew complete */ ++ unsigned char ext:1; /* has external widgets */ ++ unsigned char muted:1; /* muted for pop reduction */ ++ unsigned char suspend:1; /* was active before suspend */ ++ unsigned char pmdown:1; /* waiting for timeout */ ++ ++ /* external events */ ++ unsigned short event_flags; /* flags to specify event types */ ++ int (*event)(struct snd_soc_dapm_widget*, int); ++ ++ /* kcontrols that relate to this widget */ ++ int num_kcontrols; ++ const struct snd_kcontrol_new *kcontrols; ++ ++ /* widget input and outputs */ ++ struct list_head sources; ++ struct list_head sinks; ++}; ++ ++#endif +Index: linux-2.6-pxa-new/include/sound/soc.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/include/sound/soc.h +@@ -0,0 +1,487 @@ ++/* ++ * linux/sound/soc.h -- ALSA SoC Layer ++ * ++ * Author: Liam Girdwood ++ * Created: Aug 11th 2005 ++ * Copyright: Wolfson Microelectronics. PLC. ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#ifndef __LINUX_SND_SOC_H ++#define __LINUX_SND_SOC_H ++ ++#include <linux/platform_device.h> ++#include <linux/types.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/control.h> ++#include <sound/ac97_codec.h> ++ ++#define SND_SOC_VERSION "0.12.4" ++ ++/* ++ * Convenience kcontrol builders ++ */ ++#define SOC_SINGLE_VALUE(reg,shift,mask,invert) ((reg) | ((shift) << 8) |\ ++ ((shift) << 12) | ((mask) << 16) | ((invert) << 24)) ++#define SOC_SINGLE_VALUE_EXT(reg,mask,invert) ((reg) | ((mask) << 16) |\ ++ ((invert) << 31)) ++#define SOC_SINGLE(xname, reg, shift, mask, invert) \ ++{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, \ ++ .info = snd_soc_info_volsw, .get = snd_soc_get_volsw,\ ++ .put = snd_soc_put_volsw, \ ++ .private_value = SOC_SINGLE_VALUE(reg, shift, mask, invert) } ++#define SOC_DOUBLE(xname, reg, shift_left, shift_right, mask, invert) \ ++{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = (xname),\ ++ .info = snd_soc_info_volsw, .get = snd_soc_get_volsw, \ ++ .put = snd_soc_put_volsw, \ ++ .private_value = (reg) | ((shift_left) << 8) | \ ++ ((shift_right) << 12) | ((mask) << 16) | ((invert) << 24) } ++#define SOC_DOUBLE_R(xname, reg_left, reg_right, shift, mask, invert) \ ++{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = (xname), \ ++ .info = snd_soc_info_volsw_2r, \ ++ .get = snd_soc_get_volsw_2r, .put = snd_soc_put_volsw_2r, \ ++ .private_value = (reg_left) | ((shift) << 8) | \ ++ ((mask) << 12) | ((invert) << 20) | ((reg_right) << 24) } ++#define SOC_ENUM_DOUBLE(xreg, xshift_l, xshift_r, xmask, xtexts) \ ++{ .reg = xreg, .shift_l = xshift_l, .shift_r = xshift_r, \ ++ .mask = xmask, .texts = xtexts } ++#define SOC_ENUM_SINGLE(xreg, xshift, xmask, xtexts) \ ++ SOC_ENUM_DOUBLE(xreg, xshift, xshift, xmask, xtexts) ++#define SOC_ENUM_SINGLE_EXT(xmask, xtexts) \ ++{ .mask = xmask, .texts = xtexts } ++#define SOC_ENUM(xname, xenum) \ ++{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname,\ ++ .info = snd_soc_info_enum_double, \ ++ .get = snd_soc_get_enum_double, .put = snd_soc_put_enum_double, \ ++ .private_value = (unsigned long)&xenum } ++#define SOC_SINGLE_EXT(xname, xreg, xmask, xinvert,\ ++ xhandler_get, xhandler_put) \ ++{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, \ ++ .info = snd_soc_info_volsw_ext, \ ++ .get = xhandler_get, .put = xhandler_put, \ ++ .private_value = SOC_SINGLE_VALUE_EXT(xreg, xmask, xinvert) } ++#define SOC_SINGLE_BOOL_EXT(xname, xdata, xhandler_get, xhandler_put) \ ++{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, \ ++ .info = snd_soc_info_bool_ext, \ ++ .get = xhandler_get, .put = xhandler_put, \ ++ .private_value = xdata } ++#define SOC_ENUM_EXT(xname, xenum, xhandler_get, xhandler_put) \ ++{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, \ ++ .info = snd_soc_info_enum_ext, \ ++ .get = xhandler_get, .put = xhandler_put, \ ++ .private_value = (unsigned long)&xenum } ++ ++/* ++ * Digital Audio Interface (DAI) types ++ */ ++#define SND_SOC_DAI_AC97 0x1 ++#define SND_SOC_DAI_I2S 0x2 ++#define SND_SOC_DAI_PCM 0x4 ++ ++/* ++ * DAI hardware audio formats ++ */ ++#define SND_SOC_DAIFMT_I2S (1 << 0) /* I2S mode */ ++#define SND_SOC_DAIFMT_RIGHT_J (1 << 1) /* Right justified mode */ ++#define SND_SOC_DAIFMT_LEFT_J (1 << 2) /* Left Justified mode */ ++#define SND_SOC_DAIFMT_DSP_A (1 << 3) /* L data msb after FRM or LRC */ ++#define SND_SOC_DAIFMT_DSP_B (1 << 4) /* L data msb during FRM or LRC */ ++#define SND_SOC_DAIFMT_AC97 (1 << 5) /* AC97 */ ++ ++/* ++ * DAI hardware signal inversions ++ */ ++#define SND_SOC_DAIFMT_NB_NF (1 << 8) /* normal bit clock + frame */ ++#define SND_SOC_DAIFMT_NB_IF (1 << 9) /* normal bclk + inv frm */ ++#define SND_SOC_DAIFMT_IB_NF (1 << 10) /* invert bclk + nor frm */ ++#define SND_SOC_DAIFMT_IB_IF (1 << 11) /* invert bclk + frm */ ++ ++/* ++ * DAI hardware clock masters ++ * This is wrt the codec, the inverse is true for the interface ++ * i.e. if the codec is clk and frm master then the interface is ++ * clk and frame slave. ++ */ ++#define SND_SOC_DAIFMT_CBM_CFM (1 << 12) /* codec clk & frm master */ ++#define SND_SOC_DAIFMT_CBS_CFM (1 << 13) /* codec clk slave & frm master */ ++#define SND_SOC_DAIFMT_CBM_CFS (1 << 14) /* codec clk master & frame slave */ ++#define SND_SOC_DAIFMT_CBS_CFS (1 << 15) /* codec clk & frm slave */ ++ ++#define SND_SOC_DAIFMT_FORMAT_MASK 0x00ff ++#define SND_SOC_DAIFMT_INV_MASK 0x0f00 ++#define SND_SOC_DAIFMT_CLOCK_MASK 0xf000 ++ ++/* ++ * DAI hardware audio direction ++ */ ++#define SND_SOC_DAIDIR_PLAYBACK 0x1 ++#define SND_SOC_DAIDIR_CAPTURE 0x2 ++ ++/* ++ * DAI hardware Time Division Multiplexing (TDM) Slots ++ * Left and Right data word positions ++ * This is measured in words (sample size) and not bits. ++ */ ++#define SND_SOC_DAITDM_LRDW(l,r) ((l << 8) | r) ++ ++/* ++ * DAI hardware clock ratios ++ * bit clock can either be a generated by dividing mclk or ++ * by multiplying sample rate, hence there are 2 definitions below ++ * depending on codec type. ++ */ ++/* ratio of sample rate to mclk/sysclk */ ++#define SND_SOC_FS_ALL 0xffff /* all mclk supported */ ++ ++/* bit clock dividers */ ++#define SND_SOC_FSBD(x) (1 << (x - 1)) /* ratio mclk:bclk */ ++#define SND_SOC_FSBD_REAL(x) (ffs(x)) ++ ++/* bit clock ratio to (sample rate * channels * word size) */ ++#define SND_SOC_FSBW(x) (1 << (x - 1)) ++#define SND_SOC_FSBW_REAL(x) (ffs(x)) ++/* all bclk ratios supported */ ++#define SND_SOC_FSB_ALL ~0ULL ++ ++/* ++ * DAI hardware flags ++ */ ++/* use bfs mclk divider mode (BCLK = MCLK / x) */ ++#define SND_SOC_DAI_BFS_DIV 0x1 ++/* use bfs rate mulitplier (BCLK = RATE * x)*/ ++#define SND_SOC_DAI_BFS_RATE 0x2 ++/* use bfs rcw multiplier (BCLK = RATE * CHN * WORD SIZE) */ ++#define SND_SOC_DAI_BFS_RCW 0x4 ++/* capture and playback can use different clocks */ ++#define SND_SOC_DAI_ASYNC 0x8 ++/* can use gated BCLK */ ++#define SND_SOC_DAI_GATED 0x10 ++ ++/* ++ * AC97 codec ID's bitmask ++ */ ++#define SND_SOC_DAI_AC97_ID0 (1 << 0) ++#define SND_SOC_DAI_AC97_ID1 (1 << 1) ++#define SND_SOC_DAI_AC97_ID2 (1 << 2) ++#define SND_SOC_DAI_AC97_ID3 (1 << 3) ++ ++struct snd_soc_device; ++struct snd_soc_pcm_stream; ++struct snd_soc_ops; ++struct snd_soc_dai_mode; ++struct snd_soc_pcm_runtime; ++struct snd_soc_codec_dai; ++struct snd_soc_cpu_dai; ++struct snd_soc_codec; ++struct snd_soc_machine_config; ++struct soc_enum; ++struct snd_soc_ac97_ops; ++struct snd_soc_clock_info; ++ ++typedef int (*hw_write_t)(void *,const char* ,int); ++typedef int (*hw_read_t)(void *,char* ,int); ++ ++extern struct snd_ac97_bus_ops soc_ac97_ops; ++ ++/* pcm <-> DAI connect */ ++void snd_soc_free_pcms(struct snd_soc_device *socdev); ++int snd_soc_new_pcms(struct snd_soc_device *socdev, int idx, const char *xid); ++int snd_soc_register_card(struct snd_soc_device *socdev); ++ ++/* set runtime hw params */ ++int snd_soc_set_runtime_hwparams(struct snd_pcm_substream *substream, ++ const struct snd_pcm_hardware *hw); ++int snd_soc_get_rate(int rate); ++ ++/* codec IO */ ++#define snd_soc_read(codec, reg) codec->read(codec, reg) ++#define snd_soc_write(codec, reg, value) codec->write(codec, reg, value) ++ ++/* codec register bit access */ ++int snd_soc_update_bits(struct snd_soc_codec *codec, unsigned short reg, ++ unsigned short mask, unsigned short value); ++int snd_soc_test_bits(struct snd_soc_codec *codec, unsigned short reg, ++ unsigned short mask, unsigned short value); ++ ++int snd_soc_new_ac97_codec(struct snd_soc_codec *codec, ++ struct snd_ac97_bus_ops *ops, int num); ++void snd_soc_free_ac97_codec(struct snd_soc_codec *codec); ++ ++/* ++ *Controls ++ */ ++struct snd_kcontrol *snd_soc_cnew(const struct snd_kcontrol_new *_template, ++ void *data, char *long_name); ++int snd_soc_info_enum_double(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_info *uinfo); ++int snd_soc_info_enum_ext(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_info *uinfo); ++int snd_soc_get_enum_double(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol); ++int snd_soc_put_enum_double(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol); ++int snd_soc_info_volsw(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_info *uinfo); ++int snd_soc_info_volsw_ext(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_info *uinfo); ++int snd_soc_info_bool_ext(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_info *uinfo); ++int snd_soc_get_volsw(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol); ++int snd_soc_put_volsw(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol); ++int snd_soc_info_volsw_2r(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_info *uinfo); ++int snd_soc_get_volsw_2r(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol); ++int snd_soc_put_volsw_2r(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol); ++ ++/* SoC PCM stream information */ ++struct snd_soc_pcm_stream { ++ char *stream_name; ++ unsigned int rate_min; /* min rate */ ++ unsigned int rate_max; /* max rate */ ++ unsigned int channels_min; /* min channels */ ++ unsigned int channels_max; /* max channels */ ++ unsigned int active:1; /* stream is in use */ ++}; ++ ++/* SoC audio ops */ ++struct snd_soc_ops { ++ int (*startup)(struct snd_pcm_substream *); ++ void (*shutdown)(struct snd_pcm_substream *); ++ int (*hw_params)(struct snd_pcm_substream *, struct snd_pcm_hw_params *); ++ int (*hw_free)(struct snd_pcm_substream *); ++ int (*prepare)(struct snd_pcm_substream *); ++ int (*trigger)(struct snd_pcm_substream *, int); ++}; ++ ++/* SoC DAI hardware mode */ ++struct snd_soc_dai_mode { ++ u16 fmt; /* SND_SOC_DAIFMT_* */ ++ u16 tdm; /* SND_SOC_HWTDM_* */ ++ u64 pcmfmt; /* SNDRV_PCM_FMTBIT_* */ ++ u16 pcmrate; /* SND_SOC_HWRATE_* */ ++ u16 pcmdir:2; /* SND_SOC_HWDIR_* */ ++ u16 flags:8; /* hw flags */ ++ u16 fs; /* mclk to rate divider */ ++ u64 bfs; /* mclk to bclk dividers */ ++ unsigned long priv; /* private mode data */ ++}; ++ ++/* DAI capabilities */ ++struct snd_soc_dai_cap { ++ int num_modes; /* number of DAI modes */ ++ struct snd_soc_dai_mode *mode; /* array of supported DAI modes */ ++}; ++ ++/* SoC Codec DAI */ ++struct snd_soc_codec_dai { ++ char *name; ++ int id; ++ ++ /* DAI capabilities */ ++ struct snd_soc_pcm_stream playback; ++ struct snd_soc_pcm_stream capture; ++ struct snd_soc_dai_cap caps; ++ ++ /* DAI runtime info */ ++ struct snd_soc_dai_mode dai_runtime; ++ struct snd_soc_ops ops; ++ unsigned int (*config_sysclk)(struct snd_soc_codec_dai*, ++ struct snd_soc_clock_info *info, unsigned int clk); ++ int (*digital_mute)(struct snd_soc_codec *, ++ struct snd_soc_codec_dai*, int); ++ unsigned int mclk; /* the audio master clock */ ++ unsigned int pll_in; /* the PLL input clock */ ++ unsigned int pll_out; /* the PLL output clock */ ++ unsigned int clk_div; /* internal clock divider << 1 (for fractions) */ ++ unsigned int active; ++ unsigned char pop_wait:1; ++ ++ /* DAI private data */ ++ void *private_data; ++}; ++ ++/* SoC CPU DAI */ ++struct snd_soc_cpu_dai { ++ ++ /* DAI description */ ++ char *name; ++ unsigned int id; ++ unsigned char type; ++ ++ /* DAI callbacks */ ++ int (*probe)(struct platform_device *pdev); ++ void (*remove)(struct platform_device *pdev); ++ int (*suspend)(struct platform_device *pdev, ++ struct snd_soc_cpu_dai *cpu_dai); ++ int (*resume)(struct platform_device *pdev, ++ struct snd_soc_cpu_dai *cpu_dai); ++ unsigned int (*config_sysclk)(struct snd_soc_cpu_dai *cpu_dai, ++ struct snd_soc_clock_info *info, unsigned int clk); ++ ++ /* DAI capabilities */ ++ struct snd_soc_pcm_stream capture; ++ struct snd_soc_pcm_stream playback; ++ struct snd_soc_dai_cap caps; ++ ++ /* DAI runtime info */ ++ struct snd_soc_dai_mode dai_runtime; ++ struct snd_soc_ops ops; ++ struct snd_pcm_runtime *runtime; ++ unsigned char active:1; ++ unsigned int mclk; ++ void *dma_data; ++ ++ /* DAI private data */ ++ void *private_data; ++}; ++ ++/* SoC Audio Codec */ ++struct snd_soc_codec { ++ char *name; ++ struct module *owner; ++ struct mutex mutex; ++ ++ /* callbacks */ ++ int (*dapm_event)(struct snd_soc_codec *codec, int event); ++ ++ /* runtime */ ++ struct snd_card *card; ++ struct snd_ac97 *ac97; /* for ad-hoc ac97 devices */ ++ unsigned int active; ++ unsigned int pcm_devs; ++ void *private_data; ++ ++ /* codec IO */ ++ void *control_data; /* codec control (i2c/3wire) data */ ++ unsigned int (*read)(struct snd_soc_codec *, unsigned int); ++ int (*write)(struct snd_soc_codec *, unsigned int, unsigned int); ++ hw_write_t hw_write; ++ hw_read_t hw_read; ++ void *reg_cache; ++ short reg_cache_size; ++ short reg_cache_step; ++ ++ /* dapm */ ++ struct list_head dapm_widgets; ++ struct list_head dapm_paths; ++ unsigned int dapm_state; ++ unsigned int suspend_dapm_state; ++ ++ /* codec DAI's */ ++ struct snd_soc_codec_dai *dai; ++ unsigned int num_dai; ++}; ++ ++/* codec device */ ++struct snd_soc_codec_device { ++ int (*probe)(struct platform_device *pdev); ++ int (*remove)(struct platform_device *pdev); ++ int (*suspend)(struct platform_device *pdev, pm_message_t state); ++ int (*resume)(struct platform_device *pdev); ++}; ++ ++/* SoC platform interface */ ++struct snd_soc_platform { ++ char *name; ++ ++ int (*probe)(struct platform_device *pdev); ++ int (*remove)(struct platform_device *pdev); ++ int (*suspend)(struct platform_device *pdev, ++ struct snd_soc_cpu_dai *cpu_dai); ++ int (*resume)(struct platform_device *pdev, ++ struct snd_soc_cpu_dai *cpu_dai); ++ ++ /* pcm creation and destruction */ ++ int (*pcm_new)(struct snd_card *, struct snd_soc_codec_dai *, ++ struct snd_pcm *); ++ void (*pcm_free)(struct snd_pcm *); ++ ++ /* platform stream ops */ ++ struct snd_pcm_ops *pcm_ops; ++}; ++ ++/* SoC machine DAI configuration, glues a codec and cpu DAI together */ ++struct snd_soc_dai_link { ++ char *name; /* Codec name */ ++ char *stream_name; /* Stream name */ ++ ++ /* DAI */ ++ struct snd_soc_codec_dai *codec_dai; ++ struct snd_soc_cpu_dai *cpu_dai; ++ u32 flags; /* DAI config preference flags */ ++ ++ /* codec/machine specific init - e.g. add machine controls */ ++ int (*init)(struct snd_soc_codec *codec); ++ ++ /* audio sysclock configuration */ ++ unsigned int (*config_sysclk)(struct snd_soc_pcm_runtime *rtd, ++ struct snd_soc_clock_info *info); ++}; ++ ++/* SoC machine */ ++struct snd_soc_machine { ++ char *name; ++ ++ int (*probe)(struct platform_device *pdev); ++ int (*remove)(struct platform_device *pdev); ++ ++ /* the pre and post PM functions are used to do any PM work before and ++ * after the codec and DAI's do any PM work. */ ++ int (*suspend_pre)(struct platform_device *pdev, pm_message_t state); ++ int (*suspend_post)(struct platform_device *pdev, pm_message_t state); ++ int (*resume_pre)(struct platform_device *pdev); ++ int (*resume_post)(struct platform_device *pdev); ++ ++ /* machine stream operations */ ++ struct snd_soc_ops *ops; ++ ++ /* CPU <--> Codec DAI links */ ++ struct snd_soc_dai_link *dai_link; ++ int num_links; ++}; ++ ++/* SoC Device - the audio subsystem */ ++struct snd_soc_device { ++ struct device *dev; ++ struct snd_soc_machine *machine; ++ struct snd_soc_platform *platform; ++ struct snd_soc_codec *codec; ++ struct snd_soc_codec_device *codec_dev; ++ void *codec_data; ++}; ++ ++/* runtime channel data */ ++struct snd_soc_pcm_runtime { ++ struct snd_soc_codec_dai *codec_dai; ++ struct snd_soc_cpu_dai *cpu_dai; ++ struct snd_soc_device *socdev; ++}; ++ ++/* enumerated kcontrol */ ++struct soc_enum { ++ unsigned short reg; ++ unsigned short reg2; ++ unsigned char shift_l; ++ unsigned char shift_r; ++ unsigned int mask; ++ const char **texts; ++ void *dapm; ++}; ++ ++/* clocking configuration data */ ++struct snd_soc_clock_info { ++ unsigned int rate; ++ unsigned int fs; ++ unsigned int bclk_master; ++}; ++ ++#endif +Index: linux-2.6-pxa-new/sound/Kconfig +=================================================================== +--- linux-2.6-pxa-new.orig/sound/Kconfig ++++ linux-2.6-pxa-new/sound/Kconfig +@@ -76,6 +76,8 @@ source "sound/sparc/Kconfig" + + source "sound/parisc/Kconfig" + ++source "sound/soc/Kconfig" ++ + endmenu + + menu "Open Sound System" +Index: linux-2.6-pxa-new/sound/soc/Kconfig +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/Kconfig +@@ -0,0 +1,37 @@ ++# ++# SoC audio configuration ++# ++ ++menu "SoC audio support" ++ depends on SND!=n ++ ++config SND_SOC_AC97_BUS ++ bool ++ ++config SND_SOC ++ tristate "SoC audio support" ++ ---help--- ++ ++ If you want SoC support, you should say Y here and also to the ++ specific driver for your SoC below. You will also need to select the ++ specific codec(s) attached to the SoC ++ ++ This SoC audio support can also be built as a module. If so, the module ++ will be called snd-soc-core. ++ ++# All the supported Soc's ++menu "Soc Platforms" ++depends on SND_SOC ++source "sound/soc/pxa/Kconfig" ++source "sound/soc/at91/Kconfig" ++source "sound/soc/imx/Kconfig" ++source "sound/soc/s3c24xx/Kconfig" ++endmenu ++ ++# Supported codecs ++menu "Soc Codecs" ++depends on SND_SOC ++source "sound/soc/codecs/Kconfig" ++endmenu ++ ++endmenu +Index: linux-2.6-pxa-new/sound/soc/Makefile +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/Makefile +@@ -0,0 +1,4 @@ ++snd-soc-core-objs := soc-core.o soc-dapm.o ++ ++obj-$(CONFIG_SND_SOC) += snd-soc-core.o ++obj-$(CONFIG_SND_SOC) += pxa/ at91/ imx/ s3c24xx/ codecs/ +Index: linux-2.6-pxa-new/sound/soc/codecs/Kconfig +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/Kconfig +@@ -0,0 +1,90 @@ ++config SND_SOC_AC97_CODEC ++ tristate "SoC generic AC97 support" ++ depends SND_SOC ++ help ++ Say Y or M if you want generic AC97 support. This is not required ++ for the AC97 codecs listed below. ++ ++config SND_SOC_WM8711 ++ tristate "SoC driver for the WM8711 codec" ++ depends SND_SOC ++ help ++ Say Y or M if you want to support the WM8711 codec. ++ ++config SND_SOC_WM8510 ++ tristate "SoC driver for the WM8510 codec" ++ depends SND_SOC ++ help ++ Say Y or M if you want to support the WM8711 codec. ++ ++config SND_SOC_WM8731 ++ tristate "SoC driver for the WM8731 codec" ++ depends SND_SOC ++ help ++ Say Y or M if you want to support the WM8731 codec. ++ ++config SND_SOC_WM8750 ++ tristate "SoC driver for the WM8750 codec" ++ depends SND_SOC ++ help ++ Say Y or M if you want to support the WM8750 codec. ++ ++config SND_SOC_WM8753 ++ tristate "SoC driver for the WM8753 codec" ++ depends SND_SOC ++ help ++ Say Y or M if you want to support the WM8753 codec. ++ ++config SND_SOC_WM8772 ++ tristate "SoC driver for the WM8772 codec" ++ depends SND_SOC ++ help ++ Say Y or M if you want to support the WM8772 codec. ++ ++config SND_SOC_WM8971 ++ tristate "SoC driver for the WM8971 codec" ++ depends SND_SOC ++ help ++ Say Y or M if you want to support the WM8971 codec. ++ ++config SND_SOC_WM8976 ++ tristate "SoC driver for the WM8976 codec" ++ depends SND_SOC ++ help ++ Say Y or M if you want to support the WM8976 codec. ++ ++config SND_SOC_WM8974 ++ tristate "SoC driver for the WM8974 codec" ++ depends SND_SOC ++ help ++ Say Y or M if you want to support the WM8974 codec. ++ ++config SND_SOC_WM8980 ++ tristate "SoC driver for the WM8980 codec" ++ depends SND_SOC ++ help ++ Say Y or M if you want to support the WM8980 codec. ++ ++config SND_SOC_WM9713 ++ tristate "SoC driver for the WM9713 codec" ++ depends SND_SOC ++ help ++ Say Y or M if you want to support the WM9713 codec. ++ ++config SND_SOC_WM9712 ++ tristate "SoC driver for the WM9712 codec" ++ depends SND_SOC ++ help ++ Say Y or M if you want to support the WM9712 codec. ++ ++config SND_SOC_UDA1380 ++ tristate "SoC driver for the UDA1380 codec" ++ depends SND_SOC ++ help ++ Say Y or M if you want to support the UDA1380 codec. ++ ++config SND_SOC_AK4535 ++ tristate "SoC driver for the AK4535 codec" ++ depends SND_SOC ++ help ++ Say Y or M if you want to support the AK4535 codec. +Index: linux-2.6-pxa-new/sound/soc/codecs/Makefile +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/Makefile +@@ -0,0 +1,31 @@ ++snd-soc-ac97-objs := ac97.o ++snd-soc-wm8711-objs := wm8711.o ++snd-soc-wm8510-objs := wm8510.o ++snd-soc-wm8731-objs := wm8731.o ++snd-soc-wm8750-objs := wm8750.o ++snd-soc-wm8753-objs := wm8753.o ++snd-soc-wm8772-objs := wm8772.o ++snd-soc-wm8971-objs := wm8971.o ++snd-soc-wm8974-objs := wm8974.o ++snd-soc-wm8976-objs := wm8976.o ++snd-soc-wm8980-objs := wm8980.o ++snd-soc-uda1380-objs := uda1380.o ++snd-soc-ak4535-objs := ak4535.o ++snd-soc-wm9713-objs := wm9713.o ++snd-soc-wm9712-objs := wm9712.o ++ ++obj-$(CONFIG_SND_SOC_AC97_CODEC) += snd-soc-ac97.o ++obj-$(CONFIG_SND_SOC_WM8711) += snd-soc-wm8711.o ++obj-$(CONFIG_SND_SOC_WM8510) += snd-soc-wm8510.o ++obj-$(CONFIG_SND_SOC_WM8731) += snd-soc-wm8731.o ++obj-$(CONFIG_SND_SOC_WM8750) += snd-soc-wm8750.o ++obj-$(CONFIG_SND_SOC_WM8753) += snd-soc-wm8753.o ++obj-$(CONFIG_SND_SOC_WM8772) += snd-soc-wm8772.o ++obj-$(CONFIG_SND_SOC_WM8971) += snd-soc-wm8971.o ++obj-$(CONFIG_SND_SOC_WM8974) += snd-soc-wm8974.o ++obj-$(CONFIG_SND_SOC_WM8976) += snd-soc-wm8976.o ++obj-$(CONFIG_SND_SOC_WM8980) += snd-soc-wm8980.o ++obj-$(CONFIG_SND_SOC_UDA1380) += snd-soc-uda1380.o ++obj-$(CONFIG_SND_SOC_AK4535) += snd-soc-ak4535.o ++obj-$(CONFIG_SND_SOC_WM9713) += snd-soc-wm9713.o ++obj-$(CONFIG_SND_SOC_WM9712) += snd-soc-wm9712.o +Index: linux-2.6-pxa-new/sound/soc/codecs/ac97.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/ac97.c +@@ -0,0 +1,167 @@ ++/* ++ * ac97.c -- ALSA Soc AC97 codec support ++ * ++ * Copyright 2005 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * 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. ++ * ++ * Revision history ++ * 17th Oct 2005 Initial version. ++ * ++ * Generic AC97 support. ++ */ ++ ++#include <linux/init.h> ++#include <linux/kernel.h> ++#include <linux/device.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/ac97_codec.h> ++#include <sound/initval.h> ++#include <sound/soc.h> ++ ++#define AC97_VERSION "0.5" ++ ++#define AC97_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define AC97_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000) ++ ++/* may need to expand this */ ++static struct snd_soc_dai_mode soc_ac97[] = { ++ {0, 0, SNDRV_PCM_FMTBIT_S16_LE, AC97_RATES}, ++ {0, 0, SNDRV_PCM_FMTBIT_S18_3LE, AC97_RATES}, ++ {0, 0, SNDRV_PCM_FMTBIT_S20_3LE, AC97_RATES}, ++}; ++ ++static int ac97_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ int reg = (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ? ++ AC97_PCM_FRONT_DAC_RATE : AC97_PCM_LR_ADC_RATE; ++ return snd_ac97_set_rate(codec->ac97, reg, runtime->rate); ++} ++ ++static struct snd_soc_codec_dai ac97_dai = { ++ .name = "AC97 HiFi", ++ .playback = { ++ .stream_name = "AC97 Playback", ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .capture = { ++ .stream_name = "AC97 Capture", ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .ops = { ++ .prepare = ac97_prepare,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(soc_ac97), ++ .mode = soc_ac97,}, ++}; ++ ++static unsigned int ac97_read(struct snd_soc_codec *codec, ++ unsigned int reg) ++{ ++ return soc_ac97_ops.read(codec->ac97, reg); ++} ++ ++static int ac97_write(struct snd_soc_codec *codec, unsigned int reg, ++ unsigned int val) ++{ ++ soc_ac97_ops.write(codec->ac97, reg, val); ++ return 0; ++} ++ ++static int ac97_soc_probe(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec; ++ struct snd_ac97_bus *ac97_bus; ++ struct snd_ac97_template ac97_template; ++ int ret = 0; ++ ++ printk(KERN_INFO "AC97 SoC Audio Codec %s\n", AC97_VERSION); ++ ++ socdev->codec = kzalloc(sizeof(struct snd_soc_codec), GFP_KERNEL); ++ if (socdev->codec == NULL) ++ return -ENOMEM; ++ codec = socdev->codec; ++ mutex_init(&codec->mutex); ++ ++ codec->name = "AC97"; ++ codec->owner = THIS_MODULE; ++ codec->dai = &ac97_dai; ++ codec->num_dai = 1; ++ codec->write = ac97_write; ++ codec->read = ac97_read; ++ INIT_LIST_HEAD(&codec->dapm_widgets); ++ INIT_LIST_HEAD(&codec->dapm_paths); ++ ++ /* register pcms */ ++ ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); ++ if(ret < 0) ++ goto err; ++ ++ /* add codec as bus device for standard ac97 */ ++ ret = snd_ac97_bus(codec->card, 0, &soc_ac97_ops, NULL, &ac97_bus); ++ if(ret < 0) ++ goto bus_err; ++ ++ memset(&ac97_template, 0, sizeof(struct snd_ac97_template)); ++ ret = snd_ac97_mixer(ac97_bus, &ac97_template, &codec->ac97); ++ if(ret < 0) ++ goto bus_err; ++ ++ ret = snd_soc_register_card(socdev); ++ if (ret < 0) ++ goto bus_err; ++ return 0; ++ ++bus_err: ++ snd_soc_free_pcms(socdev); ++ ++err: ++ kfree(socdev->codec->reg_cache); ++ kfree(socdev->codec); ++ socdev->codec = NULL; ++ return ret; ++} ++ ++static int ac97_soc_remove(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ if(codec == NULL) ++ return 0; ++ ++ snd_soc_free_pcms(socdev); ++ kfree(socdev->codec->reg_cache); ++ kfree(socdev->codec); ++ ++ return 0; ++} ++ ++struct snd_soc_codec_device soc_codec_dev_ac97= { ++ .probe = ac97_soc_probe, ++ .remove = ac97_soc_remove, ++}; ++ ++EXPORT_SYMBOL_GPL(soc_codec_dev_ac97); ++ ++MODULE_DESCRIPTION("Soc Generic AC97 driver"); ++MODULE_AUTHOR("Liam Girdwood"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/codecs/ac97.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/ac97.h +@@ -0,0 +1,18 @@ ++/* ++ * linux/sound/codecs/ac97.h -- ALSA SoC Layer ++ * ++ * Author: Liam Girdwood ++ * Created: Dec 1st 2005 ++ * Copyright: Wolfson Microelectronics. PLC. ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#ifndef __LINUX_SND_SOC_AC97_H ++#define __LINUX_SND_SOC_AC97_H ++ ++extern struct snd_soc_codec_device soc_codec_dev_ac97; ++ ++#endif +Index: linux-2.6-pxa-new/sound/soc/codecs/ak4535.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/ak4535.c +@@ -0,0 +1,701 @@ ++/* ++ * ak4535.c -- AK4535 ALSA Soc Audio driver ++ * ++ * Copyright 2005 Openedhand Ltd. ++ * ++ * Author: Richard Purdie <richard@openedhand.com> ++ * ++ * Based on wm8753.c by Liam Girdwood ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/init.h> ++#include <linux/delay.h> ++#include <linux/pm.h> ++#include <linux/i2c.h> ++#include <linux/platform_device.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/pcm_params.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++#include <sound/initval.h> ++ ++#include "ak4535.h" ++ ++#define AUDIO_NAME "ak4535" ++#define AK4535_VERSION "0.3" ++ ++struct snd_soc_codec_device soc_codec_dev_ak4535; ++ ++/* ++ * ak4535 register cache ++ */ ++static const u16 ak4535_reg[AK4535_CACHEREGNUM] = { ++ 0x0000, 0x0080, 0x0000, 0x0003, ++ 0x0002, 0x0000, 0x0011, 0x0001, ++ 0x0000, 0x0040, 0x0036, 0x0010, ++ 0x0000, 0x0000, 0x0057, 0x0000, ++}; ++ ++#define AK4535_DAIFMT \ ++ (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_LEFT_J | SND_SOC_DAIFMT_CBS_CFS | \ ++ SND_SOC_DAIFMT_NB_NF) ++ ++#define AK4535_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define AK4535_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000) ++ ++static struct snd_soc_dai_mode ak4535_modes[] = { ++ /* codec frame and clock slave modes */ ++ { ++ .fmt = AK4535_DAIFMT, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = AK4535_RATES, ++ .pcmdir = AK4535_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 256, ++ .bfs = 64, ++ }, ++ { ++ .fmt = AK4535_DAIFMT, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = AK4535_RATES, ++ .pcmdir = AK4535_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 256, ++ .bfs = 32, ++ }, ++}; ++ ++/* ++ * read ak4535 register cache ++ */ ++static inline unsigned int ak4535_read_reg_cache(struct snd_soc_codec *codec, ++ unsigned int reg) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg >= AK4535_CACHEREGNUM) ++ return -1; ++ return cache[reg]; ++} ++ ++/* ++ * write ak4535 register cache ++ */ ++static inline void ak4535_write_reg_cache(struct snd_soc_codec *codec, ++ u16 reg, unsigned int value) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg >= AK4535_CACHEREGNUM) ++ return; ++ cache[reg] = value; ++} ++ ++/* ++ * write to the AK4535 register space ++ */ ++static int ak4535_write(struct snd_soc_codec *codec, unsigned int reg, ++ unsigned int value) ++{ ++ u8 data[2]; ++ ++ /* data is ++ * D15..D8 AK4535 register offset ++ * D7...D0 register data ++ */ ++ data[0] = reg & 0xff; ++ data[1] = value & 0xff; ++ ++ ak4535_write_reg_cache (codec, reg, value); ++ if (codec->hw_write(codec->control_data, data, 2) == 2) ++ return 0; ++ else ++ return -EIO; ++} ++ ++static const char *ak4535_mono_gain[] = {"+6dB", "-17dB"}; ++static const char *ak4535_mono_out[] = {"(L + R)/2", "Hi-Z"}; ++static const char *ak4535_hp_out[] = {"Stereo", "Mono"}; ++static const char *ak4535_deemp[] = {"44.1kHz", "Off", "48kHz", "32kHz"}; ++static const char *ak4535_mic_select[] = {"Internal", "External"}; ++ ++static const struct soc_enum ak4535_enum[] = { ++ SOC_ENUM_SINGLE(AK4535_SIG1, 7, 2, ak4535_mono_gain), ++ SOC_ENUM_SINGLE(AK4535_SIG1, 6, 2, ak4535_mono_out), ++ SOC_ENUM_SINGLE(AK4535_MODE2, 2, 2, ak4535_hp_out), ++ SOC_ENUM_SINGLE(AK4535_DAC, 0, 4, ak4535_deemp), ++ SOC_ENUM_SINGLE(AK4535_MIC, 1, 2, ak4535_mic_select), ++}; ++ ++static const struct snd_kcontrol_new ak4535_snd_controls[] = { ++ SOC_SINGLE("ALC2 Switch", AK4535_SIG1, 1, 1, 0), ++ SOC_ENUM("Mono 1 Output", ak4535_enum[1]), ++ SOC_ENUM("Mono 1 Gain", ak4535_enum[0]), ++ SOC_ENUM("Headphone Output", ak4535_enum[2]), ++ SOC_ENUM("Playback Deemphasis", ak4535_enum[3]), ++ SOC_SINGLE("Bass Volume", AK4535_DAC, 2, 3, 0), ++ SOC_SINGLE("Mic Boost (+20dB) Switch", AK4535_MIC, 0, 1, 0), ++ SOC_ENUM("Mic Select", ak4535_enum[4]), ++ SOC_SINGLE("ALC Operation Time", AK4535_TIMER, 0, 3, 0), ++ SOC_SINGLE("ALC Recovery Time", AK4535_TIMER, 2, 3, 0), ++ SOC_SINGLE("ALC ZC Time", AK4535_TIMER, 4, 3, 0), ++ SOC_SINGLE("ALC 1 Switch", AK4535_ALC1, 5, 1, 0), ++ SOC_SINGLE("ALC 2 Switch", AK4535_ALC1, 6, 1, 0), ++ SOC_SINGLE("ALC Volume", AK4535_ALC2, 0, 127, 0), ++ SOC_SINGLE("Capture Volume", AK4535_PGA, 0, 127, 0), ++ SOC_SINGLE("Left Playback Volume", AK4535_LATT, 0, 127, 1), ++ SOC_SINGLE("Right Playback Volume", AK4535_RATT, 0, 127, 1), ++ SOC_SINGLE("AUX Bypass Volume", AK4535_VOL, 0, 15, 0), ++ SOC_SINGLE("Mic Sidetone Volume", AK4535_VOL, 4, 7, 0), ++}; ++ ++/* add non dapm controls */ ++static int ak4535_add_controls(struct snd_soc_codec *codec) ++{ ++ int err, i; ++ ++ for (i = 0; i < ARRAY_SIZE(ak4535_snd_controls); i++) { ++ err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&ak4535_snd_controls[i],codec, NULL)); ++ if (err < 0) ++ return err; ++ } ++ ++ return 0; ++} ++ ++/* Mono 1 Mixer */ ++static const struct snd_kcontrol_new ak4535_mono1_mixer_controls[] = { ++ SOC_DAPM_SINGLE("Mic Sidetone Switch", AK4535_SIG1, 4, 1, 0), ++ SOC_DAPM_SINGLE("Mono Playback Switch", AK4535_SIG1, 5, 1, 0), ++}; ++ ++/* Stereo Mixer */ ++static const struct snd_kcontrol_new ak4535_stereo_mixer_controls[] = { ++ SOC_DAPM_SINGLE("Mic Sidetone Switch", AK4535_SIG2, 4, 1, 0), ++ SOC_DAPM_SINGLE("Playback Switch", AK4535_SIG2, 7, 1, 0), ++ SOC_DAPM_SINGLE("Aux Bypass Switch", AK4535_SIG2, 5, 1, 0), ++}; ++ ++/* Input Mixer */ ++static const struct snd_kcontrol_new ak4535_input_mixer_controls[] = { ++ SOC_DAPM_SINGLE("Mic Capture Switch", AK4535_MIC, 2, 1, 0), ++ SOC_DAPM_SINGLE("Aux Capture Switch", AK4535_MIC, 5, 1, 0), ++}; ++ ++/* Input mux */ ++static const struct snd_kcontrol_new ak4535_input_mux_control = ++ SOC_DAPM_ENUM("Input Select", ak4535_enum[0]); ++ ++/* HP L switch */ ++static const struct snd_kcontrol_new ak4535_hpl_control = ++ SOC_DAPM_SINGLE("Switch", AK4535_SIG2, 1, 1, 1); ++ ++/* HP R switch */ ++static const struct snd_kcontrol_new ak4535_hpr_control = ++ SOC_DAPM_SINGLE("Switch", AK4535_SIG2, 0, 1, 1); ++ ++/* Speaker switch */ ++static const struct snd_kcontrol_new ak4535_spk_control = ++ SOC_DAPM_SINGLE("Switch", AK4535_MODE2, 0, 0, 0); ++ ++/* mono 2 switch */ ++static const struct snd_kcontrol_new ak4535_mono2_control = ++ SOC_DAPM_SINGLE("Switch", AK4535_SIG1, 0, 1, 0); ++ ++/* Line out switch */ ++static const struct snd_kcontrol_new ak4535_line_control = ++ SOC_DAPM_SINGLE("Switch", AK4535_SIG2, 6, 1, 0); ++ ++/* ak4535 dapm widgets */ ++static const struct snd_soc_dapm_widget ak4535_dapm_widgets[] = { ++ SND_SOC_DAPM_MIXER("Stereo Mixer", SND_SOC_NOPM, 0, 0, ++ &ak4535_stereo_mixer_controls[0], ++ ARRAY_SIZE(ak4535_stereo_mixer_controls)), ++ SND_SOC_DAPM_MIXER("Mono1 Mixer", SND_SOC_NOPM, 0, 0, ++ &ak4535_mono1_mixer_controls[0], ++ ARRAY_SIZE(ak4535_mono1_mixer_controls)), ++ SND_SOC_DAPM_MIXER("Input Mixer", SND_SOC_NOPM, 0, 0, ++ &ak4535_input_mixer_controls[0], ++ ARRAY_SIZE(ak4535_mono1_mixer_controls)), ++ SND_SOC_DAPM_MUX("Input Mux", SND_SOC_NOPM, 0, 0, ++ &ak4535_input_mux_control), ++ SND_SOC_DAPM_DAC("DAC", "Playback", AK4535_PM2, 0, 0), ++ SND_SOC_DAPM_SWITCH("Mono 2 Enable", SND_SOC_NOPM, 0, 0, ++ &ak4535_mono2_control), ++ SND_SOC_DAPM_SWITCH("Speaker Enable", SND_SOC_NOPM, 0, 0, ++ &ak4535_spk_control), ++ SND_SOC_DAPM_SWITCH("Line Out Enable", SND_SOC_NOPM, 0, 0, ++ &ak4535_line_control), ++ SND_SOC_DAPM_SWITCH("Left HP Enable", SND_SOC_NOPM, 0, 0, ++ &ak4535_hpl_control), ++ SND_SOC_DAPM_SWITCH("Right HP Enable", SND_SOC_NOPM, 0, 0, ++ &ak4535_hpr_control), ++ SND_SOC_DAPM_OUTPUT("LOUT"), ++ SND_SOC_DAPM_OUTPUT("HPL"), ++ SND_SOC_DAPM_OUTPUT("ROUT"), ++ SND_SOC_DAPM_OUTPUT("HPR"), ++ SND_SOC_DAPM_OUTPUT("SPP"), ++ SND_SOC_DAPM_OUTPUT("SPN"), ++ SND_SOC_DAPM_OUTPUT("MOUT1"), ++ SND_SOC_DAPM_OUTPUT("MOUT2"), ++ SND_SOC_DAPM_OUTPUT("MICOUT"), ++ SND_SOC_DAPM_ADC("ADC", "Capture", AK4535_PM1, 0, 1), ++ SND_SOC_DAPM_PGA("Spk Amp", AK4535_PM2, 3, 0, NULL, 0), ++ SND_SOC_DAPM_PGA("HP R Amp", AK4535_PM2, 1, 0, NULL, 0), ++ SND_SOC_DAPM_PGA("HP L Amp", AK4535_PM2, 2, 0, NULL, 0), ++ SND_SOC_DAPM_PGA("Mic", AK4535_PM1, 1, 0, NULL, 0), ++ SND_SOC_DAPM_PGA("Line Out", AK4535_PM1, 4, 0, NULL, 0), ++ SND_SOC_DAPM_PGA("Mono Out", AK4535_PM1, 3, 0, NULL, 0), ++ SND_SOC_DAPM_PGA("AUX In", AK4535_PM1, 2, 0, NULL, 0), ++ ++ SND_SOC_DAPM_MICBIAS("Mic Int Bias", AK4535_MIC, 3, 0), ++ SND_SOC_DAPM_MICBIAS("Mic Ext Bias", AK4535_MIC, 4, 0), ++ SND_SOC_DAPM_INPUT("MICIN"), ++ SND_SOC_DAPM_INPUT("MICEXT"), ++ SND_SOC_DAPM_INPUT("AUX"), ++ SND_SOC_DAPM_INPUT("MIN"), ++ SND_SOC_DAPM_INPUT("AIN"), ++}; ++ ++static const char *audio_map[][3] = { ++ /*stereo mixer */ ++ {"Stereo Mixer", "Playback Switch", "DAC"}, ++ {"Stereo Mixer", "Mic Sidetone Switch", "Mic"}, ++ {"Stereo Mixer", "Aux Bypass Switch", "AUX In"}, ++ ++ /* mono1 mixer */ ++ {"Mono1 Mixer", "Mic Sidetone Switch", "Mic"}, ++ {"Mono1 Mixer", "Mono Playback Switch", "DAC"}, ++ ++ /* mono2 mixer */ ++ {"Mono2 Mixer", "Mono Playback Switch", "Stereo Mixer"}, ++ ++ /* Mic */ ++ {"AIN", NULL, "Mic"}, ++ {"Input Mux", "Internal", "Mic Int Bias"}, ++ {"Input Mux", "External", "Mic Ext Bias"}, ++ {"Mic Int Bias", NULL, "MICIN"}, ++ {"Mic Ext Bias", NULL, "MICEXT"}, ++ {"MICOUT", NULL, "Input Mux"}, ++ ++ /* line out */ ++ {"LOUT", "Switch", "Line"}, ++ {"ROUT", "Switch", "Line Out Enable"}, ++ {"Line Out Enable", NULL, "Line Out"}, ++ {"Line Out", NULL, "Stereo Mixer"}, ++ ++ /* mono1 out */ ++ {"MOUT1", NULL, "Mono Out"}, ++ {"Mono Out", NULL, "Mono Mixer"}, ++ ++ /* left HP */ ++ {"HPL", "Switch", "Left HP Enable"}, ++ {"Left HP Enable", NULL, "HP L Amp"}, ++ {"HP L Amp", NULL, "Stereo Mixer"}, ++ ++ /* right HP */ ++ {"HPR", "Switch", "Right HP Enable"}, ++ {"Right HP Enable", NULL, "HP R Amp"}, ++ {"HP R Amp", NULL, "Stereo Mixer"}, ++ ++ /* speaker */ ++ {"SPP", "Switch", "Speaker Enable"}, ++ {"SPN", "Switch", "Speaker Enable"}, ++ {"Speaker Enable", NULL, "Spk Amp"}, ++ {"Spk Amp", NULL, "MIN"}, ++ ++ /* mono 2 */ ++ {"MOUT2", "Switch", "Mono 2 Enable"}, ++ {"Mono 2 Enable", NULL, "Stereo Mixer"}, ++ ++ /* Aux In */ ++ {"Aux In", NULL, "AUX"}, ++ ++ /* ADC */ ++ {"ADC", NULL, "Input Mixer"}, ++ {"Input Mixer", "Mic Capture Switch", "Mic"}, ++ {"Input Mixer", "Aux Capture Switch", "Aux In"}, ++ ++ /* terminator */ ++ {NULL, NULL, NULL}, ++}; ++ ++static int ak4535_add_widgets(struct snd_soc_codec *codec) ++{ ++ int i; ++ ++ for(i = 0; i < ARRAY_SIZE(ak4535_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &ak4535_dapm_widgets[i]); ++ } ++ ++ /* set up audio path audio_mapnects */ ++ for(i = 0; audio_map[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, audio_map[i][0], ++ audio_map[i][1], audio_map[i][2]); ++ } ++ ++ snd_soc_dapm_new_widgets(codec); ++ return 0; ++} ++ ++static int ak4535_pcm_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u8 mode = 0, mode2; ++ int bfs; ++ ++ mode2 = ak4535_read_reg_cache(codec, AK4535_MODE2); ++ bfs = SND_SOC_FSBW_REAL(rtd->codec_dai->dai_runtime.bfs); ++ snd_assert(bfs, return -ENODEV); ++ ++ /* interface format */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_I2S: ++ mode = 0x0002; ++ break; ++ case SND_SOC_DAIFMT_LEFT_J: ++ mode = 0x0001; ++ break; ++ } ++ ++ /* set fs */ ++ switch (rtd->codec_dai->dai_runtime.fs) { ++ case 1024: ++ mode2 |= (0x3 << 5); ++ break; ++ case 512: ++ mode2 |= (0x2 << 5); ++ break; ++ case 256: ++ mode2 |= (0x1 << 5); ++ break; ++ } ++ ++ /* bfs */ ++ if (bfs == 64) ++ mode |= 0x4; ++ ++ /* set rate */ ++ ak4535_write(codec, AK4535_MODE1, mode); ++ ak4535_write(codec, AK4535_MODE2, mode2); ++ ++ return 0; ++} ++ ++static unsigned int ak4535_config_sysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ if (info->fs != 256) ++ return 0; ++ ++ /* we only support 256 FS atm */ ++ if (info->rate * info->fs == clk) { ++ dai->mclk = clk; ++ return clk; ++ } ++ ++ return 0; ++} ++ ++static int ak4535_mute(struct snd_soc_codec *codec, ++ struct snd_soc_codec_dai *dai, int mute) ++{ ++ u16 mute_reg = ak4535_read_reg_cache(codec, AK4535_DAC) & 0xffdf; ++ if (mute) ++ ak4535_write(codec, AK4535_DAC, mute_reg); ++ else ++ ak4535_write(codec, AK4535_DAC, mute_reg | 0x20); ++ return 0; ++} ++ ++static int ak4535_dapm_event(struct snd_soc_codec *codec, int event) ++{ ++ switch (event) { ++ case SNDRV_CTL_POWER_D0: /* full On */ ++ /* vref/mid, clk and osc on, dac unmute, active */ ++ case SNDRV_CTL_POWER_D1: /* partial On */ ++ case SNDRV_CTL_POWER_D2: /* partial On */ ++ break; ++ case SNDRV_CTL_POWER_D3hot: /* Off, with power */ ++ /* everything off except vref/vmid, dac mute, inactive */ ++ ak4535_write(codec, AK4535_PM1, 0x80); ++ ak4535_write(codec, AK4535_PM2, 0x0); ++ break; ++ case SNDRV_CTL_POWER_D3cold: /* Off, without power */ ++ /* everything off, inactive */ ++ ak4535_write(codec, AK4535_PM1, 0x0); ++ ak4535_write(codec, AK4535_PM2, 0x80); ++ break; ++ } ++ codec->dapm_state = event; ++ return 0; ++} ++ ++struct snd_soc_codec_dai ak4535_dai = { ++ .name = "AK4535", ++ .playback = { ++ .stream_name = "Playback", ++ .channels_min = 1, ++ .channels_max = 2, ++ }, ++ .capture = { ++ .stream_name = "Capture", ++ .channels_min = 1, ++ .channels_max = 2, ++ }, ++ .config_sysclk = ak4535_config_sysclk, ++ .digital_mute = ak4535_mute, ++ .ops = { ++ .prepare = ak4535_pcm_prepare, ++ }, ++ .caps = { ++ .num_modes = ARRAY_SIZE(ak4535_modes), ++ .mode = ak4535_modes, ++ }, ++}; ++EXPORT_SYMBOL_GPL(ak4535_dai); ++ ++static int ak4535_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ ak4535_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ return 0; ++} ++ ++static int ak4535_resume(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ int i; ++ u8 data[2]; ++ u16 *cache = codec->reg_cache; ++ ++ /* Sync reg_cache with the hardware */ ++ for (i = 0; i < ARRAY_SIZE(ak4535_reg); i++) { ++ data[0] = (i << 1) | ((cache[i] >> 8) & 0x0001); ++ data[1] = cache[i] & 0x00ff; ++ codec->hw_write(codec->control_data, data, 2); ++ } ++ ak4535_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ ak4535_dapm_event(codec, codec->suspend_dapm_state); ++ return 0; ++} ++ ++/* ++ * initialise the AK4535 driver ++ * register the mixer and dsp interfaces with the kernel ++ */ ++static int ak4535_init(struct snd_soc_device *socdev) ++{ ++ struct snd_soc_codec *codec = socdev->codec; ++ int ret = 0; ++ ++ codec->name = "AK4535"; ++ codec->owner = THIS_MODULE; ++ codec->read = ak4535_read_reg_cache; ++ codec->write = ak4535_write; ++ codec->dapm_event = ak4535_dapm_event; ++ codec->dai = &ak4535_dai; ++ codec->num_dai = 1; ++ codec->reg_cache_size = ARRAY_SIZE(ak4535_reg); ++ codec->reg_cache = ++ kzalloc(sizeof(u16) * ARRAY_SIZE(ak4535_reg), GFP_KERNEL); ++ if (codec->reg_cache == NULL) ++ return -ENOMEM; ++ memcpy(codec->reg_cache, ak4535_reg, ++ sizeof(u16) * ARRAY_SIZE(ak4535_reg)); ++ codec->reg_cache_size = sizeof(u16) * ARRAY_SIZE(ak4535_reg); ++ ++ /* register pcms */ ++ ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); ++ if (ret < 0) { ++ kfree(codec->reg_cache); ++ return ret; ++ } ++ ++ /* power on device */ ++ ak4535_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ ++ ak4535_add_controls(codec); ++ ak4535_add_widgets(codec); ++ ret = snd_soc_register_card(socdev); ++ if (ret < 0) { ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++ } ++ ++ return ret; ++} ++ ++static struct snd_soc_device *ak4535_socdev; ++ ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ ++#define I2C_DRIVERID_AK4535 0xfefe /* liam - need a proper id */ ++ ++static unsigned short normal_i2c[] = { 0, I2C_CLIENT_END }; ++ ++/* Magic definition of all other variables and things */ ++I2C_CLIENT_INSMOD; ++ ++static struct i2c_driver ak4535_i2c_driver; ++static struct i2c_client client_template; ++ ++/* If the i2c layer weren't so broken, we could pass this kind of data ++ around */ ++static int ak4535_codec_probe(struct i2c_adapter *adap, int addr, int kind) ++{ ++ struct snd_soc_device *socdev = ak4535_socdev; ++ struct ak4535_setup_data *setup = socdev->codec_data; ++ struct snd_soc_codec *codec = socdev->codec; ++ struct i2c_client *i2c; ++ int ret; ++ ++ if (addr != setup->i2c_address) ++ return -ENODEV; ++ ++ client_template.adapter = adap; ++ client_template.addr = addr; ++ ++ i2c = kzalloc(sizeof(struct i2c_client), GFP_KERNEL); ++ if (i2c == NULL){ ++ kfree(codec); ++ return -ENOMEM; ++ } ++ memcpy(i2c, &client_template, sizeof(struct i2c_client)); ++ i2c_set_clientdata(i2c, codec); ++ codec->control_data = i2c; ++ ++ ret = i2c_attach_client(i2c); ++ if (ret < 0) { ++ printk(KERN_ERR "failed to attach codec at addr %x\n", addr); ++ goto err; ++ } ++ ++ ret = ak4535_init(socdev); ++ if (ret < 0) { ++ printk(KERN_ERR "failed to initialise AK4535\n"); ++ goto err; ++ } ++ return ret; ++ ++err: ++ kfree(codec); ++ kfree(i2c); ++ return ret; ++} ++ ++static int ak4535_i2c_detach(struct i2c_client *client) ++{ ++ struct snd_soc_codec* codec = i2c_get_clientdata(client); ++ i2c_detach_client(client); ++ kfree(codec->reg_cache); ++ kfree(client); ++ ++ return 0; ++} ++ ++static int ak4535_i2c_attach(struct i2c_adapter *adap) ++{ ++ return i2c_probe(adap, &addr_data, ak4535_codec_probe); ++} ++ ++/* corgi i2c codec control layer */ ++static struct i2c_driver ak4535_i2c_driver = { ++ .driver = { ++ .name = "AK4535 I2C Codec", ++ .owner = THIS_MODULE, ++ }, ++ .id = I2C_DRIVERID_AK4535, ++ .attach_adapter = ak4535_i2c_attach, ++ .detach_client = ak4535_i2c_detach, ++ .command = NULL, ++}; ++ ++static struct i2c_client client_template = { ++ .name = "AK4535", ++ .driver = &ak4535_i2c_driver, ++}; ++#endif ++ ++static int ak4535_probe(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct ak4535_setup_data *setup; ++ struct snd_soc_codec* codec; ++ int ret = 0; ++ ++ printk(KERN_INFO "AK4535 Audio Codec %s", AK4535_VERSION); ++ ++ setup = socdev->codec_data; ++ codec = kzalloc(sizeof(struct snd_soc_codec), GFP_KERNEL); ++ if (codec == NULL) ++ return -ENOMEM; ++ ++ socdev->codec = codec; ++ mutex_init(&codec->mutex); ++ INIT_LIST_HEAD(&codec->dapm_widgets); ++ INIT_LIST_HEAD(&codec->dapm_paths); ++ ++ ak4535_socdev = socdev; ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ if (setup->i2c_address) { ++ normal_i2c[0] = setup->i2c_address; ++ codec->hw_write = (hw_write_t)i2c_master_send; ++ ret = i2c_add_driver(&ak4535_i2c_driver); ++ if (ret != 0) ++ printk(KERN_ERR "can't add i2c driver"); ++ } ++#else ++ /* Add other interfaces here */ ++#endif ++ return ret; ++} ++ ++/* power down chip */ ++static int ak4535_remove(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec* codec = socdev->codec; ++ ++ if (codec->control_data) ++ ak4535_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ i2c_del_driver(&ak4535_i2c_driver); ++#endif ++ kfree(codec); ++ ++ return 0; ++} ++ ++struct snd_soc_codec_device soc_codec_dev_ak4535 = { ++ .probe = ak4535_probe, ++ .remove = ak4535_remove, ++ .suspend = ak4535_suspend, ++ .resume = ak4535_resume, ++}; ++ ++EXPORT_SYMBOL_GPL(soc_codec_dev_ak4535); ++ ++MODULE_DESCRIPTION("Soc AK4535 driver"); ++MODULE_AUTHOR("Richard Purdie"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/codecs/ak4535.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/ak4535.h +@@ -0,0 +1,46 @@ ++/* ++ * ak4535.h -- AK4535 Soc Audio driver ++ * ++ * Copyright 2005 Openedhand Ltd. ++ * ++ * Author: Richard Purdie <richard@openedhand.com> ++ * ++ * Based on wm8753.h ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#ifndef _AK4535_H ++#define _AK4535_H ++ ++/* AK4535 register space */ ++ ++#define AK4535_PM1 0x0 ++#define AK4535_PM2 0x1 ++#define AK4535_SIG1 0x2 ++#define AK4535_SIG2 0x3 ++#define AK4535_MODE1 0x4 ++#define AK4535_MODE2 0x5 ++#define AK4535_DAC 0x6 ++#define AK4535_MIC 0x7 ++#define AK4535_TIMER 0x8 ++#define AK4535_ALC1 0x9 ++#define AK4535_ALC2 0xa ++#define AK4535_PGA 0xb ++#define AK4535_LATT 0xc ++#define AK4535_RATT 0xd ++#define AK4535_VOL 0xe ++#define AK4535_STATUS 0xf ++ ++#define AK4535_CACHEREGNUM 0x10 ++ ++struct ak4535_setup_data { ++ unsigned short i2c_address; ++}; ++ ++extern struct snd_soc_codec_dai ak4535_dai; ++extern struct snd_soc_codec_device soc_codec_dev_ak4535; ++ ++#endif +Index: linux-2.6-pxa-new/sound/soc/codecs/uda1380.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/uda1380.c +@@ -0,0 +1,582 @@ ++/* ++ * uda1380.c - Philips UDA1380 ALSA SoC audio driver ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ * ++ * Modified by Richard Purdie <richard@openedhand.com> to fit into SoC ++ * codec model. ++ * ++ * Copyright (c) 2005 Giorgio Padrin <giorgio@mandarinlogiq.org> ++ * Copyright 2005 Openedhand Ltd. ++ */ ++ ++#include <linux/module.h> ++#include <linux/init.h> ++#include <linux/types.h> ++#include <linux/string.h> ++#include <linux/slab.h> ++#include <linux/errno.h> ++#include <linux/ioctl.h> ++#include <linux/delay.h> ++#include <linux/i2c.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/control.h> ++#include <sound/initval.h> ++#include <sound/info.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++ ++#include "uda1380.h" ++ ++#define UDA1380_VERSION "0.4" ++ ++/* ++ * uda1380 register cache ++ */ ++static const u16 uda1380_reg[UDA1380_CACHEREGNUM] = { ++ 0x0502, 0x0000, 0x0000, 0x3f3f, ++ 0x0202, 0x0000, 0x0000, 0x0000, ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0000, 0xff00, 0x0000, 0x4800, ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0000, 0x8000, 0x0002, 0x0000, ++}; ++ ++#define UDA1380_DAIFMT \ ++ (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_LEFT_J | SND_SOC_DAIFMT_CBS_CFS | \ ++ SND_SOC_DAIFMT_NB_NF) ++ ++#define UDA1380_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define UDA1380_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000) ++ ++static struct snd_soc_dai_mode uda1380_modes[] = { ++ /* slave rates capture & playback */ ++ { ++ .fmt = UDA1380_DAIFMT, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = UDA1380_RATES, ++ .pcmdir = UDA1380_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 256, ++ .bfs = 64, ++ }, ++ ++ /* slave rates playback */ ++ { ++ .fmt = UDA1380_DAIFMT, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 256, ++ .bfs = 64, ++ }, ++}; ++ ++/* ++ * read uda1380 register cache ++ */ ++static inline unsigned int uda1380_read_reg_cache(struct snd_soc_codec *codec, ++ unsigned int reg) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg == UDA1380_RESET) ++ return 0; ++ if (reg >= UDA1380_CACHEREGNUM) ++ return -1; ++ return cache[reg]; ++} ++ ++/* ++ * write uda1380 register cache ++ */ ++static inline void uda1380_write_reg_cache(struct snd_soc_codec *codec, ++ u16 reg, unsigned int value) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg >= UDA1380_CACHEREGNUM) ++ return; ++ cache[reg] = value; ++} ++ ++/* ++ * write to the UDA1380 register space ++ */ ++static int uda1380_write(struct snd_soc_codec *codec, unsigned int reg, ++ unsigned int value) ++{ ++ u8 data[3]; ++ ++ /* data is ++ * data[0] is register offset ++ * data[1] is MS byte ++ * data[2] is LS byte ++ */ ++ data[0] = reg; ++ data[1] = (value & 0xff00) >> 8; ++ data[2] = value & 0x00ff; ++ ++ uda1380_write_reg_cache (codec, reg, value); ++ if (codec->hw_write(codec->control_data, data, 3) == 3) ++ return 0; ++ else ++ return -EIO; ++} ++ ++#define uda1380_reset(c) uda1380_write(c, UDA1380_RESET, 0) ++ ++/* declarations of ALSA reg_elem_REAL controls */ ++static const char *uda1380_deemp[] = {"None", "32kHz", "44.1kHz", "48kHz", ++ "96kHz"}; ++static const char *uda1380_input_sel[] = {"Line", "Mic"}; ++ ++static const struct soc_enum uda1380_enum[] = { ++ SOC_ENUM_DOUBLE(UDA1380_DEEMP, 0, 8, 5, uda1380_deemp), ++ SOC_ENUM_SINGLE(UDA1380_ADC, 3, 2, uda1380_input_sel), ++}; ++ ++static const struct snd_kcontrol_new uda1380_snd_controls[] = { ++ SOC_DOUBLE("Playback Volume", UDA1380_MVOL, 0, 8, 127, 0), ++ SOC_DOUBLE("Treble Volume", UDA1380_MODE, 4, 12, 3, 0), ++ SOC_DOUBLE("Bass Volume", UDA1380_MODE, 0, 8, 15, 0), ++ SOC_ENUM("Playback De-emphasis", uda1380_enum[0]), ++ SOC_DOUBLE("Capture Volume", UDA1380_DEC, 0, 8, 127, 0), ++ SOC_DOUBLE("Line Capture Volume", UDA1380_PGA, 0, 8, 15, 0), ++ SOC_SINGLE("Mic Capture Volume", UDA1380_PGA, 8, 11, 0), ++ SOC_DOUBLE("Playback Switch", UDA1380_DEEMP, 3, 11, 1, 0), ++ SOC_SINGLE("Capture Switch", UDA1380_PGA, 15, 1, 0), ++ SOC_SINGLE("AGC Timing", UDA1380_AGC, 8, 7, 0), ++ SOC_SINGLE("AGC Target level", UDA1380_AGC, 2, 3, 1), ++ SOC_SINGLE("AGC Switch", UDA1380_AGC, 0, 1, 0), ++}; ++ ++/* add non dapm controls */ ++static int uda1380_add_controls(struct snd_soc_codec *codec) ++{ ++ int err, i; ++ ++ for (i = 0; i < ARRAY_SIZE(uda1380_snd_controls); i++) { ++ err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&uda1380_snd_controls[i],codec, NULL)); ++ if (err < 0) ++ return err; ++ } ++ ++ return 0; ++} ++ ++/* Input mux */ ++static const struct snd_kcontrol_new uda1380_input_mux_control = ++ SOC_DAPM_ENUM("Input Select", uda1380_enum[1]); ++ ++static const struct snd_soc_dapm_widget uda1380_dapm_widgets[] = { ++ SND_SOC_DAPM_MUX("Input Mux", SND_SOC_NOPM, 0, 0, ++ &uda1380_input_mux_control), ++ SND_SOC_DAPM_PGA("Left PGA", UDA1380_PM, 3, 0, NULL, 0), ++ SND_SOC_DAPM_PGA("Right PGA", UDA1380_PM, 1, 0, NULL, 0), ++ SND_SOC_DAPM_PGA("Mic LNA", UDA1380_PM, 4, 0, NULL, 0), ++ SND_SOC_DAPM_ADC("Left ADC", "Left Capture", UDA1380_PM, 2, 0), ++ SND_SOC_DAPM_ADC("Right ADC", "Right Capture", UDA1380_PM, 0, 0), ++ SND_SOC_DAPM_INPUT("VINM"), ++ SND_SOC_DAPM_INPUT("VINL"), ++ SND_SOC_DAPM_INPUT("VINR"), ++ SND_SOC_DAPM_MIXER("Analog Mixer", UDA1380_PM, 6, 0, NULL, 0), ++ SND_SOC_DAPM_OUTPUT("VOUTLHP"), ++ SND_SOC_DAPM_OUTPUT("VOUTRHP"), ++ SND_SOC_DAPM_OUTPUT("VOUTL"), ++ SND_SOC_DAPM_OUTPUT("VOUTR"), ++ SND_SOC_DAPM_DAC("DAC", "Playback", UDA1380_PM, 10, 0), ++ SND_SOC_DAPM_PGA("HeadPhone Driver", UDA1380_PM, 13, 0, NULL, 0), ++}; ++ ++static const char *audio_map[][3] = { ++ ++ /* analog mixer setup is different from diagram for dapm */ ++ {"HeadPhone Driver", NULL, "Analog Mixer"}, ++ {"VOUTR", NULL, "Analog Mixer"}, ++ {"VOUTL", NULL, "Analog Mixer"}, ++ {"Analog Mixer", NULL, "VINR"}, ++ {"Analog Mixer", NULL, "VINL"}, ++ {"Analog Mixer", NULL, "DAC"}, ++ ++ /* headphone driver */ ++ {"VOUTLHP", NULL, "HeadPhone Driver"}, ++ {"VOUTRHP", NULL, "HeadPhone Driver"}, ++ ++ /* input mux */ ++ {"Left ADC", NULL, "Input Mux"}, ++ {"Input Mux", "Mic", "Mic LNA"}, ++ {"Input Mux", "Line", "Left PGA"}, ++ ++ /* right input */ ++ {"Right ADC", NULL, "Right PGA"}, ++ ++ /* inputs */ ++ {"Mic LNA", NULL, "VINM"}, ++ {"Left PGA", NULL, "VINL"}, ++ {"Right PGA", NULL, "VINR"}, ++ ++ /* terminator */ ++ {NULL, NULL, NULL}, ++}; ++ ++static int uda1380_add_widgets(struct snd_soc_codec *codec) ++{ ++ int i; ++ ++ for(i = 0; i < ARRAY_SIZE(uda1380_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &uda1380_dapm_widgets[i]); ++ } ++ ++ /* set up audio path audio_mapnects */ ++ for(i = 0; audio_map[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, audio_map[i][0], ++ audio_map[i][1], audio_map[i][2]); ++ } ++ ++ snd_soc_dapm_new_widgets(codec); ++ return 0; ++} ++ ++static int uda1380_pcm_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 clk = uda1380_read_reg_cache(codec, UDA1380_CLK); ++ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ uda1380_write(codec, UDA1380_CLK, R00_EN_DAC | R00_EN_INT | clk); ++ else ++ uda1380_write(codec, UDA1380_CLK, R00_EN_ADC | R00_EN_DEC | clk); ++ ++ return 0; ++} ++ ++static void uda1380_pcm_shutdown(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 clk = uda1380_read_reg_cache(codec, UDA1380_CLK); ++ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ uda1380_write(codec, UDA1380_CLK, ~(R00_EN_DAC | R00_EN_INT) & clk); ++ else ++ uda1380_write(codec, UDA1380_CLK, ~(R00_EN_ADC | R00_EN_DEC) & clk); ++} ++ ++static unsigned int uda1380_config_sysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ if(info->fs != 256) ++ return 0; ++ ++ /* we only support 256 FS atm */ ++ if(info->rate * info->fs == clk) { ++ dai->mclk = clk; ++ return clk; ++ } ++ ++ return 0; ++} ++ ++static int uda1380_mute(struct snd_soc_codec *codec, ++ struct snd_soc_codec_dai *dai, int mute) ++{ ++ u16 mute_reg = uda1380_read_reg_cache(codec, UDA1380_DEEMP) & 0xbfff; ++ if(mute) ++ uda1380_write(codec, UDA1380_DEEMP, mute_reg | 0x4000); ++ else ++ uda1380_write(codec, UDA1380_DEEMP, mute_reg); ++ return 0; ++} ++ ++static int uda1380_dapm_event(struct snd_soc_codec *codec, int event) ++{ ++ switch (event) { ++ case SNDRV_CTL_POWER_D0: /* full On */ ++ case SNDRV_CTL_POWER_D1: /* partial On */ ++ case SNDRV_CTL_POWER_D2: /* partial On */ ++ case SNDRV_CTL_POWER_D3hot: /* Off, with power */ ++ /* everything off except internal bias */ ++ uda1380_write(codec, UDA1380_PM, R02_PON_BIAS); ++ break; ++ case SNDRV_CTL_POWER_D3cold: /* Off, without power */ ++ /* everything off, inactive */ ++ uda1380_write(codec, UDA1380_PM, 0x0); ++ break; ++ } ++ codec->dapm_state = event; ++ return 0; ++} ++ ++struct snd_soc_codec_dai uda1380_dai = { ++ .name = "UDA1380", ++ .playback = { ++ .stream_name = "Playback", ++ .channels_min = 1, ++ .channels_max = 2, ++ }, ++ .capture = { ++ .stream_name = "Capture", ++ .channels_min = 1, ++ .channels_max = 2, ++ }, ++ .config_sysclk = uda1380_config_sysclk, ++ .digital_mute = uda1380_mute, ++ .ops = { ++ .prepare = uda1380_pcm_prepare, ++ .shutdown = uda1380_pcm_shutdown, ++ }, ++ .caps = { ++ .num_modes = ARRAY_SIZE(uda1380_modes), ++ .mode = uda1380_modes, ++ }, ++}; ++EXPORT_SYMBOL_GPL(uda1380_dai); ++ ++static int uda1380_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ uda1380_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ return 0; ++} ++ ++static int uda1380_resume(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ int i; ++ u8 data[2]; ++ u16 *cache = codec->reg_cache; ++ ++ /* Sync reg_cache with the hardware */ ++ for (i = 0; i < ARRAY_SIZE(uda1380_reg); i++) { ++ data[0] = (i << 1) | ((cache[i] >> 8) & 0x0001); ++ data[1] = cache[i] & 0x00ff; ++ codec->hw_write(codec->control_data, data, 2); ++ } ++ uda1380_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ uda1380_dapm_event(codec, codec->suspend_dapm_state); ++ return 0; ++} ++ ++/* ++ * initialise the UDA1380 driver ++ * register the mixer and dsp interfaces with the kernel ++ */ ++static int uda1380_init(struct snd_soc_device *socdev) ++{ ++ struct snd_soc_codec *codec = socdev->codec; ++ int ret = 0; ++ ++ codec->name = "UDA1380"; ++ codec->owner = THIS_MODULE; ++ codec->read = uda1380_read_reg_cache; ++ codec->write = uda1380_write; ++ codec->dapm_event = uda1380_dapm_event; ++ codec->dai = &uda1380_dai; ++ codec->num_dai = 1; ++ codec->reg_cache_size = ARRAY_SIZE(uda1380_reg); ++ codec->reg_cache = ++ kzalloc(sizeof(u16) * ARRAY_SIZE(uda1380_reg), GFP_KERNEL); ++ if (codec->reg_cache == NULL) ++ return -ENOMEM; ++ memcpy(codec->reg_cache, uda1380_reg, ++ sizeof(u16) * ARRAY_SIZE(uda1380_reg)); ++ codec->reg_cache_size = sizeof(u16) * ARRAY_SIZE(uda1380_reg); ++ uda1380_reset(codec); ++ ++ /* register pcms */ ++ ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); ++ if(ret < 0) { ++ kfree(codec->reg_cache); ++ return ret; ++ } ++ ++ /* power on device */ ++ uda1380_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ uda1380_write(codec, UDA1380_CLK, 0); ++ ++ /* uda1380 init */ ++ uda1380_add_controls(codec); ++ uda1380_add_widgets(codec); ++ ret = snd_soc_register_card(socdev); ++ if(ret < 0) { ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++ } ++ ++ return ret; ++} ++ ++static struct snd_soc_device *uda1380_socdev; ++ ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ ++#define I2C_DRIVERID_UDA1380 0xfefe /* liam - need a proper id */ ++ ++static unsigned short normal_i2c[] = { 0, I2C_CLIENT_END }; ++ ++/* Magic definition of all other variables and things */ ++I2C_CLIENT_INSMOD; ++ ++static struct i2c_driver uda1380_i2c_driver; ++static struct i2c_client client_template; ++ ++/* If the i2c layer weren't so broken, we could pass this kind of data ++ around */ ++ ++static int uda1380_codec_probe(struct i2c_adapter *adap, int addr, int kind) ++{ ++ struct snd_soc_device *socdev = uda1380_socdev; ++ struct uda1380_setup_data *setup = socdev->codec_data; ++ struct snd_soc_codec *codec = socdev->codec; ++ struct i2c_client *i2c; ++ int ret; ++ ++ if (addr != setup->i2c_address) ++ return -ENODEV; ++ ++ client_template.adapter = adap; ++ client_template.addr = addr; ++ ++ i2c = kzalloc(sizeof(struct i2c_client), GFP_KERNEL); ++ if (i2c == NULL){ ++ kfree(codec); ++ return -ENOMEM; ++ } ++ memcpy(i2c, &client_template, sizeof(struct i2c_client)); ++ i2c_set_clientdata(i2c, codec); ++ codec->control_data = i2c; ++ ++ ret = i2c_attach_client(i2c); ++ if(ret < 0) { ++ printk(KERN_ERR "failed to attach codec at addr %x\n", addr); ++ goto err; ++ } ++ ++ ret = uda1380_init(socdev); ++ if(ret < 0) { ++ printk(KERN_ERR "failed to initialise UDA1380\n"); ++ goto err; ++ } ++ return ret; ++ ++err: ++ kfree(codec); ++ kfree(i2c); ++ return ret; ++} ++ ++static int uda1380_i2c_detach(struct i2c_client *client) ++{ ++ struct snd_soc_codec* codec = i2c_get_clientdata(client); ++ i2c_detach_client(client); ++ kfree(codec->reg_cache); ++ kfree(client); ++ return 0; ++} ++ ++static int uda1380_i2c_attach(struct i2c_adapter *adap) ++{ ++ return i2c_probe(adap, &addr_data, uda1380_codec_probe); ++} ++ ++/* corgi i2c codec control layer */ ++static struct i2c_driver uda1380_i2c_driver = { ++ .driver = { ++ .name = "UDA1380 I2C Codec", ++ .owner = THIS_MODULE, ++ }, ++ .id = I2C_DRIVERID_UDA1380, ++ .attach_adapter = uda1380_i2c_attach, ++ .detach_client = uda1380_i2c_detach, ++ .command = NULL, ++}; ++ ++static struct i2c_client client_template = { ++ .name = "UDA1380", ++ .driver = &uda1380_i2c_driver, ++}; ++#endif ++ ++static int uda1380_probe(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct uda1380_setup_data *setup; ++ struct snd_soc_codec* codec; ++ int ret = 0; ++ ++ printk(KERN_INFO "UDA1380 Audio Codec %s", UDA1380_VERSION); ++ ++ setup = socdev->codec_data; ++ codec = kzalloc(sizeof(struct snd_soc_codec), GFP_KERNEL); ++ if (codec == NULL) ++ return -ENOMEM; ++ ++ socdev->codec = codec; ++ mutex_init(&codec->mutex); ++ INIT_LIST_HEAD(&codec->dapm_widgets); ++ INIT_LIST_HEAD(&codec->dapm_paths); ++ ++ uda1380_socdev = socdev; ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ if (setup->i2c_address) { ++ normal_i2c[0] = setup->i2c_address; ++ codec->hw_write = (hw_write_t)i2c_master_send; ++ ret = i2c_add_driver(&uda1380_i2c_driver); ++ if (ret != 0) ++ printk(KERN_ERR "can't add i2c driver"); ++ } ++#else ++ /* Add other interfaces here */ ++#endif ++ return ret; ++} ++ ++/* power down chip */ ++static int uda1380_remove(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec* codec = socdev->codec; ++ ++ if (codec->control_data) ++ uda1380_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ i2c_del_driver(&uda1380_i2c_driver); ++#endif ++ kfree(codec); ++ ++ return 0; ++} ++ ++struct snd_soc_codec_device soc_codec_dev_uda1380 = { ++ .probe = uda1380_probe, ++ .remove = uda1380_remove, ++ .suspend = uda1380_suspend, ++ .resume = uda1380_resume, ++}; ++ ++EXPORT_SYMBOL_GPL(soc_codec_dev_uda1380); ++ ++MODULE_AUTHOR("Giorgio Padrin"); ++MODULE_DESCRIPTION("Audio support for codec Philips UDA1380"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/codecs/uda1380.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/uda1380.h +@@ -0,0 +1,56 @@ ++/* ++ * Audio support for Philips UDA1380 ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ * ++ * Copyright (c) 2005 Giorgio Padrin <giorgio@mandarinlogiq.org> ++ */ ++ ++#define UDA1380_CLK 0x00 ++#define UDA1380_IFACE 0x01 ++#define UDA1380_PM 0x02 ++#define UDA1380_AMIX 0x03 ++#define UDA1380_HP 0x04 ++#define UDA1380_MVOL 0x10 ++#define UDA1380_MIXVOL 0x11 ++#define UDA1380_MODE 0x12 ++#define UDA1380_DEEMP 0x13 ++#define UDA1380_MIXER 0x14 ++#define UDA1380_INTSTAT 0x18 ++#define UDA1380_DEC 0x20 ++#define UDA1380_PGA 0x21 ++#define UDA1380_ADC 0x22 ++#define UDA1380_AGC 0x23 ++#define UDA1380_DECSTAT 0x28 ++#define UDA1380_RESET 0x7f ++ ++#define UDA1380_CACHEREGNUM 0x24 ++ ++/* Register flags */ ++#define R00_EN_ADC 0x0800 ++#define R00_EN_DEC 0x0400 ++#define R00_EN_DAC 0x0200 ++#define R00_EN_INT 0x0100 ++#define R02_PON_HP 0x2000 ++#define R02_PON_DAC 0x0400 ++#define R02_PON_BIAS 0x0100 ++#define R02_PON_LNA 0x0010 ++#define R02_PON_PGAL 0x0008 ++#define R02_PON_ADCL 0x0004 ++#define R02_PON_PGAR 0x0002 ++#define R02_PON_ADCR 0x0001 ++#define R13_MTM 0x4000 ++#define R21_MT_ADC 0x8000 ++#define R22_SEL_LNA 0x0008 ++#define R22_SEL_MIC 0x0004 ++#define R22_SKIP_DCFIL 0x0002 ++#define R23_AGC_EN 0x0001 ++ ++struct uda1380_setup_data { ++ unsigned short i2c_address; ++}; ++ ++extern struct snd_soc_codec_dai uda1380_dai; ++extern struct snd_soc_codec_device soc_codec_dev_uda1380; +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8731.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8731.c +@@ -0,0 +1,886 @@ ++/* ++ * wm8731.c -- WM8731 ALSA SoC Audio driver ++ * ++ * Copyright 2005 Openedhand Ltd. ++ * ++ * Author: Richard Purdie <richard@openedhand.com> ++ * ++ * Based on wm8753.c by Liam Girdwood ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/init.h> ++#include <linux/delay.h> ++#include <linux/pm.h> ++#include <linux/i2c.h> ++#include <linux/platform_device.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/pcm_params.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++#include <sound/initval.h> ++ ++#include "wm8731.h" ++ ++#define AUDIO_NAME "wm8731" ++#define WM8731_VERSION "0.12" ++ ++/* ++ * Debug ++ */ ++ ++#define WM8731_DEBUG 0 ++ ++#ifdef WM8731_DEBUG ++#define dbg(format, arg...) \ ++ printk(KERN_DEBUG AUDIO_NAME ": " format "\n" , ## arg) ++#else ++#define dbg(format, arg...) do {} while (0) ++#endif ++#define err(format, arg...) \ ++ printk(KERN_ERR AUDIO_NAME ": " format "\n" , ## arg) ++#define info(format, arg...) \ ++ printk(KERN_INFO AUDIO_NAME ": " format "\n" , ## arg) ++#define warn(format, arg...) \ ++ printk(KERN_WARNING AUDIO_NAME ": " format "\n" , ## arg) ++ ++struct snd_soc_codec_device soc_codec_dev_wm8731; ++ ++/* ++ * wm8731 register cache ++ * We can't read the WM8731 register space when we are ++ * using 2 wire for device control, so we cache them instead. ++ * There is no point in caching the reset register ++ */ ++static const u16 wm8731_reg[WM8731_CACHEREGNUM] = { ++ 0x0097, 0x0097, 0x0079, 0x0079, ++ 0x000a, 0x0008, 0x009f, 0x000a, ++ 0x0000, 0x0000 ++}; ++ ++#define WM8731_DAIFMT \ ++ (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_LEFT_J | SND_SOC_DAIFMT_RIGHT_J | \ ++ SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_NB_IF | SND_SOC_DAIFMT_IB_NF | \ ++ SND_SOC_DAIFMT_IB_IF) ++ ++#define WM8731_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define WM8731_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000) ++ ++#define WM8731_HIFI_BITS \ ++ (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE | \ ++ SNDRV_PCM_FMTBIT_S24_LE | SNDRV_PCM_FMTBIT_S32_LE) ++ ++static struct snd_soc_dai_mode wm8731_modes[] = { ++ /* codec frame and clock master modes */ ++ /* 8k */ ++ { ++ .fmt = WM8731_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8731_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8731_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 1536, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8731_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8731_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8731_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 2304, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8731_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8731_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8731_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 1408, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8731_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8731_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8731_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 2112, ++ .bfs = 64, ++ }, ++ ++ /* 32k */ ++ { ++ .fmt = WM8731_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8731_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = WM8731_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 384, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8731_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8731_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = WM8731_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 576, ++ .bfs = 64, ++ }, ++ ++ /* 44.1k & 48k */ ++ { ++ .fmt = WM8731_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8731_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8731_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 256, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8731_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8731_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8731_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 384, ++ .bfs = 64, ++ }, ++ ++ /* 88.2 & 96k */ ++ { ++ .fmt = WM8731_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8731_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000, ++ .pcmdir = WM8731_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 128, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8731_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8731_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000, ++ .pcmdir = WM8731_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 192, ++ .bfs = 64, ++ }, ++ ++ /* USB codec frame and clock master modes */ ++ /* 8k */ ++ { ++ .fmt = WM8731_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8731_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8731_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1500, ++ .bfs = SND_SOC_FSBD(1), ++ }, ++ ++ /* 44.1k */ ++ { ++ .fmt = WM8731_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8731_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_44100, ++ .pcmdir = WM8731_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 272, ++ .bfs = SND_SOC_FSBD(1), ++ }, ++ ++ /* 48k */ ++ { ++ .fmt = WM8731_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8731_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8731_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 250, ++ .bfs = SND_SOC_FSBD(1), ++ }, ++ ++ /* 88.2k */ ++ { ++ .fmt = WM8731_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8731_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_88200, ++ .pcmdir = WM8731_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 136, ++ .bfs = SND_SOC_FSBD(1), ++ }, ++ ++ /* 96k */ ++ { ++ .fmt = WM8731_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8731_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_96000, ++ .pcmdir = WM8731_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 125, ++ .bfs = SND_SOC_FSBD(1), ++ }, ++ ++ /* codec frame and clock slave modes */ ++ { ++ .fmt = WM8731_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = WM8731_HIFI_BITS, ++ .pcmrate = WM8731_RATES, ++ .pcmdir = WM8731_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = SND_SOC_FSB_ALL, ++ }, ++}; ++ ++/* ++ * read wm8731 register cache ++ */ ++static inline unsigned int wm8731_read_reg_cache(struct snd_soc_codec *codec, ++ unsigned int reg) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg == WM8731_RESET) ++ return 0; ++ if (reg >= WM8731_CACHEREGNUM) ++ return -1; ++ return cache[reg]; ++} ++ ++/* ++ * write wm8731 register cache ++ */ ++static inline void wm8731_write_reg_cache(struct snd_soc_codec *codec, ++ u16 reg, unsigned int value) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg >= WM8731_CACHEREGNUM) ++ return; ++ cache[reg] = value; ++} ++ ++/* ++ * write to the WM8731 register space ++ */ ++static int wm8731_write(struct snd_soc_codec *codec, unsigned int reg, ++ unsigned int value) ++{ ++ u8 data[2]; ++ ++ /* data is ++ * D15..D9 WM8731 register offset ++ * D8...D0 register data ++ */ ++ data[0] = (reg << 1) | ((value >> 8) & 0x0001); ++ data[1] = value & 0x00ff; ++ ++ wm8731_write_reg_cache (codec, reg, value); ++ if (codec->hw_write(codec->control_data, data, 2) == 2) ++ return 0; ++ else ++ return -EIO; ++} ++ ++#define wm8731_reset(c) wm8731_write(c, WM8731_RESET, 0) ++ ++static const char *wm8731_input_select[] = {"Line In", "Mic"}; ++static const char *wm8731_deemph[] = {"None", "32Khz", "44.1Khz", "48Khz"}; ++ ++static const struct soc_enum wm8731_enum[] = { ++ SOC_ENUM_SINGLE(WM8731_APANA, 2, 2, wm8731_input_select), ++ SOC_ENUM_SINGLE(WM8731_APDIGI, 1, 4, wm8731_deemph), ++}; ++ ++static const struct snd_kcontrol_new wm8731_snd_controls[] = { ++ ++SOC_DOUBLE_R("Master Playback Volume", WM8731_LOUT1V, WM8731_ROUT1V, ++ 0, 127, 0), ++SOC_DOUBLE_R("Master Playback ZC Switch", WM8731_LOUT1V, WM8731_ROUT1V, ++ 7, 1, 0), ++ ++SOC_DOUBLE_R("Capture Volume", WM8731_LINVOL, WM8731_RINVOL, 0, 31, 0), ++SOC_DOUBLE_R("Line Capture Switch", WM8731_LINVOL, WM8731_RINVOL, 7, 1, 1), ++ ++SOC_SINGLE("Mic Boost (+20dB)", WM8731_APANA, 0, 1, 0), ++SOC_SINGLE("Capture Mic Switch", WM8731_APANA, 1, 1, 1), ++ ++SOC_SINGLE("Sidetone Playback Volume", WM8731_APANA, 6, 3, 1), ++ ++SOC_SINGLE("ADC High Pass Filter Switch", WM8731_APDIGI, 0, 1, 1), ++SOC_SINGLE("Store DC Offset Switch", WM8731_APDIGI, 4, 1, 0), ++ ++SOC_ENUM("Playback De-emphasis", wm8731_enum[1]), ++}; ++ ++/* add non dapm controls */ ++static int wm8731_add_controls(struct snd_soc_codec *codec) ++{ ++ int err, i; ++ ++ for (i = 0; i < ARRAY_SIZE(wm8731_snd_controls); i++) { ++ if ((err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&wm8731_snd_controls[i],codec, NULL))) < 0) ++ return err; ++ } ++ ++ return 0; ++} ++ ++/* Output Mixer */ ++static const struct snd_kcontrol_new wm8731_output_mixer_controls[] = { ++SOC_DAPM_SINGLE("Line Bypass Switch", WM8731_APANA, 3, 1, 0), ++SOC_DAPM_SINGLE("Mic Sidetone Switch", WM8731_APANA, 5, 1, 0), ++SOC_DAPM_SINGLE("HiFi Playback Switch", WM8731_APANA, 4, 1, 0), ++}; ++ ++/* Input mux */ ++static const struct snd_kcontrol_new wm8731_input_mux_controls = ++SOC_DAPM_ENUM("Input Select", wm8731_enum[0]); ++ ++static const struct snd_soc_dapm_widget wm8731_dapm_widgets[] = { ++SND_SOC_DAPM_MIXER("Output Mixer", WM8731_PWR, 4, 1, ++ &wm8731_output_mixer_controls[0], ++ ARRAY_SIZE(wm8731_output_mixer_controls)), ++SND_SOC_DAPM_DAC("DAC", "HiFi Playback", WM8731_PWR, 3, 1), ++SND_SOC_DAPM_OUTPUT("LOUT"), ++SND_SOC_DAPM_OUTPUT("LHPOUT"), ++SND_SOC_DAPM_OUTPUT("ROUT"), ++SND_SOC_DAPM_OUTPUT("RHPOUT"), ++SND_SOC_DAPM_ADC("ADC", "HiFi Capture", WM8731_PWR, 2, 1), ++SND_SOC_DAPM_MUX("Input Mux", SND_SOC_NOPM, 0, 0, &wm8731_input_mux_controls), ++SND_SOC_DAPM_PGA("Line Input", WM8731_PWR, 0, 1, NULL, 0), ++SND_SOC_DAPM_MICBIAS("Mic Bias", WM8731_PWR, 1, 1), ++SND_SOC_DAPM_INPUT("MICIN"), ++SND_SOC_DAPM_INPUT("RLINEIN"), ++SND_SOC_DAPM_INPUT("LLINEIN"), ++}; ++ ++static const char *intercon[][3] = { ++ /* output mixer */ ++ {"Output Mixer", "Line Bypass Switch", "Line Input"}, ++ {"Output Mixer", "HiFi Playback Switch", "DAC"}, ++ {"Output Mixer", "Mic Sidetone Switch", "Mic Bias"}, ++ ++ /* outputs */ ++ {"RHPOUT", NULL, "Output Mixer"}, ++ {"ROUT", NULL, "Output Mixer"}, ++ {"LHPOUT", NULL, "Output Mixer"}, ++ {"LOUT", NULL, "Output Mixer"}, ++ ++ /* input mux */ ++ {"Input Mux", "Line In", "Line Input"}, ++ {"Input Mux", "Mic", "Mic Bias"}, ++ {"ADC", NULL, "Input Mux"}, ++ ++ /* inputs */ ++ {"Line Input", NULL, "LLINEIN"}, ++ {"Line Input", NULL, "RLINEIN"}, ++ {"Mic Bias", NULL, "MICIN"}, ++ ++ /* terminator */ ++ {NULL, NULL, NULL}, ++}; ++ ++static int wm8731_add_widgets(struct snd_soc_codec *codec) ++{ ++ int i; ++ ++ for(i = 0; i < ARRAY_SIZE(wm8731_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &wm8731_dapm_widgets[i]); ++ } ++ ++ /* set up audio path interconnects */ ++ for(i = 0; intercon[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, intercon[i][0], ++ intercon[i][1], intercon[i][2]); ++ } ++ ++ snd_soc_dapm_new_widgets(codec); ++ return 0; ++} ++ ++struct _coeff_div { ++ u32 mclk; ++ u32 rate; ++ u16 fs; ++ u8 sr:4; ++ u8 bosr:1; ++ u8 usb:1; ++}; ++ ++/* codec mclk clock divider coefficients */ ++static const struct _coeff_div coeff_div[] = { ++ /* 48k */ ++ {12288000, 48000, 256, 0x0, 0x0, 0x0}, ++ {18432000, 48000, 384, 0x0, 0x1, 0x0}, ++ {12000000, 48000, 250, 0x0, 0x0, 0x1}, ++ ++ /* 32k */ ++ {12288000, 32000, 384, 0x6, 0x0, 0x0}, ++ {18432000, 32000, 576, 0x6, 0x1, 0x0}, ++ ++ /* 8k */ ++ {12288000, 8000, 1536, 0x3, 0x0, 0x0}, ++ {18432000, 8000, 2304, 0x3, 0x1, 0x0}, ++ {11289600, 8000, 1408, 0xb, 0x0, 0x0}, ++ {16934400, 8000, 2112, 0xb, 0x1, 0x0}, ++ {12000000, 8000, 1500, 0x3, 0x0, 0x1}, ++ ++ /* 96k */ ++ {12288000, 96000, 128, 0x7, 0x0, 0x0}, ++ {18432000, 96000, 192, 0x7, 0x1, 0x0}, ++ {12000000, 96000, 125, 0x7, 0x0, 0x1}, ++ ++ /* 44.1k */ ++ {11289600, 44100, 256, 0x8, 0x0, 0x0}, ++ {16934400, 44100, 384, 0x8, 0x1, 0x0}, ++ {12000000, 44100, 272, 0x8, 0x1, 0x1}, ++ ++ /* 88.2k */ ++ {11289600, 88200, 128, 0xf, 0x0, 0x0}, ++ {16934400, 88200, 192, 0xf, 0x1, 0x0}, ++ {12000000, 88200, 136, 0xf, 0x1, 0x1}, ++}; ++ ++static inline int get_coeff(int mclk, int rate) ++{ ++ int i; ++ ++ for (i = 0; i < ARRAY_SIZE(coeff_div); i++) { ++ if (coeff_div[i].rate == rate && coeff_div[i].mclk == mclk) ++ return i; ++ } ++ return 0; ++} ++ ++/* WM8731 supports numerous clocks per sample rate */ ++static unsigned int wm8731_config_sysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ dai->mclk = 0; ++ ++ /* check that the calculated FS and rate actually match a clock from ++ * the machine driver */ ++ if (info->fs * info->rate == clk) ++ dai->mclk = clk; ++ ++ return dai->mclk; ++} ++ ++static int wm8731_pcm_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 iface = 0, srate; ++ int i = get_coeff(rtd->codec_dai->mclk, ++ snd_soc_get_rate(rtd->codec_dai->dai_runtime.pcmrate)); ++ ++ /* set master/slave audio interface */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK) { ++ case SND_SOC_DAIFMT_CBM_CFM: ++ iface |= 0x0040; ++ break; ++ case SND_SOC_DAIFMT_CBS_CFS: ++ break; ++ } ++ srate = (coeff_div[i].sr << 2) | ++ (coeff_div[i].bosr << 1) | coeff_div[i].usb; ++ wm8731_write(codec, WM8731_SRATE, srate); ++ ++ /* interface format */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_I2S: ++ iface |= 0x0002; ++ break; ++ case SND_SOC_DAIFMT_RIGHT_J: ++ break; ++ case SND_SOC_DAIFMT_LEFT_J: ++ iface |= 0x0001; ++ break; ++ case SND_SOC_DAIFMT_DSP_A: ++ iface |= 0x0003; ++ break; ++ case SND_SOC_DAIFMT_DSP_B: ++ iface |= 0x0013; ++ break; ++ } ++ ++ /* bit size */ ++ switch (rtd->codec_dai->dai_runtime.pcmfmt) { ++ case SNDRV_PCM_FMTBIT_S16_LE: ++ break; ++ case SNDRV_PCM_FMTBIT_S20_3LE: ++ iface |= 0x0004; ++ break; ++ case SNDRV_PCM_FMTBIT_S24_LE: ++ iface |= 0x0008; ++ break; ++ case SNDRV_PCM_FMTBIT_S32_LE: ++ iface |= 0x000c; ++ break; ++ } ++ ++ /* clock inversion */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_INV_MASK) { ++ case SND_SOC_DAIFMT_NB_NF: ++ break; ++ case SND_SOC_DAIFMT_IB_IF: ++ iface |= 0x0090; ++ break; ++ case SND_SOC_DAIFMT_IB_NF: ++ iface |= 0x0080; ++ break; ++ case SND_SOC_DAIFMT_NB_IF: ++ iface |= 0x0010; ++ break; ++ } ++ ++ /* set iface */ ++ wm8731_write(codec, WM8731_IFACE, iface); ++ ++ /* set active */ ++ wm8731_write(codec, WM8731_ACTIVE, 0x0001); ++ return 0; ++} ++ ++static void wm8731_shutdown(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ /* deactivate */ ++ if (!codec->active) { ++ udelay(50); ++ wm8731_write(codec, WM8731_ACTIVE, 0x0); ++ } ++} ++ ++static int wm8731_mute(struct snd_soc_codec *codec, ++ struct snd_soc_codec_dai *dai, int mute) ++{ ++ u16 mute_reg = wm8731_read_reg_cache(codec, WM8731_APDIGI) & 0xfff7; ++ if (mute) ++ wm8731_write(codec, WM8731_APDIGI, mute_reg | 0x8); ++ else ++ wm8731_write(codec, WM8731_APDIGI, mute_reg); ++ return 0; ++} ++ ++static int wm8731_dapm_event(struct snd_soc_codec *codec, int event) ++{ ++ u16 reg = wm8731_read_reg_cache(codec, WM8731_PWR) & 0xff7f; ++ ++ switch (event) { ++ case SNDRV_CTL_POWER_D0: /* full On */ ++ /* vref/mid, osc on, dac unmute */ ++ wm8731_write(codec, WM8731_PWR, reg); ++ break; ++ case SNDRV_CTL_POWER_D1: /* partial On */ ++ case SNDRV_CTL_POWER_D2: /* partial On */ ++ break; ++ case SNDRV_CTL_POWER_D3hot: /* Off, with power */ ++ /* everything off except vref/vmid, */ ++ wm8731_write(codec, WM8731_PWR, reg | 0x0040); ++ break; ++ case SNDRV_CTL_POWER_D3cold: /* Off, without power */ ++ /* everything off, dac mute, inactive */ ++ wm8731_write(codec, WM8731_ACTIVE, 0x0); ++ wm8731_write(codec, WM8731_PWR, 0xffff); ++ break; ++ } ++ codec->dapm_state = event; ++ return 0; ++} ++ ++struct snd_soc_codec_dai wm8731_dai = { ++ .name = "WM8731", ++ .playback = { ++ .stream_name = "Playback", ++ .channels_min = 1, ++ .channels_max = 2, ++ }, ++ .capture = { ++ .stream_name = "Capture", ++ .channels_min = 1, ++ .channels_max = 2, ++ }, ++ .config_sysclk = wm8731_config_sysclk, ++ .digital_mute = wm8731_mute, ++ .ops = { ++ .prepare = wm8731_pcm_prepare, ++ .shutdown = wm8731_shutdown, ++ }, ++ .caps = { ++ .num_modes = ARRAY_SIZE(wm8731_modes), ++ .mode = wm8731_modes, ++ }, ++}; ++EXPORT_SYMBOL_GPL(wm8731_dai); ++ ++static int wm8731_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ wm8731_write(codec, WM8731_ACTIVE, 0x0); ++ wm8731_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ return 0; ++} ++ ++static int wm8731_resume(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ int i; ++ u8 data[2]; ++ u16 *cache = codec->reg_cache; ++ ++ /* Sync reg_cache with the hardware */ ++ for (i = 0; i < ARRAY_SIZE(wm8731_reg); i++) { ++ data[0] = (i << 1) | ((cache[i] >> 8) & 0x0001); ++ data[1] = cache[i] & 0x00ff; ++ codec->hw_write(codec->control_data, data, 2); ++ } ++ wm8731_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ wm8731_dapm_event(codec, codec->suspend_dapm_state); ++ return 0; ++} ++ ++/* ++ * initialise the WM8731 driver ++ * register the mixer and dsp interfaces with the kernel ++ */ ++static int wm8731_init(struct snd_soc_device *socdev) ++{ ++ struct snd_soc_codec *codec = socdev->codec; ++ int reg, ret = 0; ++ ++ codec->name = "WM8731"; ++ codec->owner = THIS_MODULE; ++ codec->read = wm8731_read_reg_cache; ++ codec->write = wm8731_write; ++ codec->dapm_event = wm8731_dapm_event; ++ codec->dai = &wm8731_dai; ++ codec->num_dai = 1; ++ codec->reg_cache_size = ARRAY_SIZE(wm8731_reg); ++ ++ codec->reg_cache = ++ kzalloc(sizeof(u16) * ARRAY_SIZE(wm8731_reg), GFP_KERNEL); ++ if (codec->reg_cache == NULL) ++ return -ENOMEM; ++ memcpy(codec->reg_cache, ++ wm8731_reg, sizeof(u16) * ARRAY_SIZE(wm8731_reg)); ++ codec->reg_cache_size = sizeof(u16) * ARRAY_SIZE(wm8731_reg); ++ ++ wm8731_reset(codec); ++ ++ /* register pcms */ ++ ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); ++ if (ret < 0) { ++ kfree(codec->reg_cache); ++ return ret; ++ } ++ ++ /* power on device */ ++ wm8731_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ ++ /* set the update bits */ ++ reg = wm8731_read_reg_cache(codec, WM8731_LOUT1V); ++ wm8731_write(codec, WM8731_LOUT1V, reg | 0x0100); ++ reg = wm8731_read_reg_cache(codec, WM8731_ROUT1V); ++ wm8731_write(codec, WM8731_ROUT1V, reg | 0x0100); ++ reg = wm8731_read_reg_cache(codec, WM8731_LINVOL); ++ wm8731_write(codec, WM8731_LINVOL, reg | 0x0100); ++ reg = wm8731_read_reg_cache(codec, WM8731_RINVOL); ++ wm8731_write(codec, WM8731_RINVOL, reg | 0x0100); ++ ++ wm8731_add_controls(codec); ++ wm8731_add_widgets(codec); ++ ret = snd_soc_register_card(socdev); ++ if (ret < 0) { ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++ } ++ ++ return ret; ++} ++ ++static struct snd_soc_device *wm8731_socdev; ++ ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ ++/* ++ * WM8731 2 wire address is determined by GPIO5 ++ * state during powerup. ++ * low = 0x1a ++ * high = 0x1b ++ */ ++static unsigned short normal_i2c[] = { 0, I2C_CLIENT_END }; ++ ++/* Magic definition of all other variables and things */ ++I2C_CLIENT_INSMOD; ++ ++static struct i2c_driver wm8731_i2c_driver; ++static struct i2c_client client_template; ++ ++/* If the i2c layer weren't so broken, we could pass this kind of data ++ around */ ++ ++static int wm8731_codec_probe(struct i2c_adapter *adap, int addr, int kind) ++{ ++ struct snd_soc_device *socdev = wm8731_socdev; ++ struct wm8731_setup_data *setup = socdev->codec_data; ++ struct snd_soc_codec *codec = socdev->codec; ++ struct i2c_client *i2c; ++ int ret; ++ ++ if (addr != setup->i2c_address) ++ return -ENODEV; ++ ++ client_template.adapter = adap; ++ client_template.addr = addr; ++ ++ i2c = kzalloc(sizeof(struct i2c_client), GFP_KERNEL); ++ if (i2c == NULL) { ++ kfree(codec); ++ return -ENOMEM; ++ } ++ memcpy(i2c, &client_template, sizeof(struct i2c_client)); ++ i2c_set_clientdata(i2c, codec); ++ codec->control_data = i2c; ++ ++ ret = i2c_attach_client(i2c); ++ if (ret < 0) { ++ err("failed to attach codec at addr %x\n", addr); ++ goto err; ++ } ++ ++ ret = wm8731_init(socdev); ++ if (ret < 0) { ++ err("failed to initialise WM8731\n"); ++ goto err; ++ } ++ return ret; ++ ++err: ++ kfree(codec); ++ kfree(i2c); ++ return ret; ++} ++ ++static int wm8731_i2c_detach(struct i2c_client *client) ++{ ++ struct snd_soc_codec* codec = i2c_get_clientdata(client); ++ i2c_detach_client(client); ++ kfree(codec->reg_cache); ++ kfree(client); ++ return 0; ++} ++ ++static int wm8731_i2c_attach(struct i2c_adapter *adap) ++{ ++ return i2c_probe(adap, &addr_data, wm8731_codec_probe); ++} ++ ++/* corgi i2c codec control layer */ ++static struct i2c_driver wm8731_i2c_driver = { ++ .driver = { ++ .name = "WM8731 I2C Codec", ++ .owner = THIS_MODULE, ++ }, ++ .id = I2C_DRIVERID_WM8731, ++ .attach_adapter = wm8731_i2c_attach, ++ .detach_client = wm8731_i2c_detach, ++ .command = NULL, ++}; ++ ++static struct i2c_client client_template = { ++ .name = "WM8731", ++ .driver = &wm8731_i2c_driver, ++}; ++#endif ++ ++static int wm8731_probe(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct wm8731_setup_data *setup; ++ struct snd_soc_codec *codec; ++ int ret = 0; ++ ++ info("WM8731 Audio Codec %s", WM8731_VERSION); ++ ++ setup = socdev->codec_data; ++ codec = kzalloc(sizeof(struct snd_soc_codec), GFP_KERNEL); ++ if (codec == NULL) ++ return -ENOMEM; ++ ++ socdev->codec = codec; ++ mutex_init(&codec->mutex); ++ INIT_LIST_HEAD(&codec->dapm_widgets); ++ INIT_LIST_HEAD(&codec->dapm_paths); ++ ++ wm8731_socdev = socdev; ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ if (setup->i2c_address) { ++ normal_i2c[0] = setup->i2c_address; ++ codec->hw_write = (hw_write_t)i2c_master_send; ++ ret = i2c_add_driver(&wm8731_i2c_driver); ++ if (ret != 0) ++ printk(KERN_ERR "can't add i2c driver"); ++ } ++#else ++ /* Add other interfaces here */ ++#endif ++ return ret; ++} ++ ++/* power down chip */ ++static int wm8731_remove(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ if (codec->control_data) ++ wm8731_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ i2c_del_driver(&wm8731_i2c_driver); ++#endif ++ kfree(codec); ++ ++ return 0; ++} ++ ++struct snd_soc_codec_device soc_codec_dev_wm8731 = { ++ .probe = wm8731_probe, ++ .remove = wm8731_remove, ++ .suspend = wm8731_suspend, ++ .resume = wm8731_resume, ++}; ++ ++EXPORT_SYMBOL_GPL(soc_codec_dev_wm8731); ++ ++MODULE_DESCRIPTION("ASoC WM8731 driver"); ++MODULE_AUTHOR("Richard Purdie"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8731.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8731.h +@@ -0,0 +1,41 @@ ++/* ++ * wm8731.h -- WM8731 Soc Audio driver ++ * ++ * Copyright 2005 Openedhand Ltd. ++ * ++ * Author: Richard Purdie <richard@openedhand.com> ++ * ++ * Based on wm8753.h ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#ifndef _WM8731_H ++#define _WM8731_H ++ ++/* WM8731 register space */ ++ ++#define WM8731_LINVOL 0x00 ++#define WM8731_RINVOL 0x01 ++#define WM8731_LOUT1V 0x02 ++#define WM8731_ROUT1V 0x03 ++#define WM8731_APANA 0x04 ++#define WM8731_APDIGI 0x05 ++#define WM8731_PWR 0x06 ++#define WM8731_IFACE 0x07 ++#define WM8731_SRATE 0x08 ++#define WM8731_ACTIVE 0x09 ++#define WM8731_RESET 0x0f ++ ++#define WM8731_CACHEREGNUM 10 ++ ++struct wm8731_setup_data { ++ unsigned short i2c_address; ++}; ++ ++extern struct snd_soc_codec_dai wm8731_dai; ++extern struct snd_soc_codec_device soc_codec_dev_wm8731; ++ ++#endif +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8750.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8750.c +@@ -0,0 +1,1282 @@ ++/* ++ * wm8750.c -- WM8750 ALSA SoC audio driver ++ * ++ * Copyright 2005 Openedhand Ltd. ++ * ++ * Author: Richard Purdie <richard@openedhand.com> ++ * ++ * Based on WM8753.c ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/init.h> ++#include <linux/delay.h> ++#include <linux/pm.h> ++#include <linux/i2c.h> ++#include <linux/platform_device.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/pcm_params.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++#include <sound/initval.h> ++ ++#include "wm8750.h" ++ ++#define AUDIO_NAME "WM8750" ++#define WM8750_VERSION "0.11" ++ ++/* ++ * Debug ++ */ ++ ++#define WM8750_DEBUG 0 ++ ++#ifdef WM8750_DEBUG ++#define dbg(format, arg...) \ ++ printk(KERN_DEBUG AUDIO_NAME ": " format "\n" , ## arg) ++#else ++#define dbg(format, arg...) do {} while (0) ++#endif ++#define err(format, arg...) \ ++ printk(KERN_ERR AUDIO_NAME ": " format "\n" , ## arg) ++#define info(format, arg...) \ ++ printk(KERN_INFO AUDIO_NAME ": " format "\n" , ## arg) ++#define warn(format, arg...) \ ++ printk(KERN_WARNING AUDIO_NAME ": " format "\n" , ## arg) ++ ++static struct workqueue_struct *wm8750_workq = NULL; ++static struct work_struct wm8750_dapm_work; ++ ++/* ++ * wm8750 register cache ++ * We can't read the WM8750 register space when we ++ * are using 2 wire for device control, so we cache them instead. ++ */ ++static const u16 wm8750_reg[] = { ++ 0x0097, 0x0097, 0x0079, 0x0079, /* 0 */ ++ 0x0000, 0x0008, 0x0000, 0x000a, /* 4 */ ++ 0x0000, 0x0000, 0x00ff, 0x00ff, /* 8 */ ++ 0x000f, 0x000f, 0x0000, 0x0000, /* 12 */ ++ 0x0000, 0x007b, 0x0000, 0x0032, /* 16 */ ++ 0x0000, 0x00c3, 0x00c3, 0x00c0, /* 20 */ ++ 0x0000, 0x0000, 0x0000, 0x0000, /* 24 */ ++ 0x0000, 0x0000, 0x0000, 0x0000, /* 28 */ ++ 0x0000, 0x0000, 0x0050, 0x0050, /* 32 */ ++ 0x0050, 0x0050, 0x0050, 0x0050, /* 36 */ ++ 0x0079, 0x0079, 0x0079, /* 40 */ ++}; ++ ++#define WM8750_HIFI_DAIFMT \ ++ (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_LEFT_J | SND_SOC_DAIFMT_RIGHT_J | \ ++ SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_NB_IF | SND_SOC_DAIFMT_IB_NF | \ ++ SND_SOC_DAIFMT_IB_IF) ++ ++#define WM8750_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define WM8750_HIFI_FSB \ ++ (SND_SOC_FSBD(1) | SND_SOC_FSBD(2) | SND_SOC_FSBD(4) | \ ++ SND_SOC_FSBD(8) | SND_SOC_FSBD(16)) ++ ++#define WM8750_HIFI_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000) ++ ++#define WM8750_HIFI_BITS \ ++ (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE | \ ++ SNDRV_PCM_FMTBIT_S24_LE | SNDRV_PCM_FMTBIT_S32_LE) ++ ++static struct snd_soc_dai_mode wm8750_modes[] = { ++ /* common codec frame and clock master modes */ ++ /* 8k */ ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1536, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1408, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 2304, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 2112, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1500, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ ++ /* 11.025k */ ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_11025, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1024, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_11025, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1536, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_11025, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1088, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ ++ /* 16k */ ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_16000, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 768, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_16000, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1152, ++ .bfs = WM8750_HIFI_FSB ++ }, ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_16000, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 750, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ ++ /* 22.05k */ ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_22050, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 512, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_22050, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 768, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_22050, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 544, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ ++ /* 32k */ ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 384, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 576, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 375, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ ++ /* 44.1k & 48k */ ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 384, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_44100, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 272, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 250, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ ++ /* 88.2k & 96k */ ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 128, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 192, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_88200, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 136, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_96000, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 125, ++ .bfs = WM8750_HIFI_FSB, ++ }, ++ ++ /* codec frame and clock slave modes */ ++ { ++ .fmt = WM8750_HIFI_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = WM8750_HIFI_BITS, ++ .pcmrate = WM8750_HIFI_RATES, ++ .pcmdir = WM8750_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = SND_SOC_FSB_ALL, ++ }, ++}; ++ ++/* ++ * read wm8750 register cache ++ */ ++static inline unsigned int wm8750_read_reg_cache(struct snd_soc_codec *codec, ++ unsigned int reg) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg > WM8750_CACHE_REGNUM) ++ return -1; ++ return cache[reg]; ++} ++ ++/* ++ * write wm8750 register cache ++ */ ++static inline void wm8750_write_reg_cache(struct snd_soc_codec *codec, ++ unsigned int reg, unsigned int value) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg > WM8750_CACHE_REGNUM) ++ return; ++ cache[reg] = value; ++} ++ ++static int wm8750_write(struct snd_soc_codec *codec, unsigned int reg, ++ unsigned int value) ++{ ++ u8 data[2]; ++ ++ /* data is ++ * D15..D9 WM8753 register offset ++ * D8...D0 register data ++ */ ++ data[0] = (reg << 1) | ((value >> 8) & 0x0001); ++ data[1] = value & 0x00ff; ++ ++ wm8750_write_reg_cache (codec, reg, value); ++ if (codec->hw_write(codec->control_data, data, 2) == 2) ++ return 0; ++ else ++ return -EIO; ++} ++ ++#define wm8750_reset(c) wm8750_write(c, WM8750_RESET, 0) ++ ++/* ++ * WM8750 Controls ++ */ ++static const char *wm8750_bass[] = {"Linear Control", "Adaptive Boost"}; ++static const char *wm8750_bass_filter[] = { "130Hz @ 48kHz", "200Hz @ 48kHz" }; ++static const char *wm8750_treble[] = {"8kHz", "4kHz"}; ++static const char *wm8750_3d_lc[] = {"200Hz", "500Hz"}; ++static const char *wm8750_3d_uc[] = {"2.2kHz", "1.5kHz"}; ++static const char *wm8750_3d_func[] = {"Capture", "Playback"}; ++static const char *wm8750_alc_func[] = {"Off", "Right", "Left", "Stereo"}; ++static const char *wm8750_ng_type[] = {"Constant PGA Gain", ++ "Mute ADC Output"}; ++static const char *wm8750_line_mux[] = {"Line 1", "Line 2", "Line 3", "PGA", ++ "Differential"}; ++static const char *wm8750_pga_sel[] = {"Line 1", "Line 2", "Line 3", ++ "Differential"}; ++static const char *wm8750_out3[] = {"VREF", "ROUT1 + Vol", "MonoOut", ++ "ROUT1"}; ++static const char *wm8750_diff_sel[] = {"Line 1", "Line 2"}; ++static const char *wm8750_adcpol[] = {"Normal", "L Invert", "R Invert", ++ "L + R Invert"}; ++static const char *wm8750_deemph[] = {"None", "32Khz", "44.1Khz", "48Khz"}; ++static const char *wm8750_mono_mux[] = {"Stereo", "Mono (Left)", ++ "Mono (Right)", "Digital Mono"}; ++ ++static const struct soc_enum wm8750_enum[] = { ++SOC_ENUM_SINGLE(WM8750_BASS, 7, 2, wm8750_bass), ++SOC_ENUM_SINGLE(WM8750_BASS, 6, 2, wm8750_bass_filter), ++SOC_ENUM_SINGLE(WM8750_TREBLE, 6, 2, wm8750_treble), ++SOC_ENUM_SINGLE(WM8750_3D, 5, 2, wm8750_3d_lc), ++SOC_ENUM_SINGLE(WM8750_3D, 6, 2, wm8750_3d_uc), ++SOC_ENUM_SINGLE(WM8750_3D, 7, 2, wm8750_3d_func), ++SOC_ENUM_SINGLE(WM8750_ALC1, 7, 4, wm8750_alc_func), ++SOC_ENUM_SINGLE(WM8750_NGATE, 1, 2, wm8750_ng_type), ++SOC_ENUM_SINGLE(WM8750_LOUTM1, 0, 5, wm8750_line_mux), ++SOC_ENUM_SINGLE(WM8750_ROUTM1, 0, 5, wm8750_line_mux), ++SOC_ENUM_SINGLE(WM8750_LADCIN, 6, 4, wm8750_pga_sel), /* 10 */ ++SOC_ENUM_SINGLE(WM8750_RADCIN, 6, 4, wm8750_pga_sel), ++SOC_ENUM_SINGLE(WM8750_ADCTL2, 7, 4, wm8750_out3), ++SOC_ENUM_SINGLE(WM8750_ADCIN, 8, 2, wm8750_diff_sel), ++SOC_ENUM_SINGLE(WM8750_ADCDAC, 5, 4, wm8750_adcpol), ++SOC_ENUM_SINGLE(WM8750_ADCDAC, 1, 4, wm8750_deemph), ++SOC_ENUM_SINGLE(WM8750_ADCIN, 6, 4, wm8750_mono_mux), /* 16 */ ++ ++}; ++ ++static const struct snd_kcontrol_new wm8750_snd_controls[] = { ++ ++SOC_DOUBLE_R("Capture Volume", WM8750_LINVOL, WM8750_RINVOL, 0, 63, 0), ++SOC_DOUBLE_R("Capture ZC Switch", WM8750_LINVOL, WM8750_RINVOL, 6, 1, 0), ++SOC_DOUBLE_R("Capture Switch", WM8750_LINVOL, WM8750_RINVOL, 7, 1, 1), ++ ++SOC_DOUBLE_R("Headphone Playback ZC Switch", WM8750_LOUT1V, ++ WM8750_ROUT1V, 7, 1, 0), ++SOC_DOUBLE_R("Speaker Playback ZC Switch", WM8750_LOUT2V, ++ WM8750_ROUT2V, 7, 1, 0), ++ ++SOC_ENUM("Playback De-emphasis", wm8750_enum[15]), ++ ++SOC_ENUM("Capture Polarity", wm8750_enum[14]), ++SOC_SINGLE("Playback 6dB Attenuate", WM8750_ADCDAC, 7, 1, 0), ++SOC_SINGLE("Capture 6dB Attenuate", WM8750_ADCDAC, 8, 1, 0), ++ ++SOC_DOUBLE_R("PCM Volume", WM8750_LDAC, WM8750_RDAC, 0, 255, 0), ++ ++SOC_ENUM("Bass Boost", wm8750_enum[0]), ++SOC_ENUM("Bass Filter", wm8750_enum[1]), ++SOC_SINGLE("Bass Volume", WM8750_BASS, 0, 15, 1), ++ ++SOC_SINGLE("Treble Volume", WM8750_TREBLE, 0, 15, 0), ++SOC_ENUM("Treble Cut-off", wm8750_enum[2]), ++ ++SOC_SINGLE("3D Switch", WM8750_3D, 0, 1, 0), ++SOC_SINGLE("3D Volume", WM8750_3D, 1, 15, 0), ++SOC_ENUM("3D Lower Cut-off", wm8750_enum[3]), ++SOC_ENUM("3D Upper Cut-off", wm8750_enum[4]), ++SOC_ENUM("3D Mode", wm8750_enum[5]), ++ ++SOC_SINGLE("ALC Capture Target Volume", WM8750_ALC1, 0, 7, 0), ++SOC_SINGLE("ALC Capture Max Volume", WM8750_ALC1, 4, 7, 0), ++SOC_ENUM("ALC Capture Function", wm8750_enum[6]), ++SOC_SINGLE("ALC Capture ZC Switch", WM8750_ALC2, 7, 1, 0), ++SOC_SINGLE("ALC Capture Hold Time", WM8750_ALC2, 0, 15, 0), ++SOC_SINGLE("ALC Capture Decay Time", WM8750_ALC3, 4, 15, 0), ++SOC_SINGLE("ALC Capture Attack Time", WM8750_ALC3, 0, 15, 0), ++SOC_SINGLE("ALC Capture NG Threshold", WM8750_NGATE, 3, 31, 0), ++SOC_ENUM("ALC Capture NG Type", wm8750_enum[4]), ++SOC_SINGLE("ALC Capture NG Switch", WM8750_NGATE, 0, 1, 0), ++ ++SOC_SINGLE("Left ADC Capture Volume", WM8750_LADC, 0, 255, 0), ++SOC_SINGLE("Right ADC Capture Volume", WM8750_RADC, 0, 255, 0), ++ ++SOC_SINGLE("ZC Timeout Switch", WM8750_ADCTL1, 0, 1, 0), ++SOC_SINGLE("Playback Invert Switch", WM8750_ADCTL1, 1, 1, 0), ++ ++SOC_SINGLE("Right Speaker Playback Invert Switch", WM8750_ADCTL2, 4, 1, 0), ++ ++/* Unimplemented */ ++/* ADCDAC Bit 0 - ADCHPD */ ++/* ADCDAC Bit 4 - HPOR */ ++/* ADCTL1 Bit 2,3 - DATSEL */ ++/* ADCTL1 Bit 4,5 - DMONOMIX */ ++/* ADCTL1 Bit 6,7 - VSEL */ ++/* ADCTL2 Bit 2 - LRCM */ ++/* ADCTL2 Bit 3 - TRI */ ++/* ADCTL3 Bit 5 - HPFLREN */ ++/* ADCTL3 Bit 6 - VROI */ ++/* ADCTL3 Bit 7,8 - ADCLRM */ ++/* ADCIN Bit 4 - LDCM */ ++/* ADCIN Bit 5 - RDCM */ ++ ++SOC_DOUBLE_R("Mic Boost", WM8750_LADCIN, WM8750_RADCIN, 4, 3, 0), ++ ++SOC_DOUBLE_R("Bypass Left Playback Volume", WM8750_LOUTM1, ++ WM8750_LOUTM2, 4, 7, 1), ++SOC_DOUBLE_R("Bypass Right Playback Volume", WM8750_ROUTM1, ++ WM8750_ROUTM2, 4, 7, 1), ++SOC_DOUBLE_R("Bypass Mono Playback Volume", WM8750_MOUTM1, ++ WM8750_MOUTM2, 4, 7, 1), ++ ++SOC_SINGLE("Mono Playback ZC Switch", WM8750_MOUTV, 7, 1, 0), ++ ++SOC_DOUBLE_R("Headphone Playback Volume", WM8750_LOUT1V, WM8750_ROUT1V, ++ 0, 127, 0), ++SOC_DOUBLE_R("Speaker Playback Volume", WM8750_LOUT2V, WM8750_ROUT2V, ++ 0, 127, 0), ++ ++SOC_SINGLE("Mono Playback Volume", WM8750_MOUTV, 0, 127, 0), ++ ++}; ++ ++/* add non dapm controls */ ++static int wm8750_add_controls(struct snd_soc_codec *codec) ++{ ++ int err, i; ++ ++ for (i = 0; i < ARRAY_SIZE(wm8750_snd_controls); i++) { ++ err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&wm8750_snd_controls[i],codec, NULL)); ++ if (err < 0) ++ return err; ++ } ++ return 0; ++} ++ ++/* ++ * DAPM Controls ++ */ ++ ++/* Left Mixer */ ++static const struct snd_kcontrol_new wm8750_left_mixer_controls[] = { ++SOC_DAPM_SINGLE("Playback Switch", WM8750_LOUTM1, 8, 1, 0), ++SOC_DAPM_SINGLE("Left Bypass Switch", WM8750_LOUTM1, 7, 1, 0), ++SOC_DAPM_SINGLE("Right Playback Switch", WM8750_LOUTM2, 8, 1, 0), ++SOC_DAPM_SINGLE("Right Bypass Switch", WM8750_LOUTM2, 7, 1, 0), ++}; ++ ++/* Right Mixer */ ++static const struct snd_kcontrol_new wm8750_right_mixer_controls[] = { ++SOC_DAPM_SINGLE("Left Playback Switch", WM8750_ROUTM1, 8, 1, 0), ++SOC_DAPM_SINGLE("Left Bypass Switch", WM8750_ROUTM1, 7, 1, 0), ++SOC_DAPM_SINGLE("Playback Switch", WM8750_ROUTM2, 8, 1, 0), ++SOC_DAPM_SINGLE("Right Bypass Switch", WM8750_ROUTM2, 7, 1, 0), ++}; ++ ++/* Mono Mixer */ ++static const struct snd_kcontrol_new wm8750_mono_mixer_controls[] = { ++SOC_DAPM_SINGLE("Left Playback Switch", WM8750_MOUTM1, 8, 1, 0), ++SOC_DAPM_SINGLE("Left Bypass Switch", WM8750_MOUTM1, 7, 1, 0), ++SOC_DAPM_SINGLE("Right Playback Switch", WM8750_MOUTM2, 8, 1, 0), ++SOC_DAPM_SINGLE("Right Bypass Switch", WM8750_MOUTM2, 7, 1, 0), ++}; ++ ++/* Left Line Mux */ ++static const struct snd_kcontrol_new wm8750_left_line_controls = ++SOC_DAPM_ENUM("Route", wm8750_enum[8]); ++ ++/* Right Line Mux */ ++static const struct snd_kcontrol_new wm8750_right_line_controls = ++SOC_DAPM_ENUM("Route", wm8750_enum[9]); ++ ++/* Left PGA Mux */ ++static const struct snd_kcontrol_new wm8750_left_pga_controls = ++SOC_DAPM_ENUM("Route", wm8750_enum[10]); ++ ++/* Right PGA Mux */ ++static const struct snd_kcontrol_new wm8750_right_pga_controls = ++SOC_DAPM_ENUM("Route", wm8750_enum[11]); ++ ++/* Out 3 Mux */ ++static const struct snd_kcontrol_new wm8750_out3_controls = ++SOC_DAPM_ENUM("Route", wm8750_enum[12]); ++ ++/* Differential Mux */ ++static const struct snd_kcontrol_new wm8750_diffmux_controls = ++SOC_DAPM_ENUM("Route", wm8750_enum[13]); ++ ++/* Mono ADC Mux */ ++static const struct snd_kcontrol_new wm8750_monomux_controls = ++SOC_DAPM_ENUM("Route", wm8750_enum[16]); ++ ++static const struct snd_soc_dapm_widget wm8750_dapm_widgets[] = { ++ SND_SOC_DAPM_MIXER("Left Mixer", SND_SOC_NOPM, 0, 0, ++ &wm8750_left_mixer_controls[0], ++ ARRAY_SIZE(wm8750_left_mixer_controls)), ++ SND_SOC_DAPM_MIXER("Right Mixer", SND_SOC_NOPM, 0, 0, ++ &wm8750_right_mixer_controls[0], ++ ARRAY_SIZE(wm8750_right_mixer_controls)), ++ SND_SOC_DAPM_MIXER("Mono Mixer", WM8750_PWR2, 2, 0, ++ &wm8750_mono_mixer_controls[0], ++ ARRAY_SIZE(wm8750_mono_mixer_controls)), ++ ++ SND_SOC_DAPM_PGA("Right Out 2", WM8750_PWR2, 3, 0, NULL, 0), ++ SND_SOC_DAPM_PGA("Left Out 2", WM8750_PWR2, 4, 0, NULL, 0), ++ SND_SOC_DAPM_PGA("Right Out 1", WM8750_PWR2, 5, 0, NULL, 0), ++ SND_SOC_DAPM_PGA("Left Out 1", WM8750_PWR2, 6, 0, NULL, 0), ++ SND_SOC_DAPM_DAC("Right DAC", "Right Playback", WM8750_PWR2, 7, 0), ++ SND_SOC_DAPM_DAC("Left DAC", "Left Playback", WM8750_PWR2, 8, 0), ++ ++ SND_SOC_DAPM_MICBIAS("Mic Bias", WM8750_PWR1, 1, 0), ++ SND_SOC_DAPM_ADC("Right ADC", "Right Capture", WM8750_PWR1, 2, 0), ++ SND_SOC_DAPM_ADC("Left ADC", "Left Capture", WM8750_PWR1, 3, 0), ++ ++ SND_SOC_DAPM_MUX("Left PGA Mux", WM8750_PWR1, 5, 0, ++ &wm8750_left_pga_controls), ++ SND_SOC_DAPM_MUX("Right PGA Mux", WM8750_PWR1, 4, 0, ++ &wm8750_right_pga_controls), ++ SND_SOC_DAPM_MUX("Left Line Mux", SND_SOC_NOPM, 0, 0, ++ &wm8750_left_line_controls), ++ SND_SOC_DAPM_MUX("Right Line Mux", SND_SOC_NOPM, 0, 0, ++ &wm8750_right_line_controls), ++ ++ SND_SOC_DAPM_MUX("Out3 Mux", SND_SOC_NOPM, 0, 0, &wm8750_out3_controls), ++ SND_SOC_DAPM_PGA("Out 3", WM8750_PWR2, 1, 0, NULL, 0), ++ SND_SOC_DAPM_PGA("Mono Out 1", WM8750_PWR2, 2, 0, NULL, 0), ++ ++ SND_SOC_DAPM_MUX("Differential Mux", SND_SOC_NOPM, 0, 0, ++ &wm8750_diffmux_controls), ++ SND_SOC_DAPM_MUX("Left ADC Mux", SND_SOC_NOPM, 0, 0, ++ &wm8750_monomux_controls), ++ SND_SOC_DAPM_MUX("Right ADC Mux", SND_SOC_NOPM, 0, 0, ++ &wm8750_monomux_controls), ++ ++ SND_SOC_DAPM_OUTPUT("LOUT1"), ++ SND_SOC_DAPM_OUTPUT("ROUT1"), ++ SND_SOC_DAPM_OUTPUT("LOUT2"), ++ SND_SOC_DAPM_OUTPUT("ROUT2"), ++ SND_SOC_DAPM_OUTPUT("MONO"), ++ SND_SOC_DAPM_OUTPUT("OUT3"), ++ ++ SND_SOC_DAPM_INPUT("LINPUT1"), ++ SND_SOC_DAPM_INPUT("LINPUT2"), ++ SND_SOC_DAPM_INPUT("LINPUT3"), ++ SND_SOC_DAPM_INPUT("RINPUT1"), ++ SND_SOC_DAPM_INPUT("RINPUT2"), ++ SND_SOC_DAPM_INPUT("RINPUT3"), ++}; ++ ++static const char *audio_map[][3] = { ++ /* left mixer */ ++ {"Left Mixer", "Playback Switch", "Left DAC"}, ++ {"Left Mixer", "Left Bypass Switch", "Left Line Mux"}, ++ {"Left Mixer", "Right Playback Switch", "Right DAC"}, ++ {"Left Mixer", "Right Bypass Switch", "Right Line Mux"}, ++ ++ /* right mixer */ ++ {"Right Mixer", "Left Playback Switch", "Left DAC"}, ++ {"Right Mixer", "Left Bypass Switch", "Left Line Mux"}, ++ {"Right Mixer", "Playback Switch", "Right DAC"}, ++ {"Right Mixer", "Right Bypass Switch", "Right Line Mux"}, ++ ++ /* left out 1 */ ++ {"Left Out 1", NULL, "Left Mixer"}, ++ {"LOUT1", NULL, "Left Out 1"}, ++ ++ /* left out 2 */ ++ {"Left Out 2", NULL, "Left Mixer"}, ++ {"LOUT2", NULL, "Left Out 2"}, ++ ++ /* right out 1 */ ++ {"Right Out 1", NULL, "Right Mixer"}, ++ {"ROUT1", NULL, "Right Out 1"}, ++ ++ /* right out 2 */ ++ {"Right Out 2", NULL, "Right Mixer"}, ++ {"ROUT2", NULL, "Right Out 2"}, ++ ++ /* mono mixer */ ++ {"Mono Mixer", "Left Playback Switch", "Left DAC"}, ++ {"Mono Mixer", "Left Bypass Switch", "Left Line Mux"}, ++ {"Mono Mixer", "Right Playback Switch", "Right DAC"}, ++ {"Mono Mixer", "Right Bypass Switch", "Right Line Mux"}, ++ ++ /* mono out */ ++ {"Mono Out 1", NULL, "Mono Mixer"}, ++ {"MONO1", NULL, "Mono Out 1"}, ++ ++ /* out 3 */ ++ {"Out3 Mux", "VREF", "VREF"}, ++ {"Out3 Mux", "ROUT1 + Vol", "ROUT1"}, ++ {"Out3 Mux", "ROUT1", "Right Mixer"}, ++ {"Out3 Mux", "MonoOut", "MONO1"}, ++ {"Out 3", NULL, "Out3 Mux"}, ++ {"OUT3", NULL, "Out 3"}, ++ ++ /* Left Line Mux */ ++ {"Left Line Mux", "Line 1", "LINPUT1"}, ++ {"Left Line Mux", "Line 2", "LINPUT2"}, ++ {"Left Line Mux", "Line 3", "LINPUT3"}, ++ {"Left Line Mux", "PGA", "Left PGA Mux"}, ++ {"Left Line Mux", "Differential", "Differential Mux"}, ++ ++ /* Right Line Mux */ ++ {"Right Line Mux", "Line 1", "RINPUT1"}, ++ {"Right Line Mux", "Line 2", "RINPUT2"}, ++ {"Right Line Mux", "Line 3", "RINPUT3"}, ++ {"Right Line Mux", "PGA", "Right PGA Mux"}, ++ {"Right Line Mux", "Differential", "Differential Mux"}, ++ ++ /* Left PGA Mux */ ++ {"Left PGA Mux", "Line 1", "LINPUT1"}, ++ {"Left PGA Mux", "Line 2", "LINPUT2"}, ++ {"Left PGA Mux", "Line 3", "LINPUT3"}, ++ {"Left PGA Mux", "Differential", "Differential Mux"}, ++ ++ /* Right PGA Mux */ ++ {"Right PGA Mux", "Line 1", "RINPUT1"}, ++ {"Right PGA Mux", "Line 2", "RINPUT2"}, ++ {"Right PGA Mux", "Line 3", "RINPUT3"}, ++ {"Right PGA Mux", "Differential", "Differential Mux"}, ++ ++ /* Differential Mux */ ++ {"Differential Mux", "Line 1", "LINPUT1"}, ++ {"Differential Mux", "Line 1", "RINPUT1"}, ++ {"Differential Mux", "Line 2", "LINPUT2"}, ++ {"Differential Mux", "Line 2", "RINPUT2"}, ++ ++ /* Left ADC Mux */ ++ {"Left ADC Mux", "Stereo", "Left PGA Mux"}, ++ {"Left ADC Mux", "Mono (Left)", "Left PGA Mux"}, ++ {"Left ADC Mux", "Digital Mono", "Left PGA Mux"}, ++ ++ /* Right ADC Mux */ ++ {"Right ADC Mux", "Stereo", "Right PGA Mux"}, ++ {"Right ADC Mux", "Mono (Right)", "Right PGA Mux"}, ++ {"Right ADC Mux", "Digital Mono", "Right PGA Mux"}, ++ ++ /* ADC */ ++ {"Left ADC", NULL, "Left ADC Mux"}, ++ {"Right ADC", NULL, "Right ADC Mux"}, ++ ++ /* terminator */ ++ {NULL, NULL, NULL}, ++}; ++ ++static int wm8750_add_widgets(struct snd_soc_codec *codec) ++{ ++ int i; ++ ++ for(i = 0; i < ARRAY_SIZE(wm8750_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &wm8750_dapm_widgets[i]); ++ } ++ ++ /* set up audio path audio_mapnects */ ++ for(i = 0; audio_map[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, audio_map[i][0], ++ audio_map[i][1], audio_map[i][2]); ++ } ++ ++ snd_soc_dapm_new_widgets(codec); ++ return 0; ++} ++ ++struct _coeff_div { ++ u32 mclk; ++ u32 rate; ++ u16 fs; ++ u8 sr:5; ++ u8 usb:1; ++}; ++ ++/* codec hifi mclk clock divider coefficients */ ++static const struct _coeff_div coeff_div[] = { ++ /* 8k */ ++ {12288000, 8000, 1536, 0x6, 0x0}, ++ {11289600, 8000, 1408, 0x16, 0x0}, ++ {18432000, 8000, 2304, 0x7, 0x0}, ++ {16934400, 8000, 2112, 0x17, 0x0}, ++ {12000000, 8000, 1500, 0x6, 0x1}, ++ ++ /* 11.025k */ ++ {11289600, 11025, 1024, 0x18, 0x0}, ++ {16934400, 11025, 1536, 0x19, 0x0}, ++ {12000000, 11025, 1088, 0x19, 0x1}, ++ ++ /* 16k */ ++ {12288000, 16000, 768, 0xa, 0x0}, ++ {18432000, 16000, 1152, 0xb, 0x0}, ++ {12000000, 16000, 750, 0xa, 0x1}, ++ ++ /* 22.05k */ ++ {11289600, 22050, 512, 0x1a, 0x0}, ++ {16934400, 22050, 768, 0x1b, 0x0}, ++ {12000000, 22050, 544, 0x1b, 0x1}, ++ ++ /* 32k */ ++ {12288000, 32000, 384, 0xc, 0x0}, ++ {18432000, 32000, 576, 0xd, 0x0}, ++ {12000000, 32000, 375, 0xa, 0x1}, ++ ++ /* 44.1k */ ++ {11289600, 44100, 256, 0x10, 0x0}, ++ {16934400, 44100, 384, 0x11, 0x0}, ++ {12000000, 44100, 272, 0x11, 0x1}, ++ ++ /* 48k */ ++ {12288000, 48000, 256, 0x0, 0x0}, ++ {18432000, 48000, 384, 0x1, 0x0}, ++ {12000000, 48000, 250, 0x0, 0x1}, ++ ++ /* 88.2k */ ++ {11289600, 88200, 128, 0x1e, 0x0}, ++ {16934400, 88200, 192, 0x1f, 0x0}, ++ {12000000, 88200, 136, 0x1f, 0x1}, ++ ++ /* 96k */ ++ {12288000, 96000, 128, 0xe, 0x0}, ++ {18432000, 96000, 192, 0xf, 0x0}, ++ {12000000, 96000, 125, 0xe, 0x1}, ++}; ++ ++static inline int get_coeff(int mclk, int rate) ++{ ++ int i; ++ ++ for (i = 0; i < ARRAY_SIZE(coeff_div); i++) { ++ if (coeff_div[i].rate == rate && coeff_div[i].mclk == mclk) ++ return i; ++ } ++ ++ printk(KERN_ERR "wm8750: could not get coeff for mclk %d @ rate %d\n", ++ mclk, rate); ++ return -EINVAL; ++} ++ ++/* WM8750 supports numerous input clocks per sample rate */ ++static unsigned int wm8750_config_sysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ dai->mclk = clk; ++ return dai->mclk; ++} ++ ++static int wm8750_pcm_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 iface = 0, bfs, srate = 0; ++ int i = get_coeff(rtd->codec_dai->mclk, ++ snd_soc_get_rate(rtd->codec_dai->dai_runtime.pcmrate)); ++ ++ /* is coefficient valid ? */ ++ if (i < 0) ++ return i; ++ ++ bfs = SND_SOC_FSBD_REAL(rtd->codec_dai->dai_runtime.bfs); ++ ++ /* set master/slave audio interface */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK) { ++ case SND_SOC_DAIFMT_CBM_CFM: ++ iface = 0x0040; ++ break; ++ case SND_SOC_DAIFMT_CBS_CFS: ++ break; ++ } ++ ++ /* interface format */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_I2S: ++ iface |= 0x0002; ++ break; ++ case SND_SOC_DAIFMT_RIGHT_J: ++ break; ++ case SND_SOC_DAIFMT_LEFT_J: ++ iface |= 0x0001; ++ break; ++ case SND_SOC_DAIFMT_DSP_A: ++ iface |= 0x0003; ++ break; ++ case SND_SOC_DAIFMT_DSP_B: ++ iface |= 0x0013; ++ break; ++ } ++ ++ /* bit size */ ++ switch (rtd->codec_dai->dai_runtime.pcmfmt) { ++ case SNDRV_PCM_FMTBIT_S16_LE: ++ break; ++ case SNDRV_PCM_FMTBIT_S20_3LE: ++ iface |= 0x0004; ++ break; ++ case SNDRV_PCM_FMTBIT_S24_LE: ++ iface |= 0x0008; ++ break; ++ case SNDRV_PCM_FMTBIT_S32_LE: ++ iface |= 0x000c; ++ break; ++ } ++ ++ /* clock inversion */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_INV_MASK) { ++ case SND_SOC_DAIFMT_NB_NF: ++ break; ++ case SND_SOC_DAIFMT_IB_IF: ++ iface |= 0x0090; ++ break; ++ case SND_SOC_DAIFMT_IB_NF: ++ iface |= 0x0080; ++ break; ++ case SND_SOC_DAIFMT_NB_IF: ++ iface |= 0x0010; ++ break; ++ } ++ ++ /* set bclk divisor rate */ ++ switch (bfs) { ++ case 1: ++ break; ++ case 4: ++ srate |= (0x1 << 7); ++ break; ++ case 8: ++ srate |= (0x2 << 7); ++ break; ++ case 16: ++ srate |= (0x3 << 7); ++ break; ++ } ++ ++ /* set iface & srate */ ++ wm8750_write(codec, WM8750_IFACE, iface); ++ wm8750_write(codec, WM8750_SRATE, srate | ++ (coeff_div[i].sr << 1) | coeff_div[i].usb); ++ ++ return 0; ++} ++ ++static int wm8750_mute(struct snd_soc_codec *codec, ++ struct snd_soc_codec_dai *dai, int mute) ++{ ++ u16 mute_reg = wm8750_read_reg_cache(codec, WM8750_ADCDAC) & 0xfff7; ++ if (mute) ++ wm8750_write(codec, WM8750_ADCDAC, mute_reg | 0x8); ++ else ++ wm8750_write(codec, WM8750_ADCDAC, mute_reg); ++ return 0; ++} ++ ++static int wm8750_dapm_event(struct snd_soc_codec *codec, int event) ++{ ++ u16 pwr_reg = wm8750_read_reg_cache(codec, WM8750_PWR1) & 0xfe3e; ++ ++ switch (event) { ++ case SNDRV_CTL_POWER_D0: /* full On */ ++ /* set vmid to 50k and unmute dac */ ++ wm8750_write(codec, WM8750_PWR1, pwr_reg | 0x00c0); ++ break; ++ case SNDRV_CTL_POWER_D1: /* partial On */ ++ case SNDRV_CTL_POWER_D2: /* partial On */ ++ /* set vmid to 5k for quick power up */ ++ wm8750_write(codec, WM8750_PWR1, pwr_reg | 0x01c1); ++ break; ++ case SNDRV_CTL_POWER_D3hot: /* Off, with power */ ++ /* mute dac and set vmid to 500k, enable VREF */ ++ wm8750_write(codec, WM8750_PWR1, pwr_reg | 0x0141); ++ break; ++ case SNDRV_CTL_POWER_D3cold: /* Off, without power */ ++ wm8750_write(codec, WM8750_PWR1, 0x0001); ++ break; ++ } ++ codec->dapm_state = event; ++ return 0; ++} ++ ++struct snd_soc_codec_dai wm8750_dai = { ++ .name = "WM8750", ++ .playback = { ++ .stream_name = "Playback", ++ .channels_min = 1, ++ .channels_max = 2, ++ }, ++ .capture = { ++ .stream_name = "Capture", ++ .channels_min = 1, ++ .channels_max = 2, ++ }, ++ .config_sysclk = wm8750_config_sysclk, ++ .digital_mute = wm8750_mute, ++ .ops = { ++ .prepare = wm8750_pcm_prepare, ++ }, ++ .caps = { ++ .num_modes = ARRAY_SIZE(wm8750_modes), ++ .mode = wm8750_modes, ++ }, ++}; ++EXPORT_SYMBOL_GPL(wm8750_dai); ++ ++static void wm8750_work(void *data) ++{ ++ struct snd_soc_codec *codec = (struct snd_soc_codec *)data; ++ wm8750_dapm_event(codec, codec->dapm_state); ++} ++ ++static int wm8750_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ wm8750_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ return 0; ++} ++ ++static int wm8750_resume(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ int i; ++ u8 data[2]; ++ u16 *cache = codec->reg_cache; ++ ++ /* Sync reg_cache with the hardware */ ++ for (i = 0; i < ARRAY_SIZE(wm8750_reg); i++) { ++ if (i == WM8750_RESET) ++ continue; ++ data[0] = (i << 1) | ((cache[i] >> 8) & 0x0001); ++ data[1] = cache[i] & 0x00ff; ++ codec->hw_write(codec->control_data, data, 2); ++ } ++ ++ wm8750_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ ++ /* charge wm8750 caps */ ++ if (codec->suspend_dapm_state == SNDRV_CTL_POWER_D0) { ++ wm8750_dapm_event(codec, SNDRV_CTL_POWER_D2); ++ codec->dapm_state = SNDRV_CTL_POWER_D0; ++ queue_delayed_work(wm8750_workq, &wm8750_dapm_work, ++ msecs_to_jiffies(1000)); ++ } ++ ++ return 0; ++} ++ ++/* ++ * initialise the WM8750 driver ++ * register the mixer and dsp interfaces with the kernel ++ */ ++static int wm8750_init(struct snd_soc_device *socdev) ++{ ++ struct snd_soc_codec *codec = socdev->codec; ++ int reg, ret = 0; ++ ++ codec->name = "WM8750"; ++ codec->owner = THIS_MODULE; ++ codec->read = wm8750_read_reg_cache; ++ codec->write = wm8750_write; ++ codec->dapm_event = wm8750_dapm_event; ++ codec->dai = &wm8750_dai; ++ codec->num_dai = 1; ++ codec->reg_cache_size = ARRAY_SIZE(wm8750_reg); ++ ++ codec->reg_cache = ++ kzalloc(sizeof(u16) * ARRAY_SIZE(wm8750_reg), GFP_KERNEL); ++ if (codec->reg_cache == NULL) ++ return -ENOMEM; ++ memcpy(codec->reg_cache, wm8750_reg, ++ sizeof(u16) * ARRAY_SIZE(wm8750_reg)); ++ codec->reg_cache_size = sizeof(u16) * ARRAY_SIZE(wm8750_reg); ++ ++ wm8750_reset(codec); ++ ++ /* register pcms */ ++ ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); ++ if (ret < 0) { ++ kfree(codec->reg_cache); ++ return ret; ++ } ++ ++ /* charge output caps */ ++ wm8750_dapm_event(codec, SNDRV_CTL_POWER_D2); ++ codec->dapm_state = SNDRV_CTL_POWER_D3hot; ++ queue_delayed_work(wm8750_workq, &wm8750_dapm_work, ++ msecs_to_jiffies(1000)); ++ ++ /* set the update bits */ ++ reg = wm8750_read_reg_cache(codec, WM8750_LDAC); ++ wm8750_write(codec, WM8750_LDAC, reg | 0x0100); ++ reg = wm8750_read_reg_cache(codec, WM8750_RDAC); ++ wm8750_write(codec, WM8750_RDAC, reg | 0x0100); ++ reg = wm8750_read_reg_cache(codec, WM8750_LOUT1V); ++ wm8750_write(codec, WM8750_LOUT1V, reg | 0x0100); ++ reg = wm8750_read_reg_cache(codec, WM8750_ROUT1V); ++ wm8750_write(codec, WM8750_ROUT1V, reg | 0x0100); ++ reg = wm8750_read_reg_cache(codec, WM8750_LOUT2V); ++ wm8750_write(codec, WM8750_LOUT2V, reg | 0x0100); ++ reg = wm8750_read_reg_cache(codec, WM8750_ROUT2V); ++ wm8750_write(codec, WM8750_ROUT2V, reg | 0x0100); ++ reg = wm8750_read_reg_cache(codec, WM8750_LINVOL); ++ wm8750_write(codec, WM8750_LINVOL, reg | 0x0100); ++ reg = wm8750_read_reg_cache(codec, WM8750_RINVOL); ++ wm8750_write(codec, WM8750_RINVOL, reg | 0x0100); ++ ++ wm8750_add_controls(codec); ++ wm8750_add_widgets(codec); ++ ret = snd_soc_register_card(socdev); ++ if (ret < 0) { ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++ } ++ ++ return ret; ++} ++ ++/* If the i2c layer weren't so broken, we could pass this kind of data ++ around */ ++static struct snd_soc_device *wm8750_socdev; ++ ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ ++/* ++ * WM8731 2 wire address is determined by GPIO5 ++ * state during powerup. ++ * low = 0x1a ++ * high = 0x1b ++ */ ++static unsigned short normal_i2c[] = { 0, I2C_CLIENT_END }; ++ ++/* Magic definition of all other variables and things */ ++I2C_CLIENT_INSMOD; ++ ++static struct i2c_driver wm8750_i2c_driver; ++static struct i2c_client client_template; ++ ++static int wm8750_codec_probe(struct i2c_adapter *adap, int addr, int kind) ++{ ++ struct snd_soc_device *socdev = wm8750_socdev; ++ struct wm8750_setup_data *setup = socdev->codec_data; ++ struct snd_soc_codec *codec = socdev->codec; ++ struct i2c_client *i2c; ++ int ret; ++ ++ if (addr != setup->i2c_address) ++ return -ENODEV; ++ ++ client_template.adapter = adap; ++ client_template.addr = addr; ++ ++ i2c = kzalloc(sizeof(struct i2c_client), GFP_KERNEL); ++ if (i2c == NULL) { ++ kfree(codec); ++ return -ENOMEM; ++ } ++ memcpy(i2c, &client_template, sizeof(struct i2c_client)); ++ i2c_set_clientdata(i2c, codec); ++ codec->control_data = i2c; ++ ++ ret = i2c_attach_client(i2c); ++ if (ret < 0) { ++ err("failed to attach codec at addr %x\n", addr); ++ goto err; ++ } ++ ++ ret = wm8750_init(socdev); ++ if (ret < 0) { ++ err("failed to initialise WM8750\n"); ++ goto err; ++ } ++ return ret; ++ ++err: ++ kfree(codec); ++ kfree(i2c); ++ return ret; ++} ++ ++static int wm8750_i2c_detach(struct i2c_client *client) ++{ ++ struct snd_soc_codec *codec = i2c_get_clientdata(client); ++ i2c_detach_client(client); ++ kfree(codec->reg_cache); ++ kfree(client); ++ return 0; ++} ++ ++static int wm8750_i2c_attach(struct i2c_adapter *adap) ++{ ++ return i2c_probe(adap, &addr_data, wm8750_codec_probe); ++} ++ ++/* corgi i2c codec control layer */ ++static struct i2c_driver wm8750_i2c_driver = { ++ .driver = { ++ .name = "WM8750 I2C Codec", ++ .owner = THIS_MODULE, ++ }, ++ .id = I2C_DRIVERID_WM8750, ++ .attach_adapter = wm8750_i2c_attach, ++ .detach_client = wm8750_i2c_detach, ++ .command = NULL, ++}; ++ ++static struct i2c_client client_template = { ++ .name = "WM8750", ++ .driver = &wm8750_i2c_driver, ++}; ++#endif ++ ++static int wm8750_probe(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct wm8750_setup_data *setup = socdev->codec_data; ++ struct snd_soc_codec *codec; ++ int ret = 0; ++ ++ info("WM8750 Audio Codec %s", WM8750_VERSION); ++ codec = kzalloc(sizeof(struct snd_soc_codec), GFP_KERNEL); ++ if (codec == NULL) ++ return -ENOMEM; ++ ++ socdev->codec = codec; ++ mutex_init(&codec->mutex); ++ INIT_LIST_HEAD(&codec->dapm_widgets); ++ INIT_LIST_HEAD(&codec->dapm_paths); ++ wm8750_socdev = socdev; ++ INIT_WORK(&wm8750_dapm_work, wm8750_work, codec); ++ wm8750_workq = create_workqueue("wm8750"); ++ if (wm8750_workq == NULL) { ++ kfree(codec); ++ return -ENOMEM; ++ } ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ if (setup->i2c_address) { ++ normal_i2c[0] = setup->i2c_address; ++ codec->hw_write = (hw_write_t)i2c_master_send; ++ ret = i2c_add_driver(&wm8750_i2c_driver); ++ if (ret != 0) ++ printk(KERN_ERR "can't add i2c driver"); ++ } ++#else ++ /* Add other interfaces here */ ++#endif ++ ++ return ret; ++} ++ ++/* power down chip */ ++static int wm8750_remove(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ if (codec->control_data) ++ wm8750_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ if (wm8750_workq) ++ destroy_workqueue(wm8750_workq); ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ i2c_del_driver(&wm8750_i2c_driver); ++#endif ++ kfree(codec); ++ ++ return 0; ++} ++ ++struct snd_soc_codec_device soc_codec_dev_wm8750 = { ++ .probe = wm8750_probe, ++ .remove = wm8750_remove, ++ .suspend = wm8750_suspend, ++ .resume = wm8750_resume, ++}; ++ ++EXPORT_SYMBOL_GPL(soc_codec_dev_wm8750); ++ ++MODULE_DESCRIPTION("ASoC WM8750 driver"); ++MODULE_AUTHOR("Liam Girdwood"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8750.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8750.h +@@ -0,0 +1,66 @@ ++/* ++ * Copyright 2005 Openedhand Ltd. ++ * ++ * Author: Richard Purdie <richard@openedhand.com> ++ * ++ * Based on WM8753.h ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ * ++ */ ++ ++#ifndef _WM8750_H ++#define _WM8750_H ++ ++/* WM8750 register space */ ++ ++#define WM8750_LINVOL 0x00 ++#define WM8750_RINVOL 0x01 ++#define WM8750_LOUT1V 0x02 ++#define WM8750_ROUT1V 0x03 ++#define WM8750_ADCDAC 0x05 ++#define WM8750_IFACE 0x07 ++#define WM8750_SRATE 0x08 ++#define WM8750_LDAC 0x0a ++#define WM8750_RDAC 0x0b ++#define WM8750_BASS 0x0c ++#define WM8750_TREBLE 0x0d ++#define WM8750_RESET 0x0f ++#define WM8750_3D 0x10 ++#define WM8750_ALC1 0x11 ++#define WM8750_ALC2 0x12 ++#define WM8750_ALC3 0x13 ++#define WM8750_NGATE 0x14 ++#define WM8750_LADC 0x15 ++#define WM8750_RADC 0x16 ++#define WM8750_ADCTL1 0x17 ++#define WM8750_ADCTL2 0x18 ++#define WM8750_PWR1 0x19 ++#define WM8750_PWR2 0x1a ++#define WM8750_ADCTL3 0x1b ++#define WM8750_ADCIN 0x1f ++#define WM8750_LADCIN 0x20 ++#define WM8750_RADCIN 0x21 ++#define WM8750_LOUTM1 0x22 ++#define WM8750_LOUTM2 0x23 ++#define WM8750_ROUTM1 0x24 ++#define WM8750_ROUTM2 0x25 ++#define WM8750_MOUTM1 0x26 ++#define WM8750_MOUTM2 0x27 ++#define WM8750_LOUT2V 0x28 ++#define WM8750_ROUT2V 0x29 ++#define WM8750_MOUTV 0x2a ++ ++#define WM8750_CACHE_REGNUM 0x2a ++ ++struct wm8750_setup_data { ++ unsigned short i2c_address; ++ unsigned int mclk; ++}; ++ ++extern struct snd_soc_codec_dai wm8750_dai; ++extern struct snd_soc_codec_device soc_codec_dev_wm8750; ++ ++#endif +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8753.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8753.c +@@ -0,0 +1,2128 @@ ++/* ++ * wm8753.c -- WM8753 ALSA Soc Audio driver ++ * ++ * Copyright 2003 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * 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. ++ * ++ * Notes: ++ * The WM8753 is a low power, high quality stereo codec with integrated PCM ++ * codec designed for portable digital telephony applications. ++ * ++ * Dual DAI:- ++ * ++ * This driver support 2 DAI PCM's. This makes the default PCM available for ++ * HiFi audio (e.g. MP3, ogg) playback/capture and the other PCM available for ++ * voice. ++ * ++ * Please note that the voice PCM can be connected directly to a Bluetooth ++ * codec or GSM modem and thus cannot be read or written to, although it is ++ * available to be configured with snd_hw_params(), etc and kcontrols in the ++ * normal alsa manner. ++ * ++ * Fast DAI switching:- ++ * ++ * The driver can now fast switch between the DAI configurations via a ++ * an alsa kcontrol. This allows the PCM to remain open. ++ * ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/version.h> ++#include <linux/kernel.h> ++#include <linux/init.h> ++#include <linux/delay.h> ++#include <linux/pm.h> ++#include <linux/i2c.h> ++#include <linux/platform_device.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/pcm_params.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++#include <sound/initval.h> ++ ++#include "wm8753.h" ++ ++#define AUDIO_NAME "wm8753" ++#define WM8753_VERSION "0.16" ++ ++/* ++ * Debug ++ */ ++ ++#define WM8753_DEBUG 0 ++ ++#ifdef WM8753_DEBUG ++#define dbg(format, arg...) \ ++ printk(KERN_DEBUG AUDIO_NAME ": " format "\n" , ## arg) ++#else ++#define dbg(format, arg...) do {} while (0) ++#endif ++#define err(format, arg...) \ ++ printk(KERN_ERR AUDIO_NAME ": " format "\n" , ## arg) ++#define info(format, arg...) \ ++ printk(KERN_INFO AUDIO_NAME ": " format "\n" , ## arg) ++#define warn(format, arg...) \ ++ printk(KERN_WARNING AUDIO_NAME ": " format "\n" , ## arg) ++ ++static int caps_charge = 2000; ++module_param(caps_charge, int, 0); ++MODULE_PARM_DESC(caps_charge, "WM8753 cap charge time (msecs)"); ++ ++static struct workqueue_struct *wm8753_workq = NULL; ++static struct work_struct wm8753_dapm_work; ++static void wm8753_set_dai_mode(struct snd_soc_codec *codec, ++ unsigned int mode); ++ ++/* ++ * wm8753 register cache ++ * We can't read the WM8753 register space when we ++ * are using 2 wire for device control, so we cache them instead. ++ */ ++static const u16 wm8753_reg[] = { ++ 0x0008, 0x0000, 0x000a, 0x000a, ++ 0x0033, 0x0000, 0x0007, 0x00ff, ++ 0x00ff, 0x000f, 0x000f, 0x007b, ++ 0x0000, 0x0032, 0x0000, 0x00c3, ++ 0x00c3, 0x00c0, 0x0000, 0x0000, ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0000, 0x0000, 0x0000, 0x0055, ++ 0x0005, 0x0050, 0x0055, 0x0050, ++ 0x0055, 0x0050, 0x0055, 0x0079, ++ 0x0079, 0x0079, 0x0079, 0x0079, ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0097, 0x0097, 0x0000, 0x0004, ++ 0x0000, 0x0083, 0x0024, 0x01ba, ++ 0x0000, 0x0083, 0x0024, 0x01ba, ++ 0x0000, 0x0000 ++}; ++ ++#define WM8753_DAIFMT \ ++ (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_LEFT_J | SND_SOC_DAIFMT_RIGHT_J | \ ++ SND_SOC_DAIFMT_DSP_A | SND_SOC_DAIFMT_DSP_B | SND_SOC_DAIFMT_NB_NF | \ ++ SND_SOC_DAIFMT_NB_IF | SND_SOC_DAIFMT_IB_NF | SND_SOC_DAIFMT_IB_IF) ++ ++#define WM8753_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define WM8753_HIFI_FSB \ ++ (SND_SOC_FSBD(1) | SND_SOC_FSBD(2) | SND_SOC_FSBD(4) | \ ++ SND_SOC_FSBD(8) | SND_SOC_FSBD(16)) ++ ++#define WM8753_HIFI_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000) ++ ++#define WM8753_HIFI_BITS \ ++ (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE | \ ++ SNDRV_PCM_FMTBIT_S24_LE | SNDRV_PCM_FMTBIT_S32_LE) ++ ++/* ++ * HiFi modes ++ */ ++static struct snd_soc_dai_mode wm8753_hifi_modes[] = { ++ /* codec frame and clock master modes */ ++ /* 8k */ ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1536, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1408, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 2304, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 2112, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1500, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ ++ /* 11.025k */ ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_11025, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1024, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_11025, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1536, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_11025, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1088, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ ++ /* 16k */ ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_16000, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 768, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt= WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_16000, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1152, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_16000, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 750, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ ++ /* 22.05k */ ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_22050, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 512, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_22050, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 768, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_22050, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 544, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ ++ /* 32k */ ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 384, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 576, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 375, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ ++ /* 44.1k & 48k */ ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 384, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 250, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_44100, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 272, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ ++ /* 88.2k & 96k */ ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 128, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 192, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_88200, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 136, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_96000, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 125, ++ .bfs = WM8753_HIFI_FSB, ++ }, ++ ++ /* codec frame and clock slave modes */ ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = WM8753_HIFI_RATES, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = SND_SOC_FSB_ALL, ++ }, ++}; ++ ++#define WM8753_VOICE_FSB \ ++ (SND_SOC_FSBD(1) | SND_SOC_FSBD(2) | SND_SOC_FSBD(4) | \ ++ SND_SOC_FSBD(8) | SND_SOC_FSBD(16)) ++ ++#define WM8753_VOICE_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000) ++ ++#define WM8753_VOICE_BITS \ ++ (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE | \ ++ SNDRV_PCM_FMTBIT_S24_LE | SNDRV_PCM_FMTBIT_S32_LE) ++ ++/* ++ * Voice modes ++ */ ++static struct snd_soc_dai_mode wm8753_voice_modes[] = { ++ ++ /* master modes */ ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_VOICE_BITS, ++ .pcmrate = WM8753_VOICE_RATES, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = WM8753_VOICE_FSB, ++ }, ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM8753_VOICE_BITS, ++ .pcmrate = WM8753_VOICE_RATES, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 384, ++ .bfs = WM8753_VOICE_FSB, ++ }, ++ ++ /* slave modes */ ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = WM8753_VOICE_BITS, ++ .pcmrate = WM8753_VOICE_RATES, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = SND_SOC_FSB_ALL, ++ }, ++}; ++ ++ ++/* ++ * Mode 4 ++ */ ++static struct snd_soc_dai_mode wm8753_mixed_modes[] = { ++ /* slave modes */ ++ { ++ .fmt = WM8753_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = WM8753_HIFI_BITS, ++ .pcmrate = WM8753_HIFI_RATES, ++ .pcmdir = WM8753_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = SND_SOC_FSB_ALL, ++ }, ++}; ++ ++/* ++ * read wm8753 register cache ++ */ ++static inline unsigned int wm8753_read_reg_cache(struct snd_soc_codec *codec, ++ unsigned int reg) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg < 1 || reg > (ARRAY_SIZE(wm8753_reg) + 1)) ++ return -1; ++ return cache[reg - 1]; ++} ++ ++/* ++ * write wm8753 register cache ++ */ ++static inline void wm8753_write_reg_cache(struct snd_soc_codec *codec, ++ unsigned int reg, unsigned int value) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg < 1 || reg > 0x3f) ++ return; ++ cache[reg - 1] = value; ++} ++ ++/* ++ * write to the WM8753 register space ++ */ ++static int wm8753_write(struct snd_soc_codec *codec, unsigned int reg, ++ unsigned int value) ++{ ++ u8 data[2]; ++ ++ /* data is ++ * D15..D9 WM8753 register offset ++ * D8...D0 register data ++ */ ++ data[0] = (reg << 1) | ((value >> 8) & 0x0001); ++ data[1] = value & 0x00ff; ++ ++ wm8753_write_reg_cache (codec, reg, value); ++ if (codec->hw_write(codec->control_data, data, 2) == 2) ++ return 0; ++ else ++ return -EIO; ++} ++ ++#define wm8753_reset(c) wm8753_write(c, WM8753_RESET, 0) ++ ++/* ++ * WM8753 Controls ++ */ ++static const char *wm8753_base[] = {"Linear Control", "Adaptive Boost"}; ++static const char *wm8753_base_filter[] = ++ {"130Hz @ 48kHz", "200Hz @ 48kHz", "100Hz @ 16kHz", "400Hz @ 48kHz", ++ "100Hz @ 8kHz", "200Hz @ 8kHz"}; ++static const char *wm8753_treble[] = {"8kHz", "4kHz"}; ++static const char *wm8753_alc_func[] = {"Off", "Right", "Left", "Stereo"}; ++static const char *wm8753_ng_type[] = {"Constant PGA Gain", "Mute ADC Output"}; ++static const char *wm8753_3d_func[] = {"Capture", "Playback"}; ++static const char *wm8753_3d_uc[] = {"2.2kHz", "1.5kHz"}; ++static const char *wm8753_3d_lc[] = {"200Hz", "500Hz"}; ++static const char *wm8753_deemp[] = {"None", "32kHz", "44.1kHz", "48kHz"}; ++static const char *wm8753_mono_mix[] = {"Stereo", "Left", "Right", "Mono"}; ++static const char *wm8753_dac_phase[] = {"Non Inverted", "Inverted"}; ++static const char *wm8753_line_mix[] = {"Line 1 + 2", "Line 1 - 2", ++ "Line 1", "Line 2"}; ++static const char *wm8753_mono_mux[] = {"Line Mix", "Rx Mix"}; ++static const char *wm8753_right_mux[] = {"Line 2", "Rx Mix"}; ++static const char *wm8753_left_mux[] = {"Line 1", "Rx Mix"}; ++static const char *wm8753_rxmsel[] = {"RXP - RXN", "RXP + RXN", "RXP", "RXN"}; ++static const char *wm8753_sidetone_mux[] = {"Left PGA", "Mic 1", "Mic 2", ++ "Right PGA"}; ++static const char *wm8753_mono2_src[] = {"Inverted Mono 1", "Left", "Right", ++ "Left + Right"}; ++static const char *wm8753_out3[] = {"VREF", "ROUT2", "Left + Right"}; ++static const char *wm8753_out4[] = {"VREF", "Capture ST", "LOUT2"}; ++static const char *wm8753_radcsel[] = {"PGA", "Line or RXP-RXN", "Sidetone"}; ++static const char *wm8753_ladcsel[] = {"PGA", "Line or RXP-RXN", "Line"}; ++static const char *wm8753_mono_adc[] = {"Stereo", "Analogue Mix Left", ++ "Analogue Mix Right", "Digital Mono Mix"}; ++static const char *wm8753_adc_hp[] = {"3.4Hz @ 48kHz", "82Hz @ 16k", ++ "82Hz @ 8kHz", "170Hz @ 8kHz"}; ++static const char *wm8753_adc_filter[] = {"HiFi", "Voice"}; ++static const char *wm8753_mic_sel[] = {"Mic 1", "Mic 2", "Mic 3"}; ++static const char *wm8753_dai_mode[] = {"DAI 0", "DAI 1", "DAI 2", "DAI 3"}; ++ ++static const struct soc_enum wm8753_enum[] = { ++SOC_ENUM_SINGLE(WM8753_BASS, 7, 2, wm8753_base), // 0 ++SOC_ENUM_SINGLE(WM8753_BASS, 4, 6, wm8753_base_filter), // 1 ++SOC_ENUM_SINGLE(WM8753_TREBLE, 6, 2, wm8753_treble), // 2 ++SOC_ENUM_SINGLE(WM8753_ALC1, 7, 4, wm8753_alc_func), // 3 ++SOC_ENUM_SINGLE(WM8753_NGATE, 1, 2, wm8753_ng_type), // 4 ++SOC_ENUM_SINGLE(WM8753_3D, 7, 2, wm8753_3d_func), // 5 ++SOC_ENUM_SINGLE(WM8753_3D, 6, 2, wm8753_3d_uc), // 6 ++SOC_ENUM_SINGLE(WM8753_3D, 5, 2, wm8753_3d_lc), // 7 ++SOC_ENUM_SINGLE(WM8753_DAC, 1, 4, wm8753_deemp), // 8 ++SOC_ENUM_SINGLE(WM8753_DAC, 4, 4, wm8753_mono_mix), // 9 ++SOC_ENUM_SINGLE(WM8753_DAC, 6, 2, wm8753_dac_phase), // 10 ++SOC_ENUM_SINGLE(WM8753_INCTL1, 3, 4, wm8753_line_mix), // 11 ++SOC_ENUM_SINGLE(WM8753_INCTL1, 2, 2, wm8753_mono_mux), // 12 ++SOC_ENUM_SINGLE(WM8753_INCTL1, 1, 2, wm8753_right_mux), // 13 ++SOC_ENUM_SINGLE(WM8753_INCTL1, 0, 2, wm8753_left_mux), // 14 ++SOC_ENUM_SINGLE(WM8753_INCTL2, 6, 4, wm8753_rxmsel), // 15 ++SOC_ENUM_SINGLE(WM8753_INCTL2, 4, 4, wm8753_sidetone_mux),// 16 ++SOC_ENUM_SINGLE(WM8753_OUTCTL, 7, 4, wm8753_mono2_src), // 17 ++SOC_ENUM_SINGLE(WM8753_OUTCTL, 0, 3, wm8753_out3), // 18 ++SOC_ENUM_SINGLE(WM8753_ADCTL2, 7, 3, wm8753_out4), // 19 ++SOC_ENUM_SINGLE(WM8753_ADCIN, 2, 3, wm8753_radcsel), // 20 ++SOC_ENUM_SINGLE(WM8753_ADCIN, 0, 3, wm8753_ladcsel), // 21 ++SOC_ENUM_SINGLE(WM8753_ADCIN, 4, 4, wm8753_mono_adc), // 22 ++SOC_ENUM_SINGLE(WM8753_ADC, 2, 4, wm8753_adc_hp), // 23 ++SOC_ENUM_SINGLE(WM8753_ADC, 4, 2, wm8753_adc_filter), // 24 ++SOC_ENUM_SINGLE(WM8753_MICBIAS, 6, 3, wm8753_mic_sel), // 25 ++SOC_ENUM_SINGLE(WM8753_IOCTL, 2, 4, wm8753_dai_mode), // 26 ++}; ++ ++ ++static int wm8753_get_dai(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ int mode = wm8753_read_reg_cache(codec, WM8753_IOCTL); ++ ++ ucontrol->value.integer.value[0] = (mode & 0xc) >> 2; ++ return 0; ++} ++ ++static int wm8753_set_dai(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ int mode = wm8753_read_reg_cache(codec, WM8753_IOCTL); ++ ++ if (((mode &0xc) >> 2) == ucontrol->value.integer.value[0]) ++ return 0; ++ ++ mode &= 0xfff3; ++ mode |= (ucontrol->value.integer.value[0] << 2); ++ ++ wm8753_write(codec, WM8753_IOCTL, mode); ++ wm8753_set_dai_mode(codec, ucontrol->value.integer.value[0]); ++ return 1; ++} ++ ++static const struct snd_kcontrol_new wm8753_snd_controls[] = { ++SOC_DOUBLE_R("PCM Volume", WM8753_LDAC, WM8753_RDAC, 0, 255, 0), ++ ++SOC_DOUBLE_R("ADC Capture Volume", WM8753_LADC, WM8753_RADC, 0, 63, 0), ++SOC_DOUBLE_R("ADC Capture Switch", WM8753_LINVOL, WM8753_RINVOL, 7, 1, 0), ++SOC_DOUBLE_R("ADC Capture ZC Switch", WM8753_LINVOL, WM8753_RINVOL, 6, 1, 0), ++ ++SOC_DOUBLE_R("Headphone Playback Volume", WM8753_LOUT1V, WM8753_ROUT1V, 0, 127, 0), ++SOC_DOUBLE_R("Speaker Playback Volume", WM8753_LOUT2V, WM8753_ROUT2V, 0, 127, 0), ++ ++SOC_SINGLE("Mono Playback Volume", WM8753_MOUTV, 0, 127, 0), ++ ++SOC_DOUBLE_R("Bypass Playback Volume", WM8753_LOUTM1, WM8753_ROUTM1, 4, 7, 1), ++SOC_DOUBLE_R("Sidetone Playback Volume", WM8753_LOUTM2, WM8753_ROUTM2, 4, 7, 1), ++SOC_DOUBLE_R("Voice Playback Volume", WM8753_LOUTM2, WM8753_ROUTM2, 0, 7, 1), ++ ++SOC_DOUBLE_R("Headphone Playback ZC Switch", WM8753_LOUT1V, WM8753_ROUT1V, 7, 1, 0), ++SOC_DOUBLE_R("Speaker Playback ZC Switch", WM8753_LOUT2V, WM8753_ROUT2V, 7, 1, 0), ++ ++SOC_SINGLE("Mono Bypass Playback Volume", WM8753_MOUTM1, 4, 7, 1), ++SOC_SINGLE("Mono Sidetone Playback Volume", WM8753_MOUTM2, 4, 7, 1), ++SOC_SINGLE("Mono Voice Playback Volume", WM8753_MOUTM2, 4, 7, 1), ++SOC_SINGLE("Mono Playback ZC Switch", WM8753_MOUTV, 7, 1, 0), ++ ++SOC_ENUM("Bass Boost", wm8753_enum[0]), ++SOC_ENUM("Bass Filter", wm8753_enum[1]), ++SOC_SINGLE("Bass Volume", WM8753_BASS, 0, 7, 1), ++ ++SOC_SINGLE("Treble Volume", WM8753_TREBLE, 0, 7, 0), ++SOC_ENUM("Treble Cut-off", wm8753_enum[2]), ++ ++SOC_DOUBLE("Sidetone Capture Volume", WM8753_RECMIX1, 0, 4, 7, 1), ++SOC_SINGLE("Voice Sidetone Capture Volume", WM8753_RECMIX2, 0, 7, 1), ++ ++SOC_DOUBLE_R("Capture Volume", WM8753_LINVOL, WM8753_RINVOL, 0, 63, 0), ++SOC_DOUBLE_R("Capture ZC Switch", WM8753_LINVOL, WM8753_RINVOL, 6, 1, 0), ++SOC_DOUBLE_R("Capture Switch", WM8753_LINVOL, WM8753_RINVOL, 7, 1, 0), ++ ++SOC_ENUM("Capture Filter Select", wm8753_enum[23]), ++SOC_ENUM("Capture Filter Cut-off", wm8753_enum[24]), ++SOC_SINGLE("Capture Filter Switch", WM8753_ADC, 0, 1, 1), ++ ++SOC_SINGLE("ALC Capture Target Volume", WM8753_ALC1, 0, 7, 0), ++SOC_SINGLE("ALC Capture Max Volume", WM8753_ALC1, 4, 7, 0), ++SOC_ENUM("ALC Capture Function", wm8753_enum[3]), ++SOC_SINGLE("ALC Capture ZC Switch", WM8753_ALC2, 8, 1, 0), ++SOC_SINGLE("ALC Capture Hold Time", WM8753_ALC2, 0, 15, 1), ++SOC_SINGLE("ALC Capture Decay Time", WM8753_ALC3, 4, 15, 1), ++SOC_SINGLE("ALC Capture Attack Time", WM8753_ALC3, 0, 15, 0), ++SOC_SINGLE("ALC Capture NG Threshold", WM8753_NGATE, 3, 31, 0), ++SOC_ENUM("ALC Capture NG Type", wm8753_enum[4]), ++SOC_SINGLE("ALC Capture NG Switch", WM8753_NGATE, 0, 1, 0), ++ ++SOC_ENUM("3D Function", wm8753_enum[5]), ++SOC_ENUM("3D Upper Cut-off", wm8753_enum[6]), ++SOC_ENUM("3D Lower Cut-off", wm8753_enum[7]), ++SOC_SINGLE("3D Volume", WM8753_3D, 1, 15, 0), ++SOC_SINGLE("3D Switch", WM8753_3D, 0, 1, 0), ++ ++SOC_SINGLE("Capture 6dB Attenuate", WM8753_ADCTL1, 2, 1, 0), ++SOC_SINGLE("Playback 6dB Attenuate", WM8753_ADCTL1, 1, 1, 0), ++ ++SOC_ENUM("De-emphasis", wm8753_enum[8]), ++SOC_ENUM("Playback Mono Mix", wm8753_enum[9]), ++SOC_ENUM("Playback Phase", wm8753_enum[10]), ++ ++SOC_SINGLE("Mic2 Capture Volume", WM8753_INCTL1, 7, 3, 0), ++SOC_SINGLE("Mic1 Capture Volume", WM8753_INCTL1, 5, 3, 0), ++ ++SOC_ENUM_EXT("DAI Mode", wm8753_enum[26], wm8753_get_dai, wm8753_set_dai), ++}; ++ ++/* add non dapm controls */ ++static int wm8753_add_controls(struct snd_soc_codec *codec) ++{ ++ int err, i; ++ ++ for (i = 0; i < ARRAY_SIZE(wm8753_snd_controls); i++) { ++ err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&wm8753_snd_controls[i],codec, NULL)); ++ if (err < 0) ++ return err; ++ } ++ return 0; ++} ++ ++/* ++ * _DAPM_ Controls ++ */ ++ ++/* Left Mixer */ ++static const struct snd_kcontrol_new wm8753_left_mixer_controls[] = { ++SOC_DAPM_SINGLE("Voice Playback Switch", WM8753_LOUTM2, 8, 1, 0), ++SOC_DAPM_SINGLE("Sidetone Playback Switch", WM8753_LOUTM2, 7, 1, 0), ++SOC_DAPM_SINGLE("Left Playback Switch", WM8753_LOUTM1, 8, 1, 0), ++SOC_DAPM_SINGLE("Bypass Playback Switch", WM8753_LOUTM1, 7, 1, 0), ++}; ++ ++/* Right mixer */ ++static const struct snd_kcontrol_new wm8753_right_mixer_controls[] = { ++SOC_DAPM_SINGLE("Voice Playback Switch", WM8753_ROUTM2, 8, 1, 0), ++SOC_DAPM_SINGLE("Sidetone Playback Switch", WM8753_ROUTM2, 7, 1, 0), ++SOC_DAPM_SINGLE("Right Playback Switch", WM8753_ROUTM1, 8, 1, 0), ++SOC_DAPM_SINGLE("Bypass Playback Switch", WM8753_ROUTM1, 7, 1, 0), ++}; ++ ++/* Mono mixer */ ++static const struct snd_kcontrol_new wm8753_mono_mixer_controls[] = { ++SOC_DAPM_SINGLE("Left Playback Switch", WM8753_MOUTM1, 8, 1, 0), ++SOC_DAPM_SINGLE("Right Playback Switch", WM8753_MOUTM2, 8, 1, 0), ++SOC_DAPM_SINGLE("Voice Playback Switch", WM8753_MOUTM2, 3, 1, 0), ++SOC_DAPM_SINGLE("Sidetone Playback Switch", WM8753_MOUTM2, 7, 1, 0), ++SOC_DAPM_SINGLE("Bypass Playback Switch", WM8753_MOUTM1, 7, 1, 0), ++}; ++ ++/* Mono 2 Mux */ ++static const struct snd_kcontrol_new wm8753_mono2_controls = ++SOC_DAPM_ENUM("Route", wm8753_enum[17]); ++ ++/* Out 3 Mux */ ++static const struct snd_kcontrol_new wm8753_out3_controls = ++SOC_DAPM_ENUM("Route", wm8753_enum[18]); ++ ++/* Out 4 Mux */ ++static const struct snd_kcontrol_new wm8753_out4_controls = ++SOC_DAPM_ENUM("Route", wm8753_enum[19]); ++ ++/* ADC Mono Mix */ ++static const struct snd_kcontrol_new wm8753_adc_mono_controls = ++SOC_DAPM_ENUM("Route", wm8753_enum[22]); ++ ++/* Record mixer */ ++static const struct snd_kcontrol_new wm8753_record_mixer_controls[] = { ++SOC_DAPM_SINGLE("Voice Capture Switch", WM8753_RECMIX2, 3, 1, 0), ++SOC_DAPM_SINGLE("Left Capture Switch", WM8753_RECMIX1, 3, 1, 0), ++SOC_DAPM_SINGLE("Right Capture Switch", WM8753_RECMIX1, 7, 1, 0), ++}; ++ ++/* Left ADC mux */ ++static const struct snd_kcontrol_new wm8753_adc_left_controls = ++SOC_DAPM_ENUM("Route", wm8753_enum[21]); ++ ++/* Right ADC mux */ ++static const struct snd_kcontrol_new wm8753_adc_right_controls = ++SOC_DAPM_ENUM("Route", wm8753_enum[20]); ++ ++/* MIC mux */ ++static const struct snd_kcontrol_new wm8753_mic_mux_controls = ++SOC_DAPM_ENUM("Route", wm8753_enum[16]); ++ ++/* ALC mixer */ ++static const struct snd_kcontrol_new wm8753_alc_mixer_controls[] = { ++SOC_DAPM_SINGLE("Line Capture Switch", WM8753_INCTL2, 3, 1, 0), ++SOC_DAPM_SINGLE("Mic2 Capture Switch", WM8753_INCTL2, 2, 1, 0), ++SOC_DAPM_SINGLE("Mic1 Capture Switch", WM8753_INCTL2, 1, 1, 0), ++SOC_DAPM_SINGLE("Rx Capture Switch", WM8753_INCTL2, 0, 1, 0), ++}; ++ ++/* Left Line mux */ ++static const struct snd_kcontrol_new wm8753_line_left_controls = ++SOC_DAPM_ENUM("Route", wm8753_enum[14]); ++ ++/* Right Line mux */ ++static const struct snd_kcontrol_new wm8753_line_right_controls = ++SOC_DAPM_ENUM("Route", wm8753_enum[13]); ++ ++/* Mono Line mux */ ++static const struct snd_kcontrol_new wm8753_line_mono_controls = ++SOC_DAPM_ENUM("Route", wm8753_enum[12]); ++ ++/* Line mux and mixer */ ++static const struct snd_kcontrol_new wm8753_line_mux_mix_controls = ++SOC_DAPM_ENUM("Route", wm8753_enum[11]); ++ ++/* Rx mux and mixer */ ++static const struct snd_kcontrol_new wm8753_rx_mux_mix_controls = ++SOC_DAPM_ENUM("Route", wm8753_enum[15]); ++ ++/* Mic Selector Mux */ ++static const struct snd_kcontrol_new wm8753_mic_sel_mux_controls = ++SOC_DAPM_ENUM("Route", wm8753_enum[25]); ++ ++static const struct snd_soc_dapm_widget wm8753_dapm_widgets[] = { ++SND_SOC_DAPM_MICBIAS("Mic Bias", WM8753_PWR1, 5, 0), ++SND_SOC_DAPM_MIXER("Left Mixer", WM8753_PWR4, 0, 0, ++ &wm8753_left_mixer_controls[0], ARRAY_SIZE(wm8753_left_mixer_controls)), ++SND_SOC_DAPM_PGA("Left Out 1", WM8753_PWR3, 8, 0, NULL, 0), ++SND_SOC_DAPM_PGA("Left Out 2", WM8753_PWR3, 6, 0, NULL, 0), ++SND_SOC_DAPM_DAC("Left DAC", "Left HiFi Playback", WM8753_PWR1, 3, 0), ++SND_SOC_DAPM_OUTPUT("LOUT1"), ++SND_SOC_DAPM_OUTPUT("LOUT2"), ++SND_SOC_DAPM_MIXER("Right Mixer", WM8753_PWR4, 1, 0, ++ &wm8753_right_mixer_controls[0], ARRAY_SIZE(wm8753_right_mixer_controls)), ++SND_SOC_DAPM_PGA("Right Out 1", WM8753_PWR3, 7, 0, NULL, 0), ++SND_SOC_DAPM_PGA("Right Out 2", WM8753_PWR3, 5, 0, NULL, 0), ++SND_SOC_DAPM_DAC("Right DAC", "Right HiFi Playback", WM8753_PWR1, 2, 0), ++SND_SOC_DAPM_OUTPUT("ROUT1"), ++SND_SOC_DAPM_OUTPUT("ROUT2"), ++SND_SOC_DAPM_MIXER("Mono Mixer", WM8753_PWR4, 2, 0, ++ &wm8753_mono_mixer_controls[0], ARRAY_SIZE(wm8753_mono_mixer_controls)), ++SND_SOC_DAPM_PGA("Mono Out 1", WM8753_PWR3, 2, 0, NULL, 0), ++SND_SOC_DAPM_PGA("Mono Out 2", WM8753_PWR3, 1, 0, NULL, 0), ++SND_SOC_DAPM_DAC("Voice DAC", "Voice Playback", WM8753_PWR1, 4, 0), ++SND_SOC_DAPM_OUTPUT("MONO1"), ++SND_SOC_DAPM_MUX("Mono 2 Mux", SND_SOC_NOPM, 0, 0, &wm8753_mono2_controls), ++SND_SOC_DAPM_OUTPUT("MONO2"), ++SND_SOC_DAPM_MIXER("Out3 Left + Right", -1, 0, 0, NULL, 0), ++SND_SOC_DAPM_MUX("Out3 Mux", SND_SOC_NOPM, 0, 0, &wm8753_out3_controls), ++SND_SOC_DAPM_PGA("Out 3", WM8753_PWR3, 4, 0, NULL, 0), ++SND_SOC_DAPM_OUTPUT("OUT3"), ++SND_SOC_DAPM_MUX("Out4 Mux", SND_SOC_NOPM, 0, 0, &wm8753_out4_controls), ++SND_SOC_DAPM_PGA("Out 4", WM8753_PWR3, 3, 0, NULL, 0), ++SND_SOC_DAPM_OUTPUT("OUT4"), ++SND_SOC_DAPM_MIXER("Playback Mixer", WM8753_PWR4, 3, 0, ++ &wm8753_record_mixer_controls[0], ++ ARRAY_SIZE(wm8753_record_mixer_controls)), ++SND_SOC_DAPM_ADC("Left ADC", "Left Voice Capture", WM8753_PWR2, 3, 0), ++SND_SOC_DAPM_ADC("Right ADC", "Right Voice Capture", WM8753_PWR2, 2, 0), ++SND_SOC_DAPM_MUX("Capture Left Mixer", SND_SOC_NOPM, 0, 0, ++ &wm8753_adc_mono_controls), ++SND_SOC_DAPM_MUX("Capture Right Mixer", SND_SOC_NOPM, 0, 0, ++ &wm8753_adc_mono_controls), ++SND_SOC_DAPM_MUX("Capture Left Mux", SND_SOC_NOPM, 0, 0, ++ &wm8753_adc_left_controls), ++SND_SOC_DAPM_MUX("Capture Right Mux", SND_SOC_NOPM, 0, 0, ++ &wm8753_adc_right_controls), ++SND_SOC_DAPM_MUX("Mic Sidetone Mux", SND_SOC_NOPM, 0, 0, ++ &wm8753_mic_mux_controls), ++SND_SOC_DAPM_PGA("Left Capture Volume", WM8753_PWR2, 5, 0, NULL, 0), ++SND_SOC_DAPM_PGA("Right Capture Volume", WM8753_PWR2, 4, 0, NULL, 0), ++SND_SOC_DAPM_MIXER("ALC Mixer", WM8753_PWR2, 6, 0, ++ &wm8753_alc_mixer_controls[0], ARRAY_SIZE(wm8753_alc_mixer_controls)), ++SND_SOC_DAPM_MUX("Line Left Mux", SND_SOC_NOPM, 0, 0, ++ &wm8753_line_left_controls), ++SND_SOC_DAPM_MUX("Line Right Mux", SND_SOC_NOPM, 0, 0, ++ &wm8753_line_right_controls), ++SND_SOC_DAPM_MUX("Line Mono Mux", SND_SOC_NOPM, 0, 0, ++ &wm8753_line_mono_controls), ++SND_SOC_DAPM_MUX("Line Mixer", SND_SOC_NOPM, 0, 0, ++ &wm8753_line_mux_mix_controls), ++SND_SOC_DAPM_MUX("Rx Mixer", SND_SOC_NOPM, 0, 0, ++ &wm8753_rx_mux_mix_controls), ++SND_SOC_DAPM_PGA("Mic 1 Volume", WM8753_PWR2, 8, 0, NULL, 0), ++SND_SOC_DAPM_PGA("Mic 2 Volume", WM8753_PWR2, 7, 0, NULL, 0), ++SND_SOC_DAPM_MUX("Mic Selection Mux", SND_SOC_NOPM, 0, 0, ++ &wm8753_mic_sel_mux_controls), ++SND_SOC_DAPM_INPUT("LINE1"), ++SND_SOC_DAPM_INPUT("LINE2"), ++SND_SOC_DAPM_INPUT("RXP"), ++SND_SOC_DAPM_INPUT("RXN"), ++SND_SOC_DAPM_INPUT("ACIN"), ++SND_SOC_DAPM_INPUT("ACOP"), ++SND_SOC_DAPM_INPUT("MIC1N"), ++SND_SOC_DAPM_INPUT("MIC1"), ++SND_SOC_DAPM_INPUT("MIC2N"), ++SND_SOC_DAPM_INPUT("MIC2"), ++SND_SOC_DAPM_VMID("VREF"), ++}; ++ ++static const char *audio_map[][3] = { ++ /* left mixer */ ++ {"Left Mixer", "Left Playback Switch", "Left DAC"}, ++ {"Left Mixer", "Voice Playback Switch", "Voice DAC"}, ++ {"Left Mixer", "Sidetone Playback Switch", "Mic Sidetone Mux"}, ++ {"Left Mixer", "Bypass Playback Switch", "Line Left Mux"}, ++ ++ /* right mixer */ ++ {"Right Mixer", "Right Playback Switch", "Right DAC"}, ++ {"Right Mixer", "Voice Playback Switch", "Voice DAC"}, ++ {"Right Mixer", "Sidetone Playback Switch", "Mic Sidetone Mux"}, ++ {"Right Mixer", "Bypass Playback Switch", "Line Right Mux"}, ++ ++ /* mono mixer */ ++ {"Mono Mixer", "Voice Playback Switch", "Voice DAC"}, ++ {"Mono Mixer", "Left Playback Switch", "Left DAC"}, ++ {"Mono Mixer", "Right Playback Switch", "Right DAC"}, ++ {"Mono Mixer", "Sidetone Playback Switch", "Mic Sidetone Mux"}, ++ {"Mono Mixer", "Bypass Playback Switch", "Line Mono Mux"}, ++ ++ /* left out */ ++ {"Left Out 1", NULL, "Left Mixer"}, ++ {"Left Out 2", NULL, "Left Mixer"}, ++ {"LOUT1", NULL, "Left Out 1"}, ++ {"LOUT2", NULL, "Left Out 2"}, ++ ++ /* right out */ ++ {"Right Out 1", NULL, "Right Mixer"}, ++ {"Right Out 2", NULL, "Right Mixer"}, ++ {"ROUT1", NULL, "Right Out 1"}, ++ {"ROUT2", NULL, "Right Out 2"}, ++ ++ /* mono 1 out */ ++ {"Mono Out 1", NULL, "Mono Mixer"}, ++ {"MONO1", NULL, "Mono Out 1"}, ++ ++ /* mono 2 out */ ++ {"Mono 2 Mux", "Left + Right", "Out3 Left + Right"}, ++ {"Mono 2 Mux", "Inverted Mono 1", "MONO1"}, ++ {"Mono 2 Mux", "Left", "Left Mixer"}, ++ {"Mono 2 Mux", "Right", "Right Mixer"}, ++ {"Mono Out 2", NULL, "Mono 2 Mux"}, ++ {"MONO2", NULL, "Mono Out 2"}, ++ ++ /* out 3 */ ++ {"Out3 Left + Right", NULL, "Left Mixer"}, ++ {"Out3 Left + Right", NULL, "Right Mixer"}, ++ {"Out3 Mux", "VREF", "VREF"}, ++ {"Out3 Mux", "Left + Right", "Out3 Left + Right"}, ++ {"Out3 Mux", "ROUT2", "ROUT2"}, ++ {"Out 3", NULL, "Out3 Mux"}, ++ {"OUT3", NULL, "Out 3"}, ++ ++ /* out 4 */ ++ {"Out4 Mux", "VREF", "VREF"}, ++ {"Out4 Mux", "Capture ST", "Capture ST Mixer"}, ++ {"Out4 Mux", "LOUT2", "LOUT2"}, ++ {"Out 4", NULL, "Out4 Mux"}, ++ {"OUT4", NULL, "Out 4"}, ++ ++ /* record mixer */ ++ {"Playback Mixer", "Left Capture Switch", "Left Mixer"}, ++ {"Playback Mixer", "Voice Capture Switch", "Mono Mixer"}, ++ {"Playback Mixer", "Right Capture Switch", "Right Mixer"}, ++ ++ /* Mic/SideTone Mux */ ++ {"Mic Sidetone Mux", "Left PGA", "Left Capture Volume"}, ++ {"Mic Sidetone Mux", "Right PGA", "Right Capture Volume"}, ++ {"Mic Sidetone Mux", "Mic 1", "Mic 1 Volume"}, ++ {"Mic Sidetone Mux", "Mic 2", "Mic 2 Volume"}, ++ ++ /* Capture Left Mux */ ++ {"Capture Left Mux", "PGA", "Left Capture Volume"}, ++ {"Capture Left Mux", "Line or RXP-RXN", "Line Left Mux"}, ++ {"Capture Left Mux", "Line", "LINE1"}, ++ ++ /* Capture Right Mux */ ++ {"Capture Right Mux", "PGA", "Right Capture Volume"}, ++ {"Capture Right Mux", "Line or RXP-RXN", "Line Right Mux"}, ++ {"Capture Right Mux", "Sidetone", "Capture ST Mixer"}, ++ ++ /* Mono Capture mixer-mux */ ++ {"Capture Right Mixer", "Stereo", "Capture Right Mux"}, ++ {"Capture Left Mixer", "Analogue Mix Left", "Capture Left Mux"}, ++ {"Capture Left Mixer", "Analogue Mix Left", "Capture Right Mux"}, ++ {"Capture Right Mixer", "Analogue Mix Right", "Capture Left Mux"}, ++ {"Capture Right Mixer", "Analogue Mix Right", "Capture Right Mux"}, ++ {"Capture Left Mixer", "Digital Mono Mix", "Capture Left Mux"}, ++ {"Capture Left Mixer", "Digital Mono Mix", "Capture Right Mux"}, ++ {"Capture Right Mixer", "Digital Mono Mix", "Capture Left Mux"}, ++ {"Capture Right Mixer", "Digital Mono Mix", "Capture Right Mux"}, ++ ++ /* ADC */ ++ {"Left ADC", NULL, "Capture Left Mixer"}, ++ {"Right ADC", NULL, "Capture Right Mixer"}, ++ ++ /* Left Capture Volume */ ++ {"Left Capture Volume", NULL, "ACIN"}, ++ ++ /* Right Capture Volume */ ++ {"Right Capture Volume", NULL, "Mic 2 Volume"}, ++ ++ /* ALC Mixer */ ++ {"ALC Mixer", "Line Capture Switch", "Line Mixer"}, ++ {"ALC Mixer", "Mic2 Capture Switch", "Mic 2 Volume"}, ++ {"ALC Mixer", "Mic1 Capture Switch", "Mic 1 Volume"}, ++ {"ALC Mixer", "Rx Capture Switch", "Rx Mixer"}, ++ ++ /* Line Left Mux */ ++ {"Line Left Mux", "Line 1", "LINE1"}, ++ {"Line Left Mux", "Rx Mix", "Rx Mixer"}, ++ ++ /* Line Right Mux */ ++ {"Line Right Mux", "Line 2", "LINE2"}, ++ {"Line Right Mux", "Rx Mix", "Rx Mixer"}, ++ ++ /* Line Mono Mux */ ++ {"Line Mono Mux", "Line Mix", "Line Mixer"}, ++ {"Line Mono Mux", "Rx Mix", "Rx Mixer"}, ++ ++ /* Line Mixer/Mux */ ++ {"Line Mixer", "Line 1 + 2", "LINE1"}, ++ {"Line Mixer", "Line 1 - 2", "LINE1"}, ++ {"Line Mixer", "Line 1 + 2", "LINE2"}, ++ {"Line Mixer", "Line 1 - 2", "LINE2"}, ++ {"Line Mixer", "Line 1", "LINE1"}, ++ {"Line Mixer", "Line 2", "LINE2"}, ++ ++ /* Rx Mixer/Mux */ ++ {"Rx Mixer", "RXP - RXN", "RXP"}, ++ {"Rx Mixer", "RXP + RXN", "RXP"}, ++ {"Rx Mixer", "RXP - RXN", "RXN"}, ++ {"Rx Mixer", "RXP + RXN", "RXN"}, ++ {"Rx Mixer", "RXP", "RXP"}, ++ {"Rx Mixer", "RXN", "RXN"}, ++ ++ /* Mic 1 Volume */ ++ {"Mic 1 Volume", NULL, "MIC1N"}, ++ {"Mic 1 Volume", NULL, "Mic Selection Mux"}, ++ ++ /* Mic 2 Volume */ ++ {"Mic 2 Volume", NULL, "MIC2N"}, ++ {"Mic 2 Volume", NULL, "MIC2"}, ++ ++ /* Mic Selector Mux */ ++ {"Mic Selection Mux", "Mic 1", "MIC1"}, ++ {"Mic Selection Mux", "Mic 2", "MIC2N"}, ++ {"Mic Selection Mux", "Mic 3", "MIC2"}, ++ ++ /* ACOP */ ++ {"ACOP", NULL, "ALC Mixer"}, ++ ++ /* terminator */ ++ {NULL, NULL, NULL}, ++}; ++ ++static int wm8753_add_widgets(struct snd_soc_codec *codec) ++{ ++ int i; ++ ++ for(i = 0; i < ARRAY_SIZE(wm8753_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &wm8753_dapm_widgets[i]); ++ } ++ ++ /* set up the WM8753 audio map */ ++ for(i = 0; audio_map[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, audio_map[i][0], ++ audio_map[i][1], audio_map[i][2]); ++ } ++ ++ snd_soc_dapm_new_widgets(codec); ++ return 0; ++} ++ ++/* PLL divisors */ ++struct _pll_div { ++ u32 pll_in; /* ext clock input */ ++ u32 pll_out; /* pll out freq */ ++ u32 div2:1; ++ u32 n:4; ++ u32 k:24; ++}; ++ ++/* ++ * PLL divisors - ++ */ ++static const struct _pll_div pll_div[] = { ++ {13000000, 12288000, 0, 0x7, 0x23F54A}, ++ {13000000, 11289600, 0, 0x6, 0x3CA2F5}, ++ {12000000, 12288000, 0, 0x8, 0x0C49BA}, ++ {12000000, 11289600, 0, 0x7, 0x21B08A}, ++ {24000000, 12288000, 1, 0x8, 0x0C49BA}, ++ {24000000, 11289600, 1, 0x7, 0x21B08A}, ++ {12288000, 11289600, 0, 0x7, 0x166667}, ++ {26000000, 11289600, 1, 0x6, 0x3CA2F5}, ++ {26000000, 12288000, 1, 0x7, 0x23F54A}, ++}; ++ ++static u32 wm8753_config_pll(struct snd_soc_codec *codec, ++ struct snd_soc_codec_dai *dai, int pll) ++{ ++ u16 reg; ++ int found = 0; ++ ++ if (pll == 1) { ++ reg = wm8753_read_reg_cache(codec, WM8753_CLOCK) & 0xffef; ++ if (!dai->pll_in || !dai->mclk) { ++ /* disable PLL1 */ ++ wm8753_write(codec, WM8753_PLL1CTL1, 0x0026); ++ wm8753_write(codec, WM8753_CLOCK, reg); ++ return 0; ++ } else { ++ u16 value = 0; ++ int i = 0; ++ ++ /* if we cant match, then use good values for N and K */ ++ for (;i < ARRAY_SIZE(pll_div); i++) { ++ if (pll_div[i].pll_out == dai->pll_out && ++ pll_div[i].pll_in == dai->pll_in) { ++ found = 1; ++ break; ++ } ++ } ++ ++ if (!found) ++ goto err; ++ ++ /* set up N and K PLL divisor ratios */ ++ /* bits 8:5 = PLL_N, bits 3:0 = PLL_K[21:18] */ ++ value = (pll_div[i].n << 5) + ((pll_div[i].k & 0x3c0000) >> 18); ++ wm8753_write(codec, WM8753_PLL1CTL2, value); ++ ++ /* bits 8:0 = PLL_K[17:9] */ ++ value = (pll_div[i].k & 0x03fe00) >> 9; ++ wm8753_write(codec, WM8753_PLL1CTL3, value); ++ ++ /* bits 8:0 = PLL_K[8:0] */ ++ value = pll_div[i].k & 0x0001ff; ++ wm8753_write(codec, WM8753_PLL1CTL4, value); ++ ++ /* set PLL1 as input and enable */ ++ wm8753_write(codec, WM8753_PLL1CTL1, 0x0027 | ++ (pll_div[i].div2 << 3)); ++ wm8753_write(codec, WM8753_CLOCK, reg | 0x0010); ++ } ++ } else { ++ reg = wm8753_read_reg_cache(codec, WM8753_CLOCK) & 0xfff7; ++ if (!dai->pll_in || !dai->mclk) { ++ /* disable PLL2 */ ++ wm8753_write(codec, WM8753_PLL2CTL1, 0x0026); ++ wm8753_write(codec, WM8753_CLOCK, reg); ++ return 0; ++ } else { ++ u16 value = 0; ++ int i = 0; ++ ++ /* if we cant match, then use good values for N and K */ ++ for (;i < ARRAY_SIZE(pll_div); i++) { ++ if (pll_div[i].pll_out == dai->pll_out && ++ pll_div[i].pll_in == dai->pll_in) { ++ found = 1; ++ break; ++ } ++ } ++ ++ if (!found) ++ goto err; ++ ++ /* set up N and K PLL divisor ratios */ ++ /* bits 8:5 = PLL_N, bits 3:0 = PLL_K[21:18] */ ++ value = (pll_div[i].n << 5) + ((pll_div[i].k & 0x3c0000) >> 18); ++ wm8753_write(codec, WM8753_PLL2CTL2, value); ++ ++ /* bits 8:0 = PLL_K[17:9] */ ++ value = (pll_div[i].k & 0x03fe00) >> 9; ++ wm8753_write(codec, WM8753_PLL2CTL3, value); ++ ++ /* bits 8:0 = PLL_K[8:0] */ ++ value = pll_div[i].k & 0x0001ff; ++ wm8753_write(codec, WM8753_PLL2CTL4, value); ++ ++ /* set PLL1 as input and enable */ ++ wm8753_write(codec, WM8753_PLL2CTL1, 0x0027 | ++ (pll_div[i].div2 << 3)); ++ wm8753_write(codec, WM8753_CLOCK, reg | 0x0008); ++ } ++ } ++ ++ return dai->pll_in; ++err: ++ return 0; ++} ++ ++struct _coeff_div { ++ u32 mclk; ++ u32 rate; ++ u16 fs; ++ u8 sr:5; ++ u8 usb:1; ++}; ++ ++/* codec hifi mclk (after PLL) clock divider coefficients */ ++static const struct _coeff_div coeff_div[] = { ++ /* 8k */ ++ {12288000, 8000, 1536, 0x6, 0x0}, ++ {11289600, 8000, 1408, 0x16, 0x0}, ++ {18432000, 8000, 2304, 0x7, 0x0}, ++ {16934400, 8000, 2112, 0x17, 0x0}, ++ {12000000, 8000, 1500, 0x6, 0x1}, ++ ++ /* 11.025k */ ++ {11289600, 11025, 1024, 0x18, 0x0}, ++ {16934400, 11025, 1536, 0x19, 0x0}, ++ {12000000, 11025, 1088, 0x19, 0x1}, ++ ++ /* 16k */ ++ {12288000, 16000, 768, 0xa, 0x0}, ++ {18432000, 16000, 1152, 0xb, 0x0}, ++ {12000000, 16000, 750, 0xa, 0x1}, ++ ++ /* 22.05k */ ++ {11289600, 22050, 512, 0x1a, 0x0}, ++ {16934400, 22050, 768, 0x1b, 0x0}, ++ {12000000, 22050, 544, 0x1b, 0x1}, ++ ++ /* 32k */ ++ {12288000, 32000, 384, 0xc, 0x0}, ++ {18432000, 32000, 576, 0xd, 0x0}, ++ {12000000, 32000, 375, 0xa, 0x1}, ++ ++ /* 44.1k */ ++ {11289600, 44100, 256, 0x10, 0x0}, ++ {16934400, 44100, 384, 0x11, 0x0}, ++ {12000000, 44100, 272, 0x11, 0x1}, ++ ++ /* 48k */ ++ {12288000, 48000, 256, 0x0, 0x0}, ++ {18432000, 48000, 384, 0x1, 0x0}, ++ {12000000, 48000, 250, 0x0, 0x1}, ++ ++ /* 88.2k */ ++ {11289600, 88200, 128, 0x1e, 0x0}, ++ {16934400, 88200, 192, 0x1f, 0x0}, ++ {12000000, 88200, 136, 0x1f, 0x1}, ++ ++ /* 96k */ ++ {12288000, 96000, 128, 0xe, 0x0}, ++ {18432000, 96000, 192, 0xf, 0x0}, ++ {12000000, 96000, 125, 0xe, 0x1}, ++}; ++ ++static int get_coeff(int mclk, int rate) ++{ ++ int i; ++ ++ for (i = 0; i < ARRAY_SIZE(coeff_div); i++) { ++ if (coeff_div[i].rate == rate && coeff_div[i].mclk == mclk) ++ return i; ++ } ++ return -EINVAL; ++} ++ ++/* supported HiFi input clocks (that don't use PLL) */ ++const static int hifi_clks[] = {11289600, 12000000, 12288000, ++ 16934400, 18432000}; ++ ++/* The HiFi interface can be clocked in one of two ways:- ++ * o No PLL - MCLK is used directly. ++ * o PLL - PLL is used to generate audio MCLK from input clock. ++ * ++ * We use the direct method if we can as it saves power. ++ */ ++static unsigned int wm8753_config_i2s_sysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ int i, pll_out; ++ ++ /* is clk supported without the PLL */ ++ for(i = 0; i < ARRAY_SIZE(hifi_clks); i++) { ++ if (clk == hifi_clks[i]) { ++ dai->mclk = clk; ++ dai->pll_in = dai->pll_out = 0; ++ dai->clk_div = 1; ++ return clk; ++ } ++ } ++ ++ /* determine best PLL output speed */ ++ if (info->bclk_master & ++ (SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS)) { ++ pll_out = info->fs * info->rate; ++ } else { ++ /* calc slave clock */ ++ switch (info->rate){ ++ case 11025: ++ case 22050: ++ case 44100: ++ case 88200: ++ pll_out = 11289600; ++ break; ++ default: ++ pll_out = 12288000; ++ break; ++ } ++ } ++ ++ /* are input & output clocks supported by PLL */ ++ for (i = 0;i < ARRAY_SIZE(pll_div); i++) { ++ if (pll_div[i].pll_in == clk && pll_div[i].pll_out == pll_out) { ++ dai->pll_in = clk; ++ dai->pll_out = dai->mclk = pll_out; ++ return pll_out; ++ } ++ } ++ ++ /* this clk is not supported */ ++ return 0; ++} ++ ++/* valid PCM clock dividers * 2 */ ++static int pcm_divs[] = {2, 6, 11, 4, 8, 12, 16}; ++ ++/* The Voice interface can be clocked in one of four ways:- ++ * o No PLL - MCLK is used directly. ++ * o Div - MCLK is directly divided. ++ * o PLL - PLL is used to generate audio MCLK from input clock. ++ * o PLL & Div - PLL and post divider are used. ++ * ++ * We use the non PLL methods if we can, as it saves power. ++ */ ++ ++static unsigned int wm8753_config_pcm_sysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ int i, j, best_clk = info->fs * info->rate; ++ ++ /* can we run at this clk without the PLL ? */ ++ for (i = 0; i < ARRAY_SIZE(pcm_divs); i++) { ++ if ((best_clk >> 1) * pcm_divs[i] == clk) { ++ dai->pll_in = 0; ++ dai->clk_div = pcm_divs[i]; ++ dai->mclk = best_clk; ++ return dai->mclk; ++ } ++ } ++ ++ /* now check for PLL support */ ++ for (i = 0; i < ARRAY_SIZE(pll_div); i++) { ++ if (pll_div[i].pll_in == clk) { ++ for (j = 0; j < ARRAY_SIZE(pcm_divs); j++) { ++ if (pll_div[i].pll_out == pcm_divs[j] * (best_clk >> 1)) { ++ dai->pll_in = clk; ++ dai->pll_out = pll_div[i].pll_out; ++ dai->clk_div = pcm_divs[j]; ++ dai->mclk = best_clk; ++ return dai->mclk; ++ } ++ } ++ } ++ } ++ ++ /* this clk is not supported */ ++ return 0; ++} ++ ++/* set the format and bit size for ADC and Voice DAC */ ++static void wm8753_adc_vdac_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 voice = wm8753_read_reg_cache(codec, WM8753_PCM) & 0x01e0; ++ ++ /* interface format */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_I2S: ++ voice |= 0x0002; ++ break; ++ case SND_SOC_DAIFMT_RIGHT_J: ++ break; ++ case SND_SOC_DAIFMT_LEFT_J: ++ voice |= 0x0001; ++ break; ++ case SND_SOC_DAIFMT_DSP_A: ++ voice |= 0x0003; ++ break; ++ case SND_SOC_DAIFMT_DSP_B: ++ voice |= 0x0013; ++ break; ++ } ++ ++ /* bit size */ ++ switch (rtd->codec_dai->dai_runtime.pcmfmt) { ++ case SNDRV_PCM_FMTBIT_S16_LE: ++ break; ++ case SNDRV_PCM_FMTBIT_S20_3LE: ++ voice |= 0x0004; ++ break; ++ case SNDRV_PCM_FMTBIT_S24_LE: ++ voice |= 0x0008; ++ break; ++ case SNDRV_PCM_FMTBIT_S32_LE: ++ voice |= 0x000c; ++ break; ++ } ++ ++ wm8753_write(codec, WM8753_PCM, voice); ++} ++ ++/* configure PCM DAI */ ++static int wm8753_pcm_dai_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 voice, ioctl, srate, srate2, fs, bfs, clock; ++ unsigned int rate; ++ ++ bfs = SND_SOC_FSBD_REAL(rtd->codec_dai->dai_runtime.bfs); ++ fs = rtd->codec_dai->dai_runtime.fs; ++ rate = snd_soc_get_rate(rtd->codec_dai->dai_runtime.pcmrate); ++ voice = wm8753_read_reg_cache(codec, WM8753_PCM) & 0x001f; ++ ++ /* set master/slave audio interface */ ++ ioctl = wm8753_read_reg_cache(codec, WM8753_IOCTL) & 0x01fd; ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK) { ++ case SND_SOC_DAIFMT_CBM_CFM: ++ ioctl |= 0x0002; ++ case SND_SOC_DAIFMT_CBM_CFS: ++ voice |= 0x0040; ++ break; ++ } ++ ++ /* do we need to enable the PLL */ ++ if (rtd->codec_dai->pll_in) { ++ if (wm8753_config_pll(codec, rtd->codec_dai, 2) != ++ rtd->codec_dai->pll_in) { ++ err("could not set pll to %d --> %d", ++ rtd->codec_dai->pll_in, rtd->codec_dai->pll_out); ++ return -ENODEV; ++ } ++ } ++ ++ /* set up PCM divider */ ++ clock = wm8753_read_reg_cache(codec, WM8753_CLOCK) & 0x003f; ++ switch (rtd->codec_dai->clk_div) { ++ case 2: /* 1 */ ++ break; ++ case 6: /* 3 */ ++ clock |= (0x2 << 6); ++ break; ++ case 11: /* 5.5 */ ++ clock |= (0x3 << 6); ++ break; ++ case 4: /* 2 */ ++ clock |= (0x4 << 6); ++ break; ++ case 8: /* 4 */ ++ clock |= (0x5 << 6); ++ break; ++ case 12: /* 6 */ ++ clock |= (0x6 << 6); ++ break; ++ case 16: /* 8 */ ++ clock |= (0x7 << 6); ++ break; ++ default: ++ printk(KERN_ERR "wm8753: invalid PCM clk divider %d\n", ++ rtd->codec_dai->clk_div); ++ break; ++ } ++ wm8753_write(codec, WM8753_CLOCK, clock); ++ ++ /* set bclk divisor rate */ ++ srate2 = wm8753_read_reg_cache(codec, WM8753_SRATE2) & 0x003f; ++ switch (bfs) { ++ case 1: ++ break; ++ case 2: ++ srate2 |= (0x1 << 6); ++ break; ++ case 4: ++ srate2 |= (0x2 << 6); ++ break; ++ case 8: ++ srate2 |= (0x3 << 6); ++ break; ++ case 16: ++ srate2 |= (0x4 << 6); ++ break; ++ } ++ wm8753_write(codec, WM8753_SRATE2, srate2); ++ ++ srate = wm8753_read_reg_cache(codec, WM8753_SRATE1) & 0x017f; ++ if (rtd->codec_dai->dai_runtime.fs == 384) ++ srate |= 0x80; ++ wm8753_write(codec, WM8753_SRATE1, srate); ++ ++ /* clock inversion */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_INV_MASK) { ++ case SND_SOC_DAIFMT_IB_IF: ++ voice |= 0x0090; ++ break; ++ case SND_SOC_DAIFMT_IB_NF: ++ voice |= 0x0080; ++ break; ++ case SND_SOC_DAIFMT_NB_IF: ++ voice |= 0x0010; ++ break; ++ } ++ //printk("voice %x %x ioctl %x %x srate2 %x %x srate1 %x %x\n", ++ //WM8753_PCM, voice, WM8753_IOCTL, ioctl, WM8753_SRATE2, ++ //srate2, WM8753_SRATE1, srate); ++ ++ wm8753_write(codec, WM8753_IOCTL, ioctl); ++ wm8753_write(codec, WM8753_PCM, voice); ++ return 0; ++} ++ ++/* configure hifi DAC wordlength and format */ ++static void wm8753_hdac_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 hifi = wm8753_read_reg_cache(codec, WM8753_HIFI) & 0x01e0; ++ ++ /* interface format */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_I2S: ++ hifi |= 0x0002; ++ break; ++ case SND_SOC_DAIFMT_RIGHT_J: ++ break; ++ case SND_SOC_DAIFMT_LEFT_J: ++ hifi |= 0x0001; ++ break; ++ case SND_SOC_DAIFMT_DSP_A: ++ hifi |= 0x0003; ++ break; ++ case SND_SOC_DAIFMT_DSP_B: ++ hifi |= 0x0013; ++ break; ++ } ++ ++ /* bit size */ ++ switch (rtd->codec_dai->dai_runtime.pcmfmt) { ++ case SNDRV_PCM_FMTBIT_S16_LE: ++ break; ++ case SNDRV_PCM_FMTBIT_S20_3LE: ++ hifi |= 0x0004; ++ break; ++ case SNDRV_PCM_FMTBIT_S24_LE: ++ hifi |= 0x0008; ++ break; ++ case SNDRV_PCM_FMTBIT_S32_LE: ++ hifi |= 0x000c; ++ break; ++ } ++ ++ wm8753_write(codec, WM8753_HIFI, hifi); ++} ++ ++/* configure i2s (hifi) DAI clocking */ ++static int wm8753_i2s_dai_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 srate, bfs, hifi, ioctl; ++ unsigned int rate; ++ int i = 0; ++ ++ bfs = SND_SOC_FSBD_REAL(rtd->codec_dai->dai_runtime.bfs); ++ rate = snd_soc_get_rate(rtd->codec_dai->dai_runtime.pcmrate); ++ hifi = wm8753_read_reg_cache(codec, WM8753_HIFI) & 0x001f; ++ ++ /* is coefficient valid ? */ ++ if ((i = get_coeff(rtd->codec_dai->mclk, rate)) < 0) ++ return i; ++ ++ srate = wm8753_read_reg_cache(codec, WM8753_SRATE1) & 0x01c0; ++ wm8753_write(codec, WM8753_SRATE1, srate | (coeff_div[i].sr << 1) | ++ coeff_div[i].usb); ++ ++ /* do we need to enable the PLL */ ++ if (rtd->codec_dai->pll_in) { ++ if (wm8753_config_pll(codec, rtd->codec_dai, 1) != ++ rtd->codec_dai->pll_in) { ++ err("could not set pll to %d --> %d", ++ rtd->codec_dai->pll_in, rtd->codec_dai->pll_out); ++ return -ENODEV; ++ } ++ } ++ ++ /* set bclk divisor rate */ ++ srate = wm8753_read_reg_cache(codec, WM8753_SRATE2) & 0x01c7; ++ switch (bfs) { ++ case 1: ++ break; ++ case 2: ++ srate |= (0x1 << 3); ++ break; ++ case 4: ++ srate |= (0x2 << 3); ++ break; ++ case 8: ++ srate |= (0x3 << 3); ++ break; ++ case 16: ++ srate |= (0x4 << 3); ++ break; ++ } ++ wm8753_write(codec, WM8753_SRATE2, srate); ++ ++ /* set master/slave audio interface */ ++ ioctl = wm8753_read_reg_cache(codec, WM8753_IOCTL) & 0x00fe; ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK) { ++ case SND_SOC_DAIFMT_CBM_CFM: ++ ioctl |= 0x0001; ++ case SND_SOC_DAIFMT_CBM_CFS: ++ hifi |= 0x0040; ++ break; ++ } ++ ++ /* clock inversion */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_INV_MASK) { ++ case SND_SOC_DAIFMT_IB_IF: ++ hifi |= 0x0090; ++ break; ++ case SND_SOC_DAIFMT_IB_NF: ++ hifi |= 0x0080; ++ break; ++ case SND_SOC_DAIFMT_NB_IF: ++ hifi |= 0x0010; ++ break; ++ } ++ wm8753_write(codec, WM8753_IOCTL, ioctl); ++ wm8753_write(codec, WM8753_HIFI, hifi); ++ return 0; ++} ++ ++static int wm8753_mode1v_prepare (struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 clock; ++ ++ /* set clk source as pcmclk */ ++ clock = wm8753_read_reg_cache(codec, WM8753_CLOCK) & 0xfffb; ++ wm8753_write(codec, WM8753_CLOCK, clock); ++ ++ wm8753_adc_vdac_prepare(substream); ++ return wm8753_pcm_dai_prepare(substream); ++} ++ ++static int wm8753_mode1h_prepare (struct snd_pcm_substream *substream) ++{ ++ wm8753_hdac_prepare(substream); ++ return wm8753_i2s_dai_prepare(substream); ++} ++ ++static int wm8753_mode2_prepare (struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 clock; ++ ++ /* set clk source as pcmclk */ ++ clock = wm8753_read_reg_cache(codec, WM8753_CLOCK) & 0xfffb; ++ wm8753_write(codec, WM8753_CLOCK, clock); ++ ++ wm8753_adc_vdac_prepare(substream); ++ return wm8753_i2s_dai_prepare(substream); ++} ++ ++static int wm8753_mode3_prepare (struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 clock; ++ ++ /* set clk source as mclk */ ++ clock = wm8753_read_reg_cache(codec, WM8753_CLOCK) & 0xfffb; ++ wm8753_write(codec, WM8753_CLOCK, clock | 0x4); ++ ++ wm8753_hdac_prepare(substream); ++ wm8753_adc_vdac_prepare(substream); ++ return wm8753_i2s_dai_prepare(substream); ++} ++ ++static int wm8753_mode4_prepare (struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 clock; ++ ++ /* set clk source as mclk */ ++ clock = wm8753_read_reg_cache(codec, WM8753_CLOCK) & 0xfffb; ++ wm8753_write(codec, WM8753_CLOCK, clock | 0x4); ++ ++ wm8753_hdac_prepare(substream); ++ wm8753_adc_vdac_prepare(substream); ++ return wm8753_i2s_dai_prepare(substream); ++} ++ ++static int wm8753_mute(struct snd_soc_codec *codec, ++ struct snd_soc_codec_dai *dai, int mute) ++{ ++ u16 mute_reg = wm8753_read_reg_cache(codec, WM8753_DAC) & 0xfff7; ++ ++ /* the digital mute covers the HiFi and Voice DAC's on the WM8753. ++ * make sure we check if they are not both active when we mute */ ++ if (mute && dai->id == 1) { ++ if (!wm8753_dai[WM8753_DAI_VOICE].playback.active || ++ !wm8753_dai[WM8753_DAI_HIFI].playback.active) ++ wm8753_write(codec, WM8753_DAC, mute_reg | 0x8); ++ } else { ++ if (mute) ++ wm8753_write(codec, WM8753_DAC, mute_reg | 0x8); ++ else ++ wm8753_write(codec, WM8753_DAC, mute_reg); ++ } ++ ++ return 0; ++} ++ ++static int wm8753_dapm_event(struct snd_soc_codec *codec, int event) ++{ ++ u16 pwr_reg = wm8753_read_reg_cache(codec, WM8753_PWR1) & 0xfe3e; ++ ++ switch (event) { ++ case SNDRV_CTL_POWER_D0: /* full On */ ++ /* set vmid to 50k and unmute dac */ ++ wm8753_write(codec, WM8753_PWR1, pwr_reg | 0x00c0); ++ break; ++ case SNDRV_CTL_POWER_D1: /* partial On */ ++ case SNDRV_CTL_POWER_D2: /* partial On */ ++ /* set vmid to 5k for quick power up */ ++ wm8753_write(codec, WM8753_PWR1, pwr_reg | 0x01c1); ++ break; ++ case SNDRV_CTL_POWER_D3hot: /* Off, with power */ ++ /* mute dac and set vmid to 500k, enable VREF */ ++ wm8753_write(codec, WM8753_PWR1, pwr_reg | 0x0141); ++ break; ++ case SNDRV_CTL_POWER_D3cold: /* Off, without power */ ++ wm8753_write(codec, WM8753_PWR1, 0x0001); ++ break; ++ } ++ codec->dapm_state = event; ++ return 0; ++} ++ ++/* ++ * The WM8753 supports upto 4 different and mutually exclusive DAI ++ * configurations. This gives 2 PCM's available for use, hifi and voice. ++ * NOTE: The Voice PCM cannot play or caputure audio to the CPU as it's DAI ++ * is connected between the wm8753 and a BT codec or GSM modem. ++ * ++ * 1. Voice over PCM DAI - HIFI DAC over HIFI DAI ++ * 2. Voice over HIFI DAI - HIFI disabled ++ * 3. Voice disabled - HIFI over HIFI ++ * 4. Voice disabled - HIFI over HIFI, uses voice DAI LRC for capture ++ */ ++static const struct snd_soc_codec_dai wm8753_all_dai[] = { ++/* DAI HiFi mode 1 */ ++{ .name = "WM8753 HiFi", ++ .id = 1, ++ .playback = { ++ .stream_name = "HiFi Playback", ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .capture = { /* dummy for fast DAI switching */ ++ .stream_name = "HiFi Capture", ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .config_sysclk = wm8753_config_i2s_sysclk, ++ .digital_mute = wm8753_mute, ++ .ops = { ++ .prepare = wm8753_mode1h_prepare,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(wm8753_hifi_modes), ++ .mode = wm8753_hifi_modes,}, ++}, ++/* DAI Voice mode 1 */ ++{ .name = "WM8753 Voice", ++ .id = 1, ++ .playback = { ++ .stream_name = "Voice Playback", ++ .channels_min = 1, ++ .channels_max = 1,}, ++ .capture = { ++ .stream_name = "Voice Capture", ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .config_sysclk = wm8753_config_pcm_sysclk, ++ .digital_mute = wm8753_mute, ++ .ops = { ++ .prepare = wm8753_mode1v_prepare,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(wm8753_voice_modes), ++ .mode = wm8753_voice_modes,}, ++}, ++/* DAI HiFi mode 2 - dummy */ ++{ .name = "WM8753 HiFi", ++ .id = 2, ++}, ++/* DAI Voice mode 2 */ ++{ .name = "WM8753 Voice", ++ .id = 2, ++ .playback = { ++ .stream_name = "Voice Playback", ++ .channels_min = 1, ++ .channels_max = 1,}, ++ .capture = { ++ .stream_name = "Voice Capture", ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .config_sysclk = wm8753_config_i2s_sysclk, ++ .digital_mute = wm8753_mute, ++ .ops = { ++ .prepare = wm8753_mode2_prepare,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(wm8753_voice_modes), ++ .mode = wm8753_voice_modes,}, ++}, ++/* DAI HiFi mode 3 */ ++{ .name = "WM8753 HiFi", ++ .id = 3, ++ .playback = { ++ .stream_name = "HiFi Playback", ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .capture = { ++ .stream_name = "HiFi Capture", ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .config_sysclk = wm8753_config_i2s_sysclk, ++ .digital_mute = wm8753_mute, ++ .ops = { ++ .prepare = wm8753_mode3_prepare,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(wm8753_hifi_modes), ++ .mode = wm8753_hifi_modes,}, ++}, ++/* DAI Voice mode 3 - dummy */ ++{ .name = "WM8753 Voice", ++ .id = 3, ++}, ++/* DAI HiFi mode 4 */ ++{ .name = "WM8753 HiFi", ++ .id = 4, ++ .playback = { ++ .stream_name = "HiFi Playback", ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .capture = { ++ .stream_name = "HiFi Capture", ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .config_sysclk = wm8753_config_i2s_sysclk, ++ .digital_mute = wm8753_mute, ++ .ops = { ++ .prepare = wm8753_mode4_prepare,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(wm8753_mixed_modes), ++ .mode = wm8753_mixed_modes,}, ++}, ++/* DAI Voice mode 4 - dummy */ ++{ .name = "WM8753 Voice", ++ .id = 4, ++}, ++}; ++ ++struct snd_soc_codec_dai wm8753_dai[2]; ++EXPORT_SYMBOL_GPL(wm8753_dai); ++ ++static void wm8753_set_dai_mode(struct snd_soc_codec *codec, unsigned int mode) ++{ ++ if (mode < 4) { ++ wm8753_dai[0] = wm8753_all_dai[mode << 1]; ++ wm8753_dai[1] = wm8753_all_dai[(mode << 1) + 1]; ++ } ++} ++ ++static void wm8753_work(void *data) ++{ ++ struct snd_soc_codec *codec = (struct snd_soc_codec *)data; ++ wm8753_dapm_event(codec, codec->dapm_state); ++} ++ ++static int wm8753_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ wm8753_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ return 0; ++} ++ ++static int wm8753_resume(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ int i; ++ u8 data[2]; ++ u16 *cache = codec->reg_cache; ++ ++ /* Sync reg_cache with the hardware */ ++ for (i = 0; i < ARRAY_SIZE(wm8753_reg); i++) { ++ if (i + 1 == WM8753_RESET) ++ continue; ++ data[0] = ((i + 1) << 1) | ((cache[i] >> 8) & 0x0001); ++ data[1] = cache[i] & 0x00ff; ++ codec->hw_write(codec->control_data, data, 2); ++ } ++ ++ wm8753_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ ++ /* charge wm8753 caps */ ++ if (codec->suspend_dapm_state == SNDRV_CTL_POWER_D0) { ++ wm8753_dapm_event(codec, SNDRV_CTL_POWER_D2); ++ codec->dapm_state = SNDRV_CTL_POWER_D0; ++ queue_delayed_work(wm8753_workq, &wm8753_dapm_work, ++ msecs_to_jiffies(caps_charge)); ++ } ++ ++ return 0; ++} ++ ++/* ++ * initialise the WM8753 driver ++ * register the mixer and dsp interfaces with the kernel ++ */ ++static int wm8753_init(struct snd_soc_device *socdev) ++{ ++ struct snd_soc_codec *codec = socdev->codec; ++ int reg, ret = 0; ++ ++ codec->name = "WM8753"; ++ codec->owner = THIS_MODULE; ++ codec->read = wm8753_read_reg_cache; ++ codec->write = wm8753_write; ++ codec->dapm_event = wm8753_dapm_event; ++ codec->dai = wm8753_dai; ++ codec->num_dai = 2; ++ codec->reg_cache_size = ARRAY_SIZE(wm8753_reg); ++ ++ codec->reg_cache = ++ kzalloc(sizeof(u16) * ARRAY_SIZE(wm8753_reg), GFP_KERNEL); ++ if (codec->reg_cache == NULL) ++ return -ENOMEM; ++ memcpy(codec->reg_cache, wm8753_reg, ++ sizeof(u16) * ARRAY_SIZE(wm8753_reg)); ++ codec->reg_cache_size = sizeof(u16) * ARRAY_SIZE(wm8753_reg); ++ wm8753_set_dai_mode(codec, 0); ++ ++ wm8753_reset(codec); ++ ++ /* register pcms */ ++ ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); ++ if (ret < 0) { ++ kfree(codec->reg_cache); ++ return ret; ++ } ++ ++ /* charge output caps */ ++ wm8753_dapm_event(codec, SNDRV_CTL_POWER_D2); ++ codec->dapm_state = SNDRV_CTL_POWER_D3hot; ++ queue_delayed_work(wm8753_workq, ++ &wm8753_dapm_work, msecs_to_jiffies(caps_charge)); ++ ++ /* set the update bits */ ++ reg = wm8753_read_reg_cache(codec, WM8753_LDAC); ++ wm8753_write(codec, WM8753_LDAC, reg | 0x0100); ++ reg = wm8753_read_reg_cache(codec, WM8753_RDAC); ++ wm8753_write(codec, WM8753_RDAC, reg | 0x0100); ++ reg = wm8753_read_reg_cache(codec, WM8753_LOUT1V); ++ wm8753_write(codec, WM8753_LOUT1V, reg | 0x0100); ++ reg = wm8753_read_reg_cache(codec, WM8753_ROUT1V); ++ wm8753_write(codec, WM8753_ROUT1V, reg | 0x0100); ++ reg = wm8753_read_reg_cache(codec, WM8753_LOUT2V); ++ wm8753_write(codec, WM8753_LOUT2V, reg | 0x0100); ++ reg = wm8753_read_reg_cache(codec, WM8753_ROUT2V); ++ wm8753_write(codec, WM8753_ROUT2V, reg | 0x0100); ++ reg = wm8753_read_reg_cache(codec, WM8753_LINVOL); ++ wm8753_write(codec, WM8753_LINVOL, reg | 0x0100); ++ reg = wm8753_read_reg_cache(codec, WM8753_RINVOL); ++ wm8753_write(codec, WM8753_RINVOL, reg | 0x0100); ++ ++ wm8753_add_controls(codec); ++ wm8753_add_widgets(codec); ++ ret = snd_soc_register_card(socdev); ++ if (ret < 0) { ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++ } ++ ++ return ret; ++} ++ ++/* If the i2c layer weren't so broken, we could pass this kind of data ++ around */ ++static struct snd_soc_device *wm8753_socdev; ++ ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ ++/* ++ * WM8753 2 wire address is determined by GPIO5 ++ * state during powerup. ++ * low = 0x1a ++ * high = 0x1b ++ */ ++#define I2C_DRIVERID_WM8753 0xfefe /* liam - need a proper id */ ++ ++static unsigned short normal_i2c[] = { 0, I2C_CLIENT_END }; ++ ++/* Magic definition of all other variables and things */ ++I2C_CLIENT_INSMOD; ++ ++static struct i2c_driver wm8753_i2c_driver; ++static struct i2c_client client_template; ++ ++static int wm8753_codec_probe(struct i2c_adapter *adap, int addr, int kind) ++{ ++ struct snd_soc_device *socdev = wm8753_socdev; ++ struct wm8753_setup_data *setup = socdev->codec_data; ++ struct snd_soc_codec *codec = socdev->codec; ++ struct i2c_client *i2c; ++ int ret; ++ ++ if (addr != setup->i2c_address) ++ return -ENODEV; ++ ++ client_template.adapter = adap; ++ client_template.addr = addr; ++ ++ i2c = kzalloc(sizeof(struct i2c_client), GFP_KERNEL); ++ if (i2c == NULL){ ++ kfree(codec); ++ return -ENOMEM; ++ } ++ memcpy(i2c, &client_template, sizeof(struct i2c_client)); ++ i2c_set_clientdata(i2c, codec); ++ codec->control_data = i2c; ++ ++ ret = i2c_attach_client(i2c); ++ if (ret < 0) { ++ err("failed to attach codec at addr %x\n", addr); ++ goto err; ++ } ++ ++ ret = wm8753_init(socdev); ++ if (ret < 0) { ++ err("failed to initialise WM8753\n"); ++ goto err; ++ } ++ ++ return ret; ++ ++err: ++ kfree(codec); ++ kfree(i2c); ++ return ret; ++} ++ ++static int wm8753_i2c_detach(struct i2c_client *client) ++{ ++ struct snd_soc_codec *codec = i2c_get_clientdata(client); ++ i2c_detach_client(client); ++ kfree(codec->reg_cache); ++ kfree(client); ++ return 0; ++} ++ ++static int wm8753_i2c_attach(struct i2c_adapter *adap) ++{ ++ return i2c_probe(adap, &addr_data, wm8753_codec_probe); ++} ++ ++/* corgi i2c codec control layer */ ++static struct i2c_driver wm8753_i2c_driver = { ++ .driver = { ++ .name = "WM8753 I2C Codec", ++ .owner = THIS_MODULE, ++ }, ++ .id = I2C_DRIVERID_WM8753, ++ .attach_adapter = wm8753_i2c_attach, ++ .detach_client = wm8753_i2c_detach, ++ .command = NULL, ++}; ++ ++static struct i2c_client client_template = { ++ .name = "WM8753", ++ .driver = &wm8753_i2c_driver, ++}; ++#endif ++ ++static int wm8753_probe(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct wm8753_setup_data *setup; ++ struct snd_soc_codec *codec; ++ int ret = 0; ++ ++ info("WM8753 Audio Codec %s", WM8753_VERSION); ++ ++ setup = socdev->codec_data; ++ codec = kzalloc(sizeof(struct snd_soc_codec), GFP_KERNEL); ++ if (codec == NULL) ++ return -ENOMEM; ++ ++ socdev->codec = codec; ++ mutex_init(&codec->mutex); ++ INIT_LIST_HEAD(&codec->dapm_widgets); ++ INIT_LIST_HEAD(&codec->dapm_paths); ++ wm8753_socdev = socdev; ++ INIT_WORK(&wm8753_dapm_work, wm8753_work, codec); ++ wm8753_workq = create_workqueue("wm8753"); ++ if (wm8753_workq == NULL) { ++ kfree(codec); ++ return -ENOMEM; ++ } ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ if (setup->i2c_address) { ++ normal_i2c[0] = setup->i2c_address; ++ codec->hw_write = (hw_write_t)i2c_master_send; ++ ret = i2c_add_driver(&wm8753_i2c_driver); ++ if (ret != 0) ++ printk(KERN_ERR "can't add i2c driver"); ++ } ++#else ++ /* Add other interfaces here */ ++#endif ++ return ret; ++} ++ ++/* power down chip */ ++static int wm8753_remove(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ if (codec->control_data) ++ wm8753_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ if (wm8753_workq) ++ destroy_workqueue(wm8753_workq); ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ i2c_del_driver(&wm8753_i2c_driver); ++#endif ++ kfree(codec); ++ ++ return 0; ++} ++ ++struct snd_soc_codec_device soc_codec_dev_wm8753 = { ++ .probe = wm8753_probe, ++ .remove = wm8753_remove, ++ .suspend = wm8753_suspend, ++ .resume = wm8753_resume, ++}; ++ ++EXPORT_SYMBOL_GPL(soc_codec_dev_wm8753); ++ ++MODULE_DESCRIPTION("ASoC WM8753 driver"); ++MODULE_AUTHOR("Liam Girdwood"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8753.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8753.h +@@ -0,0 +1,91 @@ ++/* ++ * wm8753.h -- audio driver for WM8753 ++ * ++ * Copyright 2003 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * 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. ++ * ++ */ ++ ++#ifndef _WM8753_H ++#define _WM8753_H ++ ++/* WM8753 register space */ ++ ++#define WM8753_DAC 0x01 ++#define WM8753_ADC 0x02 ++#define WM8753_PCM 0x03 ++#define WM8753_HIFI 0x04 ++#define WM8753_IOCTL 0x05 ++#define WM8753_SRATE1 0x06 ++#define WM8753_SRATE2 0x07 ++#define WM8753_LDAC 0x08 ++#define WM8753_RDAC 0x09 ++#define WM8753_BASS 0x0a ++#define WM8753_TREBLE 0x0b ++#define WM8753_ALC1 0x0c ++#define WM8753_ALC2 0x0d ++#define WM8753_ALC3 0x0e ++#define WM8753_NGATE 0x0f ++#define WM8753_LADC 0x10 ++#define WM8753_RADC 0x11 ++#define WM8753_ADCTL1 0x12 ++#define WM8753_3D 0x13 ++#define WM8753_PWR1 0x14 ++#define WM8753_PWR2 0x15 ++#define WM8753_PWR3 0x16 ++#define WM8753_PWR4 0x17 ++#define WM8753_ID 0x18 ++#define WM8753_INTPOL 0x19 ++#define WM8753_INTEN 0x1a ++#define WM8753_GPIO1 0x1b ++#define WM8753_GPIO2 0x1c ++#define WM8753_RESET 0x1f ++#define WM8753_RECMIX1 0x20 ++#define WM8753_RECMIX2 0x21 ++#define WM8753_LOUTM1 0x22 ++#define WM8753_LOUTM2 0x23 ++#define WM8753_ROUTM1 0x24 ++#define WM8753_ROUTM2 0x25 ++#define WM8753_MOUTM1 0x26 ++#define WM8753_MOUTM2 0x27 ++#define WM8753_LOUT1V 0x28 ++#define WM8753_ROUT1V 0x29 ++#define WM8753_LOUT2V 0x2a ++#define WM8753_ROUT2V 0x2b ++#define WM8753_MOUTV 0x2c ++#define WM8753_OUTCTL 0x2d ++#define WM8753_ADCIN 0x2e ++#define WM8753_INCTL1 0x2f ++#define WM8753_INCTL2 0x30 ++#define WM8753_LINVOL 0x31 ++#define WM8753_RINVOL 0x32 ++#define WM8753_MICBIAS 0x33 ++#define WM8753_CLOCK 0x34 ++#define WM8753_PLL1CTL1 0x35 ++#define WM8753_PLL1CTL2 0x36 ++#define WM8753_PLL1CTL3 0x37 ++#define WM8753_PLL1CTL4 0x38 ++#define WM8753_PLL2CTL1 0x39 ++#define WM8753_PLL2CTL2 0x3a ++#define WM8753_PLL2CTL3 0x3b ++#define WM8753_PLL2CTL4 0x3c ++#define WM8753_BIASCTL 0x3d ++#define WM8753_ADCTL2 0x3f ++ ++struct wm8753_setup_data { ++ unsigned short i2c_address; ++}; ++ ++#define WM8753_DAI_HIFI 0 ++#define WM8753_DAI_VOICE 1 ++ ++extern struct snd_soc_codec_dai wm8753_dai[2]; ++extern struct snd_soc_codec_device soc_codec_dev_wm8753; ++ ++#endif +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8772.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8772.c +@@ -0,0 +1,806 @@ ++/* ++ * wm8772.c -- WM8772 ALSA Soc Audio driver ++ * ++ * Copyright 2005 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * 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. ++ * ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/version.h> ++#include <linux/kernel.h> ++#include <linux/init.h> ++#include <linux/delay.h> ++#include <linux/pm.h> ++#include <linux/i2c.h> ++ ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/pcm_params.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++#include <sound/initval.h> ++ ++#include "wm8772.h" ++ ++#define AUDIO_NAME "WM8772" ++#define WM8772_VERSION "0.3" ++ ++/* ++ * wm8772 register cache ++ * We can't read the WM8772 register space when we ++ * are using 2 wire for device control, so we cache them instead. ++ */ ++static const u16 wm8772_reg[] = { ++ 0x00ff, 0x00ff, 0x0120, 0x0000, /* 0 */ ++ 0x00ff, 0x00ff, 0x00ff, 0x00ff, /* 4 */ ++ 0x00ff, 0x0000, 0x0080, 0x0040, /* 8 */ ++ 0x0000 ++}; ++ ++#define WM8772_DAIFMT \ ++ (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_LEFT_J | SND_SOC_DAIFMT_RIGHT_J | \ ++ SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_IB_NF) ++ ++#define WM8772_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define WM8772_PRATES \ ++ (SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000 |\ ++ SNDRV_PCM_RATE_96000 | SNDRV_PCM_RATE_192000) ++ ++#define WM8772_CRATES \ ++ (SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000 |\ ++ SNDRV_PCM_RATE_96000) ++ ++static struct snd_soc_dai_mode wm8772_modes[] = { ++ /* common codec frame and clock master modes */ ++ /* 32k */ ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 768, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 512, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 384, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 256, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 192, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 128, ++ .bfs = 64, ++ }, ++ ++ /* 44.1k */ ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_44100, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 768, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_44100, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 512, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_44100, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 384, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_44100, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 256, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_44100, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 192, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_44100, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 128, ++ .bfs = 64, ++ }, ++ ++ /* 48k */ ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 768, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 512, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 384, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 256, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 192, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 128, ++ .bfs = 64, ++ }, ++ ++ /* 96k */ ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_96000, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 384, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_96000, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 256, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_96000, ++ .pcmdir = WM8772_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 192, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_96000, ++ .pcmdir = WM8772_DIR, ++ .pcmrate = SND_SOC_DAI_BFS_RATE, ++ .fs = 128, ++ .bfs = 64, ++ }, ++ ++ /* 192k */ ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_192000, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 192, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8772_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_192000, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 128, ++ .bfs = 64, ++ }, ++ ++ /* slave mode */ ++ { ++ .fmt = WM8772_DAIFMT, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = WM8772_PRATES, ++ .pcmdir = SND_SOC_DAIDIR_PLAYBACK, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = SND_SOC_FSB_ALL, ++ }, ++ { ++ .fmt = WM8772_DAIFMT, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = WM8772_CRATES, ++ .pcmdir = SND_SOC_DAIDIR_CAPTURE, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = SND_SOC_FSB_ALL, ++ }, ++}; ++ ++/* ++ * read wm8772 register cache ++ */ ++static inline unsigned int wm8772_read_reg_cache(struct snd_soc_codec * codec, ++ unsigned int reg) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg > WM8772_CACHE_REGNUM) ++ return -1; ++ return cache[reg]; ++} ++ ++/* ++ * write wm8772 register cache ++ */ ++static inline void wm8772_write_reg_cache(struct snd_soc_codec * codec, ++ unsigned int reg, unsigned int value) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg > WM8772_CACHE_REGNUM) ++ return; ++ cache[reg] = value; ++} ++ ++static int wm8772_write(struct snd_soc_codec * codec, unsigned int reg, ++ unsigned int value) ++{ ++ u8 data[2]; ++ ++ /* data is ++ * D15..D9 WM8772 register offset ++ * D8...D0 register data ++ */ ++ data[0] = (reg << 1) | ((value >> 8) & 0x0001); ++ data[1] = value & 0x00ff; ++ ++ wm8772_write_reg_cache (codec, reg, value); ++ if (codec->hw_write(codec->control_data, data, 2) == 2) ++ return 0; ++ else ++ return -1; ++} ++ ++#define wm8772_reset(c) wm8772_write(c, WM8772_RESET, 0) ++ ++/* ++ * WM8772 Controls ++ */ ++static const char *wm8772_zero_flag[] = {"All Ch", "Ch 1", "Ch 2", "Ch3"}; ++ ++static const struct soc_enum wm8772_enum[] = { ++SOC_ENUM_SINGLE(WM8772_DACCTRL, 0, 4, wm8772_zero_flag), ++}; ++ ++static const struct snd_kcontrol_new wm8772_snd_controls[] = { ++ ++SOC_SINGLE("Left1 Playback Volume", WM8772_LDAC1VOL, 0, 255, 0), ++SOC_SINGLE("Left2 Playback Volume", WM8772_LDAC2VOL, 0, 255, 0), ++SOC_SINGLE("Left3 Playback Volume", WM8772_LDAC3VOL, 0, 255, 0), ++SOC_SINGLE("Right1 Playback Volume", WM8772_RDAC1VOL, 0, 255, 0), ++SOC_SINGLE("Right1 Playback Volume", WM8772_RDAC2VOL, 0, 255, 0), ++SOC_SINGLE("Right1 Playback Volume", WM8772_RDAC3VOL, 0, 255, 0), ++SOC_SINGLE("Master Playback Volume", WM8772_MDACVOL, 0, 255, 0), ++ ++SOC_SINGLE("Playback Switch", WM8772_DACCH, 0, 1, 0), ++SOC_SINGLE("Capture Switch", WM8772_ADCCTRL, 2, 1, 0), ++ ++SOC_SINGLE("Demp1 Playback Switch", WM8772_DACCTRL, 6, 1, 0), ++SOC_SINGLE("Demp2 Playback Switch", WM8772_DACCTRL, 7, 1, 0), ++SOC_SINGLE("Demp3 Playback Switch", WM8772_DACCTRL, 8, 1, 0), ++ ++SOC_SINGLE("Phase Invert 1 Switch", WM8772_IFACE, 6, 1, 0), ++SOC_SINGLE("Phase Invert 2 Switch", WM8772_IFACE, 7, 1, 0), ++SOC_SINGLE("Phase Invert 3 Switch", WM8772_IFACE, 8, 1, 0), ++ ++SOC_SINGLE("Playback ZC Switch", WM8772_DACCTRL, 0, 1, 0), ++ ++SOC_SINGLE("Capture High Pass Switch", WM8772_ADCCTRL, 3, 1, 0), ++}; ++ ++/* add non dapm controls */ ++static int wm8772_add_controls(struct snd_soc_codec *codec) ++{ ++ int err, i; ++ ++ for (i = 0; i < ARRAY_SIZE(wm8772_snd_controls); i++) { ++ err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&wm8772_snd_controls[i],codec, NULL)); ++ if (err < 0) ++ return err; ++ } ++ return 0; ++} ++ ++/* valid wm8772 mclk frequencies */ ++static const int freq_table[5][6] = { ++ {4096000, 6144000, 8192000, 12288000, 16384000, 24576000}, ++ {5644800, 8467000, 11289600, 16934000, 22579200, 33868800}, ++ {6144000, 9216000, 12288000, 18432000, 24576000, 36864000}, ++ {12288000, 18432000, 24576000, 36864000, 0, 0}, ++ {24576000, 36864000, 0, 0, 0}, ++}; ++ ++static unsigned int check_freq(int rate, unsigned int freq) ++{ ++ int i; ++ ++ for(i = 0; i < 6; i++) { ++ if(freq == freq_table[i][rate]) ++ return freq; ++ } ++ return 0; ++} ++ ++static unsigned int wm8772_config_sysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ switch (info->rate){ ++ case 32000: ++ dai->mclk = check_freq(0, clk); ++ break; ++ case 44100: ++ dai->mclk = check_freq(1, clk); ++ break; ++ case 48000: ++ dai->mclk = check_freq(2, clk); ++ break; ++ case 96000: ++ dai->mclk = check_freq(3, clk); ++ break; ++ case 192000: ++ dai->mclk = check_freq(4, clk); ++ break; ++ default: ++ dai->mclk = 0; ++ } ++ return dai->mclk; ++} ++ ++static int wm8772_pcm_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 diface = wm8772_read_reg_cache(codec, WM8772_IFACE) & 0xffc0; ++ u16 diface_ctrl = wm8772_read_reg_cache(codec, WM8772_DACRATE) & 0xfe1f; ++ u16 aiface = 0; ++ u16 aiface_ctrl = wm8772_read_reg_cache(codec, WM8772_ADCCTRL) & 0xfcff; ++ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { ++ ++ /* set master/slave audio interface */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK) { ++ case SND_SOC_DAIFMT_CBM_CFM: ++ diface_ctrl |= 0x0010; ++ break; ++ case SND_SOC_DAIFMT_CBS_CFS: ++ break; ++ } ++ ++ /* interface format */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_I2S: ++ diface |= 0x0002; ++ break; ++ case SND_SOC_DAIFMT_RIGHT_J: ++ break; ++ case SND_SOC_DAIFMT_LEFT_J: ++ diface |= 0x0001; ++ break; ++ case SND_SOC_DAIFMT_DSP_A: ++ diface |= 0x0003; ++ break; ++ case SND_SOC_DAIFMT_DSP_B: ++ diface |= 0x0007; ++ break; ++ } ++ ++ /* bit size */ ++ switch (rtd->codec_dai->dai_runtime.pcmfmt) { ++ case SNDRV_PCM_FMTBIT_S16_LE: ++ break; ++ case SNDRV_PCM_FORMAT_S20_3LE: ++ diface |= 0x0010; ++ break; ++ case SNDRV_PCM_FORMAT_S24_3LE: ++ diface |= 0x0020; ++ break; ++ case SNDRV_PCM_FORMAT_S32_LE: ++ diface |= 0x0030; ++ break; ++ } ++ ++ /* clock inversion */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_INV_MASK) { ++ case SND_SOC_DAIFMT_NB_NF: ++ break; ++ case SND_SOC_DAIFMT_IB_NF: ++ diface |= 0x0008; ++ break; ++ } ++ ++ /* set rate */ ++ switch (rtd->codec_dai->dai_runtime.fs) { ++ case 768: ++ diface_ctrl |= (0x5 << 6); ++ break; ++ case 512: ++ diface_ctrl |= (0x4 << 6); ++ break; ++ case 384: ++ diface_ctrl |= (0x3 << 6); ++ break; ++ case 256: ++ diface_ctrl |= (0x2 << 6); ++ break; ++ case 192: ++ diface_ctrl |= (0x1 << 6); ++ break; ++ } ++ ++ wm8772_write(codec, WM8772_DACRATE, diface_ctrl); ++ wm8772_write(codec, WM8772_IFACE, diface); ++ ++ } else { ++ ++ /* set master/slave audio interface */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK) { ++ case SND_SOC_DAIFMT_CBM_CFM: ++ aiface |= 0x0010; ++ break; ++ case SND_SOC_DAIFMT_CBS_CFS: ++ break; ++ } ++ ++ /* interface format */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_I2S: ++ aiface |= 0x0002; ++ break; ++ case SND_SOC_DAIFMT_RIGHT_J: ++ break; ++ case SND_SOC_DAIFMT_LEFT_J: ++ aiface |= 0x0001; ++ break; ++ case SND_SOC_DAIFMT_DSP_A: ++ aiface |= 0x0003; ++ break; ++ case SND_SOC_DAIFMT_DSP_B: ++ aiface |= 0x0003; ++ aiface_ctrl |= 0x0010; ++ break; ++ } ++ ++ /* bit size */ ++ switch (rtd->codec_dai->dai_runtime.pcmfmt) { ++ case SNDRV_PCM_FMTBIT_S16_LE: ++ break; ++ case SNDRV_PCM_FMTBIT_S20_3LE: ++ aiface |= 0x0004; ++ break; ++ case SNDRV_PCM_FMTBIT_S24_LE: ++ aiface |= 0x0008; ++ break; ++ case SNDRV_PCM_FMTBIT_S32_LE: ++ aiface |= 0x000c; ++ break; ++ } ++ ++ /* clock inversion */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_INV_MASK) { ++ case SND_SOC_DAIFMT_NB_NF: ++ break; ++ case SND_SOC_DAIFMT_IB_NF: ++ aiface_ctrl |= 0x0020; ++ break; ++ } ++ ++ /* set rate */ ++ switch (rtd->codec_dai->dai_runtime.fs) { ++ case 768: ++ aiface |= (0x5 << 5); ++ break; ++ case 512: ++ aiface |= (0x4 << 5); ++ break; ++ case 384: ++ aiface |= (0x3 << 5); ++ break; ++ case 256: ++ aiface |= (0x2 << 5); ++ break; ++ } ++ ++ wm8772_write(codec, WM8772_ADCCTRL, aiface_ctrl); ++ wm8772_write(codec, WM8772_ADCRATE, aiface); ++ } ++ ++ return 0; ++} ++ ++static int wm8772_dapm_event(struct snd_soc_codec *codec, int event) ++{ ++ u16 master = wm8772_read_reg_cache(codec, WM8772_DACRATE) & 0xffe0; ++ ++ switch (event) { ++ case SNDRV_CTL_POWER_D0: /* full On */ ++ /* vref/mid, clk and osc on, dac unmute, active */ ++ wm8772_write(codec, WM8772_DACRATE, master); ++ break; ++ case SNDRV_CTL_POWER_D1: /* partial On */ ++ case SNDRV_CTL_POWER_D2: /* partial On */ ++ break; ++ case SNDRV_CTL_POWER_D3hot: /* Off, with power */ ++ /* everything off except vref/vmid, dac mute, inactive */ ++ wm8772_write(codec, WM8772_DACRATE, master | 0x0f); ++ break; ++ case SNDRV_CTL_POWER_D3cold: /* Off, without power */ ++ /* everything off, dac mute, inactive */ ++ wm8772_write(codec, WM8772_DACRATE, master | 0x1f); ++ break; ++ } ++ codec->dapm_state = event; ++ return 0; ++} ++ ++struct snd_soc_codec_dai wm8772_dai = { ++ .name = "WM8772", ++ .playback = { ++ .stream_name = "Playback", ++ .channels_min = 1, ++ .channels_max = 6, ++ }, ++ .capture = { ++ .stream_name = "Capture", ++ .channels_min = 1, ++ .channels_max = 2, ++ }, ++ .config_sysclk = wm8772_config_sysclk, ++ .ops = { ++ .prepare = wm8772_pcm_prepare, ++ }, ++ .caps = { ++ .num_modes = ARRAY_SIZE(wm8772_modes), ++ .mode = wm8772_modes, ++ }, ++}; ++EXPORT_SYMBOL_GPL(wm8772_dai); ++ ++static int wm8772_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ wm8772_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ return 0; ++} ++ ++static int wm8772_resume(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ int i; ++ u8 data[2]; ++ u16 *cache = codec->reg_cache; ++ ++ /* Sync reg_cache with the hardware */ ++ for (i = 0; i < ARRAY_SIZE(wm8772_reg); i++) { ++ data[0] = (i << 1) | ((cache[i] >> 8) & 0x0001); ++ data[1] = cache[i] & 0x00ff; ++ codec->hw_write(codec->control_data, data, 2); ++ } ++ wm8772_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ wm8772_dapm_event(codec, codec->suspend_dapm_state); ++ return 0; ++} ++ ++/* ++ * initialise the WM8772 driver ++ * register the mixer and dsp interfaces with the kernel ++ */ ++static int wm8772_init(struct snd_soc_device *socdev) ++{ ++ struct snd_soc_codec *codec = socdev->codec; ++ int reg, ret = 0; ++ ++ codec->name = "WM8772"; ++ codec->owner = THIS_MODULE; ++ codec->read = wm8772_read_reg_cache; ++ codec->write = wm8772_write; ++ codec->dapm_event = wm8772_dapm_event; ++ codec->dai = &wm8772_dai; ++ codec->num_dai = 1; ++ codec->reg_cache_size = ARRAY_SIZE(wm8772_reg); ++ codec->reg_cache = ++ kzalloc(sizeof(u16) * ARRAY_SIZE(wm8772_reg), GFP_KERNEL); ++ if (codec->reg_cache == NULL) ++ return -ENOMEM; ++ memcpy(codec->reg_cache, wm8772_reg, ++ sizeof(u16) * ARRAY_SIZE(wm8772_reg)); ++ codec->reg_cache_size = sizeof(u16) * ARRAY_SIZE(wm8772_reg); ++ ++ wm8772_reset(codec); ++ ++ /* register pcms */ ++ ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); ++ if(ret < 0) { ++ kfree(codec->reg_cache); ++ return ret; ++ } ++ ++ /* power on device */ ++ wm8772_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ ++ /* set the update bits */ ++ reg = wm8772_read_reg_cache(codec, WM8772_MDACVOL); ++ wm8772_write(codec, WM8772_MDACVOL, reg | 0x0100); ++ reg = wm8772_read_reg_cache(codec, WM8772_LDAC1VOL); ++ wm8772_write(codec, WM8772_LDAC1VOL, reg | 0x0100); ++ reg = wm8772_read_reg_cache(codec, WM8772_LDAC2VOL); ++ wm8772_write(codec, WM8772_LDAC2VOL, reg | 0x0100); ++ reg = wm8772_read_reg_cache(codec, WM8772_LDAC3VOL); ++ wm8772_write(codec, WM8772_LDAC3VOL, reg | 0x0100); ++ reg = wm8772_read_reg_cache(codec, WM8772_RDAC1VOL); ++ wm8772_write(codec, WM8772_RDAC1VOL, reg | 0x0100); ++ reg = wm8772_read_reg_cache(codec, WM8772_RDAC2VOL); ++ wm8772_write(codec, WM8772_RDAC2VOL, reg | 0x0100); ++ reg = wm8772_read_reg_cache(codec, WM8772_RDAC3VOL); ++ wm8772_write(codec, WM8772_RDAC3VOL, reg | 0x0100); ++ ++ wm8772_add_controls(codec); ++ ret = snd_soc_register_card(socdev); ++ if (ret < 0) { ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++ } ++ ++ return ret; ++} ++ ++static struct snd_soc_device *wm8772_socdev; ++ ++static int wm8772_probe(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct wm8772_setup_data *setup; ++ struct snd_soc_codec *codec; ++ int ret = 0; ++ ++ printk(KERN_INFO "WM8772 Audio Codec %s", WM8772_VERSION); ++ ++ setup = socdev->codec_data; ++ codec = kzalloc(sizeof(struct snd_soc_codec), GFP_KERNEL); ++ if (codec == NULL) ++ return -ENOMEM; ++ ++ socdev->codec = codec; ++ mutex_init(&codec->mutex); ++ INIT_LIST_HEAD(&codec->dapm_widgets); ++ INIT_LIST_HEAD(&codec->dapm_paths); ++ ++ wm8772_socdev = socdev; ++ ++ /* Add other interfaces here */ ++#warning do SPI device probe here and then call wm8772_init() ++ ++ return ret; ++} ++ ++/* power down chip */ ++static int wm8772_remove(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ if (codec->control_data) ++ wm8772_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ ++ snd_soc_free_pcms(socdev); ++ kfree(codec); ++ ++ return 0; ++} ++ ++struct snd_soc_codec_device soc_codec_dev_wm8772 = { ++ .probe = wm8772_probe, ++ .remove = wm8772_remove, ++ .suspend = wm8772_suspend, ++ .resume = wm8772_resume, ++}; ++ ++EXPORT_SYMBOL_GPL(soc_codec_dev_wm8772); ++ ++MODULE_DESCRIPTION("ASoC WM8772 driver"); ++MODULE_AUTHOR("Liam Girdwood"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8772.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8772.h +@@ -0,0 +1,40 @@ ++/* ++ * wm8772.h -- audio driver for WM8772 ++ * ++ * Copyright 2005 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * 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. ++ * ++ */ ++ ++#ifndef _WM8772_H ++#define _WM8772_H ++ ++/* WM8772 register space */ ++ ++#define WM8772_LDAC1VOL 0x00 ++#define WM8772_RDAC1VOL 0x01 ++#define WM8772_DACCH 0x02 ++#define WM8772_IFACE 0x03 ++#define WM8772_LDAC2VOL 0x04 ++#define WM8772_RDAC2VOL 0x05 ++#define WM8772_LDAC3VOL 0x06 ++#define WM8772_RDAC3VOL 0x07 ++#define WM8772_MDACVOL 0x08 ++#define WM8772_DACCTRL 0x09 ++#define WM8772_DACRATE 0x0a ++#define WM8772_ADCRATE 0x0b ++#define WM8772_ADCCTRL 0x0c ++#define WM8772_RESET 0x1f ++ ++#define WM8772_CACHE_REGNUM 10 ++ ++extern struct snd_soc_codec_dai wm8772_dai; ++extern struct snd_soc_codec_device soc_codec_dev_wm8772; ++ ++#endif +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8971.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8971.c +@@ -0,0 +1,1214 @@ ++/* ++ * wm8971.c -- WM8971 ALSA SoC Audio driver ++ * ++ * Copyright 2005 Lab126, Inc. ++ * ++ * Author: Kenneth Kiraly <kiraly@lab126.com> ++ * ++ * Based on wm8753.c by Liam Girdwood ++ * ++ * 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. ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/init.h> ++#include <linux/delay.h> ++#include <linux/pm.h> ++#include <linux/i2c.h> ++#include <linux/platform_device.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/pcm_params.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++#include <sound/initval.h> ++ ++#include "wm8971.h" ++ ++#define AUDIO_NAME "wm8971" ++#define WM8971_VERSION "0.8" ++ ++#undef WM8971_DEBUG ++ ++#ifdef WM8971_DEBUG ++#define dbg(format, arg...) \ ++ printk(KERN_DEBUG AUDIO_NAME ": " format "\n" , ## arg) ++#else ++#define dbg(format, arg...) do {} while (0) ++#endif ++#define err(format, arg...) \ ++ printk(KERN_ERR AUDIO_NAME ": " format "\n" , ## arg) ++#define info(format, arg...) \ ++ printk(KERN_INFO AUDIO_NAME ": " format "\n" , ## arg) ++#define warn(format, arg...) \ ++ printk(KERN_WARNING AUDIO_NAME ": " format "\n" , ## arg) ++ ++#define WM8971_REG_COUNT 43 ++ ++static struct workqueue_struct *wm8971_workq = NULL; ++static struct work_struct wm8971_dapm_work; ++ ++/* ++ * wm8971 register cache ++ * We can't read the WM8971 register space when we ++ * are using 2 wire for device control, so we cache them instead. ++ */ ++static const u16 wm8971_reg[] = { ++ 0x0097, 0x0097, 0x0079, 0x0079, /* 0 */ ++ 0x0000, 0x0008, 0x0000, 0x000a, /* 4 */ ++ 0x0000, 0x0000, 0x00ff, 0x00ff, /* 8 */ ++ 0x000f, 0x000f, 0x0000, 0x0000, /* 12 */ ++ 0x0000, 0x007b, 0x0000, 0x0032, /* 16 */ ++ 0x0000, 0x00c3, 0x00c3, 0x00c0, /* 20 */ ++ 0x0000, 0x0000, 0x0000, 0x0000, /* 24 */ ++ 0x0000, 0x0000, 0x0000, 0x0000, /* 28 */ ++ 0x0000, 0x0000, 0x0050, 0x0050, /* 32 */ ++ 0x0050, 0x0050, 0x0050, 0x0050, /* 36 */ ++ 0x0079, 0x0079, 0x0079, /* 40 */ ++}; ++ ++#define WM8971_HIFI_DAIFMT \ ++ (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_LEFT_J | SND_SOC_DAIFMT_RIGHT_J | \ ++ SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_NB_IF | SND_SOC_DAIFMT_IB_NF | \ ++ SND_SOC_DAIFMT_IB_IF) ++ ++#define WM8971_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define WM8971_HIFI_FSB \ ++ (SND_SOC_FSBD(1) | SND_SOC_FSBD(2) | SND_SOC_FSBD(4) | \ ++ SND_SOC_FSBD(8) | SND_SOC_FSBD(16)) ++ ++#define WM8971_HIFI_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000) ++ ++#define WM8971_HIFI_BITS \ ++ (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE | \ ++ SNDRV_PCM_FMTBIT_S24_LE | SNDRV_PCM_FMTBIT_S32_LE) ++ ++static struct snd_soc_dai_mode wm8971_modes[] = { ++ /* common codec frame and clock master modes */ ++ /* 8k */ ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1536, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1408, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 2304, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 2112, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1500, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ ++ /* 11.025k */ ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_11025, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1024, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_11025, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1536, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_11025, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1088, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ ++ /* 16k */ ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_16000, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 768, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_16000, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1152, ++ .bfs = WM8971_HIFI_FSB ++ }, ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_16000, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 750, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ ++ /* 22.05k */ ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_22050, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 512, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_22050, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 768, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_22050, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 544, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ ++ /* 32k */ ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 384, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 576, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 375, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ ++ /* 44.1k & 48k */ ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 384, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_44100, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 272, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 250, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ ++ /* 88.2k & 96k */ ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 128, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 192, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_88200, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 136, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_96000, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 125, ++ .bfs = WM8971_HIFI_FSB, ++ }, ++ ++ /* codec frame and clock slave modes */ ++ { ++ .fmt = WM8971_HIFI_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = WM8971_HIFI_BITS, ++ .pcmrate = WM8971_HIFI_RATES, ++ .pcmdir = WM8971_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = SND_SOC_FSB_ALL, ++ }, ++}; ++ ++static inline unsigned int wm8971_read_reg_cache(struct snd_soc_codec *codec, ++ unsigned int reg) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg < WM8971_REG_COUNT) ++ return cache[reg]; ++ ++ return -1; ++} ++ ++static inline void wm8971_write_reg_cache(struct snd_soc_codec *codec, ++ unsigned int reg, unsigned int value) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg < WM8971_REG_COUNT) ++ cache[reg] = value; ++} ++ ++static int wm8971_write(struct snd_soc_codec *codec, unsigned int reg, ++ unsigned int value) ++{ ++ u8 data[2]; ++ ++ /* data is ++ * D15..D9 WM8753 register offset ++ * D8...D0 register data ++ */ ++ data[0] = (reg << 1) | ((value >> 8) & 0x0001); ++ data[1] = value & 0x00ff; ++ ++ wm8971_write_reg_cache (codec, reg, value); ++ if (codec->hw_write(codec->control_data, data, 2) == 2) ++ return 0; ++ else ++ return -EIO; ++} ++ ++#define wm8971_reset(c) wm8971_write(c, WM8971_RESET, 0) ++ ++/* WM8971 Controls */ ++static const char *wm8971_bass[] = { "Linear Control", "Adaptive Boost" }; ++static const char *wm8971_bass_filter[] = { "130Hz @ 48kHz", ++ "200Hz @ 48kHz" }; ++static const char *wm8971_treble[] = { "8kHz", "4kHz" }; ++static const char *wm8971_alc_func[] = { "Off", "Right", "Left", "Stereo" }; ++static const char *wm8971_ng_type[] = { "Constant PGA Gain", ++ "Mute ADC Output" }; ++static const char *wm8971_deemp[] = { "None", "32kHz", "44.1kHz", "48kHz" }; ++static const char *wm8971_mono_mux[] = {"Stereo", "Mono (Left)", ++ "Mono (Right)", "Digital Mono"}; ++static const char *wm8971_dac_phase[] = { "Non Inverted", "Inverted" }; ++static const char *wm8971_lline_mux[] = {"Line", "NC", "NC", "PGA", ++ "Differential"}; ++static const char *wm8971_rline_mux[] = {"Line", "Mic", "NC", "PGA", ++ "Differential"}; ++static const char *wm8971_lpga_sel[] = {"Line", "NC", "NC", "Differential"}; ++static const char *wm8971_rpga_sel[] = {"Line", "Mic", "NC", "Differential"}; ++static const char *wm8971_adcpol[] = {"Normal", "L Invert", "R Invert", ++ "L + R Invert"}; ++ ++static const struct soc_enum wm8971_enum[] = { ++ SOC_ENUM_SINGLE(WM8971_BASS, 7, 2, wm8971_bass), /* 0 */ ++ SOC_ENUM_SINGLE(WM8971_BASS, 6, 2, wm8971_bass_filter), ++ SOC_ENUM_SINGLE(WM8971_TREBLE, 6, 2, wm8971_treble), ++ SOC_ENUM_SINGLE(WM8971_ALC1, 7, 4, wm8971_alc_func), ++ SOC_ENUM_SINGLE(WM8971_NGATE, 1, 2, wm8971_ng_type), /* 4 */ ++ SOC_ENUM_SINGLE(WM8971_ADCDAC, 1, 4, wm8971_deemp), ++ SOC_ENUM_SINGLE(WM8971_ADCTL1, 4, 4, wm8971_mono_mux), ++ SOC_ENUM_SINGLE(WM8971_ADCTL1, 1, 2, wm8971_dac_phase), ++ SOC_ENUM_SINGLE(WM8971_LOUTM1, 0, 5, wm8971_lline_mux), /* 8 */ ++ SOC_ENUM_SINGLE(WM8971_ROUTM1, 0, 5, wm8971_rline_mux), ++ SOC_ENUM_SINGLE(WM8971_LADCIN, 6, 4, wm8971_lpga_sel), ++ SOC_ENUM_SINGLE(WM8971_RADCIN, 6, 4, wm8971_rpga_sel), ++ SOC_ENUM_SINGLE(WM8971_ADCDAC, 5, 4, wm8971_adcpol), /* 12 */ ++ SOC_ENUM_SINGLE(WM8971_ADCIN, 6, 4, wm8971_mono_mux), ++}; ++ ++static const struct snd_kcontrol_new wm8971_snd_controls[] = { ++ SOC_DOUBLE_R("Capture Volume", WM8971_LINVOL, WM8971_RINVOL, 0, 63, 0), ++ SOC_DOUBLE_R("Capture ZC Switch", WM8971_LINVOL, WM8971_RINVOL, 6, 1, 0), ++ SOC_DOUBLE_R("Capture Switch", WM8971_LINVOL, WM8971_RINVOL, 7, 1, 1), ++ ++ SOC_DOUBLE_R("Headphone Playback ZC Switch", WM8971_LOUT1V, ++ WM8971_ROUT1V, 7, 1, 0), ++ SOC_DOUBLE_R("Speaker Playback ZC Switch", WM8971_LOUT2V, ++ WM8971_ROUT2V, 7, 1, 0), ++ SOC_SINGLE("Mono Playback ZC Switch", WM8971_MOUTV, 7, 1, 0), ++ ++ SOC_DOUBLE_R("PCM Volume", WM8971_LDAC, WM8971_RDAC, 0, 255, 0), ++ ++ SOC_DOUBLE_R("Bypass Left Playback Volume", WM8971_LOUTM1, ++ WM8971_LOUTM2, 4, 7, 1), ++ SOC_DOUBLE_R("Bypass Right Playback Volume", WM8971_ROUTM1, ++ WM8971_ROUTM2, 4, 7, 1), ++ SOC_DOUBLE_R("Bypass Mono Playback Volume", WM8971_MOUTM1, ++ WM8971_MOUTM2, 4, 7, 1), ++ ++ SOC_DOUBLE_R("Headphone Playback Volume", WM8971_LOUT1V, ++ WM8971_ROUT1V, 0, 127, 0), ++ SOC_DOUBLE_R("Speaker Playback Volume", WM8971_LOUT2V, ++ WM8971_ROUT2V, 0, 127, 0), ++ ++ SOC_ENUM("Bass Boost", wm8971_enum[0]), ++ SOC_ENUM("Bass Filter", wm8971_enum[1]), ++ SOC_SINGLE("Bass Volume", WM8971_BASS, 0, 7, 1), ++ ++ SOC_SINGLE("Treble Volume", WM8971_TREBLE, 0, 7, 0), ++ SOC_ENUM("Treble Cut-off", wm8971_enum[2]), ++ ++ SOC_SINGLE("Capture Filter Switch", WM8971_ADCDAC, 0, 1, 1), ++ ++ SOC_SINGLE("ALC Target Volume", WM8971_ALC1, 0, 7, 0), ++ SOC_SINGLE("ALC Max Volume", WM8971_ALC1, 4, 7, 0), ++ ++ SOC_SINGLE("ALC Capture Target Volume", WM8971_ALC1, 0, 7, 0), ++ SOC_SINGLE("ALC Capture Max Volume", WM8971_ALC1, 4, 7, 0), ++ SOC_ENUM("ALC Capture Function", wm8971_enum[3]), ++ SOC_SINGLE("ALC Capture ZC Switch", WM8971_ALC2, 7, 1, 0), ++ SOC_SINGLE("ALC Capture Hold Time", WM8971_ALC2, 0, 15, 0), ++ SOC_SINGLE("ALC Capture Decay Time", WM8971_ALC3, 4, 15, 0), ++ SOC_SINGLE("ALC Capture Attack Time", WM8971_ALC3, 0, 15, 0), ++ SOC_SINGLE("ALC Capture NG Threshold", WM8971_NGATE, 3, 31, 0), ++ SOC_ENUM("ALC Capture NG Type", wm8971_enum[4]), ++ SOC_SINGLE("ALC Capture NG Switch", WM8971_NGATE, 0, 1, 0), ++ ++ SOC_SINGLE("Capture 6dB Attenuate", WM8971_ADCDAC, 8, 1, 0), ++ SOC_SINGLE("Playback 6dB Attenuate", WM8971_ADCDAC, 7, 1, 0), ++ ++ SOC_ENUM("Playback De-emphasis", wm8971_enum[5]), ++ SOC_ENUM("Playback Function", wm8971_enum[6]), ++ SOC_ENUM("Playback Phase", wm8971_enum[7]), ++ ++ SOC_DOUBLE_R("Mic Boost", WM8971_LADCIN, WM8971_RADCIN, 4, 3, 0), ++}; ++ ++/* add non-DAPM controls */ ++static int wm8971_add_controls(struct snd_soc_codec *codec) ++{ ++ int err, i; ++ ++ for (i = 0; i < ARRAY_SIZE(wm8971_snd_controls); i++) { ++ err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&wm8971_snd_controls[i],codec, NULL)); ++ if (err < 0) ++ return err; ++ } ++ ++ return 0; ++} ++ ++/* ++ * DAPM Controls ++ */ ++ ++/* Left Mixer */ ++static const struct snd_kcontrol_new wm8971_left_mixer_controls[] = { ++SOC_DAPM_SINGLE("Playback Switch", WM8971_LOUTM1, 8, 1, 0), ++SOC_DAPM_SINGLE("Left Bypass Switch", WM8971_LOUTM1, 7, 1, 0), ++SOC_DAPM_SINGLE("Right Playback Switch", WM8971_LOUTM2, 8, 1, 0), ++SOC_DAPM_SINGLE("Right Bypass Switch", WM8971_LOUTM2, 7, 1, 0), ++}; ++ ++/* Right Mixer */ ++static const struct snd_kcontrol_new wm8971_right_mixer_controls[] = { ++SOC_DAPM_SINGLE("Left Playback Switch", WM8971_ROUTM1, 8, 1, 0), ++SOC_DAPM_SINGLE("Left Bypass Switch", WM8971_ROUTM1, 7, 1, 0), ++SOC_DAPM_SINGLE("Playback Switch", WM8971_ROUTM2, 8, 1, 0), ++SOC_DAPM_SINGLE("Right Bypass Switch", WM8971_ROUTM2, 7, 1, 0), ++}; ++ ++/* Mono Mixer */ ++static const struct snd_kcontrol_new wm8971_mono_mixer_controls[] = { ++SOC_DAPM_SINGLE("Left Playback Switch", WM8971_MOUTM1, 8, 1, 0), ++SOC_DAPM_SINGLE("Left Bypass Switch", WM8971_MOUTM1, 7, 1, 0), ++SOC_DAPM_SINGLE("Right Playback Switch", WM8971_MOUTM2, 8, 1, 0), ++SOC_DAPM_SINGLE("Right Bypass Switch", WM8971_MOUTM2, 7, 1, 0), ++}; ++ ++/* Left Line Mux */ ++static const struct snd_kcontrol_new wm8971_left_line_controls = ++SOC_DAPM_ENUM("Route", wm8971_enum[8]); ++ ++/* Right Line Mux */ ++static const struct snd_kcontrol_new wm8971_right_line_controls = ++SOC_DAPM_ENUM("Route", wm8971_enum[9]); ++ ++/* Left PGA Mux */ ++static const struct snd_kcontrol_new wm8971_left_pga_controls = ++SOC_DAPM_ENUM("Route", wm8971_enum[10]); ++ ++/* Right PGA Mux */ ++static const struct snd_kcontrol_new wm8971_right_pga_controls = ++SOC_DAPM_ENUM("Route", wm8971_enum[11]); ++ ++/* Mono ADC Mux */ ++static const struct snd_kcontrol_new wm8971_monomux_controls = ++SOC_DAPM_ENUM("Route", wm8971_enum[13]); ++ ++static const struct snd_soc_dapm_widget wm8971_dapm_widgets[] = { ++ SND_SOC_DAPM_MIXER("Left Mixer", SND_SOC_NOPM, 0, 0, ++ &wm8971_left_mixer_controls[0], ++ ARRAY_SIZE(wm8971_left_mixer_controls)), ++ SND_SOC_DAPM_MIXER("Right Mixer", SND_SOC_NOPM, 0, 0, ++ &wm8971_right_mixer_controls[0], ++ ARRAY_SIZE(wm8971_right_mixer_controls)), ++ SND_SOC_DAPM_MIXER("Mono Mixer", WM8971_PWR2, 2, 0, ++ &wm8971_mono_mixer_controls[0], ++ ARRAY_SIZE(wm8971_mono_mixer_controls)), ++ ++ SND_SOC_DAPM_PGA("Right Out 2", WM8971_PWR2, 3, 0, NULL, 0), ++ SND_SOC_DAPM_PGA("Left Out 2", WM8971_PWR2, 4, 0, NULL, 0), ++ SND_SOC_DAPM_PGA("Right Out 1", WM8971_PWR2, 5, 0, NULL, 0), ++ SND_SOC_DAPM_PGA("Left Out 1", WM8971_PWR2, 6, 0, NULL, 0), ++ SND_SOC_DAPM_DAC("Right DAC", "Right Playback", WM8971_PWR2, 7, 0), ++ SND_SOC_DAPM_DAC("Left DAC", "Left Playback", WM8971_PWR2, 8, 0), ++ SND_SOC_DAPM_PGA("Mono Out 1", WM8971_PWR2, 2, 0, NULL, 0), ++ ++ SND_SOC_DAPM_MICBIAS("Mic Bias", WM8971_PWR1, 1, 0), ++ SND_SOC_DAPM_ADC("Right ADC", "Right Capture", WM8971_PWR1, 2, 0), ++ SND_SOC_DAPM_ADC("Left ADC", "Left Capture", WM8971_PWR1, 3, 0), ++ ++ SND_SOC_DAPM_MUX("Left PGA Mux", WM8971_PWR1, 5, 0, ++ &wm8971_left_pga_controls), ++ SND_SOC_DAPM_MUX("Right PGA Mux", WM8971_PWR1, 4, 0, ++ &wm8971_right_pga_controls), ++ SND_SOC_DAPM_MUX("Left Line Mux", SND_SOC_NOPM, 0, 0, ++ &wm8971_left_line_controls), ++ SND_SOC_DAPM_MUX("Right Line Mux", SND_SOC_NOPM, 0, 0, ++ &wm8971_right_line_controls), ++ ++ SND_SOC_DAPM_MUX("Left ADC Mux", SND_SOC_NOPM, 0, 0, ++ &wm8971_monomux_controls), ++ SND_SOC_DAPM_MUX("Right ADC Mux", SND_SOC_NOPM, 0, 0, ++ &wm8971_monomux_controls), ++ ++ SND_SOC_DAPM_OUTPUT("LOUT1"), ++ SND_SOC_DAPM_OUTPUT("ROUT1"), ++ SND_SOC_DAPM_OUTPUT("LOUT2"), ++ SND_SOC_DAPM_OUTPUT("ROUT2"), ++ SND_SOC_DAPM_OUTPUT("MONO"), ++ ++ SND_SOC_DAPM_INPUT("LINPUT1"), ++ SND_SOC_DAPM_INPUT("RINPUT1"), ++ SND_SOC_DAPM_INPUT("MIC"), ++}; ++ ++static const char *audio_map[][3] = { ++ /* left mixer */ ++ {"Left Mixer", "Playback Switch", "Left DAC"}, ++ {"Left Mixer", "Left Bypass Switch", "Left Line Mux"}, ++ {"Left Mixer", "Right Playback Switch", "Right DAC"}, ++ {"Left Mixer", "Right Bypass Switch", "Right Line Mux"}, ++ ++ /* right mixer */ ++ {"Right Mixer", "Left Playback Switch", "Left DAC"}, ++ {"Right Mixer", "Left Bypass Switch", "Left Line Mux"}, ++ {"Right Mixer", "Playback Switch", "Right DAC"}, ++ {"Right Mixer", "Right Bypass Switch", "Right Line Mux"}, ++ ++ /* left out 1 */ ++ {"Left Out 1", NULL, "Left Mixer"}, ++ {"LOUT1", NULL, "Left Out 1"}, ++ ++ /* left out 2 */ ++ {"Left Out 2", NULL, "Left Mixer"}, ++ {"LOUT2", NULL, "Left Out 2"}, ++ ++ /* right out 1 */ ++ {"Right Out 1", NULL, "Right Mixer"}, ++ {"ROUT1", NULL, "Right Out 1"}, ++ ++ /* right out 2 */ ++ {"Right Out 2", NULL, "Right Mixer"}, ++ {"ROUT2", NULL, "Right Out 2"}, ++ ++ /* mono mixer */ ++ {"Mono Mixer", "Left Playback Switch", "Left DAC"}, ++ {"Mono Mixer", "Left Bypass Switch", "Left Line Mux"}, ++ {"Mono Mixer", "Right Playback Switch", "Right DAC"}, ++ {"Mono Mixer", "Right Bypass Switch", "Right Line Mux"}, ++ ++ /* mono out */ ++ {"Mono Out", NULL, "Mono Mixer"}, ++ {"MONO1", NULL, "Mono Out"}, ++ ++ /* Left Line Mux */ ++ {"Left Line Mux", "Line", "LINPUT1"}, ++ {"Left Line Mux", "PGA", "Left PGA Mux"}, ++ {"Left Line Mux", "Differential", "Differential Mux"}, ++ ++ /* Right Line Mux */ ++ {"Right Line Mux", "Line", "RINPUT1"}, ++ {"Right Line Mux", "Mic", "MIC"}, ++ {"Right Line Mux", "PGA", "Right PGA Mux"}, ++ {"Right Line Mux", "Differential", "Differential Mux"}, ++ ++ /* Left PGA Mux */ ++ {"Left PGA Mux", "Line", "LINPUT1"}, ++ {"Left PGA Mux", "Differential", "Differential Mux"}, ++ ++ /* Right PGA Mux */ ++ {"Right PGA Mux", "Line", "RINPUT1"}, ++ {"Right PGA Mux", "Differential", "Differential Mux"}, ++ ++ /* Differential Mux */ ++ {"Differential Mux", "Line", "LINPUT1"}, ++ {"Differential Mux", "Line", "RINPUT1"}, ++ ++ /* Left ADC Mux */ ++ {"Left ADC Mux", "Stereo", "Left PGA Mux"}, ++ {"Left ADC Mux", "Mono (Left)", "Left PGA Mux"}, ++ {"Left ADC Mux", "Digital Mono", "Left PGA Mux"}, ++ ++ /* Right ADC Mux */ ++ {"Right ADC Mux", "Stereo", "Right PGA Mux"}, ++ {"Right ADC Mux", "Mono (Right)", "Right PGA Mux"}, ++ {"Right ADC Mux", "Digital Mono", "Right PGA Mux"}, ++ ++ /* ADC */ ++ {"Left ADC", NULL, "Left ADC Mux"}, ++ {"Right ADC", NULL, "Right ADC Mux"}, ++ ++ /* terminator */ ++ {NULL, NULL, NULL}, ++}; ++ ++static int wm8971_add_widgets(struct snd_soc_codec *codec) ++{ ++ int i; ++ ++ for(i = 0; i < ARRAY_SIZE(wm8971_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &wm8971_dapm_widgets[i]); ++ } ++ ++ /* set up audio path audio_mapnects */ ++ for(i = 0; audio_map[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, audio_map[i][0], ++ audio_map[i][1], audio_map[i][2]); ++ } ++ ++ snd_soc_dapm_new_widgets(codec); ++ return 0; ++} ++ ++struct _coeff_div { ++ u32 mclk; ++ u32 rate; ++ u16 fs; ++ u8 sr:5; ++ u8 usb:1; ++}; ++ ++/* codec hifi mclk clock divider coefficients */ ++static const struct _coeff_div coeff_div[] = { ++ /* 8k */ ++ {12288000, 8000, 1536, 0x6, 0x0}, ++ {11289600, 8000, 1408, 0x16, 0x0}, ++ {18432000, 8000, 2304, 0x7, 0x0}, ++ {16934400, 8000, 2112, 0x17, 0x0}, ++ {12000000, 8000, 1500, 0x6, 0x1}, ++ ++ /* 11.025k */ ++ {11289600, 11025, 1024, 0x18, 0x0}, ++ {16934400, 11025, 1536, 0x19, 0x0}, ++ {12000000, 11025, 1088, 0x19, 0x1}, ++ ++ /* 16k */ ++ {12288000, 16000, 768, 0xa, 0x0}, ++ {18432000, 16000, 1152, 0xb, 0x0}, ++ {12000000, 16000, 750, 0xa, 0x1}, ++ ++ /* 22.05k */ ++ {11289600, 22050, 512, 0x1a, 0x0}, ++ {16934400, 22050, 768, 0x1b, 0x0}, ++ {12000000, 22050, 544, 0x1b, 0x1}, ++ ++ /* 32k */ ++ {12288000, 32000, 384, 0xc, 0x0}, ++ {18432000, 32000, 576, 0xd, 0x0}, ++ {12000000, 32000, 375, 0xa, 0x1}, ++ ++ /* 44.1k */ ++ {11289600, 44100, 256, 0x10, 0x0}, ++ {16934400, 44100, 384, 0x11, 0x0}, ++ {12000000, 44100, 272, 0x11, 0x1}, ++ ++ /* 48k */ ++ {12288000, 48000, 256, 0x0, 0x0}, ++ {18432000, 48000, 384, 0x1, 0x0}, ++ {12000000, 48000, 250, 0x0, 0x1}, ++ ++ /* 88.2k */ ++ {11289600, 88200, 128, 0x1e, 0x0}, ++ {16934400, 88200, 192, 0x1f, 0x0}, ++ {12000000, 88200, 136, 0x1f, 0x1}, ++ ++ /* 96k */ ++ {12288000, 96000, 128, 0xe, 0x0}, ++ {18432000, 96000, 192, 0xf, 0x0}, ++ {12000000, 96000, 125, 0xe, 0x1}, ++}; ++ ++static int get_coeff(int mclk, int rate) ++{ ++ int i; ++ ++ for (i = 0; i < ARRAY_SIZE(coeff_div); i++) { ++ if (coeff_div[i].rate == rate && coeff_div[i].mclk == mclk) ++ return i; ++ } ++ return -EINVAL; ++} ++ ++/* WM8971 supports numerous input clocks per sample rate */ ++static unsigned int wm8971_config_sysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ dai->mclk = 0; ++ ++ /* check that the calculated FS and rate actually match a clock from ++ * the machine driver */ ++ if (info->fs * info->rate == clk) ++ dai->mclk = clk; ++ ++ return dai->mclk; ++} ++ ++static int wm8971_pcm_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 iface = 0, bfs, srate = 0; ++ int i = get_coeff(rtd->codec_dai->mclk, ++ snd_soc_get_rate(rtd->codec_dai->dai_runtime.pcmrate)); ++ ++ /* is coefficient valid ? */ ++ if (i < 0) ++ return i; ++ ++ bfs = SND_SOC_FSBD_REAL(rtd->codec_dai->dai_runtime.bfs); ++ ++ /* set master/slave audio interface */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK) { ++ case SND_SOC_DAIFMT_CBM_CFM: ++ iface |= 0x0040; ++ break; ++ case SND_SOC_DAIFMT_CBS_CFS: ++ break; ++ } ++ ++ /* interface format */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_I2S: ++ iface |= 0x0002; ++ break; ++ case SND_SOC_DAIFMT_RIGHT_J: ++ break; ++ case SND_SOC_DAIFMT_LEFT_J: ++ iface |= 0x0001; ++ break; ++ case SND_SOC_DAIFMT_DSP_A: ++ iface |= 0x0003; ++ break; ++ case SND_SOC_DAIFMT_DSP_B: ++ iface |= 0x0013; ++ break; ++ } ++ ++ /* bit size */ ++ switch (rtd->codec_dai->dai_runtime.pcmfmt) { ++ case SNDRV_PCM_FMTBIT_S16_LE: ++ break; ++ case SNDRV_PCM_FMTBIT_S20_3LE: ++ iface |= 0x0004; ++ break; ++ case SNDRV_PCM_FMTBIT_S24_LE: ++ iface |= 0x0008; ++ break; ++ case SNDRV_PCM_FMTBIT_S32_LE: ++ iface |= 0x000c; ++ break; ++ } ++ ++ /* clock inversion */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_INV_MASK) { ++ case SND_SOC_DAIFMT_NB_NF: ++ break; ++ case SND_SOC_DAIFMT_IB_IF: ++ iface |= 0x0090; ++ break; ++ case SND_SOC_DAIFMT_IB_NF: ++ iface |= 0x0080; ++ break; ++ case SND_SOC_DAIFMT_NB_IF: ++ iface |= 0x0010; ++ break; ++ } ++ ++ /* set bclk divisor rate */ ++ switch (bfs) { ++ case 1: ++ break; ++ case 4: ++ srate |= (0x1 << 7); ++ break; ++ case 8: ++ srate |= (0x2 << 7); ++ break; ++ case 16: ++ srate |= (0x3 << 7); ++ break; ++ } ++ ++ /* set iface & srate */ ++ wm8971_write(codec, WM8971_AUDIO, iface); ++ wm8971_write(codec, WM8971_SRATE, srate | ++ (coeff_div[i].sr << 1) | coeff_div[i].usb); ++ return 0; ++} ++ ++static int wm8971_mute(struct snd_soc_codec *codec, ++ struct snd_soc_codec_dai *dai, int mute) ++{ ++ u16 mute_reg = wm8971_read_reg_cache(codec, WM8971_ADCDAC) & 0xfff7; ++ if (mute) ++ wm8971_write(codec, WM8971_ADCDAC, mute_reg | 0x8); ++ else ++ wm8971_write(codec, WM8971_ADCDAC, mute_reg); ++ return 0; ++} ++ ++static int wm8971_dapm_event(struct snd_soc_codec *codec, int event) ++{ ++ u16 pwr_reg = wm8971_read_reg_cache(codec, WM8971_PWR1) & 0xfe3e; ++ ++ switch (event) { ++ case SNDRV_CTL_POWER_D0: /* full On */ ++ /* set vmid to 50k and unmute dac */ ++ wm8971_write(codec, WM8971_PWR1, pwr_reg | 0x00c1); ++ break; ++ case SNDRV_CTL_POWER_D1: /* partial On */ ++ case SNDRV_CTL_POWER_D2: /* partial On */ ++ /* set vmid to 5k for quick power up */ ++ wm8971_write(codec, WM8971_PWR1, pwr_reg | 0x01c0); ++ break; ++ case SNDRV_CTL_POWER_D3hot: /* Off, with power */ ++ /* mute dac and set vmid to 500k, enable VREF */ ++ wm8971_write(codec, WM8971_PWR1, pwr_reg | 0x0140); ++ break; ++ case SNDRV_CTL_POWER_D3cold: /* Off, without power */ ++ wm8971_write(codec, WM8971_PWR1, 0x0001); ++ break; ++ } ++ codec->dapm_state = event; ++ return 0; ++} ++ ++struct snd_soc_codec_dai wm8971_dai = { ++ .name = "WM8971", ++ .playback = { ++ .stream_name = "Playback", ++ .channels_min = 1, ++ .channels_max = 2, ++ }, ++ .capture = { ++ .stream_name = "Capture", ++ .channels_min = 1, ++ .channels_max = 2, ++ }, ++ .config_sysclk = wm8971_config_sysclk, ++ .digital_mute = wm8971_mute, ++ .ops = { ++ .prepare = wm8971_pcm_prepare, ++ }, ++ .caps = { ++ .num_modes = ARRAY_SIZE(wm8971_modes), ++ .mode = wm8971_modes, ++ }, ++}; ++EXPORT_SYMBOL_GPL(wm8971_dai); ++ ++static void wm8971_work(void *data) ++{ ++ struct snd_soc_codec *codec = (struct snd_soc_codec *)data; ++ wm8971_dapm_event(codec, codec->dapm_state); ++} ++ ++static int wm8971_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ wm8971_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ return 0; ++} ++ ++static int wm8971_resume(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ int i; ++ u8 data[2]; ++ u16 *cache = codec->reg_cache; ++ ++ /* Sync reg_cache with the hardware */ ++ for (i = 0; i < ARRAY_SIZE(wm8971_reg); i++) { ++ if (i + 1 == WM8971_RESET) ++ continue; ++ data[0] = (i << 1) | ((cache[i] >> 8) & 0x0001); ++ data[1] = cache[i] & 0x00ff; ++ codec->hw_write(codec->control_data, data, 2); ++ } ++ ++ wm8971_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ ++ /* charge wm8971 caps */ ++ if (codec->suspend_dapm_state == SNDRV_CTL_POWER_D0) { ++ wm8971_dapm_event(codec, SNDRV_CTL_POWER_D2); ++ codec->dapm_state = SNDRV_CTL_POWER_D0; ++ queue_delayed_work(wm8971_workq, &wm8971_dapm_work, ++ msecs_to_jiffies(1000)); ++ } ++ ++ return 0; ++} ++ ++static int wm8971_init(struct snd_soc_device *socdev) ++{ ++ struct snd_soc_codec *codec = socdev->codec; ++ int reg, ret = 0; ++ ++ codec->name = "WM8971"; ++ codec->owner = THIS_MODULE; ++ codec->read = wm8971_read_reg_cache; ++ codec->write = wm8971_write; ++ codec->dapm_event = wm8971_dapm_event; ++ codec->dai = &wm8971_dai; ++ codec->reg_cache_size = ARRAY_SIZE(wm8971_reg); ++ codec->num_dai = 1; ++ codec->reg_cache = ++ kzalloc(sizeof(u16) * ARRAY_SIZE(wm8971_reg), GFP_KERNEL); ++ if (codec->reg_cache == NULL) ++ return -ENOMEM; ++ memcpy(codec->reg_cache, wm8971_reg, ++ sizeof(u16) * ARRAY_SIZE(wm8971_reg)); ++ codec->reg_cache_size = sizeof(u16) * ARRAY_SIZE(wm8971_reg); ++ ++ wm8971_reset(codec); ++ ++ /* register pcms */ ++ ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); ++ if (ret < 0) { ++ kfree(codec->reg_cache); ++ return ret; ++ } ++ ++ /* charge output caps */ ++ wm8971_dapm_event(codec, SNDRV_CTL_POWER_D2); ++ codec->dapm_state = SNDRV_CTL_POWER_D3hot; ++ queue_delayed_work(wm8971_workq, &wm8971_dapm_work, ++ msecs_to_jiffies(1000)); ++ ++ /* set the update bits */ ++ reg = wm8971_read_reg_cache(codec, WM8971_LDAC); ++ wm8971_write(codec, WM8971_LDAC, reg | 0x0100); ++ reg = wm8971_read_reg_cache(codec, WM8971_RDAC); ++ wm8971_write(codec, WM8971_RDAC, reg | 0x0100); ++ ++ reg = wm8971_read_reg_cache(codec, WM8971_LOUT1V); ++ wm8971_write(codec, WM8971_LOUT1V, reg | 0x0100); ++ reg = wm8971_read_reg_cache(codec, WM8971_ROUT1V); ++ wm8971_write(codec, WM8971_ROUT1V, reg | 0x0100); ++ ++ reg = wm8971_read_reg_cache(codec, WM8971_LOUT2V); ++ wm8971_write(codec, WM8971_LOUT2V, reg | 0x0100); ++ reg = wm8971_read_reg_cache(codec, WM8971_ROUT2V); ++ wm8971_write(codec, WM8971_ROUT2V, reg | 0x0100); ++ ++ reg = wm8971_read_reg_cache(codec, WM8971_LINVOL); ++ wm8971_write(codec, WM8971_LINVOL, reg | 0x0100); ++ reg = wm8971_read_reg_cache(codec, WM8971_RINVOL); ++ wm8971_write(codec, WM8971_RINVOL, reg | 0x0100); ++ ++ wm8971_add_controls(codec); ++ wm8971_add_widgets(codec); ++ ret = snd_soc_register_card(socdev); ++ if (ret < 0) { ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++ } ++ ++ return ret; ++} ++ ++/* If the i2c layer weren't so broken, we could pass this kind of data ++ around */ ++static struct snd_soc_device *wm8971_socdev; ++ ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ ++/* ++ * WM8731 2 wire address is determined by GPIO5 ++ * state during powerup. ++ * low = 0x1a ++ * high = 0x1b ++ */ ++#define I2C_DRIVERID_WM8971 0xfefe /* liam - need a proper id */ ++ ++static unsigned short normal_i2c[] = { 0, I2C_CLIENT_END }; ++ ++/* Magic definition of all other variables and things */ ++I2C_CLIENT_INSMOD; ++ ++static struct i2c_driver wm8971_i2c_driver; ++static struct i2c_client client_template; ++ ++static int wm8971_codec_probe(struct i2c_adapter *adap, int addr, int kind) ++{ ++ struct snd_soc_device *socdev = wm8971_socdev; ++ struct wm8971_setup_data *setup = socdev->codec_data; ++ struct snd_soc_codec *codec = socdev->codec; ++ struct i2c_client *i2c; ++ int ret; ++ ++ if (addr != setup->i2c_address) ++ return -ENODEV; ++ ++ client_template.adapter = adap; ++ client_template.addr = addr; ++ ++ i2c = kzalloc(sizeof(struct i2c_client), GFP_KERNEL); ++ if (i2c == NULL) { ++ kfree(codec); ++ return -ENOMEM; ++ } ++ memcpy(i2c, &client_template, sizeof(struct i2c_client)); ++ ++ i2c_set_clientdata(i2c, codec); ++ ++ codec->control_data = i2c; ++ ++ ret = i2c_attach_client(i2c); ++ if (ret < 0) { ++ err("failed to attach codec at addr %x\n", addr); ++ goto err; ++ } ++ ++ ret = wm8971_init(socdev); ++ if (ret < 0) { ++ err("failed to initialise WM8971\n"); ++ goto err; ++ } ++ return ret; ++ ++err: ++ kfree(codec); ++ kfree(i2c); ++ return ret; ++} ++ ++static int wm8971_i2c_detach(struct i2c_client *client) ++{ ++ struct snd_soc_codec* codec = i2c_get_clientdata(client); ++ i2c_detach_client(client); ++ kfree(codec->reg_cache); ++ kfree(client); ++ return 0; ++} ++ ++static int wm8971_i2c_attach(struct i2c_adapter *adap) ++{ ++ return i2c_probe(adap, &addr_data, wm8971_codec_probe); ++} ++ ++/* corgi i2c codec control layer */ ++static struct i2c_driver wm8971_i2c_driver = { ++ .driver = { ++ .name = "WM8971 I2C Codec", ++ .owner = THIS_MODULE, ++ }, ++ .id = I2C_DRIVERID_WM8971, ++ .attach_adapter = wm8971_i2c_attach, ++ .detach_client = wm8971_i2c_detach, ++ .command = NULL, ++}; ++ ++static struct i2c_client client_template = { ++ .name = "WM8971", ++ .driver = &wm8971_i2c_driver, ++}; ++#endif ++ ++static int wm8971_probe(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct wm8971_setup_data *setup; ++ struct snd_soc_codec *codec; ++ int ret = 0; ++ ++ info("WM8971 Audio Codec %s", WM8971_VERSION); ++ ++ setup = socdev->codec_data; ++ codec = kzalloc(sizeof(struct snd_soc_codec), GFP_KERNEL); ++ if (codec == NULL) ++ return -ENOMEM; ++ ++ socdev->codec = codec; ++ mutex_init(&codec->mutex); ++ INIT_LIST_HEAD(&codec->dapm_widgets); ++ INIT_LIST_HEAD(&codec->dapm_paths); ++ wm8971_socdev = socdev; ++ ++ INIT_WORK(&wm8971_dapm_work, wm8971_work, codec); ++ wm8971_workq = create_workqueue("wm8971"); ++ if (wm8971_workq == NULL) { ++ kfree(codec); ++ return -ENOMEM; ++ } ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ if (setup->i2c_address) { ++ normal_i2c[0] = setup->i2c_address; ++ codec->hw_write = (hw_write_t)i2c_master_send; ++ ret = i2c_add_driver(&wm8971_i2c_driver); ++ if (ret != 0) ++ printk(KERN_ERR "can't add i2c driver"); ++ } ++#else ++ /* Add other interfaces here */ ++#endif ++ ++ return ret; ++} ++ ++/* power down chip */ ++static int wm8971_remove(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ if (codec->control_data) ++ wm8971_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ if (wm8971_workq) ++ destroy_workqueue(wm8971_workq); ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ i2c_del_driver(&wm8971_i2c_driver); ++#endif ++ kfree(codec); ++ ++ return 0; ++} ++ ++struct snd_soc_codec_device soc_codec_dev_wm8971 = { ++ .probe = wm8971_probe, ++ .remove = wm8971_remove, ++ .suspend = wm8971_suspend, ++ .resume = wm8971_resume, ++}; ++ ++EXPORT_SYMBOL_GPL(soc_codec_dev_wm8971); ++ ++MODULE_DESCRIPTION("ASoC WM8971 driver"); ++MODULE_AUTHOR("Lab126"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8971.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8971.h +@@ -0,0 +1,61 @@ ++/* ++ * wm8971.h -- audio driver for WM8971 ++ * ++ * Copyright 2005 Lab126, Inc. ++ * ++ * Author: Kenneth Kiraly <kiraly@lab126.com> ++ * ++ * 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. ++ * ++ */ ++ ++#ifndef _WM8971_H ++#define _WM8971_H ++ ++#define WM8971_LINVOL 0x00 ++#define WM8971_RINVOL 0x01 ++#define WM8971_LOUT1V 0x02 ++#define WM8971_ROUT1V 0x03 ++#define WM8971_ADCDAC 0x05 ++#define WM8971_AUDIO 0x07 ++#define WM8971_SRATE 0x08 ++#define WM8971_LDAC 0x0a ++#define WM8971_RDAC 0x0b ++#define WM8971_BASS 0x0c ++#define WM8971_TREBLE 0x0d ++#define WM8971_RESET 0x0f ++#define WM8971_ALC1 0x11 ++#define WM8971_ALC2 0x12 ++#define WM8971_ALC3 0x13 ++#define WM8971_NGATE 0x14 ++#define WM8971_LADC 0x15 ++#define WM8971_RADC 0x16 ++#define WM8971_ADCTL1 0x17 ++#define WM8971_ADCTL2 0x18 ++#define WM8971_PWR1 0x19 ++#define WM8971_PWR2 0x1a ++#define WM8971_ADCTL3 0x1b ++#define WM8971_ADCIN 0x1f ++#define WM8971_LADCIN 0x20 ++#define WM8971_RADCIN 0x21 ++#define WM8971_LOUTM1 0x22 ++#define WM8971_LOUTM2 0x23 ++#define WM8971_ROUTM1 0x24 ++#define WM8971_ROUTM2 0x25 ++#define WM8971_MOUTM1 0x26 ++#define WM8971_MOUTM2 0x27 ++#define WM8971_LOUT2V 0x28 ++#define WM8971_ROUT2V 0x29 ++#define WM8971_MOUTV 0x2A ++ ++struct wm8971_setup_data { ++ unsigned short i2c_address; ++}; ++ ++extern struct snd_soc_codec_dai wm8971_dai; ++extern struct snd_soc_codec_device soc_codec_dev_wm8971; ++ ++#endif +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8974.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8974.c +@@ -0,0 +1,935 @@ ++/* ++ * wm8974.c -- WM8974 ALSA Soc Audio driver ++ * ++ * Copyright 2006 Wolfson Microelectronics PLC. ++ * ++ * Author: Liam Girdwood <liam.girdwood@wolfsonmicro.com> ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/version.h> ++#include <linux/kernel.h> ++#include <linux/init.h> ++#include <linux/delay.h> ++#include <linux/pm.h> ++#include <linux/i2c.h> ++#include <linux/platform_device.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/pcm_params.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++#include <sound/initval.h> ++ ++#include "wm8974.h" ++ ++#define AUDIO_NAME "wm8974" ++#define WM8974_VERSION "0.5" ++ ++/* ++ * Debug ++ */ ++ ++#define WM8974_DEBUG 0 ++ ++#ifdef WM8974_DEBUG ++#define dbg(format, arg...) \ ++ printk(KERN_DEBUG AUDIO_NAME ": " format "\n" , ## arg) ++#else ++#define dbg(format, arg...) do {} while (0) ++#endif ++#define err(format, arg...) \ ++ printk(KERN_ERR AUDIO_NAME ": " format "\n" , ## arg) ++#define info(format, arg...) \ ++ printk(KERN_INFO AUDIO_NAME ": " format "\n" , ## arg) ++#define warn(format, arg...) \ ++ printk(KERN_WARNING AUDIO_NAME ": " format "\n" , ## arg) ++ ++struct snd_soc_codec_device soc_codec_dev_wm8974; ++ ++/* ++ * wm8974 register cache ++ * We can't read the WM8974 register space when we are ++ * using 2 wire for device control, so we cache them instead. ++ */ ++static const u16 wm8974_reg[WM8974_CACHEREGNUM] = { ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0050, 0x0000, 0x0140, 0x0000, ++ 0x0000, 0x0000, 0x0000, 0x00ff, ++ 0x0000, 0x0000, 0x0100, 0x00ff, ++ 0x0000, 0x0000, 0x012c, 0x002c, ++ 0x002c, 0x002c, 0x002c, 0x0000, ++ 0x0032, 0x0000, 0x0000, 0x0000, ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0038, 0x000b, 0x0032, 0x0000, ++ 0x0008, 0x000c, 0x0093, 0x00e9, ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0003, 0x0010, 0x0000, 0x0000, ++ 0x0000, 0x0002, 0x0000, 0x0000, ++ 0x0000, 0x0000, 0x0039, 0x0000, ++ 0x0000, ++}; ++ ++#define WM8974_DAIFMT \ ++ (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_LEFT_J | SND_SOC_DAIFMT_RIGHT_J | \ ++ SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_NB_IF | SND_SOC_DAIFMT_IB_NF | \ ++ SND_SOC_DAIFMT_IB_IF) ++ ++#define WM8974_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define WM8974_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000) ++ ++#define WM8794_BCLK \ ++ (SND_SOC_FSBD(1) | SND_SOC_FSBD(2) | SND_SOC_FSBD(4) | SND_SOC_FSBD(8) |\ ++ SND_SOC_FSBD(16) | SND_SOC_FSBD(32)) ++ ++#define WM8794_HIFI_BITS \ ++ (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE | \ ++ SNDRV_PCM_FMTBIT_S24_LE | SNDRV_PCM_FMTBIT_S32_LE) ++ ++static struct snd_soc_dai_mode wm8974_modes[] = { ++ /* codec frame and clock master modes */ ++ { ++ .fmt = WM8974_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8794_HIFI_BITS, ++ .pcmrate = WM8974_RATES, ++ .pcmdir = WM8974_DIR, ++ .fs = 256, ++ .bfs = WM8794_BCLK, ++ }, ++ ++ /* codec frame and clock slave modes */ ++ { ++ .fmt = WM8974_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = WM8794_HIFI_BITS, ++ .pcmrate = WM8974_RATES, ++ .pcmdir = WM8974_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = SND_SOC_FSB_ALL, ++ }, ++}; ++ ++/* ++ * read wm8974 register cache ++ */ ++static inline unsigned int wm8974_read_reg_cache(struct snd_soc_codec * codec, ++ unsigned int reg) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg == WM8974_RESET) ++ return 0; ++ if (reg >= WM8974_CACHEREGNUM) ++ return -1; ++ return cache[reg]; ++} ++ ++/* ++ * write wm8974 register cache ++ */ ++static inline void wm8974_write_reg_cache(struct snd_soc_codec *codec, ++ u16 reg, unsigned int value) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg >= WM8974_CACHEREGNUM) ++ return; ++ cache[reg] = value; ++} ++ ++/* ++ * write to the WM8974 register space ++ */ ++static int wm8974_write(struct snd_soc_codec *codec, unsigned int reg, ++ unsigned int value) ++{ ++ u8 data[2]; ++ ++ /* data is ++ * D15..D9 WM8974 register offset ++ * D8...D0 register data ++ */ ++ data[0] = (reg << 1) | ((value >> 8) & 0x0001); ++ data[1] = value & 0x00ff; ++ ++ wm8974_write_reg_cache (codec, reg, value); ++ if (codec->hw_write(codec->control_data, data, 2) == 2) ++ return 0; ++ else ++ return -EIO; ++} ++ ++#define wm8974_reset(c) wm8974_write(c, WM8974_RESET, 0) ++ ++static const char *wm8974_companding[] = {"Off", "NC", "u-law", "A-law" }; ++static const char *wm8974_deemp[] = {"None", "32kHz", "44.1kHz", "48kHz" }; ++static const char *wm8974_eqmode[] = {"Capture", "Playback" }; ++static const char *wm8974_bw[] = {"Narrow", "Wide" }; ++static const char *wm8974_eq1[] = {"80Hz", "105Hz", "135Hz", "175Hz" }; ++static const char *wm8974_eq2[] = {"230Hz", "300Hz", "385Hz", "500Hz" }; ++static const char *wm8974_eq3[] = {"650Hz", "850Hz", "1.1kHz", "1.4kHz" }; ++static const char *wm8974_eq4[] = {"1.8kHz", "2.4kHz", "3.2kHz", "4.1kHz" }; ++static const char *wm8974_eq5[] = {"5.3kHz", "6.9kHz", "9kHz", "11.7kHz" }; ++static const char *wm8974_alc[] = {"ALC", "Limiter" }; ++ ++static const struct soc_enum wm8974_enum[] = { ++ SOC_ENUM_SINGLE(WM8974_COMP, 1, 4, wm8974_companding), /* adc */ ++ SOC_ENUM_SINGLE(WM8974_COMP, 3, 4, wm8974_companding), /* dac */ ++ SOC_ENUM_SINGLE(WM8974_DAC, 4, 4, wm8974_deemp), ++ SOC_ENUM_SINGLE(WM8974_EQ1, 8, 2, wm8974_eqmode), ++ ++ SOC_ENUM_SINGLE(WM8974_EQ1, 5, 4, wm8974_eq1), ++ SOC_ENUM_SINGLE(WM8974_EQ2, 8, 2, wm8974_bw), ++ SOC_ENUM_SINGLE(WM8974_EQ2, 5, 4, wm8974_eq2), ++ SOC_ENUM_SINGLE(WM8974_EQ3, 8, 2, wm8974_bw), ++ ++ SOC_ENUM_SINGLE(WM8974_EQ3, 5, 4, wm8974_eq3), ++ SOC_ENUM_SINGLE(WM8974_EQ4, 8, 2, wm8974_bw), ++ SOC_ENUM_SINGLE(WM8974_EQ4, 5, 4, wm8974_eq4), ++ SOC_ENUM_SINGLE(WM8974_EQ5, 8, 2, wm8974_bw), ++ ++ SOC_ENUM_SINGLE(WM8974_EQ5, 5, 4, wm8974_eq5), ++ SOC_ENUM_SINGLE(WM8974_ALC3, 8, 2, wm8974_alc), ++}; ++ ++static const struct snd_kcontrol_new wm8974_snd_controls[] = { ++ ++SOC_SINGLE("Digital Loopback Switch", WM8974_COMP, 0, 1, 0), ++ ++SOC_ENUM("DAC Companding", wm8974_enum[1]), ++SOC_ENUM("ADC Companding", wm8974_enum[0]), ++ ++SOC_ENUM("Playback De-emphasis", wm8974_enum[2]), ++SOC_SINGLE("DAC Inversion Switch", WM8974_DAC, 0, 1, 0), ++ ++SOC_SINGLE("PCM Volume", WM8974_DACVOL, 0, 127, 0), ++ ++SOC_SINGLE("High Pass Filter Switch", WM8974_ADC, 8, 1, 0), ++SOC_SINGLE("High Pass Cut Off", WM8974_ADC, 4, 7, 0), ++SOC_SINGLE("ADC Inversion Switch", WM8974_COMP, 0, 1, 0), ++ ++SOC_SINGLE("Capture Volume", WM8974_ADCVOL, 0, 127, 0), ++ ++SOC_ENUM("Equaliser Function", wm8974_enum[3]), ++SOC_ENUM("EQ1 Cut Off", wm8974_enum[4]), ++SOC_SINGLE("EQ1 Volume", WM8974_EQ1, 0, 31, 1), ++ ++SOC_ENUM("Equaliser EQ2 Bandwith", wm8974_enum[5]), ++SOC_ENUM("EQ2 Cut Off", wm8974_enum[6]), ++SOC_SINGLE("EQ2 Volume", WM8974_EQ2, 0, 31, 1), ++ ++SOC_ENUM("Equaliser EQ3 Bandwith", wm8974_enum[7]), ++SOC_ENUM("EQ3 Cut Off", wm8974_enum[8]), ++SOC_SINGLE("EQ3 Volume", WM8974_EQ3, 0, 31, 1), ++ ++SOC_ENUM("Equaliser EQ4 Bandwith", wm8974_enum[9]), ++SOC_ENUM("EQ4 Cut Off", wm8974_enum[10]), ++SOC_SINGLE("EQ4 Volume", WM8974_EQ4, 0, 31, 1), ++ ++SOC_ENUM("Equaliser EQ5 Bandwith", wm8974_enum[11]), ++SOC_ENUM("EQ5 Cut Off", wm8974_enum[12]), ++SOC_SINGLE("EQ5 Volume", WM8974_EQ5, 0, 31, 1), ++ ++SOC_SINGLE("DAC Playback Limiter Switch", WM8974_DACLIM1, 8, 1, 0), ++SOC_SINGLE("DAC Playback Limiter Decay", WM8974_DACLIM1, 4, 15, 0), ++SOC_SINGLE("DAC Playback Limiter Attack", WM8974_DACLIM1, 0, 15, 0), ++ ++SOC_SINGLE("DAC Playback Limiter Threshold", WM8974_DACLIM2, 4, 7, 0), ++SOC_SINGLE("DAC Playback Limiter Boost", WM8974_DACLIM2, 0, 15, 0), ++ ++SOC_SINGLE("ALC Enable Switch", WM8974_ALC1, 8, 1, 0), ++SOC_SINGLE("ALC Capture Max Gain", WM8974_ALC1, 3, 7, 0), ++SOC_SINGLE("ALC Capture Min Gain", WM8974_ALC1, 0, 7, 0), ++ ++SOC_SINGLE("ALC Capture ZC Switch", WM8974_ALC2, 8, 1, 0), ++SOC_SINGLE("ALC Capture Hold", WM8974_ALC2, 4, 7, 0), ++SOC_SINGLE("ALC Capture Target", WM8974_ALC2, 0, 15, 0), ++ ++SOC_ENUM("ALC Capture Mode", wm8974_enum[13]), ++SOC_SINGLE("ALC Capture Decay", WM8974_ALC3, 4, 15, 0), ++SOC_SINGLE("ALC Capture Attack", WM8974_ALC3, 0, 15, 0), ++ ++SOC_SINGLE("ALC Capture Noise Gate Switch", WM8974_NGATE, 3, 1, 0), ++SOC_SINGLE("ALC Capture Noise Gate Threshold", WM8974_NGATE, 0, 7, 0), ++ ++SOC_SINGLE("Capture PGA ZC Switch", WM8974_INPPGA, 7, 1, 0), ++SOC_SINGLE("Capture PGA Volume", WM8974_INPPGA, 0, 63, 0), ++ ++SOC_SINGLE("Speaker Playback ZC Switch", WM8974_SPKVOL, 7, 1, 0), ++SOC_SINGLE("Speaker Playback Switch", WM8974_SPKVOL, 6, 1, 1), ++SOC_SINGLE("Speaker Playback Volume", WM8974_SPKVOL, 0, 63, 0), ++ ++SOC_SINGLE("Capture Boost(+20dB)", WM8974_ADCBOOST, 8, 1, 0), ++SOC_SINGLE("Mono Playback Switch", WM8974_MONOMIX, 6, 1, 0), ++}; ++ ++/* add non dapm controls */ ++static int wm8974_add_controls(struct snd_soc_codec *codec) ++{ ++ int err, i; ++ ++ for (i = 0; i < ARRAY_SIZE(wm8974_snd_controls); i++) { ++ err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&wm8974_snd_controls[i],codec, NULL)); ++ if (err < 0) ++ return err; ++ } ++ ++ return 0; ++} ++ ++/* Speaker Output Mixer */ ++static const struct snd_kcontrol_new wm8974_speaker_mixer_controls[] = { ++SOC_DAPM_SINGLE("Line Bypass Switch", WM8974_SPKMIX, 1, 1, 0), ++SOC_DAPM_SINGLE("Aux Playback Switch", WM8974_SPKMIX, 5, 1, 0), ++SOC_DAPM_SINGLE("PCM Playback Switch", WM8974_SPKMIX, 0, 1, 1), ++}; ++ ++/* Mono Output Mixer */ ++static const struct snd_kcontrol_new wm8974_mono_mixer_controls[] = { ++SOC_DAPM_SINGLE("Line Bypass Switch", WM8974_MONOMIX, 1, 1, 0), ++SOC_DAPM_SINGLE("Aux Playback Switch", WM8974_MONOMIX, 2, 1, 0), ++SOC_DAPM_SINGLE("PCM Playback Switch", WM8974_MONOMIX, 0, 1, 1), ++}; ++ ++/* AUX Input boost vol */ ++static const struct snd_kcontrol_new wm8974_aux_boost_controls = ++SOC_DAPM_SINGLE("Aux Volume", WM8974_ADCBOOST, 0, 7, 0); ++ ++/* Mic Input boost vol */ ++static const struct snd_kcontrol_new wm8974_mic_boost_controls = ++SOC_DAPM_SINGLE("Mic Volume", WM8974_ADCBOOST, 4, 7, 0); ++ ++/* Capture boost switch */ ++static const struct snd_kcontrol_new wm8974_capture_boost_controls = ++SOC_DAPM_SINGLE("Capture Boost Switch", WM8974_INPPGA, 6, 1, 0); ++ ++/* Aux In to PGA */ ++static const struct snd_kcontrol_new wm8974_aux_capture_boost_controls = ++SOC_DAPM_SINGLE("Aux Capture Boost Switch", WM8974_INPPGA, 2, 1, 0); ++ ++/* Mic P In to PGA */ ++static const struct snd_kcontrol_new wm8974_micp_capture_boost_controls = ++SOC_DAPM_SINGLE("Mic P Capture Boost Switch", WM8974_INPPGA, 0, 1, 0); ++ ++/* Mic N In to PGA */ ++static const struct snd_kcontrol_new wm8974_micn_capture_boost_controls = ++SOC_DAPM_SINGLE("Mic N Capture Boost Switch", WM8974_INPPGA, 1, 1, 0); ++ ++static const struct snd_soc_dapm_widget wm8974_dapm_widgets[] = { ++SND_SOC_DAPM_MIXER("Speaker Mixer", WM8974_POWER3, 2, 0, ++ &wm8974_speaker_mixer_controls[0], ++ ARRAY_SIZE(wm8974_speaker_mixer_controls)), ++SND_SOC_DAPM_MIXER("Mono Mixer", WM8974_POWER3, 3, 0, ++ &wm8974_mono_mixer_controls[0], ++ ARRAY_SIZE(wm8974_mono_mixer_controls)), ++SND_SOC_DAPM_DAC("DAC", "HiFi Playback", WM8974_POWER3, 0, 0), ++SND_SOC_DAPM_ADC("ADC", "HiFi Capture", WM8974_POWER3, 0, 0), ++SND_SOC_DAPM_PGA("Aux Input", WM8974_POWER1, 6, 0, NULL, 0), ++SND_SOC_DAPM_PGA("SpkN Out", WM8974_POWER3, 5, 0, NULL, 0), ++SND_SOC_DAPM_PGA("SpkP Out", WM8974_POWER3, 6, 0, NULL, 0), ++SND_SOC_DAPM_PGA("Mono Out", WM8974_POWER3, 7, 0, NULL, 0), ++SND_SOC_DAPM_PGA("Mic PGA", WM8974_POWER2, 2, 0, NULL, 0), ++ ++SND_SOC_DAPM_PGA("Aux Boost", SND_SOC_NOPM, 0, 0, ++ &wm8974_aux_boost_controls, 1), ++SND_SOC_DAPM_PGA("Mic Boost", SND_SOC_NOPM, 0, 0, ++ &wm8974_mic_boost_controls, 1), ++SND_SOC_DAPM_SWITCH("Capture Boost", SND_SOC_NOPM, 0, 0, ++ &wm8974_capture_boost_controls), ++ ++SND_SOC_DAPM_MIXER("Boost Mixer", WM8974_POWER2, 4, 0, NULL, 0), ++ ++SND_SOC_DAPM_MICBIAS("Mic Bias", WM8974_POWER1, 4, 0), ++ ++SND_SOC_DAPM_INPUT("MICN"), ++SND_SOC_DAPM_INPUT("MICP"), ++SND_SOC_DAPM_INPUT("AUX"), ++SND_SOC_DAPM_OUTPUT("MONOOUT"), ++SND_SOC_DAPM_OUTPUT("SPKOUTP"), ++SND_SOC_DAPM_OUTPUT("SPKOUTN"), ++}; ++ ++static const char *audio_map[][3] = { ++ /* Mono output mixer */ ++ {"Mono Mixer", "PCM Playback Switch", "DAC"}, ++ {"Mono Mixer", "Aux Playback Switch", "Aux Input"}, ++ {"Mono Mixer", "Line Bypass Switch", "Boost Mixer"}, ++ ++ /* Speaker output mixer */ ++ {"Speaker Mixer", "PCM Playback Switch", "DAC"}, ++ {"Speaker Mixer", "Aux Playback Switch", "Aux Input"}, ++ {"Speaker Mixer", "Line Bypass Switch", "Boost Mixer"}, ++ ++ /* Outputs */ ++ {"Mono Out", NULL, "Mono Mixer"}, ++ {"MONOOUT", NULL, "Mono Out"}, ++ {"SpkN Out", NULL, "Speaker Mixer"}, ++ {"SpkP Out", NULL, "Speaker Mixer"}, ++ {"SPKOUTN", NULL, "SpkN Out"}, ++ {"SPKOUTP", NULL, "SpkP Out"}, ++ ++ /* Boost Mixer */ ++ {"Boost Mixer", NULL, "ADC"}, ++ {"Capture Boost Switch", "Aux Capture Boost Switch", "AUX"}, ++ {"Aux Boost", "Aux Volume", "Boost Mixer"}, ++ {"Capture Boost", "Capture Switch", "Boost Mixer"}, ++ {"Mic Boost", "Mic Volume", "Boost Mixer"}, ++ ++ /* Inputs */ ++ {"MICP", NULL, "Mic Boost"}, ++ {"MICN", NULL, "Mic PGA"}, ++ {"Mic PGA", NULL, "Capture Boost"}, ++ {"AUX", NULL, "Aux Input"}, ++ ++ /* terminator */ ++ {NULL, NULL, NULL}, ++}; ++ ++static int wm8974_add_widgets(struct snd_soc_codec *codec) ++{ ++ int i; ++ ++ for(i = 0; i < ARRAY_SIZE(wm8974_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &wm8974_dapm_widgets[i]); ++ } ++ ++ /* set up audio path audio_mapnects */ ++ for(i = 0; audio_map[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, audio_map[i][0], ++ audio_map[i][1], audio_map[i][2]); ++ } ++ ++ snd_soc_dapm_new_widgets(codec); ++ return 0; ++} ++ ++struct pll_ { ++ unsigned int in_hz, out_hz; ++ unsigned int pre:4; /* prescale - 1 */ ++ unsigned int n:4; ++ unsigned int k; ++}; ++ ++struct pll_ pll[] = { ++ {12000000, 11289600, 0, 7, 0x86c220}, ++ {12000000, 12288000, 0, 8, 0x3126e8}, ++ {13000000, 11289600, 0, 6, 0xf28bd4}, ++ {13000000, 12288000, 0, 7, 0x8fd525}, ++ {12288000, 11289600, 0, 7, 0x59999a}, ++ {11289600, 12288000, 0, 8, 0x80dee9}, ++ /* liam - add more entries */ ++}; ++ ++static int set_pll(struct snd_soc_codec *codec, unsigned int in, ++ unsigned int out) ++{ ++ int i; ++ u16 reg; ++ ++ if(out == 0) { ++ reg = wm8974_read_reg_cache(codec, WM8974_POWER1); ++ wm8974_write(codec, WM8974_POWER1, reg & 0x1df); ++ return 0; ++ } ++ ++ for(i = 0; i < ARRAY_SIZE(pll); i++) { ++ if (in == pll[i].in_hz && out == pll[i].out_hz) { ++ wm8974_write(codec, WM8974_PLLN, (pll[i].pre << 4) | pll[i].n); ++ wm8974_write(codec, WM8974_PLLK1, pll[i].k >> 18); ++ wm8974_write(codec, WM8974_PLLK1, (pll[i].k >> 9) && 0x1ff); ++ wm8974_write(codec, WM8974_PLLK1, pll[i].k && 0x1ff); ++ reg = wm8974_read_reg_cache(codec, WM8974_POWER1); ++ wm8974_write(codec, WM8974_POWER1, reg | 0x020); ++ return 0; ++ } ++ } ++ return -EINVAL; ++} ++ ++/* mclk dividers * 2 */ ++static unsigned char mclk_div[] = {2, 3, 4, 6, 8, 12, 16, 24}; ++ ++/* we need 256FS to drive the DAC's and ADC's */ ++static unsigned int wm8974_config_sysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ int i, j, best_clk = info->fs * info->rate; ++ ++ /* can we run at this clk without the PLL ? */ ++ for (i = 0; i < ARRAY_SIZE(mclk_div); i++) { ++ if ((best_clk >> 1) * mclk_div[i] == clk) { ++ dai->pll_in = 0; ++ dai->clk_div = mclk_div[i]; ++ dai->mclk = best_clk; ++ return dai->mclk; ++ } ++ } ++ ++ /* now check for PLL support */ ++ for (i = 0; i < ARRAY_SIZE(pll); i++) { ++ if (pll[i].in_hz == clk) { ++ for (j = 0; j < ARRAY_SIZE(mclk_div); j++) { ++ if (pll[i].out_hz == mclk_div[j] * (best_clk >> 1)) { ++ dai->pll_in = clk; ++ dai->pll_out = pll[i].out_hz; ++ dai->clk_div = mclk_div[j]; ++ dai->mclk = best_clk; ++ return dai->mclk; ++ } ++ } ++ } ++ } ++ ++ /* this clk is not supported */ ++ return 0; ++} ++ ++static int wm8974_pcm_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ struct snd_soc_codec_dai *dai = rtd->codec_dai; ++ u16 iface = 0, bfs, clk = 0, adn; ++ int fs = 48000 << 7, i; ++ ++ bfs = SND_SOC_FSBD_REAL(rtd->codec_dai->dai_runtime.bfs); ++ switch (bfs) { ++ case 2: ++ clk |= 0x1 << 2; ++ break; ++ case 4: ++ clk |= 0x2 << 2; ++ break; ++ case 8: ++ clk |= 0x3 << 2; ++ break; ++ case 16: ++ clk |= 0x4 << 2; ++ break; ++ case 32: ++ clk |= 0x5 << 2; ++ break; ++ } ++ ++ /* set master/slave audio interface */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK) { ++ case SND_SOC_DAIFMT_CBM_CFM: ++ clk |= 0x0001; ++ break; ++ case SND_SOC_DAIFMT_CBS_CFS: ++ break; ++ } ++ ++ /* interface format */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_I2S: ++ iface |= 0x0010; ++ break; ++ case SND_SOC_DAIFMT_RIGHT_J: ++ break; ++ case SND_SOC_DAIFMT_LEFT_J: ++ iface |= 0x0008; ++ break; ++ case SND_SOC_DAIFMT_DSP_A: ++ iface |= 0x00018; ++ break; ++ } ++ ++ /* bit size */ ++ switch (rtd->codec_dai->dai_runtime.pcmfmt) { ++ case SNDRV_PCM_FMTBIT_S16_LE: ++ break; ++ case SNDRV_PCM_FMTBIT_S20_3LE: ++ iface |= 0x0020; ++ break; ++ case SNDRV_PCM_FMTBIT_S24_LE: ++ iface |= 0x0040; ++ break; ++ case SNDRV_PCM_FMTBIT_S32_LE: ++ iface |= 0x0060; ++ break; ++ } ++ ++ /* clock inversion */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_INV_MASK) { ++ case SND_SOC_DAIFMT_NB_NF: ++ break; ++ case SND_SOC_DAIFMT_IB_IF: ++ iface |= 0x0180; ++ break; ++ case SND_SOC_DAIFMT_IB_NF: ++ iface |= 0x0100; ++ break; ++ case SND_SOC_DAIFMT_NB_IF: ++ iface |= 0x0080; ++ break; ++ } ++ ++ /* filter coefficient */ ++ adn = wm8974_read_reg_cache(codec, WM8974_ADD) & 0x1f1; ++ switch (rtd->codec_dai->dai_runtime.pcmrate) { ++ case SNDRV_PCM_RATE_8000: ++ adn |= 0x5 << 1; ++ fs = 8000 << 7; ++ break; ++ case SNDRV_PCM_RATE_11025: ++ adn |= 0x4 << 1; ++ fs = 11025 << 7; ++ break; ++ case SNDRV_PCM_RATE_16000: ++ adn |= 0x3 << 1; ++ fs = 16000 << 7; ++ break; ++ case SNDRV_PCM_RATE_22050: ++ adn |= 0x2 << 1; ++ fs = 22050 << 7; ++ break; ++ case SNDRV_PCM_RATE_32000: ++ adn |= 0x1 << 1; ++ fs = 32000 << 7; ++ break; ++ case SNDRV_PCM_RATE_44100: ++ fs = 44100 << 7; ++ break; ++ } ++ ++ /* do we need to enable the PLL */ ++ if(dai->pll_in) ++ set_pll(codec, dai->pll_in, dai->pll_out); ++ ++ /* divide the clock to 256 fs */ ++ for(i = 0; i < ARRAY_SIZE(mclk_div); i++) { ++ if (dai->clk_div == mclk_div[i]) { ++ clk |= i << 5; ++ clk &= 0xff; ++ goto set; ++ } ++ } ++ ++set: ++ /* set iface */ ++ wm8974_write(codec, WM8974_IFACE, iface); ++ wm8974_write(codec, WM8974_CLOCK, clk); ++ ++ return 0; ++} ++ ++static int wm8974_hw_free(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ set_pll(codec, 0, 0); ++ return 0; ++} ++ ++static int wm8974_mute(struct snd_soc_codec *codec, ++ struct snd_soc_codec_dai *dai, int mute) ++{ ++ u16 mute_reg = wm8974_read_reg_cache(codec, WM8974_DAC) & 0xffbf; ++ if(mute) ++ wm8974_write(codec, WM8974_DAC, mute_reg | 0x40); ++ else ++ wm8974_write(codec, WM8974_DAC, mute_reg); ++ return 0; ++} ++ ++/* liam need to make this lower power with dapm */ ++static int wm8974_dapm_event(struct snd_soc_codec *codec, int event) ++{ ++ ++ switch (event) { ++ case SNDRV_CTL_POWER_D0: /* full On */ ++ /* vref/mid, clk and osc on, dac unmute, active */ ++ wm8974_write(codec, WM8974_POWER1, 0x1ff); ++ wm8974_write(codec, WM8974_POWER2, 0x1ff); ++ wm8974_write(codec, WM8974_POWER3, 0x1ff); ++ break; ++ case SNDRV_CTL_POWER_D1: /* partial On */ ++ case SNDRV_CTL_POWER_D2: /* partial On */ ++ break; ++ case SNDRV_CTL_POWER_D3hot: /* Off, with power */ ++ /* everything off except vref/vmid, dac mute, inactive */ ++ ++ break; ++ case SNDRV_CTL_POWER_D3cold: /* Off, without power */ ++ /* everything off, dac mute, inactive */ ++ wm8974_write(codec, WM8974_POWER1, 0x0); ++ wm8974_write(codec, WM8974_POWER2, 0x0); ++ wm8974_write(codec, WM8974_POWER3, 0x0); ++ break; ++ } ++ codec->dapm_state = event; ++ return 0; ++} ++ ++struct snd_soc_codec_dai wm8974_dai = { ++ .name = "WM8974 HiFi", ++ .playback = { ++ .stream_name = "Playback", ++ .channels_min = 1, ++ .channels_max = 1, ++ }, ++ .capture = { ++ .stream_name = "Capture", ++ .channels_min = 1, ++ .channels_max = 1, ++ }, ++ .config_sysclk = wm8974_config_sysclk, ++ .digital_mute = wm8974_mute, ++ .ops = { ++ .prepare = wm8974_pcm_prepare, ++ .hw_free = wm8974_hw_free, ++ }, ++ .caps = { ++ .num_modes = ARRAY_SIZE(wm8974_modes), ++ .mode = wm8974_modes, ++ }, ++}; ++EXPORT_SYMBOL_GPL(wm8974_dai); ++ ++static int wm8974_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ wm8974_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ return 0; ++} ++ ++static int wm8974_resume(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ int i; ++ u8 data[2]; ++ u16 *cache = codec->reg_cache; ++ ++ /* Sync reg_cache with the hardware */ ++ for (i = 0; i < ARRAY_SIZE(wm8974_reg); i++) { ++ data[0] = (i << 1) | ((cache[i] >> 8) & 0x0001); ++ data[1] = cache[i] & 0x00ff; ++ codec->hw_write(codec->control_data, data, 2); ++ } ++ wm8974_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ wm8974_dapm_event(codec, codec->suspend_dapm_state); ++ return 0; ++} ++ ++/* ++ * initialise the WM8974 driver ++ * register the mixer and dsp interfaces with the kernel ++ */ ++static int wm8974_init(struct snd_soc_device *socdev) ++{ ++ struct snd_soc_codec *codec = socdev->codec; ++ int ret = 0; ++ ++ codec->name = "WM8974"; ++ codec->owner = THIS_MODULE; ++ codec->read = wm8974_read_reg_cache; ++ codec->write = wm8974_write; ++ codec->dapm_event = wm8974_dapm_event; ++ codec->dai = &wm8974_dai; ++ codec->num_dai = 1; ++ codec->reg_cache_size = ARRAY_SIZE(wm8974_reg); ++ codec->reg_cache = ++ kzalloc(sizeof(u16) * ARRAY_SIZE(wm8974_reg), GFP_KERNEL); ++ if (codec->reg_cache == NULL) ++ return -ENOMEM; ++ memcpy(codec->reg_cache, wm8974_reg, ++ sizeof(u16) * ARRAY_SIZE(wm8974_reg)); ++ codec->reg_cache_size = sizeof(u16) * ARRAY_SIZE(wm8974_reg); ++ ++ wm8974_reset(codec); ++ ++ /* register pcms */ ++ ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); ++ if(ret < 0) { ++ kfree(codec->reg_cache); ++ return ret; ++ } ++ ++ /* power on device */ ++ wm8974_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ wm8974_add_controls(codec); ++ wm8974_add_widgets(codec); ++ ret = snd_soc_register_card(socdev); ++ if(ret < 0) { ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++ } ++ ++ return ret; ++} ++ ++static struct snd_soc_device *wm8974_socdev; ++ ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ ++/* ++ * WM8974 2 wire address is 0x1a ++ */ ++#define I2C_DRIVERID_WM8974 0xfefe /* liam - need a proper id */ ++ ++static unsigned short normal_i2c[] = { 0, I2C_CLIENT_END }; ++ ++/* Magic definition of all other variables and things */ ++I2C_CLIENT_INSMOD; ++ ++static struct i2c_driver wm8974_i2c_driver; ++static struct i2c_client client_template; ++ ++/* If the i2c layer weren't so broken, we could pass this kind of data ++ around */ ++ ++static int wm8974_codec_probe(struct i2c_adapter *adap, int addr, int kind) ++{ ++ struct snd_soc_device *socdev = wm8974_socdev; ++ struct wm8974_setup_data *setup = socdev->codec_data; ++ struct snd_soc_codec *codec = socdev->codec; ++ struct i2c_client *i2c; ++ int ret; ++ ++ if (addr != setup->i2c_address) ++ return -ENODEV; ++ ++ client_template.adapter = adap; ++ client_template.addr = addr; ++ ++ i2c = kzalloc(sizeof(struct i2c_client), GFP_KERNEL); ++ if (i2c == NULL) { ++ kfree(codec); ++ return -ENOMEM; ++ } ++ memcpy(i2c, &client_template, sizeof(struct i2c_client)); ++ i2c_set_clientdata(i2c, codec); ++ codec->control_data = i2c; ++ ++ ret = i2c_attach_client(i2c); ++ if(ret < 0) { ++ err("failed to attach codec at addr %x\n", addr); ++ goto err; ++ } ++ ++ ret = wm8974_init(socdev); ++ if(ret < 0) { ++ err("failed to initialise WM8974\n"); ++ goto err; ++ } ++ return ret; ++ ++err: ++ kfree(codec); ++ kfree(i2c); ++ return ret; ++} ++ ++static int wm8974_i2c_detach(struct i2c_client *client) ++{ ++ struct snd_soc_codec *codec = i2c_get_clientdata(client); ++ i2c_detach_client(client); ++ kfree(codec->reg_cache); ++ kfree(client); ++ return 0; ++} ++ ++static int wm8974_i2c_attach(struct i2c_adapter *adap) ++{ ++ return i2c_probe(adap, &addr_data, wm8974_codec_probe); ++} ++ ++/* corgi i2c codec control layer */ ++static struct i2c_driver wm8974_i2c_driver = { ++ .driver = { ++ .name = "WM8974 I2C Codec", ++ .owner = THIS_MODULE, ++ }, ++ .id = I2C_DRIVERID_WM8974, ++ .attach_adapter = wm8974_i2c_attach, ++ .detach_client = wm8974_i2c_detach, ++ .command = NULL, ++}; ++ ++static struct i2c_client client_template = { ++ .name = "WM8974", ++ .driver = &wm8974_i2c_driver, ++}; ++#endif ++ ++static int wm8974_probe(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct wm8974_setup_data *setup; ++ struct snd_soc_codec *codec; ++ int ret = 0; ++ ++ info("WM8974 Audio Codec %s", WM8974_VERSION); ++ ++ setup = socdev->codec_data; ++ codec = kzalloc(sizeof(struct snd_soc_codec), GFP_KERNEL); ++ if (codec == NULL) ++ return -ENOMEM; ++ ++ socdev->codec = codec; ++ mutex_init(&codec->mutex); ++ INIT_LIST_HEAD(&codec->dapm_widgets); ++ INIT_LIST_HEAD(&codec->dapm_paths); ++ ++ wm8974_socdev = socdev; ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ if (setup->i2c_address) { ++ normal_i2c[0] = setup->i2c_address; ++ codec->hw_write = (hw_write_t)i2c_master_send; ++ ret = i2c_add_driver(&wm8974_i2c_driver); ++ if (ret != 0) ++ printk(KERN_ERR "can't add i2c driver"); ++ } ++#else ++ /* Add other interfaces here */ ++#endif ++ return ret; ++} ++ ++/* power down chip */ ++static int wm8974_remove(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ if (codec->control_data) ++ wm8974_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ i2c_del_driver(&wm8974_i2c_driver); ++#endif ++ kfree(codec); ++ ++ return 0; ++} ++ ++struct snd_soc_codec_device soc_codec_dev_wm8974 = { ++ .probe = wm8974_probe, ++ .remove = wm8974_remove, ++ .suspend = wm8974_suspend, ++ .resume = wm8974_resume, ++}; ++ ++EXPORT_SYMBOL_GPL(soc_codec_dev_wm8974); ++ ++MODULE_DESCRIPTION("ASoC WM8974 driver"); ++MODULE_AUTHOR("Liam Girdwood"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8974.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8974.h +@@ -0,0 +1,64 @@ ++/* ++ * wm8974.h -- WM8974 Soc Audio driver ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#ifndef _WM8974_H ++#define _WM8974_H ++ ++/* WM8974 register space */ ++ ++#define WM8974_RESET 0x0 ++#define WM8974_POWER1 0x1 ++#define WM8974_POWER2 0x2 ++#define WM8974_POWER3 0x3 ++#define WM8974_IFACE 0x4 ++#define WM8974_COMP 0x5 ++#define WM8974_CLOCK 0x6 ++#define WM8974_ADD 0x7 ++#define WM8974_GPIO 0x8 ++#define WM8974_DAC 0xa ++#define WM8974_DACVOL 0xb ++#define WM8974_ADC 0xe ++#define WM8974_ADCVOL 0xf ++#define WM8974_EQ1 0x12 ++#define WM8974_EQ2 0x13 ++#define WM8974_EQ3 0x14 ++#define WM8974_EQ4 0x15 ++#define WM8974_EQ5 0x16 ++#define WM8974_DACLIM1 0x18 ++#define WM8974_DACLIM2 0x19 ++#define WM8974_NOTCH1 0x1b ++#define WM8974_NOTCH2 0x1c ++#define WM8974_NOTCH3 0x1d ++#define WM8974_NOTCH4 0x1e ++#define WM8974_ALC1 0x20 ++#define WM8974_ALC2 0x21 ++#define WM8974_ALC3 0x22 ++#define WM8974_NGATE 0x23 ++#define WM8974_PLLN 0x24 ++#define WM8974_PLLK1 0x25 ++#define WM8974_PLLK2 0x26 ++#define WM8974_PLLK3 0x27 ++#define WM8974_ATTEN 0x28 ++#define WM8974_INPUT 0x2c ++#define WM8974_INPPGA 0x2d ++#define WM8974_ADCBOOST 0x2f ++#define WM8974_OUTPUT 0x31 ++#define WM8974_SPKMIX 0x32 ++#define WM8974_SPKVOL 0x36 ++#define WM8974_MONOMIX 0x38 ++ ++#define WM8974_CACHEREGNUM 57 ++ ++struct wm8974_setup_data { ++ unsigned short i2c_address; ++}; ++ ++extern struct snd_soc_codec_dai wm8974_dai; ++extern struct snd_soc_codec_device soc_codec_dev_wm8974; ++ ++#endif +Index: linux-2.6-pxa-new/sound/soc/codecs/wm9712.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm9712.c +@@ -0,0 +1,781 @@ ++/* ++ * wm9712.c -- ALSA Soc WM9712 codec support ++ * ++ * Copyright 2006 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * 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. ++ * ++ * Revision history ++ * 4th Feb 2006 Initial version. ++ */ ++ ++#include <linux/init.h> ++#include <linux/module.h> ++#include <linux/version.h> ++#include <linux/kernel.h> ++#include <linux/device.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/ac97_codec.h> ++#include <sound/initval.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++ ++#define WM9712_VERSION "0.4" ++ ++static unsigned int ac97_read(struct snd_soc_codec *codec, ++ unsigned int reg); ++static int ac97_write(struct snd_soc_codec *codec, ++ unsigned int reg, unsigned int val); ++ ++#define AC97_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define AC97_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000) ++ ++/* may need to expand this */ ++static struct snd_soc_dai_mode ac97_modes[] = { ++ { ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S18_3LE, ++ .pcmrate = AC97_RATES, ++ .pcmdir = AC97_DIR, ++ }, ++}; ++ ++/* ++ * WM9712 register cache ++ */ ++static const u16 wm9712_reg[] = { ++ 0x6174, 0x8000, 0x8000, 0x8000, // 6 ++ 0xf0f0, 0xaaa0, 0xc008, 0x6808, // e ++ 0xe808, 0xaaa0, 0xad00, 0x8000, // 16 ++ 0xe808, 0x3000, 0x8000, 0x0000, // 1e ++ 0x0000, 0x0000, 0x0000, 0x000f, // 26 ++ 0x0405, 0x0410, 0xbb80, 0xbb80, // 2e ++ 0x0000, 0xbb80, 0x0000, 0x0000, // 36 ++ 0x0000, 0x2000, 0x0000, 0x0000, // 3e ++ 0x0000, 0x0000, 0x0000, 0x0000, // 46 ++ 0x0000, 0x0000, 0xf83e, 0xffff, // 4e ++ 0x0000, 0x0000, 0x0000, 0xf83e, // 56 ++ 0x0008, 0x0000, 0x0000, 0x0000, // 5e ++ 0xb032, 0x3e00, 0x0000, 0x0000, // 66 ++ 0x0000, 0x0000, 0x0000, 0x0000, // 6e ++ 0x0000, 0x0000, 0x0000, 0x0006, // 76 ++ 0x0001, 0x0000, 0x574d, 0x4c12, // 7e ++ 0x0000, 0x0000 // virtual hp mixers ++}; ++ ++/* virtual HP mixers regs */ ++#define HPL_MIXER 0x80 ++#define HPR_MIXER 0x82 ++ ++static const char *wm9712_alc_select[] = {"None", "Left", "Right", "Stereo"}; ++static const char *wm9712_alc_mux[] = {"Stereo", "Left", "Right", "None"}; ++static const char *wm9712_out3_src[] = {"Left", "VREF", "Left + Right", ++ "Mono"}; ++static const char *wm9712_spk_src[] = {"Speaker Mix", "Headphone Mix"}; ++static const char *wm9712_rec_adc[] = {"Stereo", "Left", "Right", "Mute"}; ++static const char *wm9712_base[] = {"Linear Control", "Adaptive Boost"}; ++static const char *wm9712_rec_gain[] = {"+1.5dB Steps", "+0.75dB Steps"}; ++static const char *wm9712_mic[] = {"Mic 1", "Differential", "Mic 2", ++ "Stereo"}; ++static const char *wm9712_rec_sel[] = {"Mic", "NC", "NC", "Speaker Mixer", ++ "Line", "Headphone Mixer", "Phone Mixer", "Phone"}; ++static const char *wm9712_ng_type[] = {"Constant Gain", "Mute"}; ++static const char *wm9712_diff_sel[] = {"Mic", "Line"}; ++ ++static const struct soc_enum wm9712_enum[] = { ++SOC_ENUM_SINGLE(AC97_PCI_SVID, 14, 4, wm9712_alc_select), ++SOC_ENUM_SINGLE(AC97_VIDEO, 12, 4, wm9712_alc_mux), ++SOC_ENUM_SINGLE(AC97_AUX, 9, 4, wm9712_out3_src), ++SOC_ENUM_SINGLE(AC97_AUX, 8, 2, wm9712_spk_src), ++SOC_ENUM_SINGLE(AC97_REC_SEL, 12, 4, wm9712_rec_adc), ++SOC_ENUM_SINGLE(AC97_MASTER_TONE, 15, 2, wm9712_base), ++SOC_ENUM_DOUBLE(AC97_REC_GAIN, 14, 6, 2, wm9712_rec_gain), ++SOC_ENUM_SINGLE(AC97_MIC, 5, 4, wm9712_mic), ++SOC_ENUM_SINGLE(AC97_REC_SEL, 8, 8, wm9712_rec_sel), ++SOC_ENUM_SINGLE(AC97_REC_SEL, 0, 8, wm9712_rec_sel), ++SOC_ENUM_SINGLE(AC97_PCI_SVID, 5, 2, wm9712_ng_type), ++SOC_ENUM_SINGLE(0x5c, 8, 2, wm9712_diff_sel), ++}; ++ ++static const struct snd_kcontrol_new wm9712_snd_ac97_controls[] = { ++SOC_DOUBLE("Speaker Playback Volume", AC97_MASTER, 8, 0, 31, 1), ++SOC_SINGLE("Speaker Playback Switch", AC97_MASTER, 15, 1, 1), ++SOC_DOUBLE("Headphone Playback Volume", AC97_HEADPHONE, 8, 0, 31, 1), ++SOC_SINGLE("Headphone Playback Switch", AC97_HEADPHONE,15, 1, 1), ++ ++SOC_SINGLE("Speaker Playback ZC Switch", AC97_MASTER, 7, 1, 0), ++SOC_SINGLE("Speaker Playback Invert Switch", AC97_MASTER, 6, 1, 0), ++SOC_SINGLE("Headphone Playback ZC Switch", AC97_HEADPHONE, 7, 1, 0), ++SOC_SINGLE("Mono Playback ZC Switch", AC97_MASTER_MONO, 7, 1, 0), ++SOC_SINGLE("Mono Playback Volume", AC97_MASTER_MONO, 0, 31, 0), ++ ++SOC_SINGLE("ALC Target Volume", AC97_CODEC_CLASS_REV, 12, 15, 0), ++SOC_SINGLE("ALC Hold Time", AC97_CODEC_CLASS_REV, 8, 15, 0), ++SOC_SINGLE("ALC Decay Time", AC97_CODEC_CLASS_REV, 4, 15, 0), ++SOC_SINGLE("ALC Attack Time", AC97_CODEC_CLASS_REV, 0, 15, 0), ++SOC_ENUM("ALC Function", wm9712_enum[0]), ++SOC_SINGLE("ALC Max Volume", AC97_PCI_SVID, 11, 7, 0), ++SOC_SINGLE("ALC ZC Timeout", AC97_PCI_SVID, 9, 3, 1), ++SOC_SINGLE("ALC ZC Switch", AC97_PCI_SVID, 8, 1, 0), ++SOC_SINGLE("ALC NG Switch", AC97_PCI_SVID, 7, 1, 0), ++SOC_ENUM("ALC NG Type", wm9712_enum[10]), ++SOC_SINGLE("ALC NG Threshold", AC97_PCI_SVID, 0, 31, 1), ++ ++SOC_SINGLE("Mic Headphone Volume", AC97_VIDEO, 12, 7, 1), ++SOC_SINGLE("ALC Headphone Volume", AC97_VIDEO, 7, 7, 1), ++ ++SOC_SINGLE("Out3 Switch", AC97_AUX, 15, 1, 1), ++SOC_SINGLE("Out3 ZC Switch", AC97_AUX, 7, 1, 1), ++SOC_SINGLE("Out3 Volume", AC97_AUX, 0, 31, 1), ++ ++SOC_SINGLE("PCBeep Bypass Headphone Volume", AC97_PC_BEEP, 12, 7, 1), ++SOC_SINGLE("PCBeep Bypass Speaker Volume", AC97_PC_BEEP, 8, 7, 1), ++SOC_SINGLE("PCBeep Bypass Phone Volume", AC97_PC_BEEP, 4, 7, 1), ++ ++SOC_SINGLE("Aux Playback Headphone Volume", AC97_CD, 12, 7, 1), ++SOC_SINGLE("Aux Playback Speaker Volume", AC97_CD, 8, 7, 1), ++SOC_SINGLE("Aux Playback Phone Volume", AC97_CD, 4, 7, 1), ++ ++SOC_SINGLE("Phone Volume", AC97_PHONE, 0, 15, 0), ++SOC_DOUBLE("Line Capture Volume", AC97_LINE, 8, 0, 31, 1), ++ ++SOC_SINGLE("Capture 20dB Boost Switch", AC97_REC_SEL, 14, 1, 0), ++SOC_SINGLE("Capture to Phone 20dB Boost Switch", AC97_REC_SEL, 11, 1, 1), ++ ++SOC_SINGLE("3D Upper Cut-off Switch", AC97_3D_CONTROL, 5, 1, 1), ++SOC_SINGLE("3D Lower Cut-off Switch", AC97_3D_CONTROL, 4, 1, 1), ++SOC_SINGLE("3D Playback Volume", AC97_3D_CONTROL, 0, 15, 0), ++ ++SOC_ENUM("Bass Control", wm9712_enum[5]), ++SOC_SINGLE("Bass Cut-off Switch", AC97_MASTER_TONE, 12, 1, 1), ++SOC_SINGLE("Tone Cut-off Switch", AC97_MASTER_TONE, 4, 1, 1), ++SOC_SINGLE("Playback Attenuate (-6dB) Switch", AC97_MASTER_TONE, 6, 1, 0), ++SOC_SINGLE("Bass Volume", AC97_MASTER_TONE, 8, 15, 0), ++SOC_SINGLE("Treble Volume", AC97_MASTER_TONE, 0, 15, 0), ++ ++SOC_SINGLE("Capture ADC Switch", AC97_REC_GAIN, 15, 1, 1), ++SOC_ENUM("Capture Volume Steps", wm9712_enum[6]), ++SOC_DOUBLE("Capture Volume", AC97_REC_GAIN, 8, 0, 63, 1), ++SOC_SINGLE("Capture ZC Switch", AC97_REC_GAIN, 7, 1, 0), ++ ++SOC_SINGLE("Mic 1 Volume", AC97_MIC, 8, 31, 1), ++SOC_SINGLE("Mic 2 Volume", AC97_MIC, 0, 31, 1), ++SOC_SINGLE("Mic 20dB Boost Switch", AC97_MIC, 7, 1, 0), ++}; ++ ++/* add non dapm controls */ ++static int wm9712_add_controls(struct snd_soc_codec *codec) ++{ ++ int err, i; ++ ++ for (i = 0; i < ARRAY_SIZE(wm9712_snd_ac97_controls); i++) { ++ err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&wm9712_snd_ac97_controls[i],codec, NULL)); ++ if (err < 0) ++ return err; ++ } ++ return 0; ++} ++ ++/* We have to create a fake left and right HP mixers because ++ * the codec only has a single control that is shared by both channels. ++ * This makes it impossible to determine the audio path. ++ */ ++static int mixer_event (struct snd_soc_dapm_widget *w, int event) ++{ ++ u16 l, r, beep, line, phone, mic, pcm, aux; ++ ++ l = ac97_read(w->codec, HPL_MIXER); ++ r = ac97_read(w->codec, HPR_MIXER); ++ beep = ac97_read(w->codec, AC97_PC_BEEP); ++ mic = ac97_read(w->codec, AC97_VIDEO); ++ phone = ac97_read(w->codec, AC97_PHONE); ++ line = ac97_read(w->codec, AC97_LINE); ++ pcm = ac97_read(w->codec, AC97_PCM); ++ aux = ac97_read(w->codec, AC97_CD); ++ ++ if (l & 0x1 || r & 0x1) ++ ac97_write(w->codec, AC97_VIDEO, mic & 0x7fff); ++ else ++ ac97_write(w->codec, AC97_VIDEO, mic | 0x8000); ++ ++ if (l & 0x2 || r & 0x2) ++ ac97_write(w->codec, AC97_PCM, pcm & 0x7fff); ++ else ++ ac97_write(w->codec, AC97_PCM, pcm | 0x8000); ++ ++ if (l & 0x4 || r & 0x4) ++ ac97_write(w->codec, AC97_LINE, line & 0x7fff); ++ else ++ ac97_write(w->codec, AC97_LINE, line | 0x8000); ++ ++ if (l & 0x8 || r & 0x8) ++ ac97_write(w->codec, AC97_PHONE, phone & 0x7fff); ++ else ++ ac97_write(w->codec, AC97_PHONE, phone | 0x8000); ++ ++ if (l & 0x10 || r & 0x10) ++ ac97_write(w->codec, AC97_CD, aux & 0x7fff); ++ else ++ ac97_write(w->codec, AC97_CD, aux | 0x8000); ++ ++ if (l & 0x20 || r & 0x20) ++ ac97_write(w->codec, AC97_PC_BEEP, beep & 0x7fff); ++ else ++ ac97_write(w->codec, AC97_PC_BEEP, beep | 0x8000); ++ ++ return 0; ++} ++ ++/* Left Headphone Mixers */ ++static const struct snd_kcontrol_new wm9712_hpl_mixer_controls[] = { ++ SOC_DAPM_SINGLE("PCBeep Bypass Switch", HPL_MIXER, 5, 1, 0), ++ SOC_DAPM_SINGLE("Aux Playback Switch", HPL_MIXER, 4, 1, 0), ++ SOC_DAPM_SINGLE("Phone Bypass Switch", HPL_MIXER, 3, 1, 0), ++ SOC_DAPM_SINGLE("Line Bypass Switch", HPL_MIXER, 2, 1, 0), ++ SOC_DAPM_SINGLE("PCM Playback Switch", HPL_MIXER, 1, 1, 0), ++ SOC_DAPM_SINGLE("Mic Sidetone Switch", HPL_MIXER, 0, 1, 0), ++}; ++ ++/* Right Headphone Mixers */ ++static const struct snd_kcontrol_new wm9712_hpr_mixer_controls[] = { ++ SOC_DAPM_SINGLE("PCBeep Bypass Switch", HPR_MIXER, 5, 1, 0), ++ SOC_DAPM_SINGLE("Aux Playback Switch", HPR_MIXER, 4, 1, 0), ++ SOC_DAPM_SINGLE("Phone Bypass Switch", HPR_MIXER, 3, 1, 0), ++ SOC_DAPM_SINGLE("Line Bypass Switch", HPR_MIXER, 2, 1, 0), ++ SOC_DAPM_SINGLE("PCM Playback Switch", HPR_MIXER, 1, 1, 0), ++ SOC_DAPM_SINGLE("Mic Sidetone Switch", HPR_MIXER, 0, 1, 0), ++}; ++ ++/* Speaker Mixer */ ++static const struct snd_kcontrol_new wm9712_speaker_mixer_controls[] = { ++ SOC_DAPM_SINGLE("PCBeep Bypass Switch", AC97_PC_BEEP, 11, 1, 1), ++ SOC_DAPM_SINGLE("Aux Playback Switch", AC97_CD, 11, 1, 1), ++ SOC_DAPM_SINGLE("Phone Bypass Switch", AC97_PHONE, 14, 1, 1), ++ SOC_DAPM_SINGLE("Line Bypass Switch", AC97_LINE, 14, 1, 1), ++ SOC_DAPM_SINGLE("PCM Playback Switch", AC97_PCM, 14, 1, 1), ++}; ++ ++/* Phone Mixer */ ++static const struct snd_kcontrol_new wm9712_phone_mixer_controls[] = { ++ SOC_DAPM_SINGLE("PCBeep Bypass Switch", AC97_PC_BEEP, 7, 1, 1), ++ SOC_DAPM_SINGLE("Aux Playback Switch", AC97_CD, 7, 1, 1), ++ SOC_DAPM_SINGLE("Line Bypass Switch", AC97_LINE, 13, 1, 1), ++ SOC_DAPM_SINGLE("PCM Playback Switch", AC97_PCM, 13, 1, 1), ++ SOC_DAPM_SINGLE("Mic 1 Sidetone Switch", AC97_MIC, 14, 1, 1), ++ SOC_DAPM_SINGLE("Mic 2 Sidetone Switch", AC97_MIC, 13, 1, 1), ++}; ++ ++/* ALC headphone mux */ ++static const struct snd_kcontrol_new wm9712_alc_mux_controls = ++SOC_DAPM_ENUM("Route", wm9712_enum[1]); ++ ++/* out 3 mux */ ++static const struct snd_kcontrol_new wm9712_out3_mux_controls = ++SOC_DAPM_ENUM("Route", wm9712_enum[2]); ++ ++/* spk mux */ ++static const struct snd_kcontrol_new wm9712_spk_mux_controls = ++SOC_DAPM_ENUM("Route", wm9712_enum[3]); ++ ++/* Capture to Phone mux */ ++static const struct snd_kcontrol_new wm9712_capture_phone_mux_controls = ++SOC_DAPM_ENUM("Route", wm9712_enum[4]); ++ ++/* Capture left select */ ++static const struct snd_kcontrol_new wm9712_capture_selectl_controls = ++SOC_DAPM_ENUM("Route", wm9712_enum[8]); ++ ++/* Capture right select */ ++static const struct snd_kcontrol_new wm9712_capture_selectr_controls = ++SOC_DAPM_ENUM("Route", wm9712_enum[9]); ++ ++/* Mic select */ ++static const struct snd_kcontrol_new wm9712_mic_src_controls = ++SOC_DAPM_ENUM("Route", wm9712_enum[7]); ++ ++/* diff select */ ++static const struct snd_kcontrol_new wm9712_diff_sel_controls = ++SOC_DAPM_ENUM("Route", wm9712_enum[11]); ++ ++static const struct snd_soc_dapm_widget wm9712_dapm_widgets[] = { ++SND_SOC_DAPM_MUX("ALC Sidetone Mux", SND_SOC_NOPM, 0, 0, ++ &wm9712_alc_mux_controls), ++SND_SOC_DAPM_MUX("Out3 Mux", SND_SOC_NOPM, 0, 0, ++ &wm9712_out3_mux_controls), ++SND_SOC_DAPM_MUX("Speaker Mux", SND_SOC_NOPM, 0, 0, ++ &wm9712_spk_mux_controls), ++SND_SOC_DAPM_MUX("Capture Phone Mux", SND_SOC_NOPM, 0, 0, ++ &wm9712_capture_phone_mux_controls), ++SND_SOC_DAPM_MUX("Left Capture Select", SND_SOC_NOPM, 0, 0, ++ &wm9712_capture_selectl_controls), ++SND_SOC_DAPM_MUX("Right Capture Select", SND_SOC_NOPM, 0, 0, ++ &wm9712_capture_selectr_controls), ++SND_SOC_DAPM_MUX("Mic Select Source", SND_SOC_NOPM, 0, 0, ++ &wm9712_mic_src_controls), ++SND_SOC_DAPM_MUX("Differential Source", SND_SOC_NOPM, 0, 0, ++ &wm9712_diff_sel_controls), ++SND_SOC_DAPM_MIXER("AC97 Mixer", SND_SOC_NOPM, 0, 0, NULL, 0), ++SND_SOC_DAPM_MIXER_E("Left HP Mixer", AC97_INT_PAGING, 9, 1, ++ &wm9712_hpl_mixer_controls[0], ARRAY_SIZE(wm9712_hpl_mixer_controls), ++ mixer_event, SND_SOC_DAPM_POST_REG), ++SND_SOC_DAPM_MIXER_E("Right HP Mixer", AC97_INT_PAGING, 8, 1, ++ &wm9712_hpr_mixer_controls[0], ARRAY_SIZE(wm9712_hpr_mixer_controls), ++ mixer_event, SND_SOC_DAPM_POST_REG), ++SND_SOC_DAPM_MIXER("Phone Mixer", AC97_INT_PAGING, 6, 1, ++ &wm9712_phone_mixer_controls[0], ARRAY_SIZE(wm9712_phone_mixer_controls)), ++SND_SOC_DAPM_MIXER("Speaker Mixer", AC97_INT_PAGING, 7, 1, ++ &wm9712_speaker_mixer_controls[0], ++ ARRAY_SIZE(wm9712_speaker_mixer_controls)), ++SND_SOC_DAPM_MIXER("Mono Mixer", SND_SOC_NOPM, 0, 0, NULL, 0), ++SND_SOC_DAPM_DAC("Left DAC", "Left HiFi Playback", AC97_INT_PAGING, 14, 1), ++SND_SOC_DAPM_DAC("Right DAC", "Right HiFi Playback", AC97_INT_PAGING, 13, 1), ++SND_SOC_DAPM_DAC("Aux DAC", "Aux Playback", SND_SOC_NOPM, 0, 0), ++SND_SOC_DAPM_ADC("Left ADC", "Left HiFi Capture", AC97_INT_PAGING, 12, 1), ++SND_SOC_DAPM_ADC("Right ADC", "Right HiFi Capture", AC97_INT_PAGING, 11, 1), ++SND_SOC_DAPM_PGA("Headphone PGA", AC97_INT_PAGING, 4, 1, NULL, 0), ++SND_SOC_DAPM_PGA("Speaker PGA", AC97_INT_PAGING, 3, 1, NULL, 0), ++SND_SOC_DAPM_PGA("Out 3 PGA", AC97_INT_PAGING, 5, 1, NULL, 0), ++SND_SOC_DAPM_PGA("Line PGA", AC97_INT_PAGING, 2, 1, NULL, 0), ++SND_SOC_DAPM_PGA("Phone PGA", AC97_INT_PAGING, 1, 1, NULL, 0), ++SND_SOC_DAPM_PGA("Mic PGA", AC97_INT_PAGING, 0, 1, NULL, 0), ++SND_SOC_DAPM_MICBIAS("Mic Bias", AC97_INT_PAGING, 10, 1), ++SND_SOC_DAPM_OUTPUT("MONOOUT"), ++SND_SOC_DAPM_OUTPUT("HPOUTL"), ++SND_SOC_DAPM_OUTPUT("HPOUTR"), ++SND_SOC_DAPM_OUTPUT("LOUT2"), ++SND_SOC_DAPM_OUTPUT("ROUT2"), ++SND_SOC_DAPM_OUTPUT("OUT3"), ++SND_SOC_DAPM_INPUT("LINEINL"), ++SND_SOC_DAPM_INPUT("LINEINR"), ++SND_SOC_DAPM_INPUT("PHONE"), ++SND_SOC_DAPM_INPUT("PCBEEP"), ++SND_SOC_DAPM_INPUT("MIC1"), ++SND_SOC_DAPM_INPUT("MIC2"), ++}; ++ ++static const char *audio_map[][3] = { ++ /* virtual mixer - mixes left & right channels for spk and mono */ ++ {"AC97 Mixer", NULL, "Left DAC"}, ++ {"AC97 Mixer", NULL, "Right DAC"}, ++ ++ /* Left HP mixer */ ++ {"Left HP Mixer", "PCBeep Bypass Switch", "PCBEEP"}, ++ {"Left HP Mixer", "Aux Playback Switch", "Aux DAC"}, ++ {"Left HP Mixer", "Phone Bypass Switch", "Phone PGA"}, ++ {"Left HP Mixer", "Line Bypass Switch", "Line PGA"}, ++ {"Left HP Mixer", "PCM Playback Switch", "Left DAC"}, ++ {"Left HP Mixer", "Mic Sidetone Switch", "Mic PGA"}, ++ {"Left HP Mixer", NULL, "ALC Sidetone Mux"}, ++ //{"Right HP Mixer", NULL, "HP Mixer"}, ++ ++ /* Right HP mixer */ ++ {"Right HP Mixer", "PCBeep Bypass Switch", "PCBEEP"}, ++ {"Right HP Mixer", "Aux Playback Switch", "Aux DAC"}, ++ {"Right HP Mixer", "Phone Bypass Switch", "Phone PGA"}, ++ {"Right HP Mixer", "Line Bypass Switch", "Line PGA"}, ++ {"Right HP Mixer", "PCM Playback Switch", "Right DAC"}, ++ {"Right HP Mixer", "Mic Sidetone Switch", "Mic PGA"}, ++ {"Right HP Mixer", NULL, "ALC Sidetone Mux"}, ++ ++ /* speaker mixer */ ++ {"Speaker Mixer", "PCBeep Bypass Switch", "PCBEEP"}, ++ {"Speaker Mixer", "Line Bypass Switch", "Line PGA"}, ++ {"Speaker Mixer", "PCM Playback Switch", "AC97 Mixer"}, ++ {"Speaker Mixer", "Phone Bypass Switch", "Phone PGA"}, ++ {"Speaker Mixer", "Aux Playback Switch", "Aux DAC"}, ++ ++ /* Phone mixer */ ++ {"Phone Mixer", "PCBeep Bypass Switch", "PCBEEP"}, ++ {"Phone Mixer", "Line Bypass Switch", "Line PGA"}, ++ {"Phone Mixer", "Aux Playback Switch", "Aux DAC"}, ++ {"Phone Mixer", "PCM Playback Switch", "AC97 Mixer"}, ++ {"Phone Mixer", "Mic 1 Sidetone Switch", "Mic PGA"}, ++ {"Phone Mixer", "Mic 2 Sidetone Switch", "Mic PGA"}, ++ ++ /* inputs */ ++ {"Line PGA", NULL, "LINEINL"}, ++ {"Line PGA", NULL, "LINEINR"}, ++ {"Phone PGA", NULL, "PHONE"}, ++ {"Mic PGA", NULL, "MIC1"}, ++ {"Mic PGA", NULL, "MIC2"}, ++ ++ /* left capture selector */ ++ {"Left Capture Select", "Mic", "MIC1"}, ++ {"Left Capture Select", "Speaker Mixer", "Speaker Mixer"}, ++ {"Left Capture Select", "Line", "LINEINL"}, ++ {"Left Capture Select", "Headphone Mixer", "Left HP Mixer"}, ++ {"Left Capture Select", "Phone Mixer", "Phone Mixer"}, ++ {"Left Capture Select", "Phone", "PHONE"}, ++ ++ /* right capture selector */ ++ {"Right Capture Select", "Mic", "MIC2"}, ++ {"Right Capture Select", "Speaker Mixer", "Speaker Mixer"}, ++ {"Right Capture Select", "Line", "LINEINR"}, ++ {"Right Capture Select", "Headphone Mixer", "Right HP Mixer"}, ++ {"Right Capture Select", "Phone Mixer", "Phone Mixer"}, ++ {"Right Capture Select", "Phone", "PHONE"}, ++ ++ /* ALC Sidetone */ ++ {"ALC Sidetone Mux", "Stereo", "Left Capture Select"}, ++ {"ALC Sidetone Mux", "Stereo", "Right Capture Select"}, ++ {"ALC Sidetone Mux", "Left", "Left Capture Select"}, ++ {"ALC Sidetone Mux", "Right", "Right Capture Select"}, ++ ++ /* ADC's */ ++ {"Left ADC", NULL, "Left Capture Select"}, ++ {"Right ADC", NULL, "Right Capture Select"}, ++ ++ /* outputs */ ++ {"MONOOUT", NULL, "Phone Mixer"}, ++ {"HPOUTL", NULL, "Headphone PGA"}, ++ {"Headphone PGA", NULL, "Left HP Mixer"}, ++ {"HPOUTR", NULL, "Headphone PGA"}, ++ {"Headphone PGA", NULL, "Right HP Mixer"}, ++ ++ /* mono hp mixer */ ++ {"Mono HP Mixer", NULL, "Left HP Mixer"}, ++ {"Mono HP Mixer", NULL, "Right HP Mixer"}, ++ ++ /* Out3 Mux */ ++ {"Out3 Mux", "Left", "Left HP Mixer"}, ++ {"Out3 Mux", "Mono", "Phone Mixer"}, ++ {"Out3 Mux", "Left + Right", "Mono HP Mixer"}, ++ {"Out 3 PGA", NULL, "Out3 Mux"}, ++ {"OUT3", NULL, "Out 3 PGA"}, ++ ++ /* speaker Mux */ ++ {"Speaker Mux", "Speaker Mix", "Speaker Mixer"}, ++ {"Speaker Mux", "Headphone Mix", "Mono HP Mixer"}, ++ {"Speaker PGA", NULL, "Speaker Mux"}, ++ {"LOUT2", NULL, "Speaker PGA"}, ++ {"ROUT2", NULL, "Speaker PGA"}, ++ ++ {NULL, NULL, NULL}, ++}; ++ ++static int wm9712_add_widgets(struct snd_soc_codec *codec) ++{ ++ int i; ++ ++ for(i = 0; i < ARRAY_SIZE(wm9712_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &wm9712_dapm_widgets[i]); ++ } ++ ++ /* set up audio path audio_mapnects */ ++ for(i = 0; audio_map[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, audio_map[i][0], ++ audio_map[i][1], audio_map[i][2]); ++ } ++ ++ snd_soc_dapm_new_widgets(codec); ++ return 0; ++} ++ ++static unsigned int ac97_read(struct snd_soc_codec *codec, ++ unsigned int reg) ++{ ++ u16 *cache = codec->reg_cache; ++ ++ if (reg == AC97_RESET || reg == AC97_GPIO_STATUS || ++ reg == AC97_VENDOR_ID1 || reg == AC97_VENDOR_ID2 || ++ reg == AC97_REC_GAIN) ++ return soc_ac97_ops.read(codec->ac97, reg); ++ else { ++ reg = reg >> 1; ++ ++ if (reg > (ARRAY_SIZE(wm9712_reg))) ++ return -EIO; ++ ++ return cache[reg]; ++ } ++} ++ ++static int ac97_write(struct snd_soc_codec *codec, unsigned int reg, ++ unsigned int val) ++{ ++ u16 *cache = codec->reg_cache; ++ ++ soc_ac97_ops.write(codec->ac97, reg, val); ++ reg = reg >> 1; ++ if (reg <= (ARRAY_SIZE(wm9712_reg))) ++ cache[reg] = val; ++ ++ return 0; ++} ++ ++static int ac97_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ int reg; ++ u16 vra; ++ ++ vra = ac97_read(codec, AC97_EXTENDED_STATUS); ++ ac97_write(codec, AC97_EXTENDED_STATUS, vra | 0x1); ++ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ reg = AC97_PCM_FRONT_DAC_RATE; ++ else ++ reg = AC97_PCM_LR_ADC_RATE; ++ ++ return ac97_write(codec, reg, runtime->rate); ++} ++ ++static int ac97_aux_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 vra, xsle; ++ ++ vra = ac97_read(codec, AC97_EXTENDED_STATUS); ++ ac97_write(codec, AC97_EXTENDED_STATUS, vra | 0x1); ++ xsle = ac97_read(codec, AC97_PCI_SID); ++ ac97_write(codec, AC97_PCI_SID, xsle | 0x8000); ++ ++ if (substream->stream != SNDRV_PCM_STREAM_PLAYBACK) ++ return -ENODEV; ++ ++ return ac97_write(codec, AC97_PCM_SURR_DAC_RATE, runtime->rate); ++} ++ ++struct snd_soc_codec_dai wm9712_dai[] = { ++{ ++ .name = "AC97 HiFi", ++ .playback = { ++ .stream_name = "HiFi Playback", ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .capture = { ++ .stream_name = "HiFi Capture", ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .ops = { ++ .prepare = ac97_prepare,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(ac97_modes), ++ .mode = ac97_modes,}, ++ }, ++ { ++ .name = "AC97 Aux", ++ .playback = { ++ .stream_name = "Aux Playback", ++ .channels_min = 1, ++ .channels_max = 1,}, ++ .ops = { ++ .prepare = ac97_aux_prepare,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(ac97_modes), ++ .mode = ac97_modes,}, ++ }, ++}; ++EXPORT_SYMBOL_GPL(wm9712_dai); ++ ++static int wm9712_dapm_event(struct snd_soc_codec *codec, int event) ++{ ++ u16 reg; ++ ++ switch (event) { ++ case SNDRV_CTL_POWER_D0: /* full On */ ++ /* liam - maybe enable thermal shutdown */ ++ reg = ac97_read(codec, AC97_EXTENDED_MID) & 0xdfff; ++ ac97_write(codec, AC97_EXTENDED_MID, reg); ++ break; ++ case SNDRV_CTL_POWER_D1: /* partial On */ ++ case SNDRV_CTL_POWER_D2: /* partial On */ ++ break; ++ case SNDRV_CTL_POWER_D3hot: /* Off, with power */ ++ /* enable master bias and vmid */ ++ reg = ac97_read(codec, AC97_EXTENDED_MID) & 0xbbff; ++ ac97_write(codec, AC97_EXTENDED_MID, reg); ++ ac97_write(codec, AC97_POWERDOWN, 0x0000); ++ break; ++ case SNDRV_CTL_POWER_D3cold: /* Off, without power */ ++ /* disable everything including AC link */ ++ ac97_write(codec, AC97_EXTENDED_MID, 0xffff); ++ ac97_write(codec, AC97_EXTENDED_MSTATUS, 0xffff); ++ ac97_write(codec, AC97_POWERDOWN, 0xffff); ++ break; ++ } ++ codec->dapm_state = event; ++ return 0; ++} ++ ++static int wm9712_reset(struct snd_soc_codec *codec, int try_warm) ++{ ++ if (try_warm && soc_ac97_ops.warm_reset) { ++ soc_ac97_ops.warm_reset(codec->ac97); ++ if (!(ac97_read(codec, 0) & 0x8000)) ++ return 1; ++ } ++ ++ soc_ac97_ops.reset(codec->ac97); ++ if (ac97_read(codec, 0) & 0x8000) ++ goto err; ++ return 0; ++ ++err: ++ printk(KERN_ERR "WM9712 AC97 reset failed\n"); ++ return -EIO; ++} ++ ++static int wm9712_soc_suspend(struct platform_device *pdev, ++ pm_message_t state) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ wm9712_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ return 0; ++} ++ ++static int wm9712_soc_resume(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ int i, ret; ++ u16 *cache = codec->reg_cache; ++ ++ ret = wm9712_reset(codec, 1); ++ if (ret < 0){ ++ printk(KERN_ERR "could not reset AC97 codec\n"); ++ return ret; ++ } ++ ++ wm9712_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ ++ if (ret == 0) { ++ /* Sync reg_cache with the hardware after cold reset */ ++ for (i = 2; i < ARRAY_SIZE(wm9712_reg) << 1; i+=2) { ++ if (i == AC97_INT_PAGING || i == AC97_POWERDOWN || ++ (i > 0x58 && i != 0x5c)) ++ continue; ++ soc_ac97_ops.write(codec->ac97, i, cache[i>>1]); ++ } ++ } ++ ++ if (codec->suspend_dapm_state == SNDRV_CTL_POWER_D0) ++ wm9712_dapm_event(codec, SNDRV_CTL_POWER_D0); ++ ++ return ret; ++} ++ ++static int wm9712_soc_probe(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec; ++ int ret = 0; ++ ++ printk(KERN_INFO "WM9711/WM9712 SoC Audio Codec %s\n", WM9712_VERSION); ++ ++ socdev->codec = kzalloc(sizeof(struct snd_soc_codec), GFP_KERNEL); ++ if (socdev->codec == NULL) ++ return -ENOMEM; ++ codec = socdev->codec; ++ mutex_init(&codec->mutex); ++ ++ codec->reg_cache = ++ kzalloc(sizeof(u16) * ARRAY_SIZE(wm9712_reg), GFP_KERNEL); ++ if (codec->reg_cache == NULL) { ++ kfree(codec->ac97); ++ kfree(socdev->codec); ++ socdev->codec = NULL; ++ return -ENOMEM; ++ } ++ memcpy(codec->reg_cache, wm9712_reg, sizeof(u16) * ARRAY_SIZE(wm9712_reg)); ++ codec->reg_cache_size = sizeof(u16) * ARRAY_SIZE(wm9712_reg); ++ codec->reg_cache_step = 2; ++ ++ codec->name = "WM9712"; ++ codec->owner = THIS_MODULE; ++ codec->dai = wm9712_dai; ++ codec->num_dai = ARRAY_SIZE(wm9712_dai); ++ codec->write = ac97_write; ++ codec->read = ac97_read; ++ codec->dapm_event = wm9712_dapm_event; ++ INIT_LIST_HEAD(&codec->dapm_widgets); ++ INIT_LIST_HEAD(&codec->dapm_paths); ++ ++ ret = snd_soc_new_ac97_codec(codec, &soc_ac97_ops, 0); ++ if (ret < 0) ++ goto err; ++ ++ /* register pcms */ ++ ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); ++ if (ret < 0) ++ goto pcm_err; ++ ++ ret = wm9712_reset(codec, 0); ++ if (ret < 0) { ++ printk(KERN_ERR "AC97 link error\n"); ++ goto reset_err; ++ } ++ ++ /* set alc mux to none */ ++ ac97_write(codec, AC97_VIDEO, ac97_read(codec, AC97_VIDEO) | 0x3000); ++ ++ wm9712_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ wm9712_add_controls(codec); ++ wm9712_add_widgets(codec); ++ ret = snd_soc_register_card(socdev); ++ if (ret < 0) ++ goto reset_err; ++ ++ return 0; ++ ++reset_err: ++ snd_soc_free_pcms(socdev); ++ ++pcm_err: ++ snd_soc_free_ac97_codec(codec); ++ ++err: ++ kfree(socdev->codec->reg_cache); ++ kfree(socdev->codec); ++ socdev->codec = NULL; ++ return ret; ++} ++ ++static int wm9712_soc_remove(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ if (codec == NULL) ++ return 0; ++ ++ snd_soc_dapm_free(socdev); ++ snd_soc_free_pcms(socdev); ++ snd_soc_free_ac97_codec(codec); ++ kfree(codec->reg_cache); ++ kfree(codec); ++ return 0; ++} ++ ++struct snd_soc_codec_device soc_codec_dev_wm9712 = { ++ .probe = wm9712_soc_probe, ++ .remove = wm9712_soc_remove, ++ .suspend = wm9712_soc_suspend, ++ .resume = wm9712_soc_resume, ++}; ++ ++EXPORT_SYMBOL_GPL(soc_codec_dev_wm9712); ++ ++MODULE_DESCRIPTION("ASoC WM9711/WM9712 driver"); ++MODULE_AUTHOR("Liam Girdwood"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/codecs/wm9712.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm9712.h +@@ -0,0 +1,14 @@ ++/* ++ * wm9712.h -- WM9712 Soc Audio driver ++ */ ++ ++#ifndef _WM9712_H ++#define _WM9712_H ++ ++#define WM9712_DAI_AC97_HIFI 0 ++#define WM9712_DAI_AC97_AUX 1 ++ ++extern struct snd_soc_codec_dai wm9712_dai[2]; ++extern struct snd_soc_codec_device soc_codec_dev_wm9712; ++ ++#endif +Index: linux-2.6-pxa-new/sound/soc/codecs/wm9713.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm9713.c +@@ -0,0 +1,1313 @@ ++/* ++ * wm9713.c -- ALSA Soc WM9713 codec support ++ * ++ * Copyright 2006 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * 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. ++ * ++ * Revision history ++ * 4th Feb 2006 Initial version. ++ * ++ * Features:- ++ * ++ * o Support for AC97 Codec, Voice DAC and Aux DAC ++ * o Support for DAPM ++ */ ++ ++#include <linux/init.h> ++#include <linux/module.h> ++#include <linux/device.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/ac97_codec.h> ++#include <sound/initval.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++ ++#define WM9713_VERSION "0.12" ++ ++struct wm9713 { ++ u32 pll; /* current PLL frequency */ ++ u32 pll_resume; /* PLL resume frequency */ ++}; ++ ++static unsigned int ac97_read(struct snd_soc_codec *codec, ++ unsigned int reg); ++static int ac97_write(struct snd_soc_codec *codec, ++ unsigned int reg, unsigned int val); ++ ++#define AC97_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define AC97_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | \ ++ SNDRV_PCM_RATE_16000 | SNDRV_PCM_RATE_22050 | \ ++ SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000) ++ ++/* may need to expand this */ ++static struct snd_soc_dai_mode ac97_modes[] = { ++ { ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S18_3LE, ++ .pcmrate = AC97_RATES, ++ }, ++}; ++ ++#define WM9713_VOICE_DAIFMT \ ++ (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_LEFT_J | \ ++ SND_SOC_DAIFMT_RIGHT_J | SND_SOC_DAIFMT_DSP_A | \ ++ SND_SOC_DAIFMT_DSP_B | SND_SOC_DAIFMT_NB_NF | \ ++ SND_SOC_DAIFMT_NB_IF | SND_SOC_DAIFMT_IB_NF | \ ++ SND_SOC_DAIFMT_IB_IF) ++ ++#define WM9713_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define WM9713_VOICE_FSB \ ++ (SND_SOC_FSBD(1) | SND_SOC_FSBD(2) | SND_SOC_FSBD(4) | \ ++ SND_SOC_FSBD(8) | SND_SOC_FSBD(16)) ++ ++#define WM9713_VOICE_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_48000 | \ ++ SNDRV_PCM_RATE_96000) ++ ++#define WM9713_HIFI_BITS \ ++ (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE | \ ++ SNDRV_PCM_FMTBIT_S24_LE | SNDRV_PCM_FMTBIT_S32_LE) ++ ++/* ++ * Voice modes ++ */ ++static struct snd_soc_dai_mode wm9713_voice_modes[] = { ++ /* master modes */ ++ { ++ .fmt = WM9713_VOICE_DAIFMT | SND_SOC_DAIFMT_CBM_CFM | \ ++ SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = WM9713_HIFI_BITS, ++ .pcmrate = WM9713_VOICE_RATES, ++ .pcmdir = WM9713_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = WM9713_VOICE_FSB, ++ }, ++ ++ /* slave modes */ ++ { ++ .fmt = WM9713_VOICE_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = WM9713_HIFI_BITS, ++ .pcmrate = WM9713_VOICE_RATES, ++ .pcmdir = WM9713_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = SND_SOC_FSB_ALL, ++ }, ++}; ++ ++/* ++ * WM9713 register cache ++ * Reg 0x3c bit 15 is used by touch driver. ++ */ ++static const u16 wm9713_reg[] = { ++ 0x6174, 0x8080, 0x8080, 0x8080, // 6 ++ 0xc880, 0xe808, 0xe808, 0x0808, // e ++ 0x00da, 0x8000, 0xd600, 0xaaa0, // 16 ++ 0xaaa0, 0xaaa0, 0x0000, 0x0000, // 1e ++ 0x0f0f, 0x0040, 0x0000, 0x7f00, // 26 ++ 0x0405, 0x0410, 0xbb80, 0xbb80, // 2e ++ 0x0000, 0xbb80, 0x0000, 0x4523, // 36 ++ 0x0000, 0x2000, 0x7eff, 0xffff, // 3e ++ 0x0000, 0x0000, 0x0080, 0x0000, // 46 ++ 0x0000, 0x0000, 0xfffe, 0xffff, // 4e ++ 0x0000, 0x0000, 0x0000, 0xfffe, // 56 ++ 0x4000, 0x0000, 0x0000, 0x0000, // 5e ++ 0xb032, 0x3e00, 0x0000, 0x0000, // 66 ++ 0x0000, 0x0000, 0x0000, 0x0000, // 6e ++ 0x0000, 0x0000, 0x0000, 0x0006, // 76 ++ 0x0001, 0x0000, 0x574d, 0x4c13, // 7e ++ 0x0000, 0x0000, 0x0000 // virtual hp & mic mixers ++}; ++ ++/* virtual HP mixers regs */ ++#define HPL_MIXER 0x80 ++#define HPR_MIXER 0x82 ++#define MICB_MUX 0x82 ++ ++static const char *wm9713_mic_mixer[] = {"Stereo", "Mic 1", "Mic 2", "Mute"}; ++static const char *wm9713_rec_mux[] = {"Stereo", "Left", "Right", "Mute"}; ++static const char *wm9713_rec_src[] = ++ {"Mic 1", "Mic 2", "Line", "Mono In", "Headphone", "Speaker", ++ "Mono Out", "Zh"}; ++static const char *wm9713_rec_gain[] = {"+1.5dB Steps", "+0.75dB Steps"}; ++static const char *wm9713_alc_select[] = {"None", "Left", "Right", "Stereo"}; ++static const char *wm9713_mono_pga[] = {"Vmid", "Zh", "Mono", "Inv", ++ "Mono Vmid", "Inv Vmid"}; ++static const char *wm9713_spk_pga[] = ++ {"Vmid", "Zh", "Headphone", "Speaker", "Inv", "Headphone Vmid", ++ "Speaker Vmid", "Inv Vmid"}; ++static const char *wm9713_hp_pga[] = {"Vmid", "Zh", "Headphone", ++ "Headphone Vmid"}; ++static const char *wm9713_out3_pga[] = {"Vmid", "Zh", "Inv 1", "Inv 1 Vmid"}; ++static const char *wm9713_out4_pga[] = {"Vmid", "Zh", "Inv 2", "Inv 2 Vmid"}; ++static const char *wm9713_dac_inv[] = ++ {"Off", "Mono", "Speaker", "Left Headphone", "Right Headphone", ++ "Headphone Mono", "NC", "Vmid"}; ++static const char *wm9713_bass[] = {"Linear Control", "Adaptive Boost"}; ++static const char *wm9713_ng_type[] = {"Constant Gain", "Mute"}; ++static const char *wm9713_mic_select[] = {"Mic 1", "Mic 2 A", "Mic 2 B"}; ++static const char *wm9713_micb_select[] = {"MPB", "MPA"}; ++ ++static const struct soc_enum wm9713_enum[] = { ++SOC_ENUM_SINGLE(AC97_LINE, 3, 4, wm9713_mic_mixer), /* record mic mixer 0 */ ++SOC_ENUM_SINGLE(AC97_VIDEO, 14, 4, wm9713_rec_mux), /* record mux hp 1 */ ++SOC_ENUM_SINGLE(AC97_VIDEO, 9, 4, wm9713_rec_mux), /* record mux mono 2 */ ++SOC_ENUM_SINGLE(AC97_VIDEO, 3, 8, wm9713_rec_src), /* record mux left 3 */ ++SOC_ENUM_SINGLE(AC97_VIDEO, 0, 8, wm9713_rec_src), /* record mux right 4*/ ++SOC_ENUM_DOUBLE(AC97_CD, 14, 6, 2, wm9713_rec_gain), /* record step size 5 */ ++SOC_ENUM_SINGLE(AC97_PCI_SVID, 14, 4, wm9713_alc_select), /* alc source select 6*/ ++SOC_ENUM_SINGLE(AC97_REC_GAIN, 14, 4, wm9713_mono_pga), /* mono input select 7 */ ++SOC_ENUM_SINGLE(AC97_REC_GAIN, 11, 8, wm9713_spk_pga), /* speaker left input select 8 */ ++SOC_ENUM_SINGLE(AC97_REC_GAIN, 8, 8, wm9713_spk_pga), /* speaker right input select 9 */ ++SOC_ENUM_SINGLE(AC97_REC_GAIN, 6, 3, wm9713_hp_pga), /* headphone left input 10 */ ++SOC_ENUM_SINGLE(AC97_REC_GAIN, 4, 3, wm9713_hp_pga), /* headphone right input 11 */ ++SOC_ENUM_SINGLE(AC97_REC_GAIN, 2, 4, wm9713_out3_pga), /* out 3 source 12 */ ++SOC_ENUM_SINGLE(AC97_REC_GAIN, 0, 4, wm9713_out4_pga), /* out 4 source 13 */ ++SOC_ENUM_SINGLE(AC97_REC_GAIN_MIC, 13, 8, wm9713_dac_inv), /* dac invert 1 14 */ ++SOC_ENUM_SINGLE(AC97_REC_GAIN_MIC, 10, 8, wm9713_dac_inv), /* dac invert 2 15 */ ++SOC_ENUM_SINGLE(AC97_GENERAL_PURPOSE, 15, 2, wm9713_bass), /* bass control 16 */ ++SOC_ENUM_SINGLE(AC97_PCI_SVID, 5, 2, wm9713_ng_type), /* noise gate type 17 */ ++SOC_ENUM_SINGLE(AC97_3D_CONTROL, 12, 3, wm9713_mic_select), /* mic selection 18 */ ++SOC_ENUM_SINGLE(MICB_MUX, 0, 2, wm9713_micb_select), /* mic selection 19 */ ++}; ++ ++static const struct snd_kcontrol_new wm9713_snd_ac97_controls[] = { ++SOC_DOUBLE("Speaker Playback Volume", AC97_MASTER, 8, 0, 31, 1), ++SOC_DOUBLE("Speaker Playback Switch", AC97_MASTER, 15, 7, 1, 1), ++SOC_DOUBLE("Headphone Playback Volume", AC97_HEADPHONE, 8, 0, 31, 1), ++SOC_DOUBLE("Headphone Playback Switch", AC97_HEADPHONE,15, 7, 1, 1), ++SOC_DOUBLE("Line In Volume", AC97_PC_BEEP, 8, 0, 31, 1), ++SOC_DOUBLE("PCM Playback Volume", AC97_PHONE, 8, 0, 31, 1), ++SOC_SINGLE("Mic 1 Volume", AC97_MIC, 8, 31, 1), ++SOC_SINGLE("Mic 2 Volume", AC97_MIC, 0, 31, 1), ++ ++SOC_SINGLE("Mic Boost (+20dB) Switch", AC97_LINE, 5, 1, 0), ++SOC_SINGLE("Mic Headphone Mixer Volume", AC97_LINE, 0, 7, 1), ++ ++SOC_SINGLE("Capture Switch", AC97_CD, 15, 1, 1), ++SOC_ENUM("Capture Volume Steps", wm9713_enum[5]), ++SOC_DOUBLE("Capture Volume", AC97_CD, 8, 0, 63, 0), ++SOC_SINGLE("Capture ZC Switch", AC97_CD, 7, 1, 0), ++ ++SOC_SINGLE("Capture to Headphone Volume", AC97_VIDEO, 11, 7, 1), ++SOC_SINGLE("Capture to Mono Boost (+20dB) Switch", AC97_VIDEO, 8, 1, 0), ++SOC_SINGLE("Capture ADC Boost (+20dB) Switch", AC97_VIDEO, 6, 1, 0), ++ ++SOC_SINGLE("ALC Target Volume", AC97_CODEC_CLASS_REV, 12, 15, 0), ++SOC_SINGLE("ALC Hold Time", AC97_CODEC_CLASS_REV, 8, 15, 0), ++SOC_SINGLE("ALC Decay Time ", AC97_CODEC_CLASS_REV, 4, 15, 0), ++SOC_SINGLE("ALC Attack Time", AC97_CODEC_CLASS_REV, 0, 15, 0), ++SOC_ENUM("ALC Function", wm9713_enum[6]), ++SOC_SINGLE("ALC Max Volume", AC97_PCI_SVID, 11, 7, 0), ++SOC_SINGLE("ALC ZC Timeout", AC97_PCI_SVID, 9, 3, 0), ++SOC_SINGLE("ALC ZC Switch", AC97_PCI_SVID, 8, 1, 0), ++SOC_SINGLE("ALC NG Switch", AC97_PCI_SVID, 7, 1, 0), ++SOC_ENUM("ALC NG Type", wm9713_enum[17]), ++SOC_SINGLE("ALC NG Threshold", AC97_PCI_SVID, 0, 31, 0), ++ ++SOC_DOUBLE("Speaker Playback ZC Switch", AC97_MASTER, 14, 6, 1, 0), ++SOC_DOUBLE("Headphone Playback ZC Switch", AC97_HEADPHONE, 14, 6, 1, 0), ++ ++SOC_SINGLE("Out4 Playback Switch", AC97_MASTER_MONO, 15, 1, 1), ++SOC_SINGLE("Out4 Playback ZC Switch", AC97_MASTER_MONO, 14, 1, 0), ++SOC_SINGLE("Out4 Playback Volume", AC97_MASTER_MONO, 8, 63, 1), ++ ++SOC_SINGLE("Out3 Playback Switch", AC97_MASTER_MONO, 7, 1, 1), ++SOC_SINGLE("Out3 Playback ZC Switch", AC97_MASTER_MONO, 6, 1, 0), ++SOC_SINGLE("Out3 Playback Volume", AC97_MASTER_MONO, 0, 63, 1), ++ ++SOC_SINGLE("Mono Capture Volume", AC97_MASTER_TONE, 8, 31, 1), ++SOC_SINGLE("Mono Playback Switch", AC97_MASTER_TONE, 7, 1, 1), ++SOC_SINGLE("Mono Playback ZC Switch", AC97_MASTER_TONE, 6, 1, 0), ++SOC_SINGLE("Mono Playback Volume", AC97_MASTER_TONE, 0, 31, 1), ++ ++SOC_SINGLE("PC Beep Playback Headphone Volume", AC97_AUX, 12, 7, 1), ++SOC_SINGLE("PC Beep Playback Speaker Volume", AC97_AUX, 8, 7, 1), ++SOC_SINGLE("PC Beep Playback Mono Volume", AC97_AUX, 4, 7, 1), ++ ++SOC_SINGLE("Voice Playback Headphone Volume", AC97_PCM, 12, 7, 1), ++SOC_SINGLE("Voice Playback Master Volume", AC97_PCM, 8, 7, 1), ++SOC_SINGLE("Voice Playback Mono Volume", AC97_PCM, 4, 7, 1), ++ ++SOC_SINGLE("Aux Playback Headphone Volume", AC97_REC_SEL, 12, 7, 1), ++SOC_SINGLE("Aux Playback Master Volume", AC97_REC_SEL, 8, 7, 1), ++SOC_SINGLE("Aux Playback Mono Volume", AC97_REC_SEL, 4, 7, 1), ++ ++SOC_ENUM("Bass Control", wm9713_enum[16]), ++SOC_SINGLE("Bass Cut-off Switch", AC97_GENERAL_PURPOSE, 12, 1, 1), ++SOC_SINGLE("Tone Cut-off Switch", AC97_GENERAL_PURPOSE, 4, 1, 1), ++SOC_SINGLE("Playback Attenuate (-6dB) Switch", AC97_GENERAL_PURPOSE, 6, 1, 0), ++SOC_SINGLE("Bass Volume", AC97_GENERAL_PURPOSE, 8, 15, 1), ++SOC_SINGLE("Tone Volume", AC97_GENERAL_PURPOSE, 0, 15, 1), ++ ++SOC_SINGLE("3D Upper Cut-off Switch", AC97_REC_GAIN_MIC, 5, 1, 0), ++SOC_SINGLE("3D Lower Cut-off Switch", AC97_REC_GAIN_MIC, 4, 1, 0), ++SOC_SINGLE("3D Depth", AC97_REC_GAIN_MIC, 0, 15, 1), ++}; ++ ++/* add non dapm controls */ ++static int wm9713_add_controls(struct snd_soc_codec *codec) ++{ ++ int err, i; ++ ++ for (i = 0; i < ARRAY_SIZE(wm9713_snd_ac97_controls); i++) { ++ err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&wm9713_snd_ac97_controls[i],codec, NULL)); ++ if (err < 0) ++ return err; ++ } ++ return 0; ++} ++ ++/* We have to create a fake left and right HP mixers because ++ * the codec only has a single control that is shared by both channels. ++ * This makes it impossible to determine the audio path using the current ++ * register map, thus we add a new (virtual) register to help determine the ++ * audio route within the device. ++ */ ++static int mixer_event (struct snd_soc_dapm_widget *w, int event) ++{ ++ u16 l, r, beep, tone, phone, rec, pcm, aux; ++ ++ l = ac97_read(w->codec, HPL_MIXER); ++ r = ac97_read(w->codec, HPR_MIXER); ++ beep = ac97_read(w->codec, AC97_PC_BEEP); ++ tone = ac97_read(w->codec, AC97_MASTER_TONE); ++ phone = ac97_read(w->codec, AC97_PHONE); ++ rec = ac97_read(w->codec, AC97_REC_SEL); ++ pcm = ac97_read(w->codec, AC97_PCM); ++ aux = ac97_read(w->codec, AC97_AUX); ++ ++ if (event & SND_SOC_DAPM_PRE_REG) ++ return 0; ++ if (l & 0x1 || r & 0x1) ++ ac97_write(w->codec, AC97_PC_BEEP, beep & 0x7fff); ++ else ++ ac97_write(w->codec, AC97_PC_BEEP, beep | 0x8000); ++ ++ if (l & 0x2 || r & 0x2) ++ ac97_write(w->codec, AC97_MASTER_TONE, tone & 0x7fff); ++ else ++ ac97_write(w->codec, AC97_MASTER_TONE, tone | 0x8000); ++ ++ if (l & 0x4 || r & 0x4) ++ ac97_write(w->codec, AC97_PHONE, phone & 0x7fff); ++ else ++ ac97_write(w->codec, AC97_PHONE, phone | 0x8000); ++ ++ if (l & 0x8 || r & 0x8) ++ ac97_write(w->codec, AC97_REC_SEL, rec & 0x7fff); ++ else ++ ac97_write(w->codec, AC97_REC_SEL, rec | 0x8000); ++ ++ if (l & 0x10 || r & 0x10) ++ ac97_write(w->codec, AC97_PCM, pcm & 0x7fff); ++ else ++ ac97_write(w->codec, AC97_PCM, pcm | 0x8000); ++ ++ if (l & 0x20 || r & 0x20) ++ ac97_write(w->codec, AC97_AUX, aux & 0x7fff); ++ else ++ ac97_write(w->codec, AC97_AUX, aux | 0x8000); ++ ++ return 0; ++} ++ ++/* Left Headphone Mixers */ ++static const struct snd_kcontrol_new wm9713_hpl_mixer_controls[] = { ++SOC_DAPM_SINGLE("PC Beep Playback Switch", HPL_MIXER, 5, 1, 0), ++SOC_DAPM_SINGLE("Voice Playback Switch", HPL_MIXER, 4, 1, 0), ++SOC_DAPM_SINGLE("Aux Playback Switch", HPL_MIXER, 3, 1, 0), ++SOC_DAPM_SINGLE("PCM Playback Switch", HPL_MIXER, 2, 1, 0), ++SOC_DAPM_SINGLE("MonoIn Playback Switch", HPL_MIXER, 1, 1, 0), ++SOC_DAPM_SINGLE("Bypass Playback Switch", HPL_MIXER, 0, 1, 0), ++}; ++ ++/* Right Headphone Mixers */ ++static const struct snd_kcontrol_new wm9713_hpr_mixer_controls[] = { ++SOC_DAPM_SINGLE("PC Beep Playback Switch", HPR_MIXER, 5, 1, 0), ++SOC_DAPM_SINGLE("Voice Playback Switch", HPR_MIXER, 4, 1, 0), ++SOC_DAPM_SINGLE("Aux Playback Switch", HPR_MIXER, 3, 1, 0), ++SOC_DAPM_SINGLE("PCM Playback Switch", HPR_MIXER, 2, 1, 0), ++SOC_DAPM_SINGLE("MonoIn Playback Switch", HPR_MIXER, 1, 1, 0), ++SOC_DAPM_SINGLE("Bypass Playback Switch", HPR_MIXER, 0, 1, 0), ++}; ++ ++/* headphone capture mux */ ++static const struct snd_kcontrol_new wm9713_hp_rec_mux_controls = ++SOC_DAPM_ENUM("Route", wm9713_enum[1]); ++ ++/* headphone mic mux */ ++static const struct snd_kcontrol_new wm9713_hp_mic_mux_controls = ++SOC_DAPM_ENUM("Route", wm9713_enum[0]); ++ ++/* Speaker Mixer */ ++static const struct snd_kcontrol_new wm9713_speaker_mixer_controls[] = { ++SOC_DAPM_SINGLE("PC Beep Playback Switch", AC97_AUX, 11, 1, 1), ++SOC_DAPM_SINGLE("Voice Playback Switch", AC97_PCM, 11, 1, 1), ++SOC_DAPM_SINGLE("Aux Playback Switch", AC97_REC_SEL, 11, 1, 1), ++SOC_DAPM_SINGLE("PCM Playback Switch", AC97_PHONE, 14, 1, 1), ++SOC_DAPM_SINGLE("MonoIn Playback Switch", AC97_MASTER_TONE, 14, 1, 1), ++SOC_DAPM_SINGLE("Bypass Playback Switch", AC97_PC_BEEP, 14, 1, 1), ++}; ++ ++/* Mono Mixer */ ++static const struct snd_kcontrol_new wm9713_mono_mixer_controls[] = { ++SOC_DAPM_SINGLE("PC Beep Playback Switch", AC97_AUX, 7, 1, 1), ++SOC_DAPM_SINGLE("Voice Playback Switch", AC97_PCM, 7, 1, 1), ++SOC_DAPM_SINGLE("Aux Playback Switch", AC97_REC_SEL, 7, 1, 1), ++SOC_DAPM_SINGLE("PCM Playback Switch", AC97_PHONE, 13, 1, 1), ++SOC_DAPM_SINGLE("MonoIn Playback Switch", AC97_MASTER_TONE, 13, 1, 1), ++SOC_DAPM_SINGLE("Bypass Playback Switch", AC97_PC_BEEP, 13, 1, 1), ++SOC_DAPM_SINGLE("Mic 1 Sidetone Switch", AC97_LINE, 7, 1, 1), ++SOC_DAPM_SINGLE("Mic 2 Sidetone Switch", AC97_LINE, 6, 1, 1), ++}; ++ ++/* mono mic mux */ ++static const struct snd_kcontrol_new wm9713_mono_mic_mux_controls = ++SOC_DAPM_ENUM("Route", wm9713_enum[2]); ++ ++/* mono output mux */ ++static const struct snd_kcontrol_new wm9713_mono_mux_controls = ++SOC_DAPM_ENUM("Route", wm9713_enum[7]); ++ ++/* speaker left output mux */ ++static const struct snd_kcontrol_new wm9713_hp_spkl_mux_controls = ++SOC_DAPM_ENUM("Route", wm9713_enum[8]); ++ ++/* speaker right output mux */ ++static const struct snd_kcontrol_new wm9713_hp_spkr_mux_controls = ++SOC_DAPM_ENUM("Route", wm9713_enum[9]); ++ ++/* headphone left output mux */ ++static const struct snd_kcontrol_new wm9713_hpl_out_mux_controls = ++SOC_DAPM_ENUM("Route", wm9713_enum[10]); ++ ++/* headphone right output mux */ ++static const struct snd_kcontrol_new wm9713_hpr_out_mux_controls = ++SOC_DAPM_ENUM("Route", wm9713_enum[11]); ++ ++/* Out3 mux */ ++static const struct snd_kcontrol_new wm9713_out3_mux_controls = ++SOC_DAPM_ENUM("Route", wm9713_enum[12]); ++ ++/* Out4 mux */ ++static const struct snd_kcontrol_new wm9713_out4_mux_controls = ++SOC_DAPM_ENUM("Route", wm9713_enum[13]); ++ ++/* DAC inv mux 1 */ ++static const struct snd_kcontrol_new wm9713_dac_inv1_mux_controls = ++SOC_DAPM_ENUM("Route", wm9713_enum[14]); ++ ++/* DAC inv mux 2 */ ++static const struct snd_kcontrol_new wm9713_dac_inv2_mux_controls = ++SOC_DAPM_ENUM("Route", wm9713_enum[15]); ++ ++/* Capture source left */ ++static const struct snd_kcontrol_new wm9713_rec_srcl_mux_controls = ++SOC_DAPM_ENUM("Route", wm9713_enum[3]); ++ ++/* Capture source right */ ++static const struct snd_kcontrol_new wm9713_rec_srcr_mux_controls = ++SOC_DAPM_ENUM("Route", wm9713_enum[4]); ++ ++/* mic source */ ++static const struct snd_kcontrol_new wm9713_mic_sel_mux_controls = ++SOC_DAPM_ENUM("Route", wm9713_enum[18]); ++ ++/* mic source B virtual control */ ++static const struct snd_kcontrol_new wm9713_micb_sel_mux_controls = ++SOC_DAPM_ENUM("Route", wm9713_enum[19]); ++ ++static const struct snd_soc_dapm_widget wm9713_dapm_widgets[] = { ++SND_SOC_DAPM_MUX("Capture Headphone Mux", SND_SOC_NOPM, 0, 0, ++ &wm9713_hp_rec_mux_controls), ++SND_SOC_DAPM_MUX("Sidetone Mux", SND_SOC_NOPM, 0, 0, ++ &wm9713_hp_mic_mux_controls), ++SND_SOC_DAPM_MUX("Capture Mono Mux", SND_SOC_NOPM, 0, 0, ++ &wm9713_mono_mic_mux_controls), ++SND_SOC_DAPM_MUX("Mono Out Mux", SND_SOC_NOPM, 0, 0, ++ &wm9713_mono_mux_controls), ++SND_SOC_DAPM_MUX("Left Speaker Out Mux", SND_SOC_NOPM, 0, 0, ++ &wm9713_hp_spkl_mux_controls), ++SND_SOC_DAPM_MUX("Right Speaker Out Mux", SND_SOC_NOPM, 0, 0, ++ &wm9713_hp_spkr_mux_controls), ++SND_SOC_DAPM_MUX("Left Headphone Out Mux", SND_SOC_NOPM, 0, 0, ++ &wm9713_hpl_out_mux_controls), ++SND_SOC_DAPM_MUX("Right Headphone Out Mux", SND_SOC_NOPM, 0, 0, ++ &wm9713_hpr_out_mux_controls), ++SND_SOC_DAPM_MUX("Out 3 Mux", SND_SOC_NOPM, 0, 0, ++ &wm9713_out3_mux_controls), ++SND_SOC_DAPM_MUX("Out 4 Mux", SND_SOC_NOPM, 0, 0, ++ &wm9713_out4_mux_controls), ++SND_SOC_DAPM_MUX("DAC Inv Mux 1", SND_SOC_NOPM, 0, 0, ++ &wm9713_dac_inv1_mux_controls), ++SND_SOC_DAPM_MUX("DAC Inv Mux 2", SND_SOC_NOPM, 0, 0, ++ &wm9713_dac_inv2_mux_controls), ++SND_SOC_DAPM_MUX("Left Capture Source", SND_SOC_NOPM, 0, 0, ++ &wm9713_rec_srcl_mux_controls), ++SND_SOC_DAPM_MUX("Right Capture Source", SND_SOC_NOPM, 0, 0, ++ &wm9713_rec_srcr_mux_controls), ++SND_SOC_DAPM_MUX("Mic A Source", SND_SOC_NOPM, 0, 0, ++ &wm9713_mic_sel_mux_controls ), ++SND_SOC_DAPM_MUX("Mic B Source", SND_SOC_NOPM, 0, 0, ++ &wm9713_micb_sel_mux_controls ), ++SND_SOC_DAPM_MIXER_E("Left HP Mixer", AC97_EXTENDED_MID, 3, 1, ++ &wm9713_hpl_mixer_controls[0], ARRAY_SIZE(wm9713_hpl_mixer_controls), ++ mixer_event, SND_SOC_DAPM_POST_REG), ++SND_SOC_DAPM_MIXER_E("Right HP Mixer", AC97_EXTENDED_MID, 2, 1, ++ &wm9713_hpr_mixer_controls[0], ARRAY_SIZE(wm9713_hpr_mixer_controls), ++ mixer_event, SND_SOC_DAPM_POST_REG), ++SND_SOC_DAPM_MIXER("Mono Mixer", AC97_EXTENDED_MID, 0, 1, ++ &wm9713_mono_mixer_controls[0], ARRAY_SIZE(wm9713_mono_mixer_controls)), ++SND_SOC_DAPM_MIXER("Speaker Mixer", AC97_EXTENDED_MID, 1, 1, ++ &wm9713_speaker_mixer_controls[0], ++ ARRAY_SIZE(wm9713_speaker_mixer_controls)), ++SND_SOC_DAPM_DAC("Left DAC", "Left HiFi Playback", AC97_EXTENDED_MID, 7, 1), ++SND_SOC_DAPM_DAC("Right DAC", "Right HiFi Playback", AC97_EXTENDED_MID, 6, 1), ++SND_SOC_DAPM_MIXER("AC97 Mixer", SND_SOC_NOPM, 0, 0, NULL, 0), ++SND_SOC_DAPM_MIXER("HP Mixer", SND_SOC_NOPM, 0, 0, NULL, 0), ++SND_SOC_DAPM_MIXER("Capture Mixer", SND_SOC_NOPM, 0, 0, NULL, 0), ++SND_SOC_DAPM_DAC("Voice DAC", "Voice Playback", AC97_EXTENDED_MID, 12, 1), ++SND_SOC_DAPM_DAC("Aux DAC", "Aux Playback", AC97_EXTENDED_MID, 11, 1), ++SND_SOC_DAPM_ADC("Left ADC", "Left HiFi Capture", AC97_EXTENDED_MID, 5, 1), ++SND_SOC_DAPM_ADC("Right ADC", "Right HiFi Capture", AC97_EXTENDED_MID, 4, 1), ++SND_SOC_DAPM_PGA("Left Headphone", AC97_EXTENDED_MSTATUS, 10, 1, NULL, 0), ++SND_SOC_DAPM_PGA("Right Headphone", AC97_EXTENDED_MSTATUS, 9, 1, NULL, 0), ++SND_SOC_DAPM_PGA("Left Speaker", AC97_EXTENDED_MSTATUS, 8, 1, NULL, 0), ++SND_SOC_DAPM_PGA("Right Speaker", AC97_EXTENDED_MSTATUS, 7, 1, NULL, 0), ++SND_SOC_DAPM_PGA("Out 3", AC97_EXTENDED_MSTATUS, 11, 1, NULL, 0), ++SND_SOC_DAPM_PGA("Out 4", AC97_EXTENDED_MSTATUS, 12, 1, NULL, 0), ++SND_SOC_DAPM_PGA("Mono Out", AC97_EXTENDED_MSTATUS, 13, 1, NULL, 0), ++SND_SOC_DAPM_PGA("Left Line In", AC97_EXTENDED_MSTATUS, 6, 1, NULL, 0), ++SND_SOC_DAPM_PGA("Right Line In", AC97_EXTENDED_MSTATUS, 5, 1, NULL, 0), ++SND_SOC_DAPM_PGA("Mono In", AC97_EXTENDED_MSTATUS, 4, 1, NULL, 0), ++SND_SOC_DAPM_PGA("Mic A PGA", AC97_EXTENDED_MSTATUS, 3, 1, NULL, 0), ++SND_SOC_DAPM_PGA("Mic B PGA", AC97_EXTENDED_MSTATUS, 2, 1, NULL, 0), ++SND_SOC_DAPM_PGA("Mic A Pre Amp", AC97_EXTENDED_MSTATUS, 1, 1, NULL, 0), ++SND_SOC_DAPM_PGA("Mic B Pre Amp", AC97_EXTENDED_MSTATUS, 0, 1, NULL, 0), ++SND_SOC_DAPM_MICBIAS("Mic Bias", AC97_EXTENDED_MSTATUS, 14, 1), ++SND_SOC_DAPM_OUTPUT("MONO"), ++SND_SOC_DAPM_OUTPUT("HPL"), ++SND_SOC_DAPM_OUTPUT("HPR"), ++SND_SOC_DAPM_OUTPUT("SPKL"), ++SND_SOC_DAPM_OUTPUT("SPKR"), ++SND_SOC_DAPM_OUTPUT("OUT3"), ++SND_SOC_DAPM_OUTPUT("OUT4"), ++SND_SOC_DAPM_INPUT("LINEL"), ++SND_SOC_DAPM_INPUT("LINER"), ++SND_SOC_DAPM_INPUT("MONOIN"), ++SND_SOC_DAPM_INPUT("PCBEEP"), ++SND_SOC_DAPM_INPUT("MIC1"), ++SND_SOC_DAPM_INPUT("MIC2A"), ++SND_SOC_DAPM_INPUT("MIC2B"), ++SND_SOC_DAPM_VMID("VMID"), ++}; ++ ++static const char *audio_map[][3] = { ++ /* left HP mixer */ ++ {"Left HP Mixer", "PC Beep Playback Switch", "PCBEEP"}, ++ {"Left HP Mixer", "Voice Playback Switch", "Voice DAC"}, ++ {"Left HP Mixer", "Aux Playback Switch", "Aux DAC"}, ++ {"Left HP Mixer", "Bypass Playback Switch", "Left Line In"}, ++ {"Left HP Mixer", "PCM Playback Switch", "Left DAC"}, ++ {"Left HP Mixer", "MonoIn Playback Switch", "Mono In"}, ++ {"Left HP Mixer", NULL, "Capture Headphone Mux"}, ++ ++ /* right HP mixer */ ++ {"Right HP Mixer", "PC Beep Playback Switch", "PCBEEP"}, ++ {"Right HP Mixer", "Voice Playback Switch", "Voice DAC"}, ++ {"Right HP Mixer", "Aux Playback Switch", "Aux DAC"}, ++ {"Right HP Mixer", "Bypass Playback Switch", "Right Line In"}, ++ {"Right HP Mixer", "PCM Playback Switch", "Right DAC"}, ++ {"Right HP Mixer", "MonoIn Playback Switch", "Mono In"}, ++ {"Right HP Mixer", NULL, "Capture Headphone Mux"}, ++ ++ /* virtual mixer - mixes left & right channels for spk and mono */ ++ {"AC97 Mixer", NULL, "Left DAC"}, ++ {"AC97 Mixer", NULL, "Right DAC"}, ++ {"Line Mixer", NULL, "Right Line In"}, ++ {"Line Mixer", NULL, "Left Line In"}, ++ {"HP Mixer", NULL, "Left HP Mixer"}, ++ {"HP Mixer", NULL, "Right HP Mixer"}, ++ {"Capture Mixer", NULL, "Left Capture Source"}, ++ {"Capture Mixer", NULL, "Right Capture Source"}, ++ ++ /* speaker mixer */ ++ {"Speaker Mixer", "PC Beep Playback Switch", "PCBEEP"}, ++ {"Speaker Mixer", "Voice Playback Switch", "Voice DAC"}, ++ {"Speaker Mixer", "Aux Playback Switch", "Aux DAC"}, ++ {"Speaker Mixer", "Bypass Playback Switch", "Line Mixer"}, ++ {"Speaker Mixer", "PCM Playback Switch", "AC97 Mixer"}, ++ {"Speaker Mixer", "MonoIn Playback Switch", "Mono In"}, ++ ++ /* mono mixer */ ++ {"Mono Mixer", "PC Beep Playback Switch", "PCBEEP"}, ++ {"Mono Mixer", "Voice Playback Switch", "Voice DAC"}, ++ {"Mono Mixer", "Aux Playback Switch", "Aux DAC"}, ++ {"Mono Mixer", "Bypass Playback Switch", "Line Mixer"}, ++ {"Mono Mixer", "PCM Playback Switch", "AC97 Mixer"}, ++ {"Mono Mixer", NULL, "Capture Mono Mux"}, ++ ++ /* DAC inv mux 1 */ ++ {"DAC Inv Mux 1", "Mono", "Mono Mixer"}, ++ {"DAC Inv Mux 1", "Speaker", "Speaker Mixer"}, ++ {"DAC Inv Mux 1", "Left Headphone", "Left HP Mixer"}, ++ {"DAC Inv Mux 1", "Right Headphone", "Right HP Mixer"}, ++ {"DAC Inv Mux 1", "Headphone Mono", "HP Mixer"}, ++ ++ /* DAC inv mux 2 */ ++ {"DAC Inv Mux 2", "Mono", "Mono Mixer"}, ++ {"DAC Inv Mux 2", "Speaker", "Speaker Mixer"}, ++ {"DAC Inv Mux 2", "Left Headphone", "Left HP Mixer"}, ++ {"DAC Inv Mux 2", "Right Headphone", "Right HP Mixer"}, ++ {"DAC Inv Mux 2", "Headphone Mono", "HP Mixer"}, ++ ++ /* headphone left mux */ ++ {"Left Headphone Out Mux", "Headphone", "Left HP Mixer"}, ++ ++ /* headphone right mux */ ++ {"Right Headphone Out Mux", "Headphone", "Right HP Mixer"}, ++ ++ /* speaker left mux */ ++ {"Left Speaker Out Mux", "Headphone", "Left HP Mixer"}, ++ {"Left Speaker Out Mux", "Speaker", "Speaker Mixer"}, ++ {"Left Speaker Out Mux", "Inv", "DAC Inv Mux 1"}, ++ ++ /* speaker right mux */ ++ {"Right Speaker Out Mux", "Headphone", "Right HP Mixer"}, ++ {"Right Speaker Out Mux", "Speaker", "Speaker Mixer"}, ++ {"Right Speaker Out Mux", "Inv", "DAC Inv Mux 2"}, ++ ++ /* mono mux */ ++ {"Mono Out Mux", "Mono", "Mono Mixer"}, ++ {"Mono Out Mux", "Inv", "DAC Inv Mux 1"}, ++ ++ /* out 3 mux */ ++ {"Out 3 Mux", "Inv 1", "DAC Inv Mux 1"}, ++ ++ /* out 4 mux */ ++ {"Out 4 Mux", "Inv 2", "DAC Inv Mux 2"}, ++ ++ /* output pga */ ++ {"HPL", NULL, "Left Headphone"}, ++ {"Left Headphone", NULL, "Left Headphone Out Mux"}, ++ {"HPR", NULL, "Right Headphone"}, ++ {"Right Headphone", NULL, "Right Headphone Out Mux"}, ++ {"OUT3", NULL, "Out 3"}, ++ {"Out 3", NULL, "Out 3 Mux"}, ++ {"OUT4", NULL, "Out 4"}, ++ {"Out 4", NULL, "Out 4 Mux"}, ++ {"SPKL", NULL, "Left Speaker"}, ++ {"Left Speaker", NULL, "Left Speaker Out Mux"}, ++ {"SPKR", NULL, "Right Speaker"}, ++ {"Right Speaker", NULL, "Right Speaker Out Mux"}, ++ {"MONO", NULL, "Mono Out"}, ++ {"Mono Out", NULL, "Mono Out Mux"}, ++ ++ /* input pga */ ++ {"Left Line In", NULL, "LINEL"}, ++ {"Right Line In", NULL, "LINER"}, ++ {"Mono In", NULL, "MONOIN"}, ++ {"Mic A PGA", NULL, "Mic A Pre Amp"}, ++ {"Mic B PGA", NULL, "Mic B Pre Amp"}, ++ ++ /* left capture select */ ++ {"Left Capture Source", "Mic 1", "Mic A Pre Amp"}, ++ {"Left Capture Source", "Mic 2", "Mic B Pre Amp"}, ++ {"Left Capture Source", "Line", "LINEL"}, ++ {"Left Capture Source", "Mono In", "MONOIN"}, ++ {"Left Capture Source", "Headphone", "Left HP Mixer"}, ++ {"Left Capture Source", "Speaker", "Speaker Mixer"}, ++ {"Left Capture Source", "Mono Out", "Mono Mixer"}, ++ ++ /* right capture select */ ++ {"Right Capture Source", "Mic 1", "Mic A Pre Amp"}, ++ {"Right Capture Source", "Mic 2", "Mic B Pre Amp"}, ++ {"Right Capture Source", "Line", "LINER"}, ++ {"Right Capture Source", "Mono In", "MONOIN"}, ++ {"Right Capture Source", "Headphone", "Right HP Mixer"}, ++ {"Right Capture Source", "Speaker", "Speaker Mixer"}, ++ {"Right Capture Source", "Mono Out", "Mono Mixer"}, ++ ++ /* left ADC */ ++ {"Left ADC", NULL, "Left Capture Source"}, ++ ++ /* right ADC */ ++ {"Right ADC", NULL, "Right Capture Source"}, ++ ++ /* mic */ ++ {"Mic A Pre Amp", NULL, "Mic A Source"}, ++ {"Mic A Source", "Mic 1", "MIC1"}, ++ {"Mic A Source", "Mic 2 A", "MIC2A"}, ++ {"Mic A Source", "Mic 2 B", "Mic B Source"}, ++ {"Mic B Pre Amp", "MPB", "Mic B Source"}, ++ {"Mic B Source", NULL, "MIC2B"}, ++ ++ /* headphone capture */ ++ {"Capture Headphone Mux", "Stereo", "Capture Mixer"}, ++ {"Capture Headphone Mux", "Left", "Left Capture Source"}, ++ {"Capture Headphone Mux", "Right", "Right Capture Source"}, ++ ++ /* mono capture */ ++ {"Capture Mono Mux", "Stereo", "Capture Mixer"}, ++ {"Capture Mono Mux", "Left", "Left Capture Source"}, ++ {"Capture Mono Mux", "Right", "Right Capture Source"}, ++ ++ {NULL, NULL, NULL}, ++}; ++ ++static int wm9713_add_widgets(struct snd_soc_codec *codec) ++{ ++ int i; ++ ++ for(i = 0; i < ARRAY_SIZE(wm9713_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &wm9713_dapm_widgets[i]); ++ } ++ ++ /* set up audio path audio_mapnects */ ++ for(i = 0; audio_map[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, audio_map[i][0], ++ audio_map[i][1], audio_map[i][2]); ++ } ++ ++ snd_soc_dapm_new_widgets(codec); ++ return 0; ++} ++ ++static unsigned int ac97_read(struct snd_soc_codec *codec, ++ unsigned int reg) ++{ ++ u16 *cache = codec->reg_cache; ++ ++ if (reg == AC97_RESET || reg == AC97_GPIO_STATUS || ++ reg == AC97_VENDOR_ID1 || reg == AC97_VENDOR_ID2 || ++ reg == AC97_CD) ++ return soc_ac97_ops.read(codec->ac97, reg); ++ else { ++ reg = reg >> 1; ++ ++ if (reg > (ARRAY_SIZE(wm9713_reg))) ++ return -EIO; ++ ++ return cache[reg]; ++ } ++} ++ ++static int ac97_write(struct snd_soc_codec *codec, unsigned int reg, ++ unsigned int val) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg < 0x7c) ++ soc_ac97_ops.write(codec->ac97, reg, val); ++ reg = reg >> 1; ++ if (reg <= (ARRAY_SIZE(wm9713_reg))) ++ cache[reg] = val; ++ ++ return 0; ++} ++ ++struct pll_ { ++ unsigned int in_hz; ++ unsigned int lf:1; /* allows low frequency use */ ++ unsigned int sdm:1; /* allows fraction n div */ ++ unsigned int divsel:1; /* enables input clock div */ ++ unsigned int divctl:1; /* input clock divider */ ++ unsigned int n:4; ++ unsigned int k; ++}; ++ ++struct pll_ pll[] = { ++ {13000000, 0, 1, 0, 0, 7, 0x23f488}, ++ {2048000, 1, 0, 0, 0, 12, 0x0}, ++ {4096000, 1, 0, 0, 0, 6, 0x0}, ++ {12288000, 0, 0, 0, 0, 8, 0x0}, ++ /* liam - add more entries */ ++}; ++ ++/* we must have either 24.576MHz or a PLL freq */ ++static unsigned int wm9713_config_ac97sysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ int i; ++ dai->mclk = 0; ++ ++ /* first check if we can get away witout burning any PLL power */ ++ if (24576000 == clk) { ++ /* standard AC97 clock */ ++ dai->mclk = clk; ++ goto out; ++ } ++ ++ /* ok no standard clock, so we must now try the PLL */ ++ for(i = 0; i < ARRAY_SIZE(pll); i++) { ++ if (clk == pll[i].in_hz) { ++ dai->mclk = clk; /* clock out */ ++ goto out; ++ } ++ } ++ ++out: ++ return dai->mclk; ++} ++ ++/* The WM9713 voice DAC can only run at 256FS. This interface and DAC are ++ * clocked by the main AC97 clock divided down to 256 FS. ++ */ ++static unsigned int wm9713_config_vsysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ ++ int i, j, best_clk = info->fs * info->rate; ++ ++ /* can we run at this clk without the PLL ? */ ++ for (i = 1; i <= 16; i++) { ++ if (best_clk * i == clk) { ++ dai->pll_in = 0; ++ dai->clk_div = i << 1; ++ dai->mclk = best_clk; ++ return dai->mclk; ++ } ++ } ++ ++ /* now check for PLL support */ ++ for (i = 0; i < ARRAY_SIZE(pll); i++) { ++ if (pll[i].in_hz == clk) { ++ for (j = 1; j <= 16; j++) { ++ if (24576000 == j * best_clk) { ++ dai->pll_in = clk; ++ dai->pll_out = 24576000; ++ dai->clk_div = j << 1; ++ dai->mclk = best_clk; ++ return dai->mclk; ++ } ++ } ++ } ++ } ++ ++ /* this clk is not supported */ ++ return 0; ++} ++ ++u32 wm9713_set_pll(struct snd_soc_codec *codec, u32 in) ++{ ++ struct wm9713 *wm = (struct wm9713*)codec->private_data; ++ int i; ++ u16 reg, reg2; ++ ++ /* turn PLL off ? */ ++ if (in == 0) { ++ /* disable PLL power and select ext source */ ++ reg = ac97_read(codec, AC97_HANDSET_RATE); ++ ac97_write(codec, AC97_HANDSET_RATE, reg | 0x0080); ++ reg = ac97_read(codec, AC97_EXTENDED_MID); ++ ac97_write(codec, AC97_EXTENDED_MID, reg | 0x0200); ++ wm->pll = 0; ++ return 0; ++ } ++ ++ for (i = 0; i < ARRAY_SIZE(pll); i++) { ++ if (pll[i].in_hz == in) ++ goto found; ++ } ++ return -EINVAL; ++ ++found: ++ if (pll[i].sdm == 0) { ++ reg = (pll[i].n << 12) | (pll[i].lf << 11) | ++ (pll[i].divsel << 9) | (pll[i].divctl << 8); ++ ac97_write(codec, AC97_LINE1_LEVEL, reg); ++ } else { ++ /* write the fractional k to the reg 0x46 pages */ ++ reg2 = (pll[i].n << 12) | (pll[i].lf << 11) | (pll[i].sdm << 10) | ++ (pll[i].divsel << 9) | (pll[i].divctl << 8); ++ ++ reg = reg2 | (0x5 << 4) | (pll[i].k >> 20); /* K [21:20] */ ++ ac97_write(codec, AC97_LINE1_LEVEL, reg); ++ ++ reg = reg2 | (0x4 << 4) | ((pll[i].k >> 16) & 0xf); /* K [19:16] */ ++ ac97_write(codec, AC97_LINE1_LEVEL, reg); ++ ++ reg = reg2 | (0x3 << 4) | ((pll[i].k >> 12) & 0xf); /* K [15:12] */ ++ ac97_write(codec, AC97_LINE1_LEVEL, reg); ++ ++ reg = reg2 | (0x2 << 4) | ((pll[i].k >> 8) & 0xf); /* K [11:8] */ ++ ac97_write(codec, AC97_LINE1_LEVEL, reg); ++ ++ reg = reg2 | (0x1 << 4) | ((pll[i].k >> 4) & 0xf); /* K [7:4] */ ++ ac97_write(codec, AC97_LINE1_LEVEL, reg); ++ ++ reg = reg2 | (0x0 << 4) | (pll[i].k & 0xf); /* K [3:0] */ ++ ac97_write(codec, AC97_LINE1_LEVEL, reg); ++ } ++ ++ /* turn PLL on and select as source */ ++ reg = ac97_read(codec, AC97_EXTENDED_MID); ++ ac97_write(codec, AC97_EXTENDED_MID, reg & 0xfdff); ++ reg = ac97_read(codec, AC97_HANDSET_RATE); ++ ac97_write(codec, AC97_HANDSET_RATE, reg & 0xff7f); ++ /* wait 10ms AC97 link frames for the link to stabilise */ ++ schedule_timeout_interruptible(msecs_to_jiffies(10)); ++ wm->pll = in; ++ return 0; ++} ++EXPORT_SYMBOL_GPL(wm9713_set_pll); ++ ++static int wm9713_voice_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 reg = 0x8000, bfs, div, gpio; ++ ++ bfs = SND_SOC_FSBD_REAL(rtd->codec_dai->dai_runtime.bfs); ++ gpio = ac97_read(codec, AC97_GPIO_CFG) & 0xffe2; ++ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK){ ++ case SND_SOC_DAIFMT_CBM_CFM: ++ reg |= 0x4000; ++ gpio |= 0x0008; ++ break; ++ case SND_SOC_DAIFMT_CBM_CFS: ++ reg |= 0x6000; ++ gpio |= 0x000c; ++ break; ++ case SND_SOC_DAIFMT_CBS_CFS: ++ reg |= 0x0200; ++ gpio |= 0x000d; ++ break; ++ case SND_SOC_DAIFMT_CBS_CFM: ++ gpio |= 0x0009; ++ break; ++ } ++ ac97_write(codec, AC97_GPIO_CFG, gpio); ++ ++ /* enable PLL if needed */ ++ if (rtd->codec_dai->pll_in) ++ wm9713_set_pll(codec, rtd->codec_dai->pll_in); ++ ++ /* set the PCM divider */ ++ div = ac97_read(codec, AC97_HANDSET_RATE) & 0xf0ff; ++ ac97_write(codec, AC97_HANDSET_RATE, div | ++ ((rtd->codec_dai->clk_div >> 1) -1) << 8); ++ ++ /* clock inversion */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_INV_MASK) { ++ case SND_SOC_DAIFMT_IB_IF: ++ reg |= 0x00c0; ++ break; ++ case SND_SOC_DAIFMT_IB_NF: ++ reg |= 0x0080; ++ break; ++ case SND_SOC_DAIFMT_NB_IF: ++ reg |= 0x0040; ++ break; ++ } ++ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_I2S: ++ reg |= 0x0002; ++ break; ++ case SND_SOC_DAIFMT_RIGHT_J: ++ break; ++ case SND_SOC_DAIFMT_LEFT_J: ++ reg |= 0x0001; ++ break; ++ case SND_SOC_DAIFMT_DSP_A: ++ reg |= 0x0003; ++ break; ++ case SND_SOC_DAIFMT_DSP_B: ++ reg |= 0x0043; ++ break; ++ } ++ ++ switch (rtd->codec_dai->dai_runtime.pcmfmt) { ++ case SNDRV_PCM_FMTBIT_S16_LE: ++ break; ++ case SNDRV_PCM_FMTBIT_S20_3LE: ++ reg |= 0x0004; ++ break; ++ case SNDRV_PCM_FMTBIT_S24_LE: ++ reg |= 0x0008; ++ break; ++ case SNDRV_PCM_FMTBIT_S32_LE: ++ reg |= 0x000c; ++ break; ++ } ++ ++ switch (bfs) { ++ case 2: ++ reg |= (0x1 << 9); ++ break; ++ case 4: ++ reg |= (0x2 << 9); ++ break; ++ case 8: ++ reg |= (0x3 << 9); ++ break; ++ case 16: ++ reg |= (0x4 << 9); ++ break; ++ } ++ ++ /* enable PCM interface in master mode */ ++ ac97_write(codec, AC97_CENTER_LFE_MASTER, reg); ++ return 0; ++} ++ ++static void wm9713_shutdown(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ if (!codec->active) ++ wm9713_set_pll(codec, 0); ++} ++ ++static void wm9713_voiceshutdown(snd_pcm_substream_t *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 status; ++ ++ wm9713_shutdown(substream); ++ ++ /* Gracefully shut down the voice interface. */ ++ status = ac97_read(codec, AC97_EXTENDED_STATUS) | 0x1000; ++ ac97_write(codec,AC97_HANDSET_RATE,0x0280); ++ schedule_timeout_interruptible(msecs_to_jiffies(1)); ++ ac97_write(codec,AC97_HANDSET_RATE,0x0F80); ++ ac97_write(codec,AC97_EXTENDED_MID,status); ++} ++ ++static int ac97_hifi_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ int reg; ++ u16 vra; ++ ++ /* we need a 24576000Hz clock to run at the correct speed */ ++ if (rtd->codec_dai->mclk != 24576000) ++ wm9713_set_pll(codec, rtd->codec_dai->mclk); ++ ++ vra = ac97_read(codec, AC97_EXTENDED_STATUS); ++ ac97_write(codec, AC97_EXTENDED_STATUS, vra | 0x1); ++ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ reg = AC97_PCM_FRONT_DAC_RATE; ++ else ++ reg = AC97_PCM_LR_ADC_RATE; ++ ++ return ac97_write(codec, reg, runtime->rate); ++} ++ ++static int ac97_aux_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 vra, xsle; ++ ++ /* we need a 24576000Hz clock to run at the correct speed */ ++ if (rtd->codec_dai->mclk != 24576000) ++ wm9713_set_pll(codec, rtd->codec_dai->mclk); ++ ++ vra = ac97_read(codec, AC97_EXTENDED_STATUS); ++ ac97_write(codec, AC97_EXTENDED_STATUS, vra | 0x1); ++ xsle = ac97_read(codec, AC97_PCI_SID); ++ ac97_write(codec, AC97_PCI_SID, xsle | 0x8000); ++ ++ if (substream->stream != SNDRV_PCM_STREAM_PLAYBACK) ++ return -ENODEV; ++ ++ return ac97_write(codec, AC97_PCM_SURR_DAC_RATE, runtime->rate); ++} ++ ++struct snd_soc_codec_dai wm9713_dai[] = { ++{ ++ .name = "AC97 HiFi", ++ .playback = { ++ .stream_name = "HiFi Playback", ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .capture = { ++ .stream_name = "HiFi Capture", ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .config_sysclk = wm9713_config_ac97sysclk, ++ .ops = { ++ .shutdown = wm9713_shutdown, ++ .prepare = ac97_hifi_prepare,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(ac97_modes), ++ .mode = ac97_modes,},}, ++ { ++ .name = "AC97 Aux", ++ .playback = { ++ .stream_name = "Aux Playback", ++ .channels_min = 1, ++ .channels_max = 1,}, ++ .config_sysclk = wm9713_config_ac97sysclk, ++ .ops = { ++ .shutdown = wm9713_shutdown, ++ .prepare = ac97_aux_prepare,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(ac97_modes), ++ .mode = ac97_modes,} ++ }, ++ { ++ .name = "WM9713 Voice", ++ .playback = { ++ .stream_name = "Voice Playback", ++ .channels_min = 1, ++ .channels_max = 1,}, ++ .capture = { ++ .stream_name = "Voice Capture", ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .config_sysclk = wm9713_config_vsysclk, ++ .ops = { ++ .prepare = wm9713_voice_prepare, ++ .shutdown = wm9713_voiceshutdown,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(wm9713_voice_modes), ++ .mode = wm9713_voice_modes,}, ++ }, ++}; ++EXPORT_SYMBOL_GPL(wm9713_dai); ++ ++int wm9713_reset(struct snd_soc_codec *codec, int try_warm) ++{ ++ if (try_warm && soc_ac97_ops.warm_reset) { ++ soc_ac97_ops.warm_reset(codec->ac97); ++ if (!(ac97_read(codec, 0) & 0x8000)) ++ return 1; ++ } ++ ++ soc_ac97_ops.reset(codec->ac97); ++ if (ac97_read(codec, 0) & 0x8000) ++ return -EIO; ++ return 0; ++} ++EXPORT_SYMBOL_GPL(wm9713_reset); ++ ++static int wm9713_dapm_event(struct snd_soc_codec *codec, int event) ++{ ++ u16 reg; ++ ++ switch (event) { ++ case SNDRV_CTL_POWER_D0: /* full On */ ++ /* enable thermal shutdown */ ++ reg = ac97_read(codec, AC97_EXTENDED_MID) & 0x1bff; ++ ac97_write(codec, AC97_EXTENDED_MID, reg); ++ break; ++ case SNDRV_CTL_POWER_D1: /* partial On */ ++ case SNDRV_CTL_POWER_D2: /* partial On */ ++ break; ++ case SNDRV_CTL_POWER_D3hot: /* Off, with power */ ++ /* enable master bias and vmid */ ++ reg = ac97_read(codec, AC97_EXTENDED_MID) & 0x3bff; ++ ac97_write(codec, AC97_EXTENDED_MID, reg); ++ ac97_write(codec, AC97_POWERDOWN, 0x0000); ++ break; ++ case SNDRV_CTL_POWER_D3cold: /* Off, without power */ ++ /* disable everything including AC link */ ++ ac97_write(codec, AC97_EXTENDED_MID, 0xffff); ++ ac97_write(codec, AC97_EXTENDED_MSTATUS, 0xffff); ++ ac97_write(codec, AC97_POWERDOWN, 0xffff); ++ break; ++ } ++ codec->dapm_state = event; ++ return 0; ++} ++ ++static int wm9713_soc_suspend(struct platform_device *pdev, ++ pm_message_t state) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ struct wm9713 *wm = (struct wm9713*)codec->private_data; ++ ++ if (wm->pll) { ++ wm->pll_resume = wm->pll; ++ wm9713_set_pll(codec, 0); ++ } ++ wm9713_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ return 0; ++} ++ ++static int wm9713_soc_resume(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ struct wm9713 *wm = (struct wm9713*)codec->private_data; ++ int i, ret; ++ u16 *cache = codec->reg_cache; ++ ++ if ((ret = wm9713_reset(codec, 1)) < 0){ ++ printk(KERN_ERR "could not reset AC97 codec\n"); ++ return ret; ++ } ++ ++ wm9713_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ ++ /* only synchronise the codec if warm reset failed */ ++ if (ret == 0) { ++ for (i = 2; i < ARRAY_SIZE(wm9713_reg) << 1; i+=2) { ++ if (i == AC97_POWERDOWN || i == AC97_EXTENDED_MID || ++ i == AC97_EXTENDED_MSTATUS || i > 0x66) ++ continue; ++ soc_ac97_ops.write(codec->ac97, i, cache[i>>1]); ++ } ++ } ++ ++ if (wm->pll_resume) { ++ wm9713_set_pll(codec, wm->pll_resume); ++ wm->pll_resume = 0; ++ } ++ ++ if (codec->suspend_dapm_state == SNDRV_CTL_POWER_D0) ++ wm9713_dapm_event(codec, SNDRV_CTL_POWER_D0); ++ ++ return ret; ++} ++ ++static int wm9713_soc_probe(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec; ++ int ret = 0, reg; ++ ++ printk(KERN_INFO "WM9713/WM9714 SoC Audio Codec %s\n", WM9713_VERSION); ++ ++ socdev->codec = kzalloc(sizeof(struct snd_soc_codec), GFP_KERNEL); ++ if (socdev->codec == NULL) ++ return -ENOMEM; ++ codec = socdev->codec; ++ mutex_init(&codec->mutex); ++ ++ codec->reg_cache = ++ kzalloc(sizeof(u16) * ARRAY_SIZE(wm9713_reg), GFP_KERNEL); ++ if (codec->reg_cache == NULL){ ++ kfree(socdev->codec); ++ socdev->codec = NULL; ++ return -ENOMEM; ++ } ++ memcpy(codec->reg_cache, wm9713_reg, ++ sizeof(u16) * ARRAY_SIZE(wm9713_reg)); ++ codec->reg_cache_size = sizeof(u16) * ARRAY_SIZE(wm9713_reg); ++ codec->reg_cache_step = 2; ++ ++ codec->private_data = kzalloc(sizeof(struct wm9713), GFP_KERNEL); ++ if (codec->private_data == NULL) { ++ kfree(codec->reg_cache); ++ kfree(socdev->codec); ++ socdev->codec = NULL; ++ return -ENOMEM; ++ } ++ ++ codec->name = "WM9713"; ++ codec->owner = THIS_MODULE; ++ codec->dai = wm9713_dai; ++ codec->num_dai = ARRAY_SIZE(wm9713_dai); ++ codec->write = ac97_write; ++ codec->read = ac97_read; ++ codec->dapm_event = wm9713_dapm_event; ++ INIT_LIST_HEAD(&codec->dapm_widgets); ++ INIT_LIST_HEAD(&codec->dapm_paths); ++ ++ ret = snd_soc_new_ac97_codec(codec, &soc_ac97_ops, 0); ++ if (ret < 0) ++ goto err; ++ ++ /* register pcms */ ++ ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); ++ if (ret < 0) ++ goto pcm_err; ++ ++ /* do a cold reset for the controller and then try ++ * a warm reset followed by an optional cold reset for codec */ ++ wm9713_reset(codec, 0); ++ ret = wm9713_reset(codec, 1); ++ if (ret < 0) { ++ printk(KERN_ERR "AC97 link error\n"); ++ goto reset_err; ++ } ++ ++ wm9713_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ ++ /* unmute the adc - move to kcontrol */ ++ reg = ac97_read(codec, AC97_CD) & 0x7fff; ++ ac97_write(codec, AC97_CD, reg); ++ ++ wm9713_add_controls(codec); ++ wm9713_add_widgets(codec); ++ ret = snd_soc_register_card(socdev); ++ if (ret < 0) ++ goto reset_err; ++ return 0; ++ ++reset_err: ++ snd_soc_free_pcms(socdev); ++ ++pcm_err: ++ snd_soc_free_ac97_codec(codec); ++ ++err: ++ kfree(socdev->codec->private_data); ++ kfree(socdev->codec->reg_cache); ++ kfree(socdev->codec); ++ socdev->codec = NULL; ++ return ret; ++} ++ ++static int wm9713_soc_remove(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ if (codec == NULL) ++ return 0; ++ ++ snd_soc_dapm_free(socdev); ++ snd_soc_free_pcms(socdev); ++ snd_soc_free_ac97_codec(codec); ++ kfree(codec->private_data); ++ kfree(codec->reg_cache); ++ kfree(codec); ++ return 0; ++} ++ ++struct snd_soc_codec_device soc_codec_dev_wm9713= { ++ .probe = wm9713_soc_probe, ++ .remove = wm9713_soc_remove, ++ .suspend = wm9713_soc_suspend, ++ .resume = wm9713_soc_resume, ++}; ++ ++EXPORT_SYMBOL_GPL(soc_codec_dev_wm9713); ++ ++MODULE_DESCRIPTION("ASoC WM9713/WM9714 driver"); ++MODULE_AUTHOR("Liam Girdwood"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/codecs/wm9713.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm9713.h +@@ -0,0 +1,18 @@ ++/* ++ * wm9713.h -- WM9713 Soc Audio driver ++ */ ++ ++#ifndef _WM9713_H ++#define _WM9713_H ++ ++#define WM9713_DAI_AC97_HIFI 0 ++#define WM9713_DAI_AC97_AUX 1 ++#define WM9713_DAI_PCM_VOICE 2 ++ ++extern struct snd_soc_codec_device soc_codec_dev_wm9713; ++extern struct snd_soc_codec_dai wm9713_dai[3]; ++ ++u32 wm9713_set_pll(struct snd_soc_codec *codec, u32 in); ++int wm9713_reset(struct snd_soc_codec *codec, int try_warm); ++ ++#endif +Index: linux-2.6-pxa-new/sound/soc/pxa/Kconfig +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/pxa/Kconfig +@@ -0,0 +1,125 @@ ++menu "SoC Audio for the Intel PXA2xx" ++ ++config SND_PXA2xx_SOC ++ tristate "SoC Audio for the Intel PXA2xx chip" ++ depends on ARCH_PXA && SND ++ select SND_PCM ++ help ++ Say Y or M if you want to add support for codecs attached to ++ the PXA2xx AC97, I2S or SSP interface. You will also need ++ to select the audio interfaces to support below. ++ ++config SND_PXA2xx_AC97 ++ tristate ++ select SND_AC97_CODEC ++ ++config SND_PXA2xx_SOC_AC97 ++ tristate ++ select SND_AC97_BUS ++ select SND_SOC_AC97_BUS ++ ++config SND_PXA2xx_SOC_I2S ++ tristate ++ ++config SND_PXA2xx_SOC_SSP ++ tristate ++ select PXA_SSP ++ ++config SND_PXA2xx_SOC_MAINSTONE ++ tristate "SoC AC97 Audio support for Intel Mainstone" ++ depends on SND_PXA2xx_SOC && MACH_MAINSTONE ++ select SND_PXA2xx_AC97 ++ help ++ Say Y if you want to add support for generic AC97 SoC audio on Mainstone. ++ ++config SND_PXA2xx_SOC_MAINSTONE_WM8731 ++ tristate "SoC I2S Audio support for Intel Mainstone - WM8731" ++ depends on SND_PXA2xx_SOC && MACH_MAINSTONE ++ select SND_PXA2xx_SOC_I2S ++ help ++ Say Y if you want to add support for SoC audio on Mainstone ++ with the WM8731. ++ ++config SND_PXA2xx_SOC_MAINSTONE_WM8753 ++ tristate "SoC I2S/SSP Audio support for Intel Mainstone - WM8753" ++ depends on SND_PXA2xx_SOC && MACH_MAINSTONE ++ select SND_PXA2xx_SOC_I2S ++ select SND_PXA2xx_SOC_SSP ++ help ++ Say Y if you want to add support for SoC audio on Mainstone ++ with the WM8753. ++ ++config SND_PXA2xx_SOC_MAINSTONE_WM8974 ++ tristate "SoC I2S Audio support for Intel Mainstone - WM8974" ++ depends on SND_PXA2xx_SOC && MACH_MAINSTONE ++ select SND_PXA2xx_SOC_I2S ++ help ++ Say Y if you want to add support for SoC audio on Mainstone ++ with the WM8974. ++ ++config SND_PXA2xx_SOC_MAINSTONE_WM9713 ++ tristate "SoC I2S/SSP Audio support for Intel Mainstone - WM9713" ++ depends on SND_PXA2xx_SOC && MACH_MAINSTONE ++ select SND_PXA2xx_SOC_AC97 ++ select SND_PXA2xx_SOC_SSP ++ help ++ Say Y if you want to add support for SoC voice audio on Mainstone ++ with the WM9713. ++ ++config SND_MAINSTONE_BASEBAND ++ tristate "Example SoC Baseband Audio support for Intel Mainstone" ++ depends on SND_PXA2xx_SOC && MACH_MAINSTONE ++ select SND_PXA2xx_SOC_AC97 ++ help ++ Say Y if you want to add support for SoC baseband on Mainstone ++ with the WM9713 and example Baseband modem. ++ ++config SND_MAINSTONE_BLUETOOTH ++ tristate "Example SoC Bluetooth Audio support for Intel Mainstone" ++ depends on SND_PXA2xx_SOC && MACH_MAINSTONE ++ select SND_PXA2xx_SOC_I2S ++ help ++ Say Y if you want to add support for SoC bluetooth on Mainstone ++ with the WM8753 and example Bluetooth codec. ++ ++config SND_PXA2xx_SOC_MAINSTONE_WM9712 ++ tristate "SoC I2S/SSP Audio support for Intel Mainstone - WM9712" ++ depends on SND_PXA2xx_SOC && MACH_MAINSTONE ++ select SND_PXA2xx_SOC_AC97 ++ help ++ Say Y if you want to add support for SoC voice audio on Mainstone ++ with the WM9712. ++ ++config SND_PXA2xx_SOC_CORGI ++ tristate "SoC Audio support for Sharp Zaurus SL-C7x0" ++ depends on SND_PXA2xx_SOC && PXA_SHARP_C7xx ++ select SND_PXA2xx_SOC_I2S ++ help ++ Say Y if you want to add support for SoC audio on Sharp ++ Zaurus SL-C7x0 models (Corgi, Shepherd, Husky). ++ ++config SND_PXA2xx_SOC_SPITZ ++ tristate "SoC Audio support for Sharp Zaurus SL-Cxx00" ++ depends on SND_PXA2xx_SOC && PXA_SHARP_Cxx00 ++ select SND_PXA2xx_SOC_I2S ++ help ++ Say Y if you want to add support for SoC audio on Sharp ++ Zaurus SL-Cxx00 models (Spitz, Borzoi and Akita). ++ ++config SND_PXA2xx_SOC_POODLE ++ tristate "SoC Audio support for Poodle" ++ depends on SND_PXA2xx_SOC && MACH_POODLE ++ select SND_PXA2xx_SOC_I2S ++ help ++ Say Y if you want to add support for SoC audio on Sharp ++ Zaurus SL-5600 model (Poodle). ++ ++config SND_PXA2xx_SOC_TOSA ++ tristate "SoC AC97 Audio support for Tosa" ++ depends on SND_PXA2xx_SOC && MACH_TOSA ++ select SND_PXA2xx_SOC_AC97 ++ help ++ Say Y if you want to add support for SoC audio on Sharp ++ Zaurus SL-C6000x models (Tosa). ++ ++endmenu +Index: linux-2.6-pxa-new/sound/soc/pxa/Makefile +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/pxa/Makefile +@@ -0,0 +1,36 @@ ++# PXA Platform Support ++snd-soc-pxa2xx-objs := pxa2xx-pcm.o ++snd-soc-pxa2xx-ac97-objs := pxa2xx-ac97.o ++snd-soc-pxa2xx-i2s-objs := pxa2xx-i2s.o ++snd-soc-pxa2xx-ssp-objs := pxa2xx-ssp.o ++ ++obj-$(CONFIG_SND_PXA2xx_SOC) += snd-soc-pxa2xx.o ++obj-$(CONFIG_SND_PXA2xx_SOC_AC97) += snd-soc-pxa2xx-ac97.o ++obj-$(CONFIG_SND_PXA2xx_SOC_I2S) += snd-soc-pxa2xx-i2s.o ++obj-$(CONFIG_SND_PXA2xx_SOC_SSP) += snd-soc-pxa2xx-ssp.o ++ ++# PXA Machine Support ++snd-soc-corgi-objs := corgi.o ++snd-soc-mainstone-wm8731-objs := mainstone_wm8731.o ++snd-soc-mainstone-wm8753-objs := mainstone_wm8753.o ++snd-soc-mainstone-wm8974-objs := mainstone_wm8974.o ++snd-soc-mainstone-wm9713-objs := mainstone_wm9713.o ++snd-soc-mainstone-wm9712-objs := mainstone_wm9712.o ++snd-soc-mainstone-baseband-objs := mainstone_baseband.o ++snd-soc-mainstone-bluetooth-objs := mainstone_bluetooth.o ++snd-soc-poodle-objs := poodle.o ++snd-soc-tosa-objs := tosa.o ++snd-soc-spitz-objs := spitz.o ++ ++obj-$(CONFIG_SND_PXA2xx_SOC_CORGI) += snd-soc-corgi.o ++obj-$(CONFIG_SND_PXA2xx_SOC_MAINSTONE_WM8731) += snd-soc-mainstone-wm8731.o ++obj-$(CONFIG_SND_PXA2xx_SOC_MAINSTONE_WM8753) += snd-soc-mainstone-wm8753.o ++obj-$(CONFIG_SND_PXA2xx_SOC_MAINSTONE_WM8974) += snd-soc-mainstone-wm8974.o ++obj-$(CONFIG_SND_PXA2xx_SOC_MAINSTONE_WM9713) += snd-soc-mainstone-wm9713.o ++obj-$(CONFIG_SND_PXA2xx_SOC_MAINSTONE_WM9712) += snd-soc-mainstone-wm9712.o ++obj-$(CONFIG_SND_MAINSTONE_BASEBAND) += snd-soc-mainstone-baseband.o ++obj-$(CONFIG_SND_MAINSTONE_BLUETOOTH) += snd-soc-mainstone-bluetooth.o ++obj-$(CONFIG_SND_PXA2xx_SOC_POODLE) += snd-soc-poodle.o ++obj-$(CONFIG_SND_PXA2xx_SOC_TOSA) += snd-soc-tosa.o ++obj-$(CONFIG_SND_PXA2xx_SOC_SPITZ) += snd-soc-spitz.o ++ +Index: linux-2.6-pxa-new/sound/soc/pxa/corgi.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/pxa/corgi.c +@@ -0,0 +1,361 @@ ++/* ++ * corgi.c -- SoC audio for Corgi ++ * ++ * Copyright 2005 Wolfson Microelectronics PLC. ++ * Copyright 2005 Openedhand Ltd. ++ * ++ * Authors: Liam Girdwood <liam.girdwood@wolfsonmicro.com> ++ * Richard Purdie <richard@openedhand.com> ++ * ++ * 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. ++ * ++ * Revision history ++ * 30th Nov 2005 Initial version. ++ * ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/timer.h> ++#include <linux/interrupt.h> ++#include <linux/platform_device.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++ ++#include <asm/mach-types.h> ++#include <asm/hardware/scoop.h> ++#include <asm/arch/pxa-regs.h> ++#include <asm/arch/hardware.h> ++#include <asm/arch/corgi.h> ++#include <asm/arch/audio.h> ++ ++#include "../codecs/wm8731.h" ++#include "pxa2xx-pcm.h" ++ ++#define CORGI_HP 0 ++#define CORGI_MIC 1 ++#define CORGI_LINE 2 ++#define CORGI_HEADSET 3 ++#define CORGI_HP_OFF 4 ++#define CORGI_SPK_ON 0 ++#define CORGI_SPK_OFF 1 ++ ++ /* audio clock in Hz - rounded from 12.235MHz */ ++#define CORGI_AUDIO_CLOCK 12288000 ++ ++static int corgi_jack_func; ++static int corgi_spk_func; ++ ++static void corgi_ext_control(struct snd_soc_codec *codec) ++{ ++ int spk = 0, mic = 0, line = 0, hp = 0, hs = 0; ++ ++ /* set up jack connection */ ++ switch (corgi_jack_func) { ++ case CORGI_HP: ++ hp = 1; ++ /* set = unmute headphone */ ++ set_scoop_gpio(&corgiscoop_device.dev, CORGI_SCP_MUTE_L); ++ set_scoop_gpio(&corgiscoop_device.dev, CORGI_SCP_MUTE_R); ++ break; ++ case CORGI_MIC: ++ mic = 1; ++ /* reset = mute headphone */ ++ reset_scoop_gpio(&corgiscoop_device.dev, CORGI_SCP_MUTE_L); ++ reset_scoop_gpio(&corgiscoop_device.dev, CORGI_SCP_MUTE_R); ++ break; ++ case CORGI_LINE: ++ line = 1; ++ reset_scoop_gpio(&corgiscoop_device.dev, CORGI_SCP_MUTE_L); ++ reset_scoop_gpio(&corgiscoop_device.dev, CORGI_SCP_MUTE_R); ++ break; ++ case CORGI_HEADSET: ++ hs = 1; ++ mic = 1; ++ reset_scoop_gpio(&corgiscoop_device.dev, CORGI_SCP_MUTE_L); ++ set_scoop_gpio(&corgiscoop_device.dev, CORGI_SCP_MUTE_R); ++ break; ++ } ++ ++ if (corgi_spk_func == CORGI_SPK_ON) ++ spk = 1; ++ ++ /* set the enpoints to their new connetion states */ ++ snd_soc_dapm_set_endpoint(codec, "Ext Spk", spk); ++ snd_soc_dapm_set_endpoint(codec, "Mic Jack", mic); ++ snd_soc_dapm_set_endpoint(codec, "Line Jack", line); ++ snd_soc_dapm_set_endpoint(codec, "Headphone Jack", hp); ++ snd_soc_dapm_set_endpoint(codec, "Headset Jack", hs); ++ ++ /* signal a DAPM event */ ++ snd_soc_dapm_sync_endpoints(codec); ++} ++ ++static int corgi_startup(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_codec *codec = rtd->socdev->codec; ++ ++ /* check the jack status at stream startup */ ++ corgi_ext_control(codec); ++ return 0; ++} ++ ++/* we need to unmute the HP at shutdown as the mute burns power on corgi */ ++static int corgi_shutdown(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_codec *codec = rtd->socdev->codec; ++ ++ /* set = unmute headphone */ ++ set_scoop_gpio(&corgiscoop_device.dev, CORGI_SCP_MUTE_L); ++ set_scoop_gpio(&corgiscoop_device.dev, CORGI_SCP_MUTE_R); ++ return 0; ++} ++ ++static struct snd_soc_ops corgi_ops = { ++ .startup = corgi_startup, ++ .shutdown = corgi_shutdown, ++}; ++ ++static int corgi_get_jack(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ ucontrol->value.integer.value[0] = corgi_jack_func; ++ return 0; ++} ++ ++static int corgi_set_jack(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ ++ if (corgi_jack_func == ucontrol->value.integer.value[0]) ++ return 0; ++ ++ corgi_jack_func = ucontrol->value.integer.value[0]; ++ corgi_ext_control(codec); ++ return 1; ++} ++ ++static int corgi_get_spk(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ ucontrol->value.integer.value[0] = corgi_spk_func; ++ return 0; ++} ++ ++static int corgi_set_spk(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ ++ if (corgi_spk_func == ucontrol->value.integer.value[0]) ++ return 0; ++ ++ corgi_spk_func = ucontrol->value.integer.value[0]; ++ corgi_ext_control(codec); ++ return 1; ++} ++ ++static int corgi_amp_event(struct snd_soc_dapm_widget *w, int event) ++{ ++ if (SND_SOC_DAPM_EVENT_ON(event)) ++ set_scoop_gpio(&corgiscoop_device.dev, CORGI_SCP_APM_ON); ++ else ++ reset_scoop_gpio(&corgiscoop_device.dev, CORGI_SCP_APM_ON); ++ ++ return 0; ++} ++ ++static int corgi_mic_event(struct snd_soc_dapm_widget *w, int event) ++{ ++ if (SND_SOC_DAPM_EVENT_ON(event)) ++ set_scoop_gpio(&corgiscoop_device.dev, CORGI_SCP_MIC_BIAS); ++ else ++ reset_scoop_gpio(&corgiscoop_device.dev, CORGI_SCP_MIC_BIAS); ++ ++ return 0; ++} ++ ++/* corgi machine dapm widgets */ ++static const struct snd_soc_dapm_widget wm8731_dapm_widgets[] = { ++SND_SOC_DAPM_HP("Headphone Jack", NULL), ++SND_SOC_DAPM_MIC("Mic Jack", corgi_mic_event), ++SND_SOC_DAPM_SPK("Ext Spk", corgi_amp_event), ++SND_SOC_DAPM_LINE("Line Jack", NULL), ++SND_SOC_DAPM_HP("Headset Jack", NULL), ++}; ++ ++/* Corgi machine audio map (connections to the codec pins) */ ++static const char *audio_map[][3] = { ++ ++ /* headset Jack - in = micin, out = LHPOUT*/ ++ {"Headset Jack", NULL, "LHPOUT"}, ++ ++ /* headphone connected to LHPOUT1, RHPOUT1 */ ++ {"Headphone Jack", NULL, "LHPOUT"}, ++ {"Headphone Jack", NULL, "RHPOUT"}, ++ ++ /* speaker connected to LOUT, ROUT */ ++ {"Ext Spk", NULL, "ROUT"}, ++ {"Ext Spk", NULL, "LOUT"}, ++ ++ /* mic is connected to MICIN (via right channel of headphone jack) */ ++ {"MICIN", NULL, "Mic Jack"}, ++ ++ /* Same as the above but no mic bias for line signals */ ++ {"MICIN", NULL, "Line Jack"}, ++ ++ {NULL, NULL, NULL}, ++}; ++ ++static const char *jack_function[] = {"Headphone", "Mic", "Line", "Headset", ++ "Off"}; ++static const char *spk_function[] = {"On", "Off"}; ++static const struct soc_enum corgi_enum[] = { ++ SOC_ENUM_SINGLE_EXT(5, jack_function), ++ SOC_ENUM_SINGLE_EXT(2, spk_function), ++}; ++ ++static const struct snd_kcontrol_new wm8731_corgi_controls[] = { ++ SOC_ENUM_EXT("Jack Function", corgi_enum[0], corgi_get_jack, ++ corgi_set_jack), ++ SOC_ENUM_EXT("Speaker Function", corgi_enum[1], corgi_get_spk, ++ corgi_set_spk), ++}; ++ ++/* ++ * Logic for a wm8731 as connected on a Sharp SL-C7x0 Device ++ */ ++static int corgi_wm8731_init(struct snd_soc_codec *codec) ++{ ++ int i, err; ++ ++ snd_soc_dapm_set_endpoint(codec, "LLINEIN", 0); ++ snd_soc_dapm_set_endpoint(codec, "RLINEIN", 0); ++ ++ /* Add corgi specific controls */ ++ for (i = 0; i < ARRAY_SIZE(wm8731_corgi_controls); i++) { ++ err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&wm8731_corgi_controls[i],codec, NULL)); ++ if (err < 0) ++ return err; ++ } ++ ++ /* Add corgi specific widgets */ ++ for(i = 0; i < ARRAY_SIZE(wm8731_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &wm8731_dapm_widgets[i]); ++ } ++ ++ /* Set up corgi specific audio path audio_map */ ++ for(i = 0; audio_map[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, audio_map[i][0], ++ audio_map[i][1], audio_map[i][2]); ++ } ++ ++ snd_soc_dapm_sync_endpoints(codec); ++ return 0; ++} ++ ++static unsigned int corgi_config_sysclk(struct snd_soc_pcm_runtime *rtd, ++ struct snd_soc_clock_info *info) ++{ ++ if (info->bclk_master & SND_SOC_DAIFMT_CBS_CFS) { ++ /* pxa2xx is i2s master */ ++ switch (info->rate) { ++ case 44100: ++ case 88200: ++ /* configure codec digital filters for 44.1, 88.2 */ ++ rtd->codec_dai->config_sysclk(rtd->codec_dai, info, ++ 11289600); ++ break; ++ default: ++ /* configure codec digital filters for all other rates */ ++ rtd->codec_dai->config_sysclk(rtd->codec_dai, info, ++ CORGI_AUDIO_CLOCK); ++ break; ++ } ++ /* config pxa i2s as master */ ++ return rtd->cpu_dai->config_sysclk(rtd->cpu_dai, info, ++ CORGI_AUDIO_CLOCK); ++ } else { ++ /* codec is i2s master - ++ * only configure codec DAI clock and filters */ ++ return rtd->codec_dai->config_sysclk(rtd->codec_dai, info, ++ CORGI_AUDIO_CLOCK); ++ } ++} ++ ++/* corgi digital audio interface glue - connects codec <--> CPU */ ++static struct snd_soc_dai_link corgi_dai = { ++ .name = "WM8731", ++ .stream_name = "WM8731", ++ .cpu_dai = &pxa_i2s_dai, ++ .codec_dai = &wm8731_dai, ++ .init = corgi_wm8731_init, ++ .config_sysclk = corgi_config_sysclk, ++}; ++ ++/* corgi audio machine driver */ ++static struct snd_soc_machine snd_soc_machine_corgi = { ++ .name = "Corgi", ++ .dai_link = &corgi_dai, ++ .num_links = 1, ++ .ops = &corgi_ops, ++}; ++ ++/* corgi audio private data */ ++static struct wm8731_setup_data corgi_wm8731_setup = { ++ .i2c_address = 0x1b, ++}; ++ ++/* corgi audio subsystem */ ++static struct snd_soc_device corgi_snd_devdata = { ++ .machine = &snd_soc_machine_corgi, ++ .platform = &pxa2xx_soc_platform, ++ .codec_dev = &soc_codec_dev_wm8731, ++ .codec_data = &corgi_wm8731_setup, ++}; ++ ++static struct platform_device *corgi_snd_device; ++ ++static int __init corgi_init(void) ++{ ++ int ret; ++ ++ if (!(machine_is_corgi() || machine_is_shepherd() || machine_is_husky())) ++ return -ENODEV; ++ ++ corgi_snd_device = platform_device_alloc("soc-audio", -1); ++ if (!corgi_snd_device) ++ return -ENOMEM; ++ ++ platform_set_drvdata(corgi_snd_device, &corgi_snd_devdata); ++ corgi_snd_devdata.dev = &corgi_snd_device->dev; ++ ret = platform_device_add(corgi_snd_device); ++ ++ if (ret) ++ platform_device_put(corgi_snd_device); ++ ++ return ret; ++} ++ ++static void __exit corgi_exit(void) ++{ ++ platform_device_unregister(corgi_snd_device); ++} ++ ++module_init(corgi_init); ++module_exit(corgi_exit); ++ ++/* Module information */ ++MODULE_AUTHOR("Richard Purdie"); ++MODULE_DESCRIPTION("ALSA SoC Corgi"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/pxa/mainstone.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/pxa/mainstone.c +@@ -0,0 +1,126 @@ ++/* ++ * mainstone.c -- SoC audio for Mainstone ++ * ++ * Copyright 2005 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * Mainstone audio amplifier code taken from arch/arm/mach-pxa/mainstone.c ++ * Copyright: MontaVista Software Inc. ++ * ++ * 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. ++ * ++ * Revision history ++ * 30th Oct 2005 Initial version. ++ * ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/device.h> ++#include <linux/i2c.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++ ++#include <asm/arch/pxa-regs.h> ++#include <asm/arch/mainstone.h> ++#include <asm/arch/audio.h> ++ ++#include "../codecs/ac97.h" ++#include "pxa2xx-pcm.h" ++ ++static struct snd_soc_machine mainstone; ++static long mst_audio_suspend_mask; ++ ++static int mainstone_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ mst_audio_suspend_mask = MST_MSCWR2; ++ MST_MSCWR2 |= MST_MSCWR2_AC97_SPKROFF; ++ return 0; ++} ++ ++static int mainstone_resume(struct platform_device *pdev) ++{ ++ MST_MSCWR2 &= mst_audio_suspend_mask | ~MST_MSCWR2_AC97_SPKROFF; ++ return 0; ++} ++ ++static int mainstone_probe(struct platform_device *pdev) ++{ ++ MST_MSCWR2 &= ~MST_MSCWR2_AC97_SPKROFF; ++ return 0; ++} ++ ++static int mainstone_remove(struct platform_device *pdev) ++{ ++ MST_MSCWR2 |= MST_MSCWR2_AC97_SPKROFF; ++ return 0; ++} ++ ++static struct snd_soc_machine_config codecs[] = { ++{ ++ .name = "AC97", ++ .sname = "AC97 HiFi", ++ .iface = &pxa_ac97_interface[0], ++}, ++{ ++ .name = "AC97 Aux", ++ .sname = "AC97 Aux", ++ .iface = &pxa_ac97_interface[1], ++}, ++}; ++ ++static struct snd_soc_machine mainstone = { ++ .name = "Mainstone", ++ .probe = mainstone_probe, ++ .remove = mainstone_remove, ++ .suspend_pre = mainstone_suspend, ++ .resume_post = mainstone_resume, ++ .config = codecs, ++ .nconfigs = ARRAY_SIZE(codecs), ++}; ++ ++static struct snd_soc_device mainstone_snd_devdata = { ++ .machine = &mainstone, ++ .platform = &pxa2xx_soc_platform, ++ .codec_dev = &soc_codec_dev_ac97, ++}; ++ ++static struct platform_device *mainstone_snd_device; ++ ++static int __init mainstone_init(void) ++{ ++ int ret; ++ ++ mainstone_snd_device = platform_device_alloc("soc-audio", -1); ++ if (!mainstone_snd_device) ++ return -ENOMEM; ++ ++ platform_set_drvdata(mainstone_snd_device, &mainstone_snd_devdata); ++ mainstone_snd_devdata.dev = &mainstone_snd_device->dev; ++ ret = platform_device_add(mainstone_snd_device); ++ ++ if (ret) ++ platform_device_put(mainstone_snd_device); ++ ++ return ret; ++} ++ ++static void __exit mainstone_exit(void) ++{ ++ platform_device_unregister(mainstone_snd_device); ++} ++ ++module_init(mainstone_init); ++module_exit(mainstone_exit); ++ ++/* Module information */ ++MODULE_AUTHOR("Liam Girdwood, liam.girdwood@wolfsonmicro.com, www.wolfsonmicro.com"); ++MODULE_DESCRIPTION("ALSA SoC Mainstone"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/pxa/mainstone_baseband.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/pxa/mainstone_baseband.c +@@ -0,0 +1,249 @@ ++/* ++ * mainstone_baseband.c ++ * Mainstone Example Baseband modem -- ALSA Soc Audio Layer ++ * ++ * Copyright 2006 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * 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. ++ * ++ * Revision history ++ * 15th Apr 2006 Initial version. ++ * ++ * This is example code to demonstrate connecting a baseband modem to the PCM ++ * DAI on the WM9713 codec on the Intel Mainstone platform. It is by no means ++ * complete as it requires code to control the modem. ++ * ++ * The architecture consists of the WM9713 AC97 DAI connected to the PXA27x ++ * AC97 controller and the WM9713 PCM DAI connected to the basebands DAI. The ++ * baseband is controlled via a serial port. Audio is routed between the PXA27x ++ * and the baseband via internal WM9713 analog paths. ++ * ++ * This driver is not the baseband modem driver. This driver only calls ++ * functions from the Baseband driver to set up it's PCM DAI. ++ * ++ * It's intended to use this driver as follows:- ++ * ++ * 1. open() WM9713 PCM audio device. ++ * 2. open() serial device (for AT commands). ++ * 3. configure PCM audio device (rate etc) - sets up WM9713 PCM DAI, ++ * this will also set up the baseband PCM DAI (via calling baseband driver). ++ * 4. send any further AT commands to set up baseband. ++ * 5. configure codec audio mixer paths. ++ * 6. open(), configure and read/write AC97 audio device - to Tx/Rx voice ++ * ++ * The PCM audio device is opened but IO is never performed on it as the IO is ++ * directly between the codec and the baseband (and not the CPU). ++ * ++ * TODO: ++ * o Implement callbacks ++ */ ++ ++#include <linux/init.h> ++#include <linux/module.h> ++#include <linux/platform_device.h> ++ ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++ ++#include <asm/hardware.h> ++#include <asm/arch/pxa-regs.h> ++#include <asm/arch/audio.h> ++#include <asm/arch/ssp.h> ++ ++#include "../codecs/wm9713.h" ++#include "pxa2xx-pcm.h" ++ ++static struct snd_soc_machine mainstone; ++ ++#define BASEBAND_XXX_DAIFMT \ ++ (SND_SOC_DAIFMT_DSP_B | SND_SOC_DAIFMT_CBS_CFS |\ ++ SND_SOC_DAIFMT_NB_NF) ++ ++#define BASEBAND_XXX_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++/* ++ * PCM modes - 8k 16bit mono baseband modem is master ++ */ ++static struct snd_soc_dai_mode mainstone_example_modes[] = { ++ /* port master clk & frame modes */ ++ {BASEBAND_XXX_DAIFMT, SND_SOC_DAITDM_LRDW(0,0), SNDRV_PCM_FORMAT_S16_LE, ++ SNDRV_PCM_RATE_8000, BASEBAND_XXX_DIR, SND_SOC_DAI_BFS_RATE, 256, 64}, ++}; ++ ++/* Do specific baseband PCM voice startup here */ ++static int mainstone_baseband_startup(struct snd_pcm_substream *substream) ++{ ++ return 0; ++} ++ ++/* Do specific baseband PCM voice shutdown here */ ++static void mainstone_baseband_shutdown (struct snd_pcm_substream *substream) ++{ ++} ++ ++/* Do specific baseband modem PCM voice hw params init here */ ++static int mainstone_baseband_hw_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ return 0; ++} ++ ++/* Do specific baseband modem PCM voice hw params free here */ ++static int mainstone_baseband_hw_free(struct snd_pcm_substream *substream) ++{ ++ return 0; ++} ++ ++static struct snd_soc_cpu_dai mainstone_example_dai[] = { ++ { .name = "Baseband", ++ .id = 0, ++ .type = SND_SOC_DAI_PCM, ++ .playback = { ++ .channels_min = 1, ++ .channels_max = 1,}, ++ .capture = { ++ .channels_min = 1, ++ .channels_max = 1,}, ++ .ops = { ++ .startup = mainstone_baseband_startup, ++ .shutdown = mainstone_baseband_shutdown, ++ .hw_params = mainstone_baseband_hw_params, ++ .hw_free = mainstone_baseband_hw_free, ++ }, ++ .caps = { ++ .mode = mainstone_example_modes, ++ .num_modes = ARRAY_SIZE(mainstone_example_modes),}, ++ }, ++}; ++ ++/* do we need to do any thing on the mainstone when the stream is ++ * started and stopped ++ */ ++static int mainstone_startup(struct snd_pcm_substream *substream) ++{ ++ return 0; ++} ++ ++static void mainstone_shutdown(struct snd_pcm_substream *substream) ++{ ++} ++ ++static struct snd_soc_ops mainstone_ops = { ++ .startup = mainstone_startup, ++ .shutdown = mainstone_shutdown, ++}; ++ ++/* PM */ ++static int mainstone_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ return 0; ++} ++ ++static int mainstone_resume(struct platform_device *pdev) ++{ ++ return 0; ++} ++ ++static int mainstone_probe(struct platform_device *pdev) ++{ ++ return 0; ++} ++ ++static int mainstone_remove(struct platform_device *pdev) ++{ ++ return 0; ++} ++ ++static int mainstone_wm9713_init(struct snd_soc_codec *codec) ++{ ++ return 0; ++} ++ ++unsigned int mainstone_config_sysclk(struct snd_soc_pcm_runtime *rtd, ++ struct snd_soc_clock_info *info) ++{ ++ /* wm8753 has pll that generates mclk from 13MHz xtal */ ++ return rtd->codec_dai->config_sysclk(rtd->codec_dai, info, 13000000); ++} ++ ++/* the physical audio connections between the WM9713, Baseband and pxa2xx */ ++static struct snd_soc_dai_link mainstone_dai[] = { ++{ ++ .name = "AC97", ++ .stream_name = "AC97 HiFi", ++ .cpu_dai = &pxa_ac97_dai[PXA2XX_DAI_AC97_HIFI], ++ .codec_dai = &wm9713_dai[WM9713_DAI_AC97_HIFI], ++ .init = mainstone_wm9713_init, ++}, ++{ ++ .name = "AC97 Aux", ++ .stream_name = "AC97 Aux", ++ .cpu_dai = &pxa_ac97_dai[PXA2XX_DAI_AC97_AUX], ++ .codec_dai = &wm9713_dai[WM9713_DAI_AC97_AUX], ++}, ++{ ++ .name = "Baseband", ++ .stream_name = "Voice", ++ .cpu_dai = mainstone_example_dai, ++ .codec_dai = &wm9713_dai[WM9713_DAI_PCM_VOICE], ++ .config_sysclk = mainstone_config_sysclk, ++}, ++}; ++ ++static struct snd_soc_machine mainstone = { ++ .name = "Mainstone", ++ .probe = mainstone_probe, ++ .remove = mainstone_remove, ++ .suspend_pre = mainstone_suspend, ++ .resume_post = mainstone_resume, ++ .ops = &mainstone_ops, ++ .dai_link = mainstone_dai, ++ .num_links = ARRAY_SIZE(mainstone_dai), ++}; ++ ++static struct snd_soc_device mainstone_snd_ac97_devdata = { ++ .machine = &mainstone, ++ .platform = &pxa2xx_soc_platform, ++ .codec_dev = &soc_codec_dev_wm9713, ++}; ++ ++static struct platform_device *mainstone_snd_ac97_device; ++ ++static int __init mainstone_init(void) ++{ ++ int ret; ++ ++ mainstone_snd_ac97_device = platform_device_alloc("soc-audio", -1); ++ if (!mainstone_snd_ac97_device) ++ return -ENOMEM; ++ ++ platform_set_drvdata(mainstone_snd_ac97_device, &mainstone_snd_ac97_devdata); ++ mainstone_snd_ac97_devdata.dev = &mainstone_snd_ac97_device->dev; ++ ++ if((ret = platform_device_add(mainstone_snd_ac97_device)) != 0) ++ platform_device_put(mainstone_snd_ac97_device); ++ ++ return ret; ++} ++ ++static void __exit mainstone_exit(void) ++{ ++ platform_device_unregister(mainstone_snd_ac97_device); ++} ++ ++module_init(mainstone_init); ++module_exit(mainstone_exit); ++ ++/* Module information */ ++MODULE_AUTHOR("Liam Girdwood, liam.girdwood@wolfsonmicro.com, www.wolfsonmicro.com"); ++MODULE_DESCRIPTION("Mainstone Example Baseband PCM Interface"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/pxa/mainstone_bluetooth.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/pxa/mainstone_bluetooth.c +@@ -0,0 +1,399 @@ ++/* ++ * mainstone_bluetooth.c ++ * Mainstone Example Bluetooth -- ALSA Soc Audio Layer ++ * ++ * Copyright 2006 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * 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. ++ * ++ * Revision history ++ * 15th May 2006 Initial version. ++ * ++ * This is example code to demonstrate connecting a bluetooth codec to the PCM ++ * DAI on the WM8753 codec on the Intel Mainstone platform. It is by no means ++ * complete as it requires code to control the BT codec. ++ * ++ * The architecture consists of the WM8753 HIFI DAI connected to the PXA27x ++ * I2S controller and the WM8753 PCM DAI connected to the bluetooth DAI. The ++ * bluetooth codec and wm8753 are controlled via I2C. Audio is routed between ++ * the PXA27x and the bluetooth via internal WM8753 analog paths. ++ * ++ * This example supports the following audio input/outputs. ++ * ++ * o Board mounted Mic and Speaker (spk has amplifier) ++ * o Headphones via jack socket ++ * o BT source and sink ++ * ++ * This driver is not the bluetooth codec driver. This driver only calls ++ * functions from the Bluetooth driver to set up it's PCM DAI. ++ * ++ * It's intended to use the driver as follows:- ++ * ++ * 1. open() WM8753 PCM audio device. ++ * 2. configure PCM audio device (rate etc) - sets up WM8753 PCM DAI, ++ * this should also set up the BT codec DAI (via calling bt driver). ++ * 3. configure codec audio mixer paths. ++ * 4. open(), configure and read/write HIFI audio device - to Tx/Rx voice ++ * ++ * The PCM audio device is opened but IO is never performed on it as the IO is ++ * directly between the codec and the BT codec (and not the CPU). ++ * ++ * TODO: ++ * o Implement callbacks ++ */ ++ ++#include <linux/init.h> ++#include <linux/module.h> ++#include <linux/platform_device.h> ++ ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++ ++#include <asm/hardware.h> ++#include <asm/arch/pxa-regs.h> ++#include <asm/arch/audio.h> ++#include <asm/arch/ssp.h> ++ ++#include "../codecs/wm8753.h" ++#include "pxa2xx-pcm.h" ++ ++static struct snd_soc_machine mainstone; ++ ++#define BLUETOOTH_DAIFMT \ ++ (SND_SOC_DAIFMT_DSP_B | SND_SOC_DAIFMT_CBS_CFS |\ ++ SND_SOC_DAIFMT_NB_NF) ++ ++#define BLUETOOTH_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++/* ++ * PCM modes - 8k 16bit mono BT codec is master ++ */ ++static struct snd_soc_dai_mode mainstone_bt_modes[] = { ++ /* port master clk & frame modes */ ++ {BLUETOOTH_DAIFMT, SND_SOC_DAITDM_LRDW(0,0), SNDRV_PCM_FORMAT_S16_LE, ++ SNDRV_PCM_RATE_8000, BLUETOOTH_DIR, SND_SOC_DAI_BFS_RATE, 256, 64}, ++}; ++ ++/* Do specific bluetooth PCM startup here */ ++static int mainstone_bt_startup(struct snd_pcm_substream *substream) ++{ ++ return 0; ++} ++ ++/* Do specific bluetooth PCM shutdown here */ ++static void mainstone_bt_shutdown (struct snd_pcm_substream *substream) ++{ ++} ++ ++/* Do pecific bluetooth PCM hw params init here */ ++static int mainstone_bt_hw_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ return 0; ++} ++ ++/* Do specific bluetooth PCM hw params free here */ ++static int mainstone_bt_hw_free(struct snd_pcm_substream *substream) ++{ ++ return 0; ++} ++ ++static struct snd_soc_cpu_dai mainstone_bt_dai[] = { ++ { .name = "Bluetooth", ++ .id = 0, ++ .type = SND_SOC_DAI_PCM, ++ .playback = { ++ .channels_min = 1, ++ .channels_max = 1,}, ++ .capture = { ++ .channels_min = 1, ++ .channels_max = 1,}, ++ .ops = { ++ .startup = mainstone_bt_startup, ++ .shutdown = mainstone_bt_shutdown, ++ .hw_params = mainstone_bt_hw_params, ++ .hw_free = mainstone_bt_hw_free, ++ }, ++ .caps = { ++ .mode = mainstone_bt_modes, ++ .num_modes = ARRAY_SIZE(mainstone_bt_modes),}, ++ }, ++}; ++ ++/* do we need to do any thing on the mainstone when the stream is ++ * started and stopped ++ */ ++static int mainstone_startup(struct snd_pcm_substream *substream) ++{ ++ return 0; ++} ++ ++static void mainstone_shutdown(struct snd_pcm_substream *substream) ++{ ++} ++ ++static struct snd_soc_ops mainstone_ops = { ++ .startup = mainstone_startup, ++ .shutdown = mainstone_shutdown, ++}; ++ ++/* PM */ ++static int mainstone_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ return 0; ++} ++ ++static int mainstone_resume(struct platform_device *pdev) ++{ ++ return 0; ++} ++ ++static int mainstone_probe(struct platform_device *pdev) ++{ ++ return 0; ++} ++ ++static int mainstone_remove(struct platform_device *pdev) ++{ ++ return 0; ++} ++ ++/* ++ * Machine audio functions. ++ * ++ * The machine now has 3 extra audio controls. ++ * ++ * Jack function: Sets function (device plugged into Jack) to nothing (Off) ++ * or Headphones. ++ * ++ * Mic function: Set the on board Mic to On or Off ++ * Spk function: Set the on board Spk to On or Off ++ * ++ * example: BT playback (of far end) and capture (of near end) ++ * Set Mic and Speaker to On, open BT alsa interface as above and set up ++ * internal audio paths. ++ */ ++ ++static int machine_jack_func = 0; ++static int machine_spk_func = 0; ++static int machine_mic_func = 0; ++ ++static int machine_get_jack(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ ucontrol->value.integer.value[0] = machine_jack_func; ++ return 0; ++} ++ ++static int machine_set_jack(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ machine_jack_func = ucontrol->value.integer.value[0]; ++ snd_soc_dapm_set_endpoint(codec, "Headphone Jack", machine_jack_func); ++ return 0; ++} ++ ++static int machine_get_spk(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ ucontrol->value.integer.value[0] = machine_spk_func; ++ return 0; ++} ++ ++static int machine_set_spk(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ machine_spk_func = ucontrol->value.integer.value[0]; ++ snd_soc_dapm_set_endpoint(codec, "Spk", machine_spk_func); ++ return 0; ++} ++ ++static int machine_get_mic(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ ucontrol->value.integer.value[0] = machine_spk_func; ++ return 0; ++} ++ ++static int machine_set_mic(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ machine_spk_func = ucontrol->value.integer.value[0]; ++ snd_soc_dapm_set_endpoint(codec, "Mic", machine_mic_func); ++ return 0; ++} ++ ++/* turns on board speaker amp on/off */ ++static int machine_amp_event(struct snd_soc_dapm_widget *w, int event) ++{ ++#if 0 ++ if (SND_SOC_DAPM_EVENT_ON(event)) ++ /* on */ ++ else ++ /* off */ ++#endif ++ return 0; ++} ++ ++/* machine dapm widgets */ ++static const struct snd_soc_dapm_widget machine_dapm_widgets[] = { ++SND_SOC_DAPM_HP("Headphone Jack", NULL), ++SND_SOC_DAPM_SPK("Spk", machine_amp_event), ++SND_SOC_DAPM_MIC("Mic", NULL), ++}; ++ ++/* machine connections to the codec pins */ ++static const char* audio_map[][3] = { ++ ++ /* headphone connected to LOUT1, ROUT1 */ ++ {"Headphone Jack", NULL, "LOUT"}, ++ {"Headphone Jack", NULL, "ROUT"}, ++ ++ /* speaker connected to LOUT2, ROUT2 */ ++ {"Spk", NULL, "ROUT2"}, ++ {"Spk", NULL, "LOUT2"}, ++ ++ /* mic is connected to MIC1 (via Mic Bias) */ ++ {"MIC1", NULL, "Mic Bias"}, ++ {"Mic Bias", NULL, "Mic"}, ++ ++ {NULL, NULL, NULL}, ++}; ++ ++static const char* jack_function[] = {"Off", "Headphone"}; ++static const char* spk_function[] = {"Off", "On"}; ++static const char* mic_function[] = {"Off", "On"}; ++static const struct soc_enum machine_ctl_enum[] = { ++ SOC_ENUM_SINGLE_EXT(2, jack_function), ++ SOC_ENUM_SINGLE_EXT(2, spk_function), ++ SOC_ENUM_SINGLE_EXT(2, mic_function), ++}; ++ ++static const struct snd_kcontrol_new wm8753_machine_controls[] = { ++ SOC_ENUM_EXT("Jack Function", machine_ctl_enum[0], machine_get_jack, machine_set_jack), ++ SOC_ENUM_EXT("Speaker Function", machine_ctl_enum[1], machine_get_spk, machine_set_spk), ++ SOC_ENUM_EXT("Mic Function", machine_ctl_enum[2], machine_get_mic, machine_set_mic), ++}; ++ ++static int mainstone_wm8753_init(struct snd_soc_codec *codec) ++{ ++ int i, err; ++ ++ /* not used on this machine - e.g. will never be powered up */ ++ snd_soc_dapm_set_endpoint(codec, "OUT3", 0); ++ snd_soc_dapm_set_endpoint(codec, "OUT4", 0); ++ snd_soc_dapm_set_endpoint(codec, "MONO2", 0); ++ snd_soc_dapm_set_endpoint(codec, "MONO1", 0); ++ snd_soc_dapm_set_endpoint(codec, "LINE1", 0); ++ snd_soc_dapm_set_endpoint(codec, "LINE2", 0); ++ snd_soc_dapm_set_endpoint(codec, "RXP", 0); ++ snd_soc_dapm_set_endpoint(codec, "RXN", 0); ++ snd_soc_dapm_set_endpoint(codec, "MIC2", 0); ++ ++ /* Add machine specific controls */ ++ for (i = 0; i < ARRAY_SIZE(wm8753_machine_controls); i++) { ++ if ((err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&wm8753_machine_controls[i],codec, NULL))) < 0) ++ return err; ++ } ++ ++ /* Add machine specific widgets */ ++ for(i = 0; i < ARRAY_SIZE(machine_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &machine_dapm_widgets[i]); ++ } ++ ++ /* Set up machine specific audio path audio_mapnects */ ++ for(i = 0; audio_map[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, audio_map[i][0], audio_map[i][1], audio_map[i][2]); ++ } ++ ++ snd_soc_dapm_sync_endpoints(codec); ++ return 0; ++} ++ ++/* this configures the clocking between the WM8753 and the BT codec */ ++unsigned int mainstone_config_sysclk(struct snd_soc_pcm_runtime *rtd, ++ struct snd_soc_clock_info *info) ++{ ++ /* wm8753 has pll that generates mclk from 13MHz xtal */ ++ return rtd->codec_dai->config_sysclk(rtd->codec_dai, info, 13000000); ++} ++ ++static struct snd_soc_dai_link mainstone_dai[] = { ++{ /* Hifi Playback - for similatious use with voice below */ ++ .name = "WM8753", ++ .stream_name = "WM8753 HiFi", ++ .cpu_dai = &pxa_i2s_dai, ++ .codec_dai = &wm8753_dai[WM8753_DAI_HIFI], ++ .init = mainstone_wm8753_init, ++ .config_sysclk = mainstone_config_sysclk, ++}, ++{ /* Voice via BT */ ++ .name = "Bluetooth", ++ .stream_name = "Voice", ++ .cpu_dai = mainstone_bt_dai, ++ .codec_dai = &wm8753_dai[WM8753_DAI_VOICE], ++ .config_sysclk = mainstone_config_sysclk, ++}, ++}; ++ ++static struct snd_soc_machine mainstone = { ++ .name = "Mainstone", ++ .probe = mainstone_probe, ++ .remove = mainstone_remove, ++ .suspend_pre = mainstone_suspend, ++ .resume_post = mainstone_resume, ++ .ops = &mainstone_ops, ++ .dai_link = mainstone_dai, ++ .num_links = ARRAY_SIZE(mainstone_dai), ++}; ++ ++static struct snd_soc_device mainstone_snd_wm8753_devdata = { ++ .machine = &mainstone, ++ .platform = &pxa2xx_soc_platform, ++ .codec_dev = &soc_codec_dev_wm8753, ++}; ++ ++static struct platform_device *mainstone_snd_wm8753_device; ++ ++static int __init mainstone_init(void) ++{ ++ int ret; ++ ++ mainstone_snd_wm8753_device = platform_device_alloc("soc-audio", -1); ++ if (!mainstone_snd_wm8753_device) ++ return -ENOMEM; ++ ++ platform_set_drvdata(mainstone_snd_wm8753_device, &mainstone_snd_wm8753_devdata); ++ mainstone_snd_wm8753_devdata.dev = &mainstone_snd_wm8753_device->dev; ++ ++ if((ret = platform_device_add(mainstone_snd_wm8753_device)) != 0) ++ platform_device_put(mainstone_snd_wm8753_device); ++ ++ return ret; ++} ++ ++static void __exit mainstone_exit(void) ++{ ++ platform_device_unregister(mainstone_snd_wm8753_device); ++} ++ ++module_init(mainstone_init); ++module_exit(mainstone_exit); ++ ++/* Module information */ ++MODULE_AUTHOR("Liam Girdwood, liam.girdwood@wolfsonmicro.com, www.wolfsonmicro.com"); ++MODULE_DESCRIPTION("Mainstone Example Bluetooth PCM Interface"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/pxa/mainstone_wm8731.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/pxa/mainstone_wm8731.c +@@ -0,0 +1,156 @@ ++/* ++ * mainstone.c -- SoC audio for Mainstone ++ * ++ * Copyright 2005 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * Mainstone audio amplifier code taken from arch/arm/mach-pxa/mainstone.c ++ * Copyright: MontaVista Software Inc. ++ * ++ * 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. ++ * ++ * Revision history ++ * 5th June 2006 Initial version. ++ * ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/device.h> ++#include <linux/i2c.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++ ++#include <asm/arch/pxa-regs.h> ++#include <asm/arch/mainstone.h> ++#include <asm/arch/audio.h> ++ ++#include "../codecs/wm8731.h" ++#include "pxa2xx-pcm.h" ++ ++static struct snd_soc_machine mainstone; ++ ++ ++static const struct snd_soc_dapm_widget dapm_widgets[] = { ++ SND_SOC_DAPM_MIC("Int Mic", NULL), ++ SND_SOC_DAPM_SPK("Ext Spk", NULL), ++}; ++ ++static const char* intercon[][3] = { ++ ++ /* speaker connected to LHPOUT */ ++ {"Ext Spk", NULL, "LHPOUT"}, ++ ++ /* mic is connected to Mic Jack, with WM8731 Mic Bias */ ++ {"MICIN", NULL, "Mic Bias"}, ++ {"Mic Bias", NULL, "Int Mic"}, ++ ++ /* terminator */ ++ {NULL, NULL, NULL}, ++}; ++ ++/* ++ * Logic for a wm8731 as connected on a Endrelia ETI-B1 board. ++ */ ++static int mainstone_wm8731_init(struct snd_soc_codec *codec) ++{ ++ int i; ++ ++ ++ /* Add specific widgets */ ++ for(i = 0; i < ARRAY_SIZE(dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &dapm_widgets[i]); ++ } ++ ++ /* Set up specific audio path interconnects */ ++ for(i = 0; intercon[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, intercon[i][0], intercon[i][1], intercon[i][2]); ++ } ++ ++ /* not connected */ ++ snd_soc_dapm_set_endpoint(codec, "RLINEIN", 0); ++ snd_soc_dapm_set_endpoint(codec, "LLINEIN", 0); ++ ++ /* always connected */ ++ snd_soc_dapm_set_endpoint(codec, "Int Mic", 1); ++ snd_soc_dapm_set_endpoint(codec, "Ext Spk", 1); ++ ++ snd_soc_dapm_sync_endpoints(codec); ++ ++ return 0; ++} ++ ++unsigned int mainstone_config_sysclk(struct snd_soc_pcm_runtime *rtd, ++ struct snd_soc_clock_info *info) ++{ ++ /* we have a 12.288MHz crystal */ ++ return rtd->codec_dai->config_sysclk(rtd->codec_dai, info, 12288000); ++} ++ ++static struct snd_soc_dai_link mainstone_dai[] = { ++{ ++ .name = "WM8731", ++ .stream_name = "WM8731 HiFi", ++ .cpu_dai = &pxa_i2s_dai, ++ .codec_dai = &wm8731_dai, ++ .init = mainstone_wm8731_init, ++ .config_sysclk = mainstone_config_sysclk, ++}, ++}; ++ ++static struct snd_soc_machine mainstone = { ++ .name = "Mainstone", ++ .dai_link = mainstone_dai, ++ .num_links = ARRAY_SIZE(mainstone_dai), ++}; ++ ++static struct wm8731_setup_data corgi_wm8731_setup = { ++ .i2c_address = 0x1b, ++}; ++ ++static struct snd_soc_device mainstone_snd_devdata = { ++ .machine = &mainstone, ++ .platform = &pxa2xx_soc_platform, ++ .codec_dev = &soc_codec_dev_wm8731, ++ .codec_data = &corgi_wm8731_setup, ++}; ++ ++static struct platform_device *mainstone_snd_device; ++ ++static int __init mainstone_init(void) ++{ ++ int ret; ++ ++ mainstone_snd_device = platform_device_alloc("soc-audio", -1); ++ if (!mainstone_snd_device) ++ return -ENOMEM; ++ ++ platform_set_drvdata(mainstone_snd_device, &mainstone_snd_devdata); ++ mainstone_snd_devdata.dev = &mainstone_snd_device->dev; ++ ret = platform_device_add(mainstone_snd_device); ++ ++ if (ret) ++ platform_device_put(mainstone_snd_device); ++ ++ return ret; ++} ++ ++static void __exit mainstone_exit(void) ++{ ++ platform_device_unregister(mainstone_snd_device); ++} ++ ++module_init(mainstone_init); ++module_exit(mainstone_exit); ++ ++/* Module information */ ++MODULE_AUTHOR("Liam Girdwood, liam.girdwood@wolfsonmicro.com, www.wolfsonmicro.com"); ++MODULE_DESCRIPTION("ALSA SoC WM8731 Mainstone"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/pxa/mainstone_wm8753.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/pxa/mainstone_wm8753.c +@@ -0,0 +1,226 @@ ++/* ++ * mainstone.c -- SoC audio for Mainstone ++ * ++ * Copyright 2005 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * Mainstone audio amplifier code taken from arch/arm/mach-pxa/mainstone.c ++ * Copyright: MontaVista Software Inc. ++ * ++ * 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. ++ * ++ * Revision history ++ * 30th Oct 2005 Initial version. ++ * ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/device.h> ++#include <linux/i2c.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++ ++#include <asm/arch/pxa-regs.h> ++#include <asm/arch/mainstone.h> ++#include <asm/arch/audio.h> ++ ++#include "../codecs/wm8753.h" ++#include "pxa2xx-pcm.h" ++ ++static struct snd_soc_machine mainstone; ++ ++static int mainstone_startup(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ ++ if(rtd->cpu_dai->type == SND_SOC_DAI_PCM && rtd->cpu_dai->id == 1) { ++ /* enable USB on the go MUX so we can use SSPFRM2 */ ++ MST_MSCWR2 |= MST_MSCWR2_USB_OTG_SEL; ++ MST_MSCWR2 &= ~MST_MSCWR2_USB_OTG_RST; ++ } ++ return 0; ++} ++ ++static void mainstone_shutdown(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ ++ if(rtd->cpu_dai->type == SND_SOC_DAI_PCM && rtd->cpu_dai->id == 1) { ++ /* disable USB on the go MUX so we can use ttyS0 */ ++ MST_MSCWR2 &= ~MST_MSCWR2_USB_OTG_SEL; ++ MST_MSCWR2 |= MST_MSCWR2_USB_OTG_RST; ++ } ++} ++ ++static struct snd_soc_ops mainstone_ops = { ++ .startup = mainstone_startup, ++ .shutdown = mainstone_shutdown, ++}; ++ ++static long mst_audio_suspend_mask; ++ ++static int mainstone_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ mst_audio_suspend_mask = MST_MSCWR2; ++ MST_MSCWR2 |= MST_MSCWR2_AC97_SPKROFF; ++ return 0; ++} ++ ++static int mainstone_resume(struct platform_device *pdev) ++{ ++ MST_MSCWR2 &= mst_audio_suspend_mask | ~MST_MSCWR2_AC97_SPKROFF; ++ return 0; ++} ++ ++static int mainstone_probe(struct platform_device *pdev) ++{ ++ MST_MSCWR2 &= ~MST_MSCWR2_AC97_SPKROFF; ++ return 0; ++} ++ ++static int mainstone_remove(struct platform_device *pdev) ++{ ++ MST_MSCWR2 |= MST_MSCWR2_AC97_SPKROFF; ++ return 0; ++} ++ ++/* example machine audio_mapnections */ ++static const char* audio_map[][3] = { ++ ++ /* mic is connected to mic1 - with bias */ ++ {"MIC1", NULL, "Mic Bias"}, ++ {"MIC1N", NULL, "Mic Bias"}, ++ {"Mic Bias", NULL, "Mic1 Jack"}, ++ {"Mic Bias", NULL, "Mic1 Jack"}, ++ ++ {"ACIN", NULL, "ACOP"}, ++ {NULL, NULL, NULL}, ++}; ++ ++/* headphone detect support on my board */ ++static const char * hp_pol[] = {"Headphone", "Speaker"}; ++static const struct soc_enum wm8753_enum = ++ SOC_ENUM_SINGLE(WM8753_OUTCTL, 1, 2, hp_pol); ++ ++static const struct snd_kcontrol_new wm8753_mainstone_controls[] = { ++ SOC_SINGLE("Headphone Detect Switch", WM8753_OUTCTL, 6, 1, 0), ++ SOC_ENUM("Headphone Detect Polarity", wm8753_enum), ++}; ++ ++/* ++ * This is an example machine initialisation for a wm8753 connected to a ++ * Mainstone II. It is missing logic to detect hp/mic insertions and logic ++ * to re-route the audio in such an event. ++ */ ++static int mainstone_wm8753_init(struct snd_soc_codec *codec) ++{ ++ int i, err; ++ ++ /* set up mainstone codec pins */ ++ snd_soc_dapm_set_endpoint(codec, "RXP", 0); ++ snd_soc_dapm_set_endpoint(codec, "RXN", 0); ++ snd_soc_dapm_set_endpoint(codec, "MIC2", 0); ++ ++ /* add mainstone specific controls */ ++ for (i = 0; i < ARRAY_SIZE(wm8753_mainstone_controls); i++) { ++ if ((err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&wm8753_mainstone_controls[i],codec, NULL))) < 0) ++ return err; ++ } ++ ++ /* set up mainstone specific audio path audio_mapnects */ ++ for(i = 0; audio_map[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, audio_map[i][0], audio_map[i][1], audio_map[i][2]); ++ } ++ ++ snd_soc_dapm_sync_endpoints(codec); ++ return 0; ++} ++ ++unsigned int mainstone_config_sysclk(struct snd_soc_pcm_runtime *rtd, ++ struct snd_soc_clock_info *info) ++{ ++ /* wm8753 has pll that generates mclk from 13MHz xtal */ ++ return rtd->codec_dai->config_sysclk(rtd->codec_dai, info, 13000000); ++} ++ ++static struct snd_soc_dai_link mainstone_dai[] = { ++{ /* Hifi Playback - for similatious use with voice below */ ++ .name = "WM8753", ++ .stream_name = "WM8753 HiFi", ++ .cpu_dai = &pxa_i2s_dai, ++ .codec_dai = &wm8753_dai[WM8753_DAI_HIFI], ++ .init = mainstone_wm8753_init, ++ .config_sysclk = mainstone_config_sysclk, ++}, ++{ /* Voice via BT */ ++ .name = "Bluetooth", ++ .stream_name = "Voice", ++ .cpu_dai = &pxa_ssp_dai[1], ++ .codec_dai = &wm8753_dai[WM8753_DAI_VOICE], ++ .config_sysclk = mainstone_config_sysclk, ++}, ++}; ++ ++static struct snd_soc_machine mainstone = { ++ .name = "Mainstone", ++ .probe = mainstone_probe, ++ .remove = mainstone_remove, ++ .suspend_pre = mainstone_suspend, ++ .resume_post = mainstone_resume, ++ .ops = &mainstone_ops, ++ .dai_link = mainstone_dai, ++ .num_links = ARRAY_SIZE(mainstone_dai), ++}; ++ ++static struct wm8753_setup_data mainstone_wm8753_setup = { ++ .i2c_address = 0x1a, ++}; ++ ++static struct snd_soc_device mainstone_snd_devdata = { ++ .machine = &mainstone, ++ .platform = &pxa2xx_soc_platform, ++ .codec_dev = &soc_codec_dev_wm8753, ++ .codec_data = &mainstone_wm8753_setup, ++}; ++ ++static struct platform_device *mainstone_snd_device; ++ ++static int __init mainstone_init(void) ++{ ++ int ret; ++ ++ mainstone_snd_device = platform_device_alloc("soc-audio", -1); ++ if (!mainstone_snd_device) ++ return -ENOMEM; ++ ++ platform_set_drvdata(mainstone_snd_device, &mainstone_snd_devdata); ++ mainstone_snd_devdata.dev = &mainstone_snd_device->dev; ++ ret = platform_device_add(mainstone_snd_device); ++ ++ if (ret) ++ platform_device_put(mainstone_snd_device); ++ ++ return ret; ++} ++ ++static void __exit mainstone_exit(void) ++{ ++ platform_device_unregister(mainstone_snd_device); ++} ++ ++module_init(mainstone_init); ++module_exit(mainstone_exit); ++ ++/* Module information */ ++MODULE_AUTHOR("Liam Girdwood, liam.girdwood@wolfsonmicro.com, www.wolfsonmicro.com"); ++MODULE_DESCRIPTION("ALSA SoC WM8753 Mainstone"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/pxa/mainstone_wm8974.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/pxa/mainstone_wm8974.c +@@ -0,0 +1,112 @@ ++/* ++ * mainstone.c -- SoC audio for Mainstone ++ * ++ * Copyright 2005 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * Mainstone audio amplifier code taken from arch/arm/mach-pxa/mainstone.c ++ * Copyright: MontaVista Software Inc. ++ * ++ * 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. ++ * ++ * Revision history ++ * 30th Oct 2005 Initial version. ++ * ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/device.h> ++#include <linux/i2c.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++ ++#include <asm/arch/pxa-regs.h> ++#include <asm/arch/mainstone.h> ++#include <asm/arch/audio.h> ++ ++#include "../codecs/wm8974.h" ++#include "pxa2xx-pcm.h" ++ ++static struct snd_soc_machine mainstone; ++ ++static int mainstone_wm8974_init(struct snd_soc_codec *codec) ++{ ++ return 0; ++} ++ ++unsigned int mainstone_config_sysclk(struct snd_soc_pcm_runtime *rtd, ++ struct snd_soc_clock_info *info) ++{ ++ /* we have a PLL */ ++ return rtd->codec_dai->config_sysclk(rtd->codec_dai, info, 12288000); ++ ++} ++ ++static struct snd_soc_dai_link mainstone_dai[] = { ++{ ++ .name = "WM8974", ++ .stream_name = "WM8974 HiFi", ++ .cpu_dai = &pxa_i2s_dai, ++ .codec_dai = &wm8974_dai, ++ .init = mainstone_wm8974_init, ++ .config_sysclk = mainstone_config_sysclk, ++}, ++}; ++ ++static struct snd_soc_machine mainstone = { ++ .name = "Mainstone", ++ .dai_link = mainstone_dai, ++ .num_links = ARRAY_SIZE(mainstone_dai), ++}; ++ ++static struct wm8974_setup_data mainstone_wm8974_setup = { ++ .i2c_address = 0x1a, ++}; ++ ++static struct snd_soc_device mainstone_snd_devdata = { ++ .machine = &mainstone, ++ .platform = &pxa2xx_soc_platform, ++ .codec_dev = &soc_codec_dev_wm8974, ++ .codec_data = &mainstone_wm8974_setup, ++}; ++ ++static struct platform_device *mainstone_snd_device; ++ ++static int __init mainstone_init(void) ++{ ++ int ret; ++ ++ mainstone_snd_device = platform_device_alloc("soc-audio", -1); ++ if (!mainstone_snd_device) ++ return -ENOMEM; ++ ++ platform_set_drvdata(mainstone_snd_device, &mainstone_snd_devdata); ++ mainstone_snd_devdata.dev = &mainstone_snd_device->dev; ++ ret = platform_device_add(mainstone_snd_device); ++ ++ if (ret) ++ platform_device_put(mainstone_snd_device); ++ ++ return ret; ++} ++ ++static void __exit mainstone_exit(void) ++{ ++ platform_device_unregister(mainstone_snd_device); ++} ++ ++module_init(mainstone_init); ++module_exit(mainstone_exit); ++ ++/* Module information */ ++MODULE_AUTHOR("Liam Girdwood, liam.girdwood@wolfsonmicro.com, www.wolfsonmicro.com"); ++MODULE_DESCRIPTION("ALSA SoC Mainstone"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/pxa/mainstone_wm9712.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/pxa/mainstone_wm9712.c +@@ -0,0 +1,171 @@ ++/* ++ * mainstone.c -- SoC audio for Mainstone ++ * ++ * Copyright 2006 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * Mainstone audio amplifier code taken from arch/arm/mach-pxa/mainstone.c ++ * Copyright: MontaVista Software Inc. ++ * ++ * 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. ++ * ++ * Revision history ++ * 29th Jan 2006 Initial version. ++ * ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/device.h> ++#include <linux/i2c.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++ ++#include <asm/arch/pxa-regs.h> ++#include <asm/arch/mainstone.h> ++#include <asm/arch/audio.h> ++ ++#include "../codecs/wm9712.h" ++#include "pxa2xx-pcm.h" ++ ++static struct snd_soc_machine mainstone; ++static long mst_audio_suspend_mask; ++ ++static int mainstone_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ mst_audio_suspend_mask = MST_MSCWR2; ++ MST_MSCWR2 |= MST_MSCWR2_AC97_SPKROFF; ++ return 0; ++} ++ ++static int mainstone_resume(struct platform_device *pdev) ++{ ++ MST_MSCWR2 &= mst_audio_suspend_mask | ~MST_MSCWR2_AC97_SPKROFF; ++ return 0; ++} ++ ++static int mainstone_probe(struct platform_device *pdev) ++{ ++ MST_MSCWR2 &= ~MST_MSCWR2_AC97_SPKROFF; ++ return 0; ++} ++ ++static int mainstone_remove(struct platform_device *pdev) ++{ ++ MST_MSCWR2 |= MST_MSCWR2_AC97_SPKROFF; ++ return 0; ++} ++ ++/* mainstone machine dapm widgets */ ++static const struct snd_soc_dapm_widget mainstone_dapm_widgets[] = { ++ SND_SOC_DAPM_MIC("Mic (Internal)", NULL), ++}; ++ ++/* example machine interconnections */ ++static const char* intercon[][3] = { ++ ++ /* mic is connected to mic1 - with bias */ ++ {"MIC1", NULL, "Mic Bias"}, ++ {"Mic Bias", NULL, "Mic (Internal)"}, ++ ++ {NULL, NULL, NULL}, ++}; ++ ++/* ++ * This is an example machine initialisation for a wm8753 connected to a ++ * Mainstone II. It is missing logic to detect hp/mic insertions and logic ++ * to re-route the audio in such an event. ++ */ ++static int mainstone_wm9712_init(struct snd_soc_codec *codec) ++{ ++ int i; ++ ++ /* set up mainstone codec pins */ ++ snd_soc_dapm_set_endpoint(codec, "RXP", 0); ++ snd_soc_dapm_set_endpoint(codec, "RXN", 0); ++ //snd_soc_dapm_set_endpoint(codec, "MIC2", 0); ++ ++ /* Add mainstone specific widgets */ ++ for(i = 0; i < ARRAY_SIZE(mainstone_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &mainstone_dapm_widgets[i]); ++ } ++ ++ /* set up mainstone specific audio path interconnects */ ++ for(i = 0; intercon[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, intercon[i][0], intercon[i][1], intercon[i][2]); ++ } ++ ++ snd_soc_dapm_sync_endpoints(codec); ++ return 0; ++} ++ ++static struct snd_soc_dai_link mainstone_dai[] = { ++{ ++ .name = "AC97", ++ .stream_name = "AC97 HiFi", ++ .cpu_dai = &pxa_ac97_dai[PXA2XX_DAI_AC97_HIFI], ++ .codec_dai = &wm9712_dai[WM9712_DAI_AC97_HIFI], ++ .init = mainstone_wm9712_init, ++}, ++{ ++ .name = "AC97 Aux", ++ .stream_name = "AC97 Aux", ++ .cpu_dai = &pxa_ac97_dai[PXA2XX_DAI_AC97_AUX], ++ .codec_dai = &wm9712_dai[WM9712_DAI_AC97_AUX], ++}, ++}; ++ ++static struct snd_soc_machine mainstone = { ++ .name = "Mainstone", ++ .probe = mainstone_probe, ++ .remove = mainstone_remove, ++ .suspend_pre = mainstone_suspend, ++ .resume_post = mainstone_resume, ++ .dai_link = mainstone_dai, ++ .num_links = ARRAY_SIZE(mainstone_dai), ++}; ++ ++static struct snd_soc_device mainstone_snd_ac97_devdata = { ++ .machine = &mainstone, ++ .platform = &pxa2xx_soc_platform, ++ .codec_dev = &soc_codec_dev_wm9712, ++}; ++ ++static struct platform_device *mainstone_snd_ac97_device; ++ ++static int __init mainstone_init(void) ++{ ++ int ret; ++ ++ mainstone_snd_ac97_device = platform_device_alloc("soc-audio", -1); ++ if (!mainstone_snd_ac97_device) ++ return -ENOMEM; ++ ++ platform_set_drvdata(mainstone_snd_ac97_device, &mainstone_snd_ac97_devdata); ++ mainstone_snd_ac97_devdata.dev = &mainstone_snd_ac97_device->dev; ++ ++ if((ret = platform_device_add(mainstone_snd_ac97_device)) != 0) ++ platform_device_put(mainstone_snd_ac97_device); ++ ++ return ret; ++} ++ ++static void __exit mainstone_exit(void) ++{ ++ platform_device_unregister(mainstone_snd_ac97_device); ++} ++ ++module_init(mainstone_init); ++module_exit(mainstone_exit); ++ ++/* Module information */ ++MODULE_AUTHOR("Liam Girdwood, liam.girdwood@wolfsonmicro.com, www.wolfsonmicro.com"); ++MODULE_DESCRIPTION("ALSA SoC WM9712 Mainstone"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/pxa/mainstone_wm9713.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/pxa/mainstone_wm9713.c +@@ -0,0 +1,263 @@ ++/* ++ * mainstone.c -- SoC audio for Mainstone ++ * ++ * Copyright 2006 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * Mainstone audio amplifier code taken from arch/arm/mach-pxa/mainstone.c ++ * Copyright: MontaVista Software Inc. ++ * ++ * 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. ++ * ++ * Revision history ++ * 29th Jan 2006 Initial version. ++ * ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/device.h> ++#include <linux/i2c.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++ ++#include <asm/arch/pxa-regs.h> ++#include <asm/arch/mainstone.h> ++#include <asm/arch/audio.h> ++ ++#include "../codecs/wm9713.h" ++#include "pxa2xx-pcm.h" ++ ++static struct snd_soc_machine mainstone; ++ ++static int mainstone_startup(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ ++ if(rtd->cpu_dai->type == SND_SOC_DAI_PCM && rtd->cpu_dai->id == 1) { ++ /* enable USB on the go MUX so we can use SSPFRM2 */ ++ MST_MSCWR2 |= MST_MSCWR2_USB_OTG_SEL; ++ MST_MSCWR2 &= ~MST_MSCWR2_USB_OTG_RST; ++ } ++ return 0; ++} ++ ++static void mainstone_shutdown(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ ++ if(rtd->cpu_dai->type == SND_SOC_DAI_PCM && rtd->cpu_dai->id == 1) { ++ /* disable USB on the go MUX so we can use ttyS0 */ ++ MST_MSCWR2 &= ~MST_MSCWR2_USB_OTG_SEL; ++ MST_MSCWR2 |= MST_MSCWR2_USB_OTG_RST; ++ } ++} ++ ++static struct snd_soc_ops mainstone_ops = { ++ .startup = mainstone_startup, ++ .shutdown = mainstone_shutdown, ++}; ++ ++static int test = 0; ++static int get_test(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ ucontrol->value.integer.value[0] = test; ++ return 0; ++} ++ ++static int set_test(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ ++ test = ucontrol->value.integer.value[0]; ++ if(test) { ++ ++ } else { ++ ++ } ++ return 0; ++} ++ ++static long mst_audio_suspend_mask; ++ ++static int mainstone_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ mst_audio_suspend_mask = MST_MSCWR2; ++ MST_MSCWR2 |= MST_MSCWR2_AC97_SPKROFF; ++ return 0; ++} ++ ++static int mainstone_resume(struct platform_device *pdev) ++{ ++ MST_MSCWR2 &= mst_audio_suspend_mask | ~MST_MSCWR2_AC97_SPKROFF; ++ return 0; ++} ++ ++static int mainstone_probe(struct platform_device *pdev) ++{ ++ MST_MSCWR2 &= ~MST_MSCWR2_AC97_SPKROFF; ++ return 0; ++} ++ ++static int mainstone_remove(struct platform_device *pdev) ++{ ++ MST_MSCWR2 |= MST_MSCWR2_AC97_SPKROFF; ++ return 0; ++} ++ ++static const char* test_function[] = {"Off", "On"}; ++static const struct soc_enum mainstone_enum[] = { ++ SOC_ENUM_SINGLE_EXT(2, test_function), ++}; ++ ++static const struct snd_kcontrol_new mainstone_controls[] = { ++ SOC_ENUM_EXT("ATest Function", mainstone_enum[0], get_test, set_test), ++}; ++ ++/* mainstone machine dapm widgets */ ++static const struct snd_soc_dapm_widget mainstone_dapm_widgets[] = { ++ SND_SOC_DAPM_MIC("Mic 1", NULL), ++ SND_SOC_DAPM_MIC("Mic 2", NULL), ++ SND_SOC_DAPM_MIC("Mic 3", NULL), ++}; ++ ++/* example machine audio_mapnections */ ++static const char* audio_map[][3] = { ++ ++ /* mic is connected to mic1 - with bias */ ++ {"MIC1", NULL, "Mic Bias"}, ++ {"Mic Bias", NULL, "Mic 1"}, ++ /* mic is connected to mic2A - with bias */ ++ {"MIC2A", NULL, "Mic Bias"}, ++ {"Mic Bias", NULL, "Mic 2"}, ++ /* mic is connected to mic2B - with bias */ ++ {"MIC2B", NULL, "Mic Bias"}, ++ {"Mic Bias", NULL, "Mic 3"}, ++ ++ {NULL, NULL, NULL}, ++}; ++ ++/* ++ * This is an example machine initialisation for a wm9713 connected to a ++ * Mainstone II. It is missing logic to detect hp/mic insertions and logic ++ * to re-route the audio in such an event. ++ */ ++static int mainstone_wm9713_init(struct snd_soc_codec *codec) ++{ ++ int i, err; ++ ++ /* set up mainstone codec pins */ ++ snd_soc_dapm_set_endpoint(codec, "RXP", 0); ++ snd_soc_dapm_set_endpoint(codec, "RXN", 0); ++ //snd_soc_dapm_set_endpoint(codec, "MIC2", 0); ++ ++ /* Add test specific controls */ ++ for (i = 0; i < ARRAY_SIZE(mainstone_controls); i++) { ++ if ((err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&mainstone_controls[i],codec, NULL))) < 0) ++ return err; ++ } ++ ++ /* Add mainstone specific widgets */ ++ for(i = 0; i < ARRAY_SIZE(mainstone_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &mainstone_dapm_widgets[i]); ++ } ++ ++ /* set up mainstone specific audio path audio_mapnects */ ++ for(i = 0; audio_map[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, audio_map[i][0], audio_map[i][1], audio_map[i][2]); ++ } ++ ++ snd_soc_dapm_sync_endpoints(codec); ++ return 0; ++} ++ ++/* configure the system audio clock */ ++unsigned int mainstone_config_sysclk(struct snd_soc_pcm_runtime *rtd, ++ struct snd_soc_clock_info *info) ++{ ++ return rtd->codec_dai->config_sysclk(rtd->codec_dai, info, 24576000); ++} ++ ++static struct snd_soc_dai_link mainstone_dai[] = { ++{ ++ .name = "AC97", ++ .stream_name = "AC97 HiFi", ++ .cpu_dai = &pxa_ac97_dai[PXA2XX_DAI_AC97_HIFI], ++ .codec_dai = &wm9713_dai[WM9713_DAI_AC97_HIFI], ++ .init = mainstone_wm9713_init, ++ .config_sysclk = mainstone_config_sysclk, ++}, ++{ ++ .name = "AC97 Aux", ++ .stream_name = "AC97 Aux", ++ .cpu_dai = &pxa_ac97_dai[PXA2XX_DAI_AC97_AUX], ++ .codec_dai = &wm9713_dai[WM9713_DAI_AC97_AUX], ++ .config_sysclk = mainstone_config_sysclk, ++}, ++{ ++ .name = "WM9713", ++ .stream_name = "WM9713 Voice", ++ .cpu_dai = &pxa_ssp_dai[PXA2XX_DAI_SSP2], ++ .codec_dai = &wm9713_dai[WM9713_DAI_PCM_VOICE], ++ .config_sysclk = mainstone_config_sysclk, ++}, ++}; ++ ++static struct snd_soc_machine mainstone = { ++ .name = "Mainstone", ++ .probe = mainstone_probe, ++ .remove = mainstone_remove, ++ .suspend_pre = mainstone_suspend, ++ .resume_post = mainstone_resume, ++ .ops = &mainstone_ops, ++ .dai_link = mainstone_dai, ++ .num_links = ARRAY_SIZE(mainstone_dai), ++}; ++ ++static struct snd_soc_device mainstone_snd_ac97_devdata = { ++ .machine = &mainstone, ++ .platform = &pxa2xx_soc_platform, ++ .codec_dev = &soc_codec_dev_wm9713, ++}; ++ ++static struct platform_device *mainstone_snd_ac97_device; ++ ++static int __init mainstone_init(void) ++{ ++ int ret; ++ ++ mainstone_snd_ac97_device = platform_device_alloc("soc-audio", -1); ++ if (!mainstone_snd_ac97_device) ++ return -ENOMEM; ++ ++ platform_set_drvdata(mainstone_snd_ac97_device, &mainstone_snd_ac97_devdata); ++ mainstone_snd_ac97_devdata.dev = &mainstone_snd_ac97_device->dev; ++ ++ if((ret = platform_device_add(mainstone_snd_ac97_device)) != 0) ++ platform_device_put(mainstone_snd_ac97_device); ++ ++ return ret; ++} ++ ++static void __exit mainstone_exit(void) ++{ ++ platform_device_unregister(mainstone_snd_ac97_device); ++} ++ ++module_init(mainstone_init); ++module_exit(mainstone_exit); ++ ++/* Module information */ ++MODULE_AUTHOR("Liam Girdwood, liam.girdwood@wolfsonmicro.com, www.wolfsonmicro.com"); ++MODULE_DESCRIPTION("ALSA SoC WM9713 Mainstone"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/pxa/poodle.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/pxa/poodle.c +@@ -0,0 +1,329 @@ ++/* ++ * poodle.c -- SoC audio for Poodle ++ * ++ * Copyright 2005 Wolfson Microelectronics PLC. ++ * Copyright 2005 Openedhand Ltd. ++ * ++ * Authors: Liam Girdwood <liam.girdwood@wolfsonmicro.com> ++ * Richard Purdie <richard@openedhand.com> ++ * ++ * 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. ++ * ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/timer.h> ++#include <linux/interrupt.h> ++#include <linux/platform_device.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++ ++#include <asm/mach-types.h> ++#include <asm/hardware/locomo.h> ++#include <asm/arch/pxa-regs.h> ++#include <asm/arch/hardware.h> ++#include <asm/arch/poodle.h> ++#include <asm/arch/audio.h> ++ ++#include "../codecs/wm8731.h" ++#include "pxa2xx-pcm.h" ++ ++#define POODLE_HP 1 ++#define POODLE_HP_OFF 0 ++#define POODLE_SPK_ON 1 ++#define POODLE_SPK_OFF 0 ++ ++ /* audio clock in Hz - rounded from 12.235MHz */ ++#define POODLE_AUDIO_CLOCK 12288000 ++ ++static int poodle_jack_func; ++static int poodle_spk_func; ++ ++static void poodle_ext_control(struct snd_soc_codec *codec) ++{ ++ int spk = 0; ++ ++ /* set up jack connection */ ++ if (poodle_jack_func == POODLE_HP) { ++ /* set = unmute headphone */ ++ locomo_gpio_write(&poodle_locomo_device.dev, ++ POODLE_LOCOMO_GPIO_MUTE_L, 1); ++ locomo_gpio_write(&poodle_locomo_device.dev, ++ POODLE_LOCOMO_GPIO_MUTE_R, 1); ++ snd_soc_dapm_set_endpoint(codec, "Headphone Jack", 1); ++ } else { ++ locomo_gpio_write(&poodle_locomo_device.dev, ++ POODLE_LOCOMO_GPIO_MUTE_L, 0); ++ locomo_gpio_write(&poodle_locomo_device.dev, ++ POODLE_LOCOMO_GPIO_MUTE_R, 0); ++ snd_soc_dapm_set_endpoint(codec, "Headphone Jack", 0); ++ } ++ ++ if (poodle_spk_func == POODLE_SPK_ON) ++ spk = 1; ++ ++ /* set the enpoints to their new connetion states */ ++ snd_soc_dapm_set_endpoint(codec, "Ext Spk", spk); ++ ++ /* signal a DAPM event */ ++ snd_soc_dapm_sync_endpoints(codec); ++} ++ ++static int poodle_startup(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_codec *codec = rtd->socdev->codec; ++ ++ /* check the jack status at stream startup */ ++ poodle_ext_control(codec); ++ return 0; ++} ++ ++/* we need to unmute the HP at shutdown as the mute burns power on poodle */ ++static int poodle_shutdown(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_codec *codec = rtd->socdev->codec; ++ ++ /* set = unmute headphone */ ++ locomo_gpio_write(&poodle_locomo_device.dev, ++ POODLE_LOCOMO_GPIO_MUTE_L, 1); ++ locomo_gpio_write(&poodle_locomo_device.dev, ++ POODLE_LOCOMO_GPIO_MUTE_R, 1); ++ return 0; ++} ++ ++static struct snd_soc_ops poodle_ops = { ++ .startup = poodle_startup, ++ .shutdown = poodle_shutdown, ++}; ++ ++static int poodle_get_jack(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ ucontrol->value.integer.value[0] = poodle_jack_func; ++ return 0; ++} ++ ++static int poodle_set_jack(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ ++ if (poodle_jack_func == ucontrol->value.integer.value[0]) ++ return 0; ++ ++ poodle_jack_func = ucontrol->value.integer.value[0]; ++ poodle_ext_control(codec); ++ return 1; ++} ++ ++static int poodle_get_spk(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ ucontrol->value.integer.value[0] = poodle_spk_func; ++ return 0; ++} ++ ++static int poodle_set_spk(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ ++ if (poodle_spk_func == ucontrol->value.integer.value[0]) ++ return 0; ++ ++ poodle_spk_func = ucontrol->value.integer.value[0]; ++ poodle_ext_control(codec); ++ return 1; ++} ++ ++static int poodle_amp_event(struct snd_soc_dapm_widget *w, int event) ++{ ++ if (SND_SOC_DAPM_EVENT_ON(event)) ++ locomo_gpio_write(&poodle_locomo_device.dev, ++ POODLE_LOCOMO_GPIO_AMP_ON, 0); ++ else ++ locomo_gpio_write(&poodle_locomo_device.dev, ++ POODLE_LOCOMO_GPIO_AMP_ON, 1); ++ ++ return 0; ++} ++ ++/* poodle machine dapm widgets */ ++static const struct snd_soc_dapm_widget wm8731_dapm_widgets[] = { ++SND_SOC_DAPM_HP("Headphone Jack", NULL), ++SND_SOC_DAPM_SPK("Ext Spk", poodle_amp_event), ++}; ++ ++/* Corgi machine audio_mapnections to the codec pins */ ++static const char *audio_map[][3] = { ++ ++ /* headphone connected to LHPOUT1, RHPOUT1 */ ++ {"Headphone Jack", NULL, "LHPOUT"}, ++ {"Headphone Jack", NULL, "RHPOUT"}, ++ ++ /* speaker connected to LOUT, ROUT */ ++ {"Ext Spk", NULL, "ROUT"}, ++ {"Ext Spk", NULL, "LOUT"}, ++ ++ {NULL, NULL, NULL}, ++}; ++ ++static const char *jack_function[] = {"Off", "Headphone"}; ++static const char *spk_function[] = {"Off", "On"}; ++static const struct soc_enum poodle_enum[] = { ++ SOC_ENUM_SINGLE_EXT(2, jack_function), ++ SOC_ENUM_SINGLE_EXT(2, spk_function), ++}; ++ ++static const snd_kcontrol_new_t wm8731_poodle_controls[] = { ++ SOC_ENUM_EXT("Jack Function", poodle_enum[0], poodle_get_jack, ++ poodle_set_jack), ++ SOC_ENUM_EXT("Speaker Function", poodle_enum[1], poodle_get_spk, ++ poodle_set_spk), ++}; ++ ++/* ++ * Logic for a wm8731 as connected on a Sharp SL-C7x0 Device ++ */ ++static int poodle_wm8731_init(struct snd_soc_codec *codec) ++{ ++ int i, err; ++ ++ snd_soc_dapm_set_endpoint(codec, "LLINEIN", 0); ++ snd_soc_dapm_set_endpoint(codec, "RLINEIN", 0); ++ snd_soc_dapm_set_endpoint(codec, "MICIN", 1); ++ ++ /* Add poodle specific controls */ ++ for (i = 0; i < ARRAY_SIZE(wm8731_poodle_controls); i++) { ++ err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&wm8731_poodle_controls[i],codec, NULL)); ++ if (err < 0) ++ return err; ++ } ++ ++ /* Add poodle specific widgets */ ++ for (i = 0; i < ARRAY_SIZE(wm8731_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &wm8731_dapm_widgets[i]); ++ } ++ ++ /* Set up poodle specific audio path audio_map */ ++ for (i = 0; audio_map[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, audio_map[i][0], ++ audio_map[i][1], audio_map[i][2]); ++ } ++ ++ snd_soc_dapm_sync_endpoints(codec); ++ return 0; ++} ++ ++static unsigned int poodle_config_sysclk(struct snd_soc_pcm_runtime *rtd, ++ struct snd_soc_clock_info *info) ++{ ++ if (info->bclk_master & SND_SOC_DAIFMT_CBS_CFS) { ++ /* pxa2xx is i2s master */ ++ switch (info->rate) { ++ case 44100: ++ case 88200: ++ /* configure codec digital filters for 44.1, 88.2 */ ++ rtd->codec_dai->config_sysclk(rtd->codec_dai, info, ++ 11289600); ++ break; ++ default: ++ /* configure codec digital filters for all other rates */ ++ rtd->codec_dai->config_sysclk(rtd->codec_dai, info, ++ POODLE_AUDIO_CLOCK); ++ break; ++ } ++ return rtd->cpu_dai->config_sysclk(rtd->cpu_dai, info, ++ POODLE_AUDIO_CLOCK); ++ } else { ++ /* codec is i2s master - ++ * only configure codec DAI clock and filters */ ++ return rtd->codec_dai->config_sysclk(rtd->codec_dai, info, ++ POODLE_AUDIO_CLOCK); ++ } ++} ++ ++/* poodle digital audio interface glue - connects codec <--> CPU */ ++static struct snd_soc_dai_link poodle_dai = { ++ .name = "WM8731", ++ .stream_name = "WM8731", ++ .cpu_dai = &pxa_i2s_dai, ++ .codec_dai = &wm8731_dai, ++ .init = poodle_wm8731_init, ++ .config_sysclk = poodle_config_sysclk, ++}; ++ ++/* poodle audio machine driver */ ++static struct snd_soc_machine snd_soc_machine_poodle = { ++ .name = "Poodle", ++ .dai_link = &poodle_dai, ++ .num_links = 1, ++ .ops = &poodle_ops, ++}; ++ ++/* poodle audio private data */ ++static struct wm8731_setup_data poodle_wm8731_setup = { ++ .i2c_address = 0x1b, ++}; ++ ++/* poodle audio subsystem */ ++static struct snd_soc_device poodle_snd_devdata = { ++ .machine = &snd_soc_machine_poodle, ++ .platform = &pxa2xx_soc_platform, ++ .codec_dev = &soc_codec_dev_wm8731, ++ .codec_data = &poodle_wm8731_setup, ++}; ++ ++static struct platform_device *poodle_snd_device; ++ ++static int __init poodle_init(void) ++{ ++ int ret; ++ ++ if (!machine_is_poodle()) ++ return -ENODEV; ++ ++ locomo_gpio_set_dir(&poodle_locomo_device.dev, ++ POODLE_LOCOMO_GPIO_AMP_ON, 0); ++ /* should we mute HP at startup - burning power ?*/ ++ locomo_gpio_set_dir(&poodle_locomo_device.dev, ++ POODLE_LOCOMO_GPIO_MUTE_L, 0); ++ locomo_gpio_set_dir(&poodle_locomo_device.dev, ++ POODLE_LOCOMO_GPIO_MUTE_R, 0); ++ ++ poodle_snd_device = platform_device_alloc("soc-audio", -1); ++ if (!poodle_snd_device) ++ return -ENOMEM; ++ ++ platform_set_drvdata(poodle_snd_device, &poodle_snd_devdata); ++ poodle_snd_devdata.dev = &poodle_snd_device->dev; ++ ret = platform_device_add(poodle_snd_device); ++ ++ if (ret) ++ platform_device_put(poodle_snd_device); ++ ++ return ret; ++} ++ ++static void __exit poodle_exit(void) ++{ ++ platform_device_unregister(poodle_snd_device); ++} ++ ++module_init(poodle_init); ++module_exit(poodle_exit); ++ ++/* Module information */ ++MODULE_AUTHOR("Richard Purdie"); ++MODULE_DESCRIPTION("ALSA SoC Poodle"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/pxa/pxa2xx-ac97.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/pxa/pxa2xx-ac97.c +@@ -0,0 +1,437 @@ ++/* ++ * linux/sound/pxa2xx-ac97.c -- AC97 support for the Intel PXA2xx chip. ++ * ++ * Author: Nicolas Pitre ++ * Created: Dec 02, 2004 ++ * Copyright: MontaVista Software Inc. ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#include <linux/init.h> ++#include <linux/module.h> ++#include <linux/platform_device.h> ++#include <linux/interrupt.h> ++#include <linux/wait.h> ++#include <linux/delay.h> ++ ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/ac97_codec.h> ++#include <sound/initval.h> ++#include <sound/soc.h> ++ ++#include <asm/irq.h> ++#include <linux/mutex.h> ++#include <asm/hardware.h> ++#include <asm/arch/pxa-regs.h> ++#include <asm/arch/audio.h> ++ ++#include "pxa2xx-pcm.h" ++ ++static DEFINE_MUTEX(car_mutex); ++static DECLARE_WAIT_QUEUE_HEAD(gsr_wq); ++static volatile long gsr_bits; ++ ++#define AC97_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define AC97_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000) ++ ++/* may need to expand this */ ++static struct snd_soc_dai_mode pxa2xx_ac97_modes[] = { ++ { ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = AC97_RATES, ++ .pcmdir = AC97_DIR, ++ }, ++}; ++ ++/* ++ * Beware PXA27x bugs: ++ * ++ * o Slot 12 read from modem space will hang controller. ++ * o CDONE, SDONE interrupt fails after any slot 12 IO. ++ * ++ * We therefore have an hybrid approach for waiting on SDONE (interrupt or ++ * 1 jiffy timeout if interrupt never comes). ++ */ ++ ++static unsigned short pxa2xx_ac97_read(struct snd_ac97 *ac97, ++ unsigned short reg) ++{ ++ unsigned short val = -1; ++ volatile u32 *reg_addr; ++ ++ mutex_lock(&car_mutex); ++ ++ /* set up primary or secondary codec/modem space */ ++#ifdef CONFIG_PXA27x ++ reg_addr = ac97->num ? &SAC_REG_BASE : &PAC_REG_BASE; ++#else ++ if (reg == AC97_GPIO_STATUS) ++ reg_addr = ac97->num ? &SMC_REG_BASE : &PMC_REG_BASE; ++ else ++ reg_addr = ac97->num ? &SAC_REG_BASE : &PAC_REG_BASE; ++#endif ++ reg_addr += (reg >> 1); ++ ++#ifndef CONFIG_PXA27x ++ if (reg == AC97_GPIO_STATUS) { ++ /* read from controller cache */ ++ val = *reg_addr; ++ goto out; ++ } ++#endif ++ ++ /* start read access across the ac97 link */ ++ GSR = GSR_CDONE | GSR_SDONE; ++ gsr_bits = 0; ++ val = *reg_addr; ++ ++ wait_event_timeout(gsr_wq, (GSR | gsr_bits) & GSR_SDONE, 1); ++ if (!((GSR | gsr_bits) & GSR_SDONE)) { ++ printk(KERN_ERR "%s: read error (ac97_reg=%x GSR=%#lx)\n", ++ __FUNCTION__, reg, GSR | gsr_bits); ++ val = -1; ++ goto out; ++ } ++ ++ /* valid data now */ ++ GSR = GSR_CDONE | GSR_SDONE; ++ gsr_bits = 0; ++ val = *reg_addr; ++ /* but we've just started another cycle... */ ++ wait_event_timeout(gsr_wq, (GSR | gsr_bits) & GSR_SDONE, 1); ++ ++out: mutex_unlock(&car_mutex); ++ return val; ++} ++ ++static void pxa2xx_ac97_write(struct snd_ac97 *ac97, unsigned short reg, ++ unsigned short val) ++{ ++ volatile u32 *reg_addr; ++ ++ mutex_lock(&car_mutex); ++ ++ /* set up primary or secondary codec/modem space */ ++#ifdef CONFIG_PXA27x ++ reg_addr = ac97->num ? &SAC_REG_BASE : &PAC_REG_BASE; ++#else ++ if (reg == AC97_GPIO_STATUS) ++ reg_addr = ac97->num ? &SMC_REG_BASE : &PMC_REG_BASE; ++ else ++ reg_addr = ac97->num ? &SAC_REG_BASE : &PAC_REG_BASE; ++#endif ++ reg_addr += (reg >> 1); ++ ++ GSR = GSR_CDONE | GSR_SDONE; ++ gsr_bits = 0; ++ *reg_addr = val; ++ wait_event_timeout(gsr_wq, (GSR | gsr_bits) & GSR_CDONE, 1); ++ if (!((GSR | gsr_bits) & GSR_CDONE)) ++ printk(KERN_ERR "%s: write error (ac97_reg=%x GSR=%#lx)\n", ++ __FUNCTION__, reg, GSR | gsr_bits); ++ ++ mutex_unlock(&car_mutex); ++} ++ ++static void pxa2xx_ac97_warm_reset(struct snd_ac97 *ac97) ++{ ++ gsr_bits = 0; ++ ++#ifdef CONFIG_PXA27x ++ /* warm reset broken on Bulverde, ++ so manually keep AC97 reset high */ ++ pxa_gpio_mode(113 | GPIO_OUT | GPIO_DFLT_HIGH); ++ udelay(10); ++ GCR |= GCR_WARM_RST; ++ pxa_gpio_mode(113 | GPIO_ALT_FN_2_OUT); ++ udelay(500); ++#else ++ GCR |= GCR_WARM_RST | GCR_PRIRDY_IEN | GCR_SECRDY_IEN; ++ wait_event_timeout(gsr_wq, gsr_bits & (GSR_PCR | GSR_SCR), 1); ++#endif ++ ++ if (!((GSR | gsr_bits) & (GSR_PCR | GSR_SCR))) ++ printk(KERN_INFO "%s: warm reset timeout (GSR=%#lx)\n", ++ __FUNCTION__, gsr_bits); ++ ++ GCR &= ~(GCR_PRIRDY_IEN|GCR_SECRDY_IEN); ++ GCR |= GCR_SDONE_IE|GCR_CDONE_IE; ++} ++ ++static void pxa2xx_ac97_cold_reset(struct snd_ac97 *ac97) ++{ ++ GCR &= GCR_COLD_RST; /* clear everything but nCRST */ ++ GCR &= ~GCR_COLD_RST; /* then assert nCRST */ ++ ++ gsr_bits = 0; ++#ifdef CONFIG_PXA27x ++ /* PXA27x Developers Manual section 13.5.2.2.1 */ ++ pxa_set_cken(1 << 31, 1); ++ udelay(5); ++ pxa_set_cken(1 << 31, 0); ++ GCR = GCR_COLD_RST; ++ udelay(50); ++#else ++ GCR = GCR_COLD_RST; ++ GCR |= GCR_CDONE_IE|GCR_SDONE_IE; ++ wait_event_timeout(gsr_wq, gsr_bits & (GSR_PCR | GSR_SCR), 1); ++#endif ++ ++ if (!((GSR | gsr_bits) & (GSR_PCR | GSR_SCR))) ++ printk(KERN_INFO "%s: cold reset timeout (GSR=%#lx)\n", ++ __FUNCTION__, gsr_bits); ++ ++ GCR &= ~(GCR_PRIRDY_IEN|GCR_SECRDY_IEN); ++ GCR |= GCR_SDONE_IE|GCR_CDONE_IE; ++} ++ ++static irqreturn_t pxa2xx_ac97_irq(int irq, void *dev_id) ++{ ++ long status; ++ ++ status = GSR; ++ if (status) { ++ GSR = status; ++ gsr_bits |= status; ++ wake_up(&gsr_wq); ++ ++#ifdef CONFIG_PXA27x ++ /* Although we don't use those we still need to clear them ++ since they tend to spuriously trigger when MMC is used ++ (hardware bug? go figure)... */ ++ MISR = MISR_EOC; ++ PISR = PISR_EOC; ++ MCSR = MCSR_EOC; ++#endif ++ ++ return IRQ_HANDLED; ++ } ++ ++ return IRQ_NONE; ++} ++ ++struct snd_ac97_bus_ops soc_ac97_ops = { ++ .read = pxa2xx_ac97_read, ++ .write = pxa2xx_ac97_write, ++ .warm_reset = pxa2xx_ac97_warm_reset, ++ .reset = pxa2xx_ac97_cold_reset, ++}; ++ ++static struct pxa2xx_pcm_dma_params pxa2xx_ac97_pcm_stereo_out = { ++ .name = "AC97 PCM Stereo out", ++ .dev_addr = __PREG(PCDR), ++ .drcmr = &DRCMRTXPCDR, ++ .dcmd = DCMD_INCSRCADDR | DCMD_FLOWTRG | ++ DCMD_BURST32 | DCMD_WIDTH4, ++}; ++ ++static struct pxa2xx_pcm_dma_params pxa2xx_ac97_pcm_stereo_in = { ++ .name = "AC97 PCM Stereo in", ++ .dev_addr = __PREG(PCDR), ++ .drcmr = &DRCMRRXPCDR, ++ .dcmd = DCMD_INCTRGADDR | DCMD_FLOWSRC | ++ DCMD_BURST32 | DCMD_WIDTH4, ++}; ++ ++static struct pxa2xx_pcm_dma_params pxa2xx_ac97_pcm_aux_mono_out = { ++ .name = "AC97 Aux PCM (Slot 5) Mono out", ++ .dev_addr = __PREG(MODR), ++ .drcmr = &DRCMRTXMODR, ++ .dcmd = DCMD_INCSRCADDR | DCMD_FLOWTRG | ++ DCMD_BURST16 | DCMD_WIDTH2, ++}; ++ ++static struct pxa2xx_pcm_dma_params pxa2xx_ac97_pcm_aux_mono_in = { ++ .name = "AC97 Aux PCM (Slot 5) Mono in", ++ .dev_addr = __PREG(MODR), ++ .drcmr = &DRCMRRXMODR, ++ .dcmd = DCMD_INCTRGADDR | DCMD_FLOWSRC | ++ DCMD_BURST16 | DCMD_WIDTH2, ++}; ++ ++static struct pxa2xx_pcm_dma_params pxa2xx_ac97_pcm_mic_mono_in = { ++ .name = "AC97 Mic PCM (Slot 6) Mono in", ++ .dev_addr = __PREG(MCDR), ++ .drcmr = &DRCMRRXMCDR, ++ .dcmd = DCMD_INCTRGADDR | DCMD_FLOWSRC | ++ DCMD_BURST16 | DCMD_WIDTH2, ++}; ++ ++#ifdef CONFIG_PM ++static int pxa2xx_ac97_suspend(struct platform_device *pdev, ++ struct snd_soc_cpu_dai *dai) ++{ ++ GCR |= GCR_ACLINK_OFF; ++ pxa_set_cken(CKEN2_AC97, 0); ++ return 0; ++} ++ ++static int pxa2xx_ac97_resume(struct platform_device *pdev, ++ struct snd_soc_cpu_dai *dai) ++{ ++ pxa_gpio_mode(GPIO31_SYNC_AC97_MD); ++ pxa_gpio_mode(GPIO30_SDATA_OUT_AC97_MD); ++ pxa_gpio_mode(GPIO28_BITCLK_AC97_MD); ++ pxa_gpio_mode(GPIO29_SDATA_IN_AC97_MD); ++#ifdef CONFIG_PXA27x ++ /* Use GPIO 113 as AC97 Reset on Bulverde */ ++ pxa_gpio_mode(113 | GPIO_ALT_FN_2_OUT); ++#endif ++ pxa_set_cken(CKEN2_AC97, 1); ++ return 0; ++} ++ ++#else ++#define pxa2xx_ac97_suspend NULL ++#define pxa2xx_ac97_resume NULL ++#endif ++ ++static int pxa2xx_ac97_probe(struct platform_device *pdev) ++{ ++ int ret; ++ ++ ret = request_irq(IRQ_AC97, pxa2xx_ac97_irq, IRQF_DISABLED, "AC97", NULL); ++ if (ret < 0) ++ goto err; ++ ++ pxa_gpio_mode(GPIO31_SYNC_AC97_MD); ++ pxa_gpio_mode(GPIO30_SDATA_OUT_AC97_MD); ++ pxa_gpio_mode(GPIO28_BITCLK_AC97_MD); ++ pxa_gpio_mode(GPIO29_SDATA_IN_AC97_MD); ++#ifdef CONFIG_PXA27x ++ /* Use GPIO 113 as AC97 Reset on Bulverde */ ++ pxa_gpio_mode(113 | GPIO_ALT_FN_2_OUT); ++#endif ++ pxa_set_cken(CKEN2_AC97, 1); ++ return 0; ++ ++ err: ++ if (CKEN & CKEN2_AC97) { ++ GCR |= GCR_ACLINK_OFF; ++ free_irq(IRQ_AC97, NULL); ++ pxa_set_cken(CKEN2_AC97, 0); ++ } ++ return ret; ++} ++ ++static void pxa2xx_ac97_remove(struct platform_device *pdev) ++{ ++ GCR |= GCR_ACLINK_OFF; ++ free_irq(IRQ_AC97, NULL); ++ pxa_set_cken(CKEN2_AC97, 0); ++} ++ ++static int pxa2xx_ac97_hw_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ rtd->cpu_dai->dma_data = &pxa2xx_ac97_pcm_stereo_out; ++ else ++ rtd->cpu_dai->dma_data = &pxa2xx_ac97_pcm_stereo_in; ++ ++ return 0; ++} ++ ++static int pxa2xx_ac97_hw_aux_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ rtd->cpu_dai->dma_data = &pxa2xx_ac97_pcm_aux_mono_out; ++ else ++ rtd->cpu_dai->dma_data = &pxa2xx_ac97_pcm_aux_mono_in; ++ ++ return 0; ++} ++ ++static int pxa2xx_ac97_hw_mic_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ return -ENODEV; ++ else ++ rtd->cpu_dai->dma_data = &pxa2xx_ac97_pcm_mic_mono_in; ++ ++ return 0; ++} ++ ++/* ++ * There is only 1 physical AC97 interface for pxa2xx, but it ++ * has extra fifo's that can be used for aux DACs and ADCs. ++ */ ++struct snd_soc_cpu_dai pxa_ac97_dai[] = { ++{ ++ .name = "pxa2xx-ac97", ++ .id = 0, ++ .type = SND_SOC_DAI_AC97, ++ .probe = pxa2xx_ac97_probe, ++ .remove = pxa2xx_ac97_remove, ++ .suspend = pxa2xx_ac97_suspend, ++ .resume = pxa2xx_ac97_resume, ++ .playback = { ++ .stream_name = "AC97 Playback", ++ .channels_min = 2, ++ .channels_max = 2,}, ++ .capture = { ++ .stream_name = "AC97 Capture", ++ .channels_min = 2, ++ .channels_max = 2,}, ++ .ops = { ++ .hw_params = pxa2xx_ac97_hw_params,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(pxa2xx_ac97_modes), ++ .mode = pxa2xx_ac97_modes,}, ++}, ++{ ++ .name = "pxa2xx-ac97-aux", ++ .id = 1, ++ .type = SND_SOC_DAI_AC97, ++ .playback = { ++ .stream_name = "AC97 Aux Playback", ++ .channels_min = 1, ++ .channels_max = 1,}, ++ .capture = { ++ .stream_name = "AC97 Aux Capture", ++ .channels_min = 1, ++ .channels_max = 1,}, ++ .ops = { ++ .hw_params = pxa2xx_ac97_hw_aux_params,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(pxa2xx_ac97_modes), ++ .mode = pxa2xx_ac97_modes,}, ++}, ++{ ++ .name = "pxa2xx-ac97-mic", ++ .id = 2, ++ .type = SND_SOC_DAI_AC97, ++ .capture = { ++ .stream_name = "AC97 Mic Capture", ++ .channels_min = 1, ++ .channels_max = 1,}, ++ .ops = { ++ .hw_params = pxa2xx_ac97_hw_mic_params,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(pxa2xx_ac97_modes), ++ .mode = pxa2xx_ac97_modes,},}, ++}; ++ ++EXPORT_SYMBOL_GPL(pxa_ac97_dai); ++EXPORT_SYMBOL_GPL(soc_ac97_ops); ++ ++MODULE_AUTHOR("Nicolas Pitre"); ++MODULE_DESCRIPTION("AC97 driver for the Intel PXA2xx chip"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/pxa/pxa2xx-i2s.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/pxa/pxa2xx-i2s.c +@@ -0,0 +1,354 @@ ++/* ++ * pxa2xx-i2s.c -- ALSA Soc Audio Layer ++ * ++ * Copyright 2005 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * 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. ++ * ++ * Revision history ++ * 12th Aug 2005 Initial version. ++ */ ++ ++#include <linux/init.h> ++#include <linux/module.h> ++#include <linux/device.h> ++#include <linux/delay.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/initval.h> ++#include <sound/soc.h> ++ ++#include <asm/hardware.h> ++#include <asm/arch/pxa-regs.h> ++#include <asm/arch/audio.h> ++ ++#include "pxa2xx-pcm.h" ++ ++/* used to disable sysclk if external crystal is used */ ++static int extclk; ++module_param(extclk, int, 0); ++MODULE_PARM_DESC(extclk, "set to 1 to disable pxa2xx i2s sysclk"); ++ ++struct pxa_i2s_port { ++ u32 sadiv; ++ u32 sacr0; ++ u32 sacr1; ++ u32 saimr; ++ int master; ++}; ++static struct pxa_i2s_port pxa_i2s; ++ ++#define PXA_I2S_DAIFMT \ ++ (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_LEFT_J | SND_SOC_DAIFMT_NB_NF) ++ ++#define PXA_I2S_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define PXA_I2S_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000) ++ ++/* priv is divider */ ++static struct snd_soc_dai_mode pxa2xx_i2s_modes[] = { ++ /* pxa2xx I2S frame and clock master modes */ ++ { ++ .fmt = PXA_I2S_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = PXA_I2S_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = SND_SOC_FSBD(4), ++ .priv = 0x48, ++ }, ++ { ++ .fmt = PXA_I2S_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_11025, ++ .pcmdir = PXA_I2S_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = SND_SOC_FSBD(4), ++ .priv = 0x34, ++ }, ++ { ++ .fmt = PXA_I2S_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_16000, ++ .pcmdir = PXA_I2S_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = SND_SOC_FSBD(4), ++ .priv = 0x24, ++ }, ++ { ++ .fmt = PXA_I2S_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_22050, ++ .pcmdir = PXA_I2S_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = SND_SOC_FSBD(4), ++ .priv = 0x1a, ++ }, ++ { ++ .fmt = PXA_I2S_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_44100, ++ .pcmdir = PXA_I2S_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = SND_SOC_FSBD(4), ++ .priv = 0xd, ++ }, ++ { ++ .fmt = PXA_I2S_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_48000, ++ .pcmdir = PXA_I2S_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = SND_SOC_FSBD(4), ++ .priv = 0xc, ++ }, ++ ++ /* pxa2xx I2S frame master and clock slave mode */ ++ { ++ .fmt = PXA_I2S_DAIFMT | SND_SOC_DAIFMT_CBM_CFS, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = PXA_I2S_RATES, ++ .pcmdir = PXA_I2S_DIR, ++ .fs = SND_SOC_FS_ALL, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .bfs = 64, ++ .priv = 0x48, ++ }, ++}; ++ ++static struct pxa2xx_pcm_dma_params pxa2xx_i2s_pcm_stereo_out = { ++ .name = "I2S PCM Stereo out", ++ .dev_addr = __PREG(SADR), ++ .drcmr = &DRCMRTXSADR, ++ .dcmd = DCMD_INCSRCADDR | DCMD_FLOWTRG | ++ DCMD_BURST32 | DCMD_WIDTH4, ++}; ++ ++static struct pxa2xx_pcm_dma_params pxa2xx_i2s_pcm_stereo_in = { ++ .name = "I2S PCM Stereo in", ++ .dev_addr = __PREG(SADR), ++ .drcmr = &DRCMRRXSADR, ++ .dcmd = DCMD_INCTRGADDR | DCMD_FLOWSRC | ++ DCMD_BURST32 | DCMD_WIDTH4, ++}; ++ ++static struct pxa2xx_gpio gpio_bus[] = { ++ { /* I2S SoC Slave */ ++ .rx = GPIO29_SDATA_IN_I2S_MD, ++ .tx = GPIO30_SDATA_OUT_I2S_MD, ++ .clk = GPIO28_BITCLK_IN_I2S_MD, ++ .frm = GPIO31_SYNC_I2S_MD, ++ }, ++ { /* I2S SoC Master */ ++#ifdef CONFIG_PXA27x ++ .sys = GPIO113_I2S_SYSCLK_MD, ++#else ++ .sys = GPIO32_SYSCLK_I2S_MD, ++#endif ++ .rx = GPIO29_SDATA_IN_I2S_MD, ++ .tx = GPIO30_SDATA_OUT_I2S_MD, ++ .clk = GPIO28_BITCLK_OUT_I2S_MD, ++ .frm = GPIO31_SYNC_I2S_MD, ++ }, ++}; ++ ++static int pxa2xx_i2s_startup(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ ++ if (!rtd->cpu_dai->active) { ++ SACR0 |= SACR0_RST; ++ SACR0 = 0; ++ } ++ ++ return 0; ++} ++ ++/* wait for I2S controller to be ready */ ++static int pxa_i2s_wait(void) ++{ ++ int i; ++ ++ /* flush the Rx FIFO */ ++ for(i = 0; i < 16; i++) ++ SADR; ++ return 0; ++} ++ ++static int pxa2xx_i2s_hw_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ ++ pxa_i2s.master = 0; ++ if (rtd->cpu_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CBS_CFS) ++ pxa_i2s.master = 1; ++ ++ if (pxa_i2s.master && !extclk) ++ pxa_gpio_mode(gpio_bus[pxa_i2s.master].sys); ++ ++ pxa_gpio_mode(gpio_bus[pxa_i2s.master].rx); ++ pxa_gpio_mode(gpio_bus[pxa_i2s.master].tx); ++ pxa_gpio_mode(gpio_bus[pxa_i2s.master].frm); ++ pxa_gpio_mode(gpio_bus[pxa_i2s.master].clk); ++ pxa_set_cken(CKEN8_I2S, 1); ++ pxa_i2s_wait(); ++ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ rtd->cpu_dai->dma_data = &pxa2xx_i2s_pcm_stereo_out; ++ else ++ rtd->cpu_dai->dma_data = &pxa2xx_i2s_pcm_stereo_in; ++ ++ /* is port used by another stream */ ++ if (!(SACR0 & SACR0_ENB)) { ++ ++ SACR0 = 0; ++ SACR1 = 0; ++ if (pxa_i2s.master) ++ SACR0 |= SACR0_BCKD; ++ ++ SACR0 |= SACR0_RFTH(14) | SACR0_TFTH(1); ++ ++ if (rtd->cpu_dai->dai_runtime.fmt & SND_SOC_DAIFMT_LEFT_J) ++ SACR1 |= SACR1_AMSL; ++ } ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ SAIMR |= SAIMR_TFS; ++ else ++ SAIMR |= SAIMR_RFS; ++ ++ SADIV = rtd->cpu_dai->dai_runtime.priv; ++ return 0; ++} ++ ++static int pxa2xx_i2s_trigger(struct snd_pcm_substream *substream, int cmd) ++{ ++ int ret = 0; ++ ++ switch (cmd) { ++ case SNDRV_PCM_TRIGGER_START: ++ SACR0 |= SACR0_ENB; ++ break; ++ case SNDRV_PCM_TRIGGER_RESUME: ++ case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: ++ case SNDRV_PCM_TRIGGER_STOP: ++ case SNDRV_PCM_TRIGGER_SUSPEND: ++ case SNDRV_PCM_TRIGGER_PAUSE_PUSH: ++ break; ++ default: ++ ret = -EINVAL; ++ } ++ ++ return ret; ++} ++ ++static void pxa2xx_i2s_shutdown(struct snd_pcm_substream *substream) ++{ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { ++ SACR1 |= SACR1_DRPL; ++ SAIMR &= ~SAIMR_TFS; ++ } else { ++ SACR1 |= SACR1_DREC; ++ SAIMR &= ~SAIMR_RFS; ++ } ++ ++ if (SACR1 & (SACR1_DREC | SACR1_DRPL)) { ++ SACR0 &= ~SACR0_ENB; ++ pxa_i2s_wait(); ++ pxa_set_cken(CKEN8_I2S, 0); ++ } ++} ++ ++#ifdef CONFIG_PM ++static int pxa2xx_i2s_suspend(struct platform_device *dev, ++ struct snd_soc_cpu_dai *dai) ++{ ++ if (!dai->active) ++ return 0; ++ ++ /* store registers */ ++ pxa_i2s.sacr0 = SACR0; ++ pxa_i2s.sacr1 = SACR1; ++ pxa_i2s.saimr = SAIMR; ++ pxa_i2s.sadiv = SADIV; ++ ++ /* deactivate link */ ++ SACR0 &= ~SACR0_ENB; ++ pxa_i2s_wait(); ++ return 0; ++} ++ ++static int pxa2xx_i2s_resume(struct platform_device *pdev, ++ struct snd_soc_cpu_dai *dai) ++{ ++ if (!dai->active) ++ return 0; ++ ++ pxa_i2s_wait(); ++ ++ SACR0 = pxa_i2s.sacr0 &= ~SACR0_ENB; ++ SACR1 = pxa_i2s.sacr1; ++ SAIMR = pxa_i2s.saimr; ++ SADIV = pxa_i2s.sadiv; ++ SACR0 |= SACR0_ENB; ++ ++ return 0; ++} ++ ++#else ++#define pxa2xx_i2s_suspend NULL ++#define pxa2xx_i2s_resume NULL ++#endif ++ ++/* pxa2xx I2S sysclock is always 256 FS */ ++static unsigned int pxa_i2s_config_sysclk(struct snd_soc_cpu_dai *iface, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ return info->rate << 8; ++} ++ ++struct snd_soc_cpu_dai pxa_i2s_dai = { ++ .name = "pxa2xx-i2s", ++ .id = 0, ++ .type = SND_SOC_DAI_I2S, ++ .suspend = pxa2xx_i2s_suspend, ++ .resume = pxa2xx_i2s_resume, ++ .config_sysclk = pxa_i2s_config_sysclk, ++ .playback = { ++ .channels_min = 2, ++ .channels_max = 2,}, ++ .capture = { ++ .channels_min = 2, ++ .channels_max = 2,}, ++ .ops = { ++ .startup = pxa2xx_i2s_startup, ++ .shutdown = pxa2xx_i2s_shutdown, ++ .trigger = pxa2xx_i2s_trigger, ++ .hw_params = pxa2xx_i2s_hw_params,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(pxa2xx_i2s_modes), ++ .mode = pxa2xx_i2s_modes,}, ++}; ++ ++EXPORT_SYMBOL_GPL(pxa_i2s_dai); ++ ++/* Module information */ ++MODULE_AUTHOR("Liam Girdwood, liam.girdwood@wolfsonmicro.com, www.wolfsonmicro.com"); ++MODULE_DESCRIPTION("pxa2xx I2S SoC Interface"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/pxa/pxa2xx-pcm.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/pxa/pxa2xx-pcm.c +@@ -0,0 +1,363 @@ ++/* ++ * linux/sound/arm/pxa2xx-pcm.c -- ALSA PCM interface for the Intel PXA2xx chip ++ * ++ * Author: Nicolas Pitre ++ * Created: Nov 30, 2004 ++ * Copyright: (C) 2004 MontaVista Software, Inc. ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#include <linux/module.h> ++#include <linux/init.h> ++#include <linux/platform_device.h> ++#include <linux/slab.h> ++#include <linux/dma-mapping.h> ++ ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/pcm_params.h> ++#include <sound/soc.h> ++ ++#include <asm/dma.h> ++#include <asm/hardware.h> ++#include <asm/arch/pxa-regs.h> ++#include <asm/arch/audio.h> ++ ++#include "pxa2xx-pcm.h" ++ ++static const struct snd_pcm_hardware pxa2xx_pcm_hardware = { ++ .info = SNDRV_PCM_INFO_MMAP | ++ SNDRV_PCM_INFO_MMAP_VALID | ++ SNDRV_PCM_INFO_INTERLEAVED | ++ SNDRV_PCM_INFO_PAUSE | ++ SNDRV_PCM_INFO_RESUME, ++ .formats = SNDRV_PCM_FMTBIT_S16_LE | ++ SNDRV_PCM_FMTBIT_S24_LE | ++ SNDRV_PCM_FMTBIT_S32_LE, ++ .period_bytes_min = 32, ++ .period_bytes_max = 8192 - 32, ++ .periods_min = 1, ++ .periods_max = PAGE_SIZE/sizeof(pxa_dma_desc), ++ .buffer_bytes_max = 128 * 1024, ++ .fifo_size = 32, ++}; ++ ++struct pxa2xx_runtime_data { ++ int dma_ch; ++ struct pxa2xx_pcm_dma_params *params; ++ pxa_dma_desc *dma_desc_array; ++ dma_addr_t dma_desc_array_phys; ++}; ++ ++static void pxa2xx_pcm_dma_irq(int dma_ch, void *dev_id) ++{ ++ struct snd_pcm_substream *substream = dev_id; ++ struct pxa2xx_runtime_data *prtd = substream->runtime->private_data; ++ int dcsr; ++ ++ dcsr = DCSR(dma_ch); ++ DCSR(dma_ch) = dcsr & ~DCSR_STOPIRQEN; ++ ++ if (dcsr & DCSR_ENDINTR) { ++ snd_pcm_period_elapsed(substream); ++ } else { ++ printk( KERN_ERR "%s: DMA error on channel %d (DCSR=%#x)\n", ++ prtd->params->name, dma_ch, dcsr ); ++ } ++} ++ ++static int pxa2xx_pcm_hw_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct pxa2xx_runtime_data *prtd = runtime->private_data; ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct pxa2xx_pcm_dma_params *dma = rtd->cpu_dai->dma_data; ++ size_t totsize = params_buffer_bytes(params); ++ size_t period = params_period_bytes(params); ++ pxa_dma_desc *dma_desc; ++ dma_addr_t dma_buff_phys, next_desc_phys; ++ int ret; ++ ++ /* this may get called several times by oss emulation ++ * with different params */ ++ if (prtd->params == NULL) { ++ prtd->params = dma; ++ ret = pxa_request_dma(prtd->params->name, DMA_PRIO_LOW, ++ pxa2xx_pcm_dma_irq, substream); ++ if (ret < 0) ++ return ret; ++ prtd->dma_ch = ret; ++ } else if (prtd->params != dma) { ++ pxa_free_dma(prtd->dma_ch); ++ prtd->params = dma; ++ ret = pxa_request_dma(prtd->params->name, DMA_PRIO_LOW, ++ pxa2xx_pcm_dma_irq, substream); ++ if (ret < 0) ++ return ret; ++ prtd->dma_ch = ret; ++ } ++ ++ snd_pcm_set_runtime_buffer(substream, &substream->dma_buffer); ++ runtime->dma_bytes = totsize; ++ ++ dma_desc = prtd->dma_desc_array; ++ next_desc_phys = prtd->dma_desc_array_phys; ++ dma_buff_phys = runtime->dma_addr; ++ do { ++ next_desc_phys += sizeof(pxa_dma_desc); ++ dma_desc->ddadr = next_desc_phys; ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { ++ dma_desc->dsadr = dma_buff_phys; ++ dma_desc->dtadr = prtd->params->dev_addr; ++ } else { ++ dma_desc->dsadr = prtd->params->dev_addr; ++ dma_desc->dtadr = dma_buff_phys; ++ } ++ if (period > totsize) ++ period = totsize; ++ dma_desc->dcmd = prtd->params->dcmd | period | DCMD_ENDIRQEN; ++ dma_desc++; ++ dma_buff_phys += period; ++ } while (totsize -= period); ++ dma_desc[-1].ddadr = prtd->dma_desc_array_phys; ++ ++ return 0; ++} ++ ++static int pxa2xx_pcm_hw_free(struct snd_pcm_substream *substream) ++{ ++ struct pxa2xx_runtime_data *prtd = substream->runtime->private_data; ++ ++ if (prtd && prtd->params) ++ *prtd->params->drcmr = 0; ++ ++ if (prtd->dma_ch) { ++ snd_pcm_set_runtime_buffer(substream, NULL); ++ pxa_free_dma(prtd->dma_ch); ++ prtd->dma_ch = 0; ++ } ++ ++ return 0; ++} ++ ++static int pxa2xx_pcm_prepare(struct snd_pcm_substream *substream) ++{ ++ struct pxa2xx_runtime_data *prtd = substream->runtime->private_data; ++ ++ DCSR(prtd->dma_ch) &= ~DCSR_RUN; ++ DCSR(prtd->dma_ch) = 0; ++ DCMD(prtd->dma_ch) = 0; ++ *prtd->params->drcmr = prtd->dma_ch | DRCMR_MAPVLD; ++ ++ return 0; ++} ++ ++static int pxa2xx_pcm_trigger(struct snd_pcm_substream *substream, int cmd) ++{ ++ struct pxa2xx_runtime_data *prtd = substream->runtime->private_data; ++ int ret = 0; ++ ++ switch (cmd) { ++ case SNDRV_PCM_TRIGGER_START: ++ DDADR(prtd->dma_ch) = prtd->dma_desc_array_phys; ++ DCSR(prtd->dma_ch) = DCSR_RUN; ++ break; ++ ++ case SNDRV_PCM_TRIGGER_STOP: ++ case SNDRV_PCM_TRIGGER_SUSPEND: ++ case SNDRV_PCM_TRIGGER_PAUSE_PUSH: ++ DCSR(prtd->dma_ch) &= ~DCSR_RUN; ++ break; ++ ++ case SNDRV_PCM_TRIGGER_RESUME: ++ DCSR(prtd->dma_ch) |= DCSR_RUN; ++ break; ++ case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: ++ DDADR(prtd->dma_ch) = prtd->dma_desc_array_phys; ++ DCSR(prtd->dma_ch) |= DCSR_RUN; ++ break; ++ ++ default: ++ ret = -EINVAL; ++ } ++ ++ return ret; ++} ++ ++static snd_pcm_uframes_t ++pxa2xx_pcm_pointer(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct pxa2xx_runtime_data *prtd = runtime->private_data; ++ ++ dma_addr_t ptr = (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ? ++ DSADR(prtd->dma_ch) : DTADR(prtd->dma_ch); ++ snd_pcm_uframes_t x = bytes_to_frames(runtime, ptr - runtime->dma_addr); ++ ++ if (x == runtime->buffer_size) ++ x = 0; ++ return x; ++} ++ ++static int pxa2xx_pcm_open(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct pxa2xx_runtime_data *prtd; ++ int ret; ++ ++ snd_soc_set_runtime_hwparams(substream, &pxa2xx_pcm_hardware); ++ ++ /* ++ * For mysterious reasons (and despite what the manual says) ++ * playback samples are lost if the DMA count is not a multiple ++ * of the DMA burst size. Let's add a rule to enforce that. ++ */ ++ ret = snd_pcm_hw_constraint_step(runtime, 0, ++ SNDRV_PCM_HW_PARAM_PERIOD_BYTES, 32); ++ if (ret) ++ goto out; ++ ++ ret = snd_pcm_hw_constraint_step(runtime, 0, ++ SNDRV_PCM_HW_PARAM_BUFFER_BYTES, 32); ++ if (ret) ++ goto out; ++ ++ prtd = kzalloc(sizeof(struct pxa2xx_runtime_data), GFP_KERNEL); ++ if (prtd == NULL) { ++ ret = -ENOMEM; ++ goto out; ++ } ++ ++ prtd->dma_desc_array = ++ dma_alloc_writecombine(substream->pcm->card->dev, PAGE_SIZE, ++ &prtd->dma_desc_array_phys, GFP_KERNEL); ++ if (!prtd->dma_desc_array) { ++ ret = -ENOMEM; ++ goto err1; ++ } ++ ++ runtime->private_data = prtd; ++ return 0; ++ ++ err1: ++ kfree(prtd); ++ out: ++ return ret; ++} ++ ++static int pxa2xx_pcm_close(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct pxa2xx_runtime_data *prtd = runtime->private_data; ++ ++ dma_free_writecombine(substream->pcm->card->dev, PAGE_SIZE, ++ prtd->dma_desc_array, prtd->dma_desc_array_phys); ++ kfree(prtd); ++ return 0; ++} ++ ++static int pxa2xx_pcm_mmap(struct snd_pcm_substream *substream, ++ struct vm_area_struct *vma) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ return dma_mmap_writecombine(substream->pcm->card->dev, vma, ++ runtime->dma_area, ++ runtime->dma_addr, ++ runtime->dma_bytes); ++} ++ ++struct snd_pcm_ops pxa2xx_pcm_ops = { ++ .open = pxa2xx_pcm_open, ++ .close = pxa2xx_pcm_close, ++ .ioctl = snd_pcm_lib_ioctl, ++ .hw_params = pxa2xx_pcm_hw_params, ++ .hw_free = pxa2xx_pcm_hw_free, ++ .prepare = pxa2xx_pcm_prepare, ++ .trigger = pxa2xx_pcm_trigger, ++ .pointer = pxa2xx_pcm_pointer, ++ .mmap = pxa2xx_pcm_mmap, ++}; ++ ++static int pxa2xx_pcm_preallocate_dma_buffer(struct snd_pcm *pcm, int stream) ++{ ++ struct snd_pcm_substream *substream = pcm->streams[stream].substream; ++ struct snd_dma_buffer *buf = &substream->dma_buffer; ++ size_t size = pxa2xx_pcm_hardware.buffer_bytes_max; ++ buf->dev.type = SNDRV_DMA_TYPE_DEV; ++ buf->dev.dev = pcm->card->dev; ++ buf->private_data = NULL; ++ buf->area = dma_alloc_writecombine(pcm->card->dev, size, ++ &buf->addr, GFP_KERNEL); ++ if (!buf->area) ++ return -ENOMEM; ++ buf->bytes = size; ++ return 0; ++} ++ ++static void pxa2xx_pcm_free_dma_buffers(struct snd_pcm *pcm) ++{ ++ struct snd_pcm_substream *substream; ++ struct snd_dma_buffer *buf; ++ int stream; ++ ++ for (stream = 0; stream < 2; stream++) { ++ substream = pcm->streams[stream].substream; ++ if (!substream) ++ continue; ++ ++ buf = &substream->dma_buffer; ++ if (!buf->area) ++ continue; ++ ++ dma_free_writecombine(pcm->card->dev, buf->bytes, ++ buf->area, buf->addr); ++ buf->area = NULL; ++ } ++} ++ ++static u64 pxa2xx_pcm_dmamask = DMA_32BIT_MASK; ++ ++int pxa2xx_pcm_new(struct snd_card *card, struct snd_soc_codec_dai *dai, ++ struct snd_pcm *pcm) ++{ ++ int ret = 0; ++ ++ if (!card->dev->dma_mask) ++ card->dev->dma_mask = &pxa2xx_pcm_dmamask; ++ if (!card->dev->coherent_dma_mask) ++ card->dev->coherent_dma_mask = DMA_32BIT_MASK; ++ ++ if (dai->playback.channels_min) { ++ ret = pxa2xx_pcm_preallocate_dma_buffer(pcm, ++ SNDRV_PCM_STREAM_PLAYBACK); ++ if (ret) ++ goto out; ++ } ++ ++ if (dai->capture.channels_min) { ++ ret = pxa2xx_pcm_preallocate_dma_buffer(pcm, ++ SNDRV_PCM_STREAM_CAPTURE); ++ if (ret) ++ goto out; ++ } ++ out: ++ return ret; ++} ++ ++struct snd_soc_platform pxa2xx_soc_platform = { ++ .name = "pxa2xx-audio", ++ .pcm_ops = &pxa2xx_pcm_ops, ++ .pcm_new = pxa2xx_pcm_new, ++ .pcm_free = pxa2xx_pcm_free_dma_buffers, ++}; ++ ++EXPORT_SYMBOL_GPL(pxa2xx_soc_platform); ++ ++MODULE_AUTHOR("Nicolas Pitre"); ++MODULE_DESCRIPTION("Intel PXA2xx PCM DMA module"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/pxa/pxa2xx-pcm.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/pxa/pxa2xx-pcm.h +@@ -0,0 +1,48 @@ ++/* ++ * linux/sound/arm/pxa2xx-pcm.h -- ALSA PCM interface for the Intel PXA2xx chip ++ * ++ * Author: Nicolas Pitre ++ * Created: Nov 30, 2004 ++ * Copyright: MontaVista Software, Inc. ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#ifndef _PXA2XX_PCM_H ++#define _PXA2XX_PCM_H ++ ++struct pxa2xx_pcm_dma_params { ++ char *name; /* stream identifier */ ++ u32 dcmd; /* DMA descriptor dcmd field */ ++ volatile u32 *drcmr; /* the DMA request channel to use */ ++ u32 dev_addr; /* device physical address for DMA */ ++}; ++ ++struct pxa2xx_gpio { ++ u32 sys; ++ u32 rx; ++ u32 tx; ++ u32 clk; ++ u32 frm; ++}; ++ ++/* pxa2xx DAI ID's */ ++#define PXA2XX_DAI_AC97_HIFI 0 ++#define PXA2XX_DAI_AC97_AUX 1 ++#define PXA2XX_DAI_AC97_MIC 2 ++#define PXA2XX_DAI_I2S 0 ++#define PXA2XX_DAI_SSP1 0 ++#define PXA2XX_DAI_SSP2 1 ++#define PXA2XX_DAI_SSP3 2 ++ ++extern struct snd_soc_cpu_dai pxa_ac97_dai[3]; ++extern struct snd_soc_cpu_dai pxa_i2s_dai; ++extern struct snd_soc_cpu_dai pxa_ssp_dai[3]; ++ ++/* platform data */ ++extern struct snd_soc_platform pxa2xx_soc_platform; ++extern struct snd_ac97_bus_ops pxa2xx_ac97_ops; ++ ++#endif +Index: linux-2.6-pxa-new/sound/soc/pxa/pxa2xx-ssp.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/pxa/pxa2xx-ssp.c +@@ -0,0 +1,767 @@ ++/* ++ * pxa2xx-ssp.c -- ALSA Soc Audio Layer ++ * ++ * Copyright 2005 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * 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. ++ * ++ * Revision history ++ * 12th Aug 2005 Initial version. ++ * ++ * TODO: ++ * o Fix master mode (bug) ++ * o Fix resume (bug) ++ * o Add support for other clocks ++ * o Test network mode for > 16bit sample size ++ */ ++ ++#include <linux/init.h> ++#include <linux/module.h> ++#include <linux/platform_device.h> ++ ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/initval.h> ++#include <sound/soc.h> ++ ++#include <asm/hardware.h> ++#include <asm/arch/pxa-regs.h> ++#include <asm/arch/audio.h> ++#include <asm/arch/ssp.h> ++ ++#include "pxa2xx-pcm.h" ++ ++/* ++ * SSP sysclock frequency in Hz ++ * Neither default pxa2xx PLL clocks are good for audio, hence pxa27x ++ * has audio clock. I would recommend using the pxa27x audio clock or an ++ * external clock or making the codec master to gurantee better sample rates. ++ */ ++#ifdef CONFIG_PXA27x ++static int sysclk[3] = {13000000, 13000000, 13000000}; ++#else ++static int sysclk[3] = {1843200, 1843200, 1843200}; ++#endif ++module_param_array(sysclk, int, NULL, 0); ++MODULE_PARM_DESC(sysclk, "sysclk frequency in Hz"); ++ ++/* ++ * SSP sysclock source. ++ * sysclk is ignored if audio clock is used ++ */ ++#ifdef CONFIG_PXA27x ++static int clksrc[3] = {0, 0, 0}; ++#else ++static int clksrc[3] = {0, 0, 0}; ++#endif ++module_param_array(clksrc, int, NULL, 0); ++MODULE_PARM_DESC(clksrc, ++ "sysclk source, 0 = internal PLL, 1 = ext, 2 = network, 3 = audio clock"); ++ ++/* ++ * SSP GPIO's ++ */ ++#define GPIO26_SSP1RX_MD (26 | GPIO_ALT_FN_1_IN) ++#define GPIO25_SSP1TX_MD (25 | GPIO_ALT_FN_2_OUT) ++#define GPIO23_SSP1CLKS_MD (23 | GPIO_ALT_FN_2_IN) ++#define GPIO24_SSP1FRMS_MD (24 | GPIO_ALT_FN_2_IN) ++#define GPIO23_SSP1CLKM_MD (23 | GPIO_ALT_FN_2_OUT) ++#define GPIO24_SSP1FRMM_MD (24 | GPIO_ALT_FN_2_OUT) ++ ++#define GPIO11_SSP2RX_MD (11 | GPIO_ALT_FN_2_IN) ++#define GPIO13_SSP2TX_MD (13 | GPIO_ALT_FN_1_OUT) ++#define GPIO22_SSP2CLKS_MD (22 | GPIO_ALT_FN_3_IN) ++#define GPIO88_SSP2FRMS_MD (88 | GPIO_ALT_FN_3_IN) ++#define GPIO22_SSP2CLKM_MD (22 | GPIO_ALT_FN_3_OUT) ++#define GPIO88_SSP2FRMM_MD (88 | GPIO_ALT_FN_3_OUT) ++ ++#define GPIO82_SSP3RX_MD (82 | GPIO_ALT_FN_1_IN) ++#define GPIO81_SSP3TX_MD (81 | GPIO_ALT_FN_1_OUT) ++#define GPIO84_SSP3CLKS_MD (84 | GPIO_ALT_FN_1_IN) ++#define GPIO83_SSP3FRMS_MD (83 | GPIO_ALT_FN_1_IN) ++#define GPIO84_SSP3CLKM_MD (84 | GPIO_ALT_FN_1_OUT) ++#define GPIO83_SSP3FRMM_MD (83 | GPIO_ALT_FN_1_OUT) ++ ++#define PXA_SSP_MDAIFMT \ ++ (SND_SOC_DAIFMT_DSP_B |SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBS_CFS | SND_SOC_DAIFMT_CBM_CFS | \ ++ SND_SOC_DAIFMT_CBS_CFM | SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_NB_IF) ++ ++#define PXA_SSP_SDAIFMT \ ++ (SND_SOC_DAIFMT_DSP_B |SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_CBM_CFS | \ ++ SND_SOC_DAIFMT_CBS_CFM | SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_NB_IF) ++ ++#define PXA_SSP_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define PXA_SSP_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000 | \ ++ SNDRV_PCM_RATE_176400 | SNDRV_PCM_RATE_192000) ++ ++#define PXA_SSP_BITS \ ++ (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE | \ ++ SNDRV_PCM_FMTBIT_S24_LE | SNDRV_PCM_FMTBIT_S32_LE) ++ ++/* ++ * SSP modes ++ */ ++static struct snd_soc_dai_mode pxa2xx_ssp_modes[] = { ++ /* port slave clk & frame modes */ ++ { ++ .fmt = PXA_SSP_SDAIFMT, ++ .pcmfmt = PXA_SSP_BITS, ++ .pcmrate = PXA_SSP_RATES, ++ .pcmdir = PXA_SSP_DIR, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = SND_SOC_FSB_ALL, ++ }, ++ ++ /* port master clk & frame modes */ ++#ifdef CONFIG_PXA27x ++ { ++ .fmt = PXA_SSP_MDAIFMT, ++ .pcmfmt = PXA_SSP_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = PXA_SSP_DIR, ++ .flags = SND_SOC_DAI_BFS_RCW, ++ .fs = 256, ++ .bfs = SND_SOC_FSBW(1), ++ }, ++ { ++ .fmt = PXA_SSP_MDAIFMT, ++ .pcmfmt = PXA_SSP_BITS, ++ .pcmrate = SNDRV_PCM_RATE_11025, ++ .pcmdir = PXA_SSP_DIR, ++ .flags = SND_SOC_DAI_BFS_RCW, ++ .fs = 256, ++ .bfs = SND_SOC_FSBW(1), ++ }, ++ { ++ .fmt = PXA_SSP_MDAIFMT, ++ .pcmfmt = PXA_SSP_BITS, ++ .pcmrate = SNDRV_PCM_RATE_16000, ++ .pcmdir = PXA_SSP_DIR, ++ .flags = SND_SOC_DAI_BFS_RCW, ++ .fs = 256, ++ .bfs = SND_SOC_FSBW(1), ++ }, ++ { ++ .fmt = PXA_SSP_MDAIFMT, ++ .pcmfmt = PXA_SSP_BITS, ++ .pcmrate = SNDRV_PCM_RATE_22050, ++ .pcmdir = PXA_SSP_DIR, ++ .flags = SND_SOC_DAI_BFS_RCW, ++ .fs = 256, ++ .bfs = SND_SOC_FSBW(1), ++ }, ++ { ++ .fmt = PXA_SSP_MDAIFMT, ++ .pcmfmt = PXA_SSP_BITS, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = PXA_SSP_DIR, ++ .flags = SND_SOC_DAI_BFS_RCW, ++ .fs = 256, ++ .bfs = SND_SOC_FSBW(1), ++ }, ++ { ++ .fmt = PXA_SSP_MDAIFMT, ++ .pcmfmt = PXA_SSP_BITS, ++ .pcmrate = SNDRV_PCM_RATE_44100, ++ .pcmdir = PXA_SSP_DIR, ++ .flags = SND_SOC_DAI_BFS_RCW, ++ .fs = 256, ++ .bfs = SND_SOC_FSBW(1), ++ }, ++ { ++ .fmt = PXA_SSP_MDAIFMT, ++ .pcmfmt = PXA_SSP_BITS, ++ .pcmrate = SNDRV_PCM_RATE_48000, ++ .pcmdir = PXA_SSP_DIR, ++ .flags = SND_SOC_DAI_BFS_RCW, ++ .fs = 256, ++ .bfs = SND_SOC_FSBW(1), ++ }, ++ { ++ .fmt = PXA_SSP_MDAIFMT, ++ .pcmfmt = PXA_SSP_BITS, ++ .pcmrate = SNDRV_PCM_RATE_88200, ++ .pcmdir = PXA_SSP_DIR, ++ .flags = SND_SOC_DAI_BFS_RCW, ++ .fs = 128, ++ .bfs = SND_SOC_FSBW(1), ++ }, ++ { ++ .fmt = PXA_SSP_MDAIFMT, ++ .pcmfmt = PXA_SSP_BITS, ++ .pcmrate = SNDRV_PCM_RATE_96000, ++ .pcmdir = PXA_SSP_DIR, ++ .flags = SND_SOC_DAI_BFS_RCW, ++ .fs = 128, ++ .bfs = SND_SOC_FSBW(1), ++ }, ++#endif ++}; ++ ++static struct ssp_dev ssp[3]; ++#ifdef CONFIG_PM ++static struct ssp_state ssp_state[3]; ++#endif ++ ++static struct pxa2xx_pcm_dma_params pxa2xx_ssp1_pcm_mono_out = { ++ .name = "SSP1 PCM Mono out", ++ .dev_addr = __PREG(SSDR_P1), ++ .drcmr = &DRCMRTXSSDR, ++ .dcmd = DCMD_INCSRCADDR | DCMD_FLOWTRG | ++ DCMD_BURST16 | DCMD_WIDTH2, ++}; ++ ++static struct pxa2xx_pcm_dma_params pxa2xx_ssp1_pcm_mono_in = { ++ .name = "SSP1 PCM Mono in", ++ .dev_addr = __PREG(SSDR_P1), ++ .drcmr = &DRCMRRXSSDR, ++ .dcmd = DCMD_INCTRGADDR | DCMD_FLOWSRC | ++ DCMD_BURST16 | DCMD_WIDTH2, ++}; ++ ++static struct pxa2xx_pcm_dma_params pxa2xx_ssp1_pcm_stereo_out = { ++ .name = "SSP1 PCM Stereo out", ++ .dev_addr = __PREG(SSDR_P1), ++ .drcmr = &DRCMRTXSSDR, ++ .dcmd = DCMD_INCSRCADDR | DCMD_FLOWTRG | ++ DCMD_BURST16 | DCMD_WIDTH4, ++}; ++ ++static struct pxa2xx_pcm_dma_params pxa2xx_ssp1_pcm_stereo_in = { ++ .name = "SSP1 PCM Stereo in", ++ .dev_addr = __PREG(SSDR_P1), ++ .drcmr = &DRCMRRXSSDR, ++ .dcmd = DCMD_INCTRGADDR | DCMD_FLOWSRC | ++ DCMD_BURST16 | DCMD_WIDTH4, ++}; ++ ++static struct pxa2xx_pcm_dma_params pxa2xx_ssp2_pcm_mono_out = { ++ .name = "SSP2 PCM Mono out", ++ .dev_addr = __PREG(SSDR_P2), ++ .drcmr = &DRCMRTXSS2DR, ++ .dcmd = DCMD_INCSRCADDR | DCMD_FLOWTRG | ++ DCMD_BURST16 | DCMD_WIDTH2, ++}; ++ ++static struct pxa2xx_pcm_dma_params pxa2xx_ssp2_pcm_mono_in = { ++ .name = "SSP2 PCM Mono in", ++ .dev_addr = __PREG(SSDR_P2), ++ .drcmr = &DRCMRRXSS2DR, ++ .dcmd = DCMD_INCTRGADDR | DCMD_FLOWSRC | ++ DCMD_BURST16 | DCMD_WIDTH2, ++}; ++ ++static struct pxa2xx_pcm_dma_params pxa2xx_ssp2_pcm_stereo_out = { ++ .name = "SSP2 PCM Stereo out", ++ .dev_addr = __PREG(SSDR_P2), ++ .drcmr = &DRCMRTXSS2DR, ++ .dcmd = DCMD_INCSRCADDR | DCMD_FLOWTRG | ++ DCMD_BURST16 | DCMD_WIDTH4, ++}; ++ ++static struct pxa2xx_pcm_dma_params pxa2xx_ssp2_pcm_stereo_in = { ++ .name = "SSP2 PCM Stereo in", ++ .dev_addr = __PREG(SSDR_P2), ++ .drcmr = &DRCMRRXSS2DR, ++ .dcmd = DCMD_INCTRGADDR | DCMD_FLOWSRC | ++ DCMD_BURST16 | DCMD_WIDTH4, ++}; ++ ++static struct pxa2xx_pcm_dma_params pxa2xx_ssp3_pcm_mono_out = { ++ .name = "SSP3 PCM Mono out", ++ .dev_addr = __PREG(SSDR_P3), ++ .drcmr = &DRCMRTXSS3DR, ++ .dcmd = DCMD_INCSRCADDR | DCMD_FLOWTRG | ++ DCMD_BURST16 | DCMD_WIDTH2, ++}; ++ ++static struct pxa2xx_pcm_dma_params pxa2xx_ssp3_pcm_mono_in = { ++ .name = "SSP3 PCM Mono in", ++ .dev_addr = __PREG(SSDR_P3), ++ .drcmr = &DRCMRRXSS3DR, ++ .dcmd = DCMD_INCTRGADDR | DCMD_FLOWSRC | ++ DCMD_BURST16 | DCMD_WIDTH2, ++}; ++ ++static struct pxa2xx_pcm_dma_params pxa2xx_ssp3_pcm_stereo_out = { ++ .name = "SSP3 PCM Stereo out", ++ .dev_addr = __PREG(SSDR_P3), ++ .drcmr = &DRCMRTXSS3DR, ++ .dcmd = DCMD_INCSRCADDR | DCMD_FLOWTRG | ++ DCMD_BURST16 | DCMD_WIDTH4, ++}; ++ ++static struct pxa2xx_pcm_dma_params pxa2xx_ssp3_pcm_stereo_in = { ++ .name = "SSP3 PCM Stereo in", ++ .dev_addr = __PREG(SSDR_P3), ++ .drcmr = &DRCMRRXSS3DR, ++ .dcmd = DCMD_INCTRGADDR | DCMD_FLOWSRC | ++ DCMD_BURST16 | DCMD_WIDTH4, ++}; ++ ++static struct pxa2xx_pcm_dma_params *ssp_dma_params[3][4] = { ++ {&pxa2xx_ssp1_pcm_mono_out, &pxa2xx_ssp1_pcm_mono_in, ++ &pxa2xx_ssp1_pcm_stereo_out,&pxa2xx_ssp1_pcm_stereo_in,}, ++ {&pxa2xx_ssp2_pcm_mono_out, &pxa2xx_ssp2_pcm_mono_in, ++ &pxa2xx_ssp2_pcm_stereo_out, &pxa2xx_ssp2_pcm_stereo_in,}, ++ {&pxa2xx_ssp3_pcm_mono_out, &pxa2xx_ssp3_pcm_mono_in, ++ &pxa2xx_ssp3_pcm_stereo_out,&pxa2xx_ssp3_pcm_stereo_in,}, ++}; ++ ++static struct pxa2xx_gpio ssp_gpios[3][4] = { ++ {{ /* SSP1 SND_SOC_DAIFMT_CBM_CFM */ ++ .rx = GPIO26_SSP1RX_MD, ++ .tx = GPIO25_SSP1TX_MD, ++ .clk = (23 | GPIO_ALT_FN_2_IN), ++ .frm = (24 | GPIO_ALT_FN_2_IN), ++ }, ++ { /* SSP1 SND_SOC_DAIFMT_CBS_CFS */ ++ .rx = GPIO26_SSP1RX_MD, ++ .tx = GPIO25_SSP1TX_MD, ++ .clk = (23 | GPIO_ALT_FN_2_OUT), ++ .frm = (24 | GPIO_ALT_FN_2_OUT), ++ }, ++ { /* SSP1 SND_SOC_DAIFMT_CBS_CFM */ ++ .rx = GPIO26_SSP1RX_MD, ++ .tx = GPIO25_SSP1TX_MD, ++ .clk = (23 | GPIO_ALT_FN_2_OUT), ++ .frm = (24 | GPIO_ALT_FN_2_IN), ++ }, ++ { /* SSP1 SND_SOC_DAIFMT_CBM_CFS */ ++ .rx = GPIO26_SSP1RX_MD, ++ .tx = GPIO25_SSP1TX_MD, ++ .clk = (23 | GPIO_ALT_FN_2_IN), ++ .frm = (24 | GPIO_ALT_FN_2_OUT), ++ }}, ++ {{ /* SSP2 SND_SOC_DAIFMT_CBM_CFM */ ++ .rx = GPIO11_SSP2RX_MD, ++ .tx = GPIO13_SSP2TX_MD, ++ .clk = (22 | GPIO_ALT_FN_3_IN), ++ .frm = (88 | GPIO_ALT_FN_3_IN), ++ }, ++ { /* SSP2 SND_SOC_DAIFMT_CBS_CFS */ ++ .rx = GPIO11_SSP2RX_MD, ++ .tx = GPIO13_SSP2TX_MD, ++ .clk = (22 | GPIO_ALT_FN_3_OUT), ++ .frm = (88 | GPIO_ALT_FN_3_OUT), ++ }, ++ { /* SSP2 SND_SOC_DAIFMT_CBS_CFM */ ++ .rx = GPIO11_SSP2RX_MD, ++ .tx = GPIO13_SSP2TX_MD, ++ .clk = (22 | GPIO_ALT_FN_3_OUT), ++ .frm = (88 | GPIO_ALT_FN_3_IN), ++ }, ++ { /* SSP2 SND_SOC_DAIFMT_CBM_CFS */ ++ .rx = GPIO11_SSP2RX_MD, ++ .tx = GPIO13_SSP2TX_MD, ++ .clk = (22 | GPIO_ALT_FN_3_IN), ++ .frm = (88 | GPIO_ALT_FN_3_OUT), ++ }}, ++ {{ /* SSP3 SND_SOC_DAIFMT_CBM_CFM */ ++ .rx = GPIO82_SSP3RX_MD, ++ .tx = GPIO81_SSP3TX_MD, ++ .clk = (84 | GPIO_ALT_FN_3_IN), ++ .frm = (83 | GPIO_ALT_FN_3_IN), ++ }, ++ { /* SSP3 SND_SOC_DAIFMT_CBS_CFS */ ++ .rx = GPIO82_SSP3RX_MD, ++ .tx = GPIO81_SSP3TX_MD, ++ .clk = (84 | GPIO_ALT_FN_3_OUT), ++ .frm = (83 | GPIO_ALT_FN_3_OUT), ++ }, ++ { /* SSP3 SND_SOC_DAIFMT_CBS_CFM */ ++ .rx = GPIO82_SSP3RX_MD, ++ .tx = GPIO81_SSP3TX_MD, ++ .clk = (84 | GPIO_ALT_FN_3_OUT), ++ .frm = (83 | GPIO_ALT_FN_3_IN), ++ }, ++ { /* SSP3 SND_SOC_DAIFMT_CBM_CFS */ ++ .rx = GPIO82_SSP3RX_MD, ++ .tx = GPIO81_SSP3TX_MD, ++ .clk = (84 | GPIO_ALT_FN_3_IN), ++ .frm = (83 | GPIO_ALT_FN_3_OUT), ++ }}, ++}; ++ ++static int pxa2xx_ssp_startup(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ int ret = 0; ++ ++ if (!rtd->cpu_dai->active) { ++ ret = ssp_init (&ssp[rtd->cpu_dai->id], rtd->cpu_dai->id + 1, ++ SSP_NO_IRQ); ++ if (ret < 0) ++ return ret; ++ ssp_disable(&ssp[rtd->cpu_dai->id]); ++ } ++ return ret; ++} ++ ++static void pxa2xx_ssp_shutdown(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ ++ if (!rtd->cpu_dai->active) { ++ ssp_disable(&ssp[rtd->cpu_dai->id]); ++ ssp_exit(&ssp[rtd->cpu_dai->id]); ++ } ++} ++ ++#ifdef CONFIG_PM ++ ++#if defined (CONFIG_PXA27x) ++static int cken[3] = {CKEN23_SSP1, CKEN3_SSP2, CKEN4_SSP3}; ++#else ++static int cken[3] = {CKEN3_SSP, CKEN9_NSSP, CKEN10_ASSP}; ++#endif ++ ++static int pxa2xx_ssp_suspend(struct platform_device *pdev, ++ struct snd_soc_cpu_dai *dai) ++{ ++ if (!dai->active) ++ return 0; ++ ++ ssp_save_state(&ssp[dai->id], &ssp_state[dai->id]); ++ pxa_set_cken(cken[dai->id], 0); ++ return 0; ++} ++ ++static int pxa2xx_ssp_resume(struct platform_device *pdev, ++ struct snd_soc_cpu_dai *dai) ++{ ++ if (!dai->active) ++ return 0; ++ ++ pxa_set_cken(cken[dai->id], 1); ++ ssp_restore_state(&ssp[dai->id], &ssp_state[dai->id]); ++ ssp_enable(&ssp[dai->id]); ++ ++ return 0; ++} ++ ++#else ++#define pxa2xx_ssp_suspend NULL ++#define pxa2xx_ssp_resume NULL ++#endif ++ ++/* todo - check clk source and PLL before returning clock rate */ ++static unsigned int pxa_ssp_config_sysclk(struct snd_soc_cpu_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ /* audio clock ? (divide by 1) */ ++ if (clksrc[dai->id] == 3) { ++ switch(info->rate){ ++ case 8000: ++ case 16000: ++ case 32000: ++ case 48000: ++ case 96000: ++ return 12288000; ++ break; ++ case 11025: ++ case 22050: ++ case 44100: ++ case 88200: ++ return 11289600; ++ break; ++ } ++ } ++ ++ /* pll */ ++ return sysclk[dai->id]; ++} ++ ++#ifdef CONFIG_PXA27x ++static u32 pxa27x_set_audio_clk(unsigned int rate, unsigned int fs) ++{ ++ u32 aclk = 0, div = 0; ++ ++ if (rate == 0 || fs == 0) ++ return 0; ++ ++ switch(rate){ ++ case 8000: ++ case 16000: ++ case 32000: ++ case 48000: ++ case 96000: ++ aclk = 0x2 << 4; ++ div = 12288000 / (rate * fs); ++ break; ++ case 11025: ++ case 22050: ++ case 44100: ++ case 88200: ++ aclk = 0x1 << 4; ++ div = 11289600 / (rate * fs); ++ break; ++ } ++ ++ aclk |= ffs(div) - 1; ++ return aclk; ++} ++#endif ++ ++static inline int get_scr(int srate, int id) ++{ ++ if (srate == 0) ++ return 0; ++ return (sysclk[id] / srate) - 1; ++} ++ ++static int pxa2xx_ssp_hw_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ int fmt = 0, dma = 0, fs, chn = params_channels(params); ++ u32 ssp_mode = 0, ssp_setup = 0, psp_mode = 0, rate = 0; ++ ++ fs = rtd->cpu_dai->dai_runtime.fs; ++ ++ /* select correct DMA params */ ++ if (substream->stream != SNDRV_PCM_STREAM_PLAYBACK) ++ dma = 1; ++ if (chn == 2 || rtd->cpu_dai->dai_runtime.pcmfmt != PXA_SSP_BITS) ++ dma += 2; ++ rtd->cpu_dai->dma_data = ssp_dma_params[rtd->cpu_dai->id][dma]; ++ ++ /* is port used by another stream */ ++ if (SSCR0 & SSCR0_SSE) ++ return 0; ++ ++ /* bit size */ ++ switch(rtd->cpu_dai->dai_runtime.pcmfmt) { ++ case SNDRV_PCM_FMTBIT_S16_LE: ++ ssp_mode |=SSCR0_DataSize(16); ++ break; ++ case SNDRV_PCM_FMTBIT_S24_LE: ++ ssp_mode |=(SSCR0_EDSS | SSCR0_DataSize(8)); ++ /* use network mode for stereo samples > 16 bits */ ++ if (chn == 2) { ++ ssp_mode |= (SSCR0_MOD | SSCR0_SlotsPerFrm(2) << 24); ++ /* active slots 0,1 */ ++ SSTSA_P(rtd->cpu_dai->id +1) = 0x3; ++ SSRSA_P(rtd->cpu_dai->id +1) = 0x3; ++ } ++ break; ++ case SNDRV_PCM_FMTBIT_S32_LE: ++ ssp_mode |= (SSCR0_EDSS | SSCR0_DataSize(16)); ++ /* use network mode for stereo samples > 16 bits */ ++ if (chn == 2) { ++ ssp_mode |= (SSCR0_MOD | SSCR0_SlotsPerFrm(2) << 24); ++ /* active slots 0,1 */ ++ SSTSA_P(rtd->cpu_dai->id +1) = 0x3; ++ SSRSA_P(rtd->cpu_dai->id +1) = 0x3; ++ } ++ break; ++ } ++ ++ ssp_mode |= SSCR0_PSP; ++ ssp_setup = SSCR1_RxTresh(14) | SSCR1_TxTresh(1) | ++ SSCR1_TRAIL | SSCR1_RWOT; ++ ++ switch(rtd->cpu_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK) { ++ case SND_SOC_DAIFMT_CBM_CFM: ++ ssp_setup |= (SSCR1_SCLKDIR | SSCR1_SFRMDIR); ++ break; ++ case SND_SOC_DAIFMT_CBM_CFS: ++ ssp_setup |= SSCR1_SCLKDIR; ++ break; ++ case SND_SOC_DAIFMT_CBS_CFM: ++ ssp_setup |= SSCR1_SFRMDIR; ++ break; ++ } ++ ++ switch(rtd->cpu_dai->dai_runtime.fmt) { ++ case SND_SOC_DAIFMT_CBS_CFS: ++ fmt = 1; ++ break; ++ case SND_SOC_DAIFMT_CBS_CFM: ++ fmt = 2; ++ break; ++ case SND_SOC_DAIFMT_CBM_CFS: ++ fmt = 3; ++ break; ++ } ++ ++ pxa_gpio_mode(ssp_gpios[rtd->cpu_dai->id][fmt].rx); ++ pxa_gpio_mode(ssp_gpios[rtd->cpu_dai->id][fmt].tx); ++ pxa_gpio_mode(ssp_gpios[rtd->cpu_dai->id][fmt].frm); ++ pxa_gpio_mode(ssp_gpios[rtd->cpu_dai->id][fmt].clk); ++ ++ switch (rtd->cpu_dai->dai_runtime.fmt & SND_SOC_DAIFMT_INV_MASK) { ++ case SND_SOC_DAIFMT_NB_NF: ++ psp_mode |= SSPSP_SFRMP | SSPSP_FSRT; ++ break; ++ } ++ ++ if (rtd->cpu_dai->dai_runtime.fmt & SND_SOC_DAIFMT_DSP_A) ++ psp_mode |= SSPSP_SCMODE(2); ++ if (rtd->cpu_dai->dai_runtime.fmt & SND_SOC_DAIFMT_DSP_B) ++ psp_mode |= SSPSP_SCMODE(3); ++ ++ switch(clksrc[rtd->cpu_dai->id]) { ++ case 2: /* network clock */ ++ ssp_mode |= SSCR0_NCS | SSCR0_MOD; ++ case 1: /* external clock */ ++ ssp_mode |= SSCR0_ECS; ++ case 0: /* internal clock */ ++ rate = get_scr(snd_soc_get_rate(rtd->cpu_dai->dai_runtime.pcmrate), ++ rtd->cpu_dai->id); ++ break; ++#ifdef CONFIG_PXA27x ++ case 3: /* audio clock */ ++ ssp_mode |= (1 << 30); ++ SSACD_P(rtd->cpu_dai->id) = (0x1 << 3) | ++ pxa27x_set_audio_clk( ++ snd_soc_get_rate(rtd->cpu_dai->dai_runtime.pcmrate), fs); ++ break; ++#endif ++ } ++ ++ ssp_config(&ssp[rtd->cpu_dai->id], ssp_mode, ssp_setup, psp_mode, ++ SSCR0_SerClkDiv(rate)); ++#if 0 ++ printk("SSCR0 %x SSCR1 %x SSTO %x SSPSP %x SSSR %x\n", ++ SSCR0_P(rtd->cpu_dai->id+1), SSCR1_P(rtd->cpu_dai->id+1), ++ SSTO_P(rtd->cpu_dai->id+1), SSPSP_P(rtd->cpu_dai->id+1), ++ SSSR_P(rtd->cpu_dai->id+1)); ++#endif ++ return 0; ++} ++ ++static int pxa2xx_ssp_trigger(struct snd_pcm_substream *substream, int cmd) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ int ret = 0; ++ ++ switch (cmd) { ++ case SNDRV_PCM_TRIGGER_RESUME: ++ ssp_enable(&ssp[rtd->cpu_dai->id]); ++ break; ++ case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ SSCR1_P(rtd->cpu_dai->id+1) |= SSCR1_TSRE; ++ else ++ SSCR1_P(rtd->cpu_dai->id+1) |= SSCR1_RSRE; ++ SSSR_P(rtd->cpu_dai->id+1) |= SSSR_P(rtd->cpu_dai->id+1); ++ break; ++ case SNDRV_PCM_TRIGGER_START: ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ SSCR1_P(rtd->cpu_dai->id+1) |= SSCR1_TSRE; ++ else ++ SSCR1_P(rtd->cpu_dai->id+1) |= SSCR1_RSRE; ++ ssp_enable(&ssp[rtd->cpu_dai->id]); ++ break; ++ case SNDRV_PCM_TRIGGER_STOP: ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ SSCR1_P(rtd->cpu_dai->id+1) &= ~SSCR1_TSRE; ++ else ++ SSCR1_P(rtd->cpu_dai->id+1) &= ~SSCR1_RSRE; ++ break; ++ case SNDRV_PCM_TRIGGER_SUSPEND: ++ ssp_disable(&ssp[rtd->cpu_dai->id]); ++ break; ++ case SNDRV_PCM_TRIGGER_PAUSE_PUSH: ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ SSCR1_P(rtd->cpu_dai->id+1) &= ~SSCR1_TSRE; ++ else ++ SSCR1_P(rtd->cpu_dai->id+1) &= ~SSCR1_RSRE; ++ break; ++ ++ default: ++ ret = -EINVAL; ++ } ++#if 0 ++ printk("SSCR0 %x SSCR1 %x SSTO %x SSPSP %x SSSR %x\n", ++ SSCR0_P(rtd->cpu_dai->id+1), SSCR1_P(rtd->cpu_dai->id+1), ++ SSTO_P(rtd->cpu_dai->id+1), SSPSP_P(rtd->cpu_dai->id+1), ++ SSSR_P(rtd->cpu_dai->id+1)); ++#endif ++ return ret; ++} ++ ++struct snd_soc_cpu_dai pxa_ssp_dai[] = { ++ { .name = "pxa2xx-ssp1", ++ .id = 0, ++ .type = SND_SOC_DAI_PCM, ++ .suspend = pxa2xx_ssp_suspend, ++ .resume = pxa2xx_ssp_resume, ++ .config_sysclk = pxa_ssp_config_sysclk, ++ .playback = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .capture = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .ops = { ++ .startup = pxa2xx_ssp_startup, ++ .shutdown = pxa2xx_ssp_shutdown, ++ .trigger = pxa2xx_ssp_trigger, ++ .hw_params = pxa2xx_ssp_hw_params,}, ++ .caps = { ++ .mode = pxa2xx_ssp_modes, ++ .num_modes = ARRAY_SIZE(pxa2xx_ssp_modes),}, ++ }, ++ { .name = "pxa2xx-ssp2", ++ .id = 1, ++ .type = SND_SOC_DAI_PCM, ++ .suspend = pxa2xx_ssp_suspend, ++ .resume = pxa2xx_ssp_resume, ++ .config_sysclk = pxa_ssp_config_sysclk, ++ .playback = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .capture = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .ops = { ++ .startup = pxa2xx_ssp_startup, ++ .shutdown = pxa2xx_ssp_shutdown, ++ .trigger = pxa2xx_ssp_trigger, ++ .hw_params = pxa2xx_ssp_hw_params,}, ++ .caps = { ++ .mode = pxa2xx_ssp_modes, ++ .num_modes = ARRAY_SIZE(pxa2xx_ssp_modes),}, ++ }, ++ { .name = "pxa2xx-ssp3", ++ .id = 2, ++ .type = SND_SOC_DAI_PCM, ++ .suspend = pxa2xx_ssp_suspend, ++ .resume = pxa2xx_ssp_resume, ++ .config_sysclk = pxa_ssp_config_sysclk, ++ .playback = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .capture = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .ops = { ++ .startup = pxa2xx_ssp_startup, ++ .shutdown = pxa2xx_ssp_shutdown, ++ .trigger = pxa2xx_ssp_trigger, ++ .hw_params = pxa2xx_ssp_hw_params,}, ++ .caps = { ++ .mode = pxa2xx_ssp_modes, ++ .num_modes = ARRAY_SIZE(pxa2xx_ssp_modes),}, ++ }, ++}; ++ ++EXPORT_SYMBOL_GPL(pxa_ssp_dai); ++ ++/* Module information */ ++MODULE_AUTHOR("Liam Girdwood, liam.girdwood@wolfsonmicro.com, www.wolfsonmicro.com"); ++MODULE_DESCRIPTION("pxa2xx SSP/PCM SoC Interface"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/pxa/spitz.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/pxa/spitz.c +@@ -0,0 +1,374 @@ ++/* ++ * spitz.c -- SoC audio for Sharp SL-Cxx00 models Spitz, Borzoi and Akita ++ * ++ * Copyright 2005 Wolfson Microelectronics PLC. ++ * Copyright 2005 Openedhand Ltd. ++ * ++ * Authors: Liam Girdwood <liam.girdwood@wolfsonmicro.com> ++ * Richard Purdie <richard@openedhand.com> ++ * ++ * 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. ++ * ++ * Revision history ++ * 30th Nov 2005 Initial version. ++ * ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/timer.h> ++#include <linux/interrupt.h> ++#include <linux/platform_device.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++ ++#include <asm/mach-types.h> ++#include <asm/hardware/scoop.h> ++#include <asm/arch/pxa-regs.h> ++#include <asm/arch/hardware.h> ++#include <asm/arch/akita.h> ++#include <asm/arch/spitz.h> ++#include <asm/mach-types.h> ++#include "../codecs/wm8750.h" ++#include "pxa2xx-pcm.h" ++ ++#define SPITZ_HP 0 ++#define SPITZ_MIC 1 ++#define SPITZ_LINE 2 ++#define SPITZ_HEADSET 3 ++#define SPITZ_HP_OFF 4 ++#define SPITZ_SPK_ON 0 ++#define SPITZ_SPK_OFF 1 ++ ++ /* audio clock in Hz - rounded from 12.235MHz */ ++#define SPITZ_AUDIO_CLOCK 12288000 ++ ++static int spitz_jack_func; ++static int spitz_spk_func; ++ ++static void spitz_ext_control(struct snd_soc_codec *codec) ++{ ++ if (spitz_spk_func == SPITZ_SPK_ON) ++ snd_soc_dapm_set_endpoint(codec, "Ext Spk", 1); ++ else ++ snd_soc_dapm_set_endpoint(codec, "Ext Spk", 0); ++ ++ /* set up jack connection */ ++ switch (spitz_jack_func) { ++ case SPITZ_HP: ++ /* enable and unmute hp jack, disable mic bias */ ++ snd_soc_dapm_set_endpoint(codec, "Headset Jack", 0); ++ snd_soc_dapm_set_endpoint(codec, "Mic Jack", 0); ++ snd_soc_dapm_set_endpoint(codec, "Line Jack", 0); ++ snd_soc_dapm_set_endpoint(codec, "Headphone Jack", 1); ++ set_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_L); ++ set_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_R); ++ break; ++ case SPITZ_MIC: ++ /* enable mic jack and bias, mute hp */ ++ snd_soc_dapm_set_endpoint(codec, "Headphone Jack", 0); ++ snd_soc_dapm_set_endpoint(codec, "Headset Jack", 0); ++ snd_soc_dapm_set_endpoint(codec, "Line Jack", 0); ++ snd_soc_dapm_set_endpoint(codec, "Mic Jack", 1); ++ reset_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_L); ++ reset_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_R); ++ break; ++ case SPITZ_LINE: ++ /* enable line jack, disable mic bias and mute hp */ ++ snd_soc_dapm_set_endpoint(codec, "Headphone Jack", 0); ++ snd_soc_dapm_set_endpoint(codec, "Headset Jack", 0); ++ snd_soc_dapm_set_endpoint(codec, "Mic Jack", 0); ++ snd_soc_dapm_set_endpoint(codec, "Line Jack", 1); ++ reset_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_L); ++ reset_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_R); ++ break; ++ case SPITZ_HEADSET: ++ /* enable and unmute headset jack enable mic bias, mute L hp */ ++ snd_soc_dapm_set_endpoint(codec, "Headphone Jack", 0); ++ snd_soc_dapm_set_endpoint(codec, "Mic Jack", 1); ++ snd_soc_dapm_set_endpoint(codec, "Line Jack", 0); ++ snd_soc_dapm_set_endpoint(codec, "Headset Jack", 1); ++ reset_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_L); ++ set_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_R); ++ break; ++ case SPITZ_HP_OFF: ++ ++ /* jack removed, everything off */ ++ snd_soc_dapm_set_endpoint(codec, "Headphone Jack", 0); ++ snd_soc_dapm_set_endpoint(codec, "Headset Jack", 0); ++ snd_soc_dapm_set_endpoint(codec, "Mic Jack", 0); ++ snd_soc_dapm_set_endpoint(codec, "Line Jack", 0); ++ reset_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_L); ++ reset_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_R); ++ break; ++ } ++ snd_soc_dapm_sync_endpoints(codec); ++} ++ ++static int spitz_startup(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_codec *codec = rtd->socdev->codec; ++ ++ /* check the jack status at stream startup */ ++ spitz_ext_control(codec); ++ return 0; ++} ++ ++static struct snd_soc_ops spitz_ops = { ++ .startup = spitz_startup, ++}; ++ ++static int spitz_get_jack(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ ucontrol->value.integer.value[0] = spitz_jack_func; ++ return 0; ++} ++ ++static int spitz_set_jack(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ ++ if (spitz_jack_func == ucontrol->value.integer.value[0]) ++ return 0; ++ ++ spitz_jack_func = ucontrol->value.integer.value[0]; ++ spitz_ext_control(codec); ++ return 1; ++} ++ ++static int spitz_get_spk(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ ucontrol->value.integer.value[0] = spitz_spk_func; ++ return 0; ++} ++ ++static int spitz_set_spk(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ ++ if (spitz_spk_func == ucontrol->value.integer.value[0]) ++ return 0; ++ ++ spitz_spk_func = ucontrol->value.integer.value[0]; ++ spitz_ext_control(codec); ++ return 1; ++} ++ ++static int spitz_mic_bias(struct snd_soc_dapm_widget *w, int event) ++{ ++ if (machine_is_borzoi() || machine_is_spitz()) { ++ if (SND_SOC_DAPM_EVENT_ON(event)) ++ set_scoop_gpio(&spitzscoop2_device.dev, ++ SPITZ_SCP2_MIC_BIAS); ++ else ++ reset_scoop_gpio(&spitzscoop2_device.dev, ++ SPITZ_SCP2_MIC_BIAS); ++ } ++ ++ if (machine_is_akita()) { ++ if (SND_SOC_DAPM_EVENT_ON(event)) ++ akita_set_ioexp(&akitaioexp_device.dev, ++ AKITA_IOEXP_MIC_BIAS); ++ else ++ akita_reset_ioexp(&akitaioexp_device.dev, ++ AKITA_IOEXP_MIC_BIAS); ++ } ++ return 0; ++} ++ ++/* spitz machine dapm widgets */ ++static const struct snd_soc_dapm_widget wm8750_dapm_widgets[] = { ++ SND_SOC_DAPM_HP("Headphone Jack", NULL), ++ SND_SOC_DAPM_MIC("Mic Jack", spitz_mic_bias), ++ SND_SOC_DAPM_SPK("Ext Spk", NULL), ++ SND_SOC_DAPM_LINE("Line Jack", NULL), ++ ++ /* headset is a mic and mono headphone */ ++ SND_SOC_DAPM_HP("Headset Jack", NULL), ++}; ++ ++/* Spitz machine audio_map */ ++static const char *audio_map[][3] = { ++ ++ /* headphone connected to LOUT1, ROUT1 */ ++ {"Headphone Jack", NULL, "LOUT1"}, ++ {"Headphone Jack", NULL, "ROUT1"}, ++ ++ /* headset connected to ROUT1 and LINPUT1 with bias (def below) */ ++ {"Headset Jack", NULL, "ROUT1"}, ++ ++ /* ext speaker connected to LOUT2, ROUT2 */ ++ {"Ext Spk", NULL , "ROUT2"}, ++ {"Ext Spk", NULL , "LOUT2"}, ++ ++ /* mic is connected to input 1 - with bias */ ++ {"LINPUT1", NULL, "Mic Bias"}, ++ {"Mic Bias", NULL, "Mic Jack"}, ++ ++ /* line is connected to input 1 - no bias */ ++ {"LINPUT1", NULL, "Line Jack"}, ++ ++ {NULL, NULL, NULL}, ++}; ++ ++static const char *jack_function[] = {"Headphone", "Mic", "Line", "Headset", ++ "Off"}; ++static const char *spk_function[] = {"On", "Off"}; ++static const struct soc_enum spitz_enum[] = { ++ SOC_ENUM_SINGLE_EXT(5, jack_function), ++ SOC_ENUM_SINGLE_EXT(2, spk_function), ++}; ++ ++static const struct snd_kcontrol_new wm8750_spitz_controls[] = { ++ SOC_ENUM_EXT("Jack Function", spitz_enum[0], spitz_get_jack, ++ spitz_set_jack), ++ SOC_ENUM_EXT("Speaker Function", spitz_enum[1], spitz_get_spk, ++ spitz_set_spk), ++}; ++ ++/* ++ * Logic for a wm8750 as connected on a Sharp SL-Cxx00 Device ++ */ ++static int spitz_wm8750_init(struct snd_soc_codec *codec) ++{ ++ int i, err; ++ ++ /* NC codec pins */ ++ snd_soc_dapm_set_endpoint(codec, "RINPUT1", 0); ++ snd_soc_dapm_set_endpoint(codec, "LINPUT2", 0); ++ snd_soc_dapm_set_endpoint(codec, "RINPUT2", 0); ++ snd_soc_dapm_set_endpoint(codec, "LINPUT3", 0); ++ snd_soc_dapm_set_endpoint(codec, "RINPUT3", 0); ++ snd_soc_dapm_set_endpoint(codec, "OUT3", 0); ++ snd_soc_dapm_set_endpoint(codec, "MONO", 0); ++ ++ /* Add spitz specific controls */ ++ for (i = 0; i < ARRAY_SIZE(wm8750_spitz_controls); i++) { ++ err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&wm8750_spitz_controls[i], codec, NULL)); ++ if (err < 0) ++ return err; ++ } ++ ++ /* Add spitz specific widgets */ ++ for (i = 0; i < ARRAY_SIZE(wm8750_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &wm8750_dapm_widgets[i]); ++ } ++ ++ /* Set up spitz specific audio path audio_map */ ++ for (i = 0; audio_map[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, audio_map[i][0], ++ audio_map[i][1], audio_map[i][2]); ++ } ++ ++ snd_soc_dapm_sync_endpoints(codec); ++ return 0; ++} ++ ++static unsigned int spitz_config_sysclk(struct snd_soc_pcm_runtime *rtd, ++ struct snd_soc_clock_info *info) ++{ ++ if (info->bclk_master & SND_SOC_DAIFMT_CBS_CFS) { ++ /* pxa2xx is i2s master */ ++ switch (info->rate) { ++ case 11025: ++ case 22050: ++ case 44100: ++ case 88200: ++ /* configure codec digital filters ++ * for 11.025, 22.05, 44.1, 88.2 */ ++ rtd->codec_dai->config_sysclk(rtd->codec_dai, info, ++ 11289600); ++ break; ++ default: ++ /* configure codec digital filters for all other rates */ ++ rtd->codec_dai->config_sysclk(rtd->codec_dai, info, ++ SPITZ_AUDIO_CLOCK); ++ break; ++ } ++ /* configure pxa2xx i2s interface clocks as master */ ++ return rtd->cpu_dai->config_sysclk(rtd->cpu_dai, info, ++ SPITZ_AUDIO_CLOCK); ++ } else { ++ /* codec is i2s master - only configure codec DAI clock */ ++ return rtd->codec_dai->config_sysclk(rtd->codec_dai, info, ++ SPITZ_AUDIO_CLOCK); ++ } ++} ++ ++/* spitz digital audio interface glue - connects codec <--> CPU */ ++static struct snd_soc_dai_link spitz_dai = { ++ .name = "wm8750", ++ .stream_name = "WM8750", ++ .cpu_dai = &pxa_i2s_dai, ++ .codec_dai = &wm8750_dai, ++ .init = spitz_wm8750_init, ++ .config_sysclk = spitz_config_sysclk, ++}; ++ ++/* spitz audio machine driver */ ++static struct snd_soc_machine snd_soc_machine_spitz = { ++ .name = "Spitz", ++ .dai_link = &spitz_dai, ++ .num_links = 1, ++ .ops = &spitz_ops, ++}; ++ ++/* spitz audio private data */ ++static struct wm8750_setup_data spitz_wm8750_setup = { ++ .i2c_address = 0x1b, ++}; ++ ++/* spitz audio subsystem */ ++static struct snd_soc_device spitz_snd_devdata = { ++ .machine = &snd_soc_machine_spitz, ++ .platform = &pxa2xx_soc_platform, ++ .codec_dev = &soc_codec_dev_wm8750, ++ .codec_data = &spitz_wm8750_setup, ++}; ++ ++static struct platform_device *spitz_snd_device; ++ ++static int __init spitz_init(void) ++{ ++ int ret; ++ ++ if (!(machine_is_spitz() || machine_is_borzoi() || machine_is_akita())) ++ return -ENODEV; ++ ++ spitz_snd_device = platform_device_alloc("soc-audio", -1); ++ if (!spitz_snd_device) ++ return -ENOMEM; ++ ++ platform_set_drvdata(spitz_snd_device, &spitz_snd_devdata); ++ spitz_snd_devdata.dev = &spitz_snd_device->dev; ++ ret = platform_device_add(spitz_snd_device); ++ ++ if (ret) ++ platform_device_put(spitz_snd_device); ++ ++ return ret; ++} ++ ++static void __exit spitz_exit(void) ++{ ++ platform_device_unregister(spitz_snd_device); ++} ++ ++module_init(spitz_init); ++module_exit(spitz_exit); ++ ++MODULE_AUTHOR("Richard Purdie"); ++MODULE_DESCRIPTION("ALSA SoC Spitz"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/pxa/tosa.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/pxa/tosa.c +@@ -0,0 +1,287 @@ ++/* ++ * tosa.c -- SoC audio for Tosa ++ * ++ * Copyright 2005 Wolfson Microelectronics PLC. ++ * Copyright 2005 Openedhand Ltd. ++ * ++ * Authors: Liam Girdwood <liam.girdwood@wolfsonmicro.com> ++ * Richard Purdie <richard@openedhand.com> ++ * ++ * 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. ++ * ++ * Revision history ++ * 30th Nov 2005 Initial version. ++ * ++ * GPIO's ++ * 1 - Jack Insertion ++ * 5 - Hookswitch (headset answer/hang up switch) ++ * ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/device.h> ++ ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++ ++#include <asm/mach-types.h> ++#include <asm/hardware/tmio.h> ++#include <asm/arch/pxa-regs.h> ++#include <asm/arch/hardware.h> ++#include <asm/arch/audio.h> ++#include <asm/arch/tosa.h> ++ ++#include "../codecs/wm9712.h" ++#include "pxa2xx-pcm.h" ++ ++static struct snd_soc_machine tosa; ++ ++#define TOSA_HP 0 ++#define TOSA_MIC_INT 1 ++#define TOSA_HEADSET 2 ++#define TOSA_HP_OFF 3 ++#define TOSA_SPK_ON 0 ++#define TOSA_SPK_OFF 1 ++ ++static int tosa_jack_func; ++static int tosa_spk_func; ++ ++static void tosa_ext_control(struct snd_soc_codec *codec) ++{ ++ int spk = 0, mic_int = 0, hp = 0, hs = 0; ++ ++ /* set up jack connection */ ++ switch (tosa_jack_func) { ++ case TOSA_HP: ++ hp = 1; ++ break; ++ case TOSA_MIC_INT: ++ mic_int = 1; ++ break; ++ case TOSA_HEADSET: ++ hs = 1; ++ break; ++ } ++ ++ if (tosa_spk_func == TOSA_SPK_ON) ++ spk = 1; ++ ++ snd_soc_dapm_set_endpoint(codec, "Speaker", spk); ++ snd_soc_dapm_set_endpoint(codec, "Mic (Internal)", mic_int); ++ snd_soc_dapm_set_endpoint(codec, "Headphone Jack", hp); ++ snd_soc_dapm_set_endpoint(codec, "Headset Jack", hs); ++ snd_soc_dapm_sync_endpoints(codec); ++} ++ ++static int tosa_startup(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_codec *codec = rtd->socdev->codec; ++ ++ /* check the jack status at stream startup */ ++ tosa_ext_control(codec); ++ return 0; ++} ++ ++static struct snd_soc_ops tosa_ops = { ++ .startup = tosa_startup, ++}; ++ ++static int tosa_get_jack(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ ucontrol->value.integer.value[0] = tosa_jack_func; ++ return 0; ++} ++ ++static int tosa_set_jack(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ ++ if (tosa_jack_func == ucontrol->value.integer.value[0]) ++ return 0; ++ ++ tosa_jack_func = ucontrol->value.integer.value[0]; ++ tosa_ext_control(codec); ++ return 1; ++} ++ ++static int tosa_get_spk(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ ucontrol->value.integer.value[0] = tosa_spk_func; ++ return 0; ++} ++ ++static int tosa_set_spk(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ ++ if (tosa_spk_func == ucontrol->value.integer.value[0]) ++ return 0; ++ ++ tosa_spk_func = ucontrol->value.integer.value[0]; ++ tosa_ext_control(codec); ++ return 1; ++} ++ ++/* tosa dapm event handlers */ ++static int tosa_hp_event(struct snd_soc_dapm_widget *w, int event) ++{ ++ if (SND_SOC_DAPM_EVENT_ON(event)) ++ set_tc6393_gpio(&tc6393_device.dev,TOSA_TC6393_L_MUTE); ++ else ++ reset_tc6393_gpio(&tc6393_device.dev,TOSA_TC6393_L_MUTE); ++ return 0; ++} ++ ++/* tosa machine dapm widgets */ ++static const struct snd_soc_dapm_widget tosa_dapm_widgets[] = { ++SND_SOC_DAPM_HP("Headphone Jack", tosa_hp_event), ++SND_SOC_DAPM_HP("Headset Jack", NULL), ++SND_SOC_DAPM_MIC("Mic (Internal)", NULL), ++SND_SOC_DAPM_SPK("Speaker", NULL), ++}; ++ ++/* tosa audio map */ ++static const char *audio_map[][3] = { ++ ++ /* headphone connected to HPOUTL, HPOUTR */ ++ {"Headphone Jack", NULL, "HPOUTL"}, ++ {"Headphone Jack", NULL, "HPOUTR"}, ++ ++ /* ext speaker connected to LOUT2, ROUT2 */ ++ {"Speaker", NULL, "LOUT2"}, ++ {"Speaker", NULL, "ROUT2"}, ++ ++ /* internal mic is connected to mic1, mic2 differential - with bias */ ++ {"MIC1", NULL, "Mic Bias"}, ++ {"MIC2", NULL, "Mic Bias"}, ++ {"Mic Bias", NULL, "Mic (Internal)"}, ++ ++ /* headset is connected to HPOUTR, and LINEINR with bias */ ++ {"Headset Jack", NULL, "HPOUTR"}, ++ {"LINEINR", NULL, "Mic Bias"}, ++ {"Mic Bias", NULL, "Headset Jack"}, ++ ++ {NULL, NULL, NULL}, ++}; ++ ++static const char *jack_function[] = {"Headphone", "Mic", "Line", "Headset", ++ "Off"}; ++static const char *spk_function[] = {"On", "Off"}; ++static const struct soc_enum tosa_enum[] = { ++ SOC_ENUM_SINGLE_EXT(5, jack_function), ++ SOC_ENUM_SINGLE_EXT(2, spk_function), ++}; ++ ++static const struct snd_kcontrol_new tosa_controls[] = { ++ SOC_ENUM_EXT("Jack Function", tosa_enum[0], tosa_get_jack, ++ tosa_set_jack), ++ SOC_ENUM_EXT("Speaker Function", tosa_enum[1], tosa_get_spk, ++ tosa_set_spk), ++}; ++ ++static int tosa_ac97_init(struct snd_soc_codec *codec) ++{ ++ int i, err; ++ ++ snd_soc_dapm_set_endpoint(codec, "OUT3", 0); ++ snd_soc_dapm_set_endpoint(codec, "MONOOUT", 0); ++ ++ /* add tosa specific controls */ ++ for (i = 0; i < ARRAY_SIZE(tosa_controls); i++) { ++ err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&tosa_controls[i],codec, NULL)); ++ if (err < 0) ++ return err; ++ } ++ ++ /* add tosa specific widgets */ ++ for (i = 0; i < ARRAY_SIZE(tosa_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &tosa_dapm_widgets[i]); ++ } ++ ++ /* set up tosa specific audio path audio_map */ ++ for (i = 0; audio_map[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, audio_map[i][0], ++ audio_map[i][1], audio_map[i][2]); ++ } ++ ++ snd_soc_dapm_sync_endpoints(codec); ++ return 0; ++} ++ ++static struct snd_soc_dai_link tosa_dai[] = { ++{ ++ .name = "AC97", ++ .stream_name = "AC97 HiFi", ++ .cpu_dai = &pxa_ac97_dai[PXA2XX_DAI_AC97_HIFI], ++ .codec_dai = &wm9712_dai[WM9712_DAI_AC97_HIFI], ++ .init = tosa_ac97_init, ++}, ++{ ++ .name = "AC97 Aux", ++ .stream_name = "AC97 Aux", ++ .cpu_dai = &pxa_ac97_dai[PXA2XX_DAI_AC97_AUX], ++ .codec_dai = &wm9712_dai[WM9712_DAI_AC97_AUX], ++}, ++}; ++ ++static struct snd_soc_machine tosa = { ++ .name = "Tosa", ++ .dai_link = tosa_dai, ++ .num_links = ARRAY_SIZE(tosa_dai), ++ .ops = &tosa_ops, ++}; ++ ++static struct snd_soc_device tosa_snd_devdata = { ++ .machine = &tosa, ++ .platform = &pxa2xx_soc_platform, ++ .codec_dev = &soc_codec_dev_wm9712, ++}; ++ ++static struct platform_device *tosa_snd_device; ++ ++static int __init tosa_init(void) ++{ ++ int ret; ++ ++ if (!machine_is_tosa()) ++ return -ENODEV; ++ ++ tosa_snd_device = platform_device_alloc("soc-audio", -1); ++ if (!tosa_snd_device) ++ return -ENOMEM; ++ ++ platform_set_drvdata(tosa_snd_device, &tosa_snd_devdata); ++ tosa_snd_devdata.dev = &tosa_snd_device->dev; ++ ret = platform_device_add(tosa_snd_device); ++ ++ if (ret) ++ platform_device_put(tosa_snd_device); ++ ++ return ret; ++} ++ ++static void __exit tosa_exit(void) ++{ ++ platform_device_unregister(tosa_snd_device); ++} ++ ++module_init(tosa_init); ++module_exit(tosa_exit); ++ ++/* Module information */ ++MODULE_AUTHOR("Richard Purdie"); ++MODULE_DESCRIPTION("ALSA SoC Tosa"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/soc-dapm.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/soc-dapm.c +@@ -0,0 +1,1327 @@ ++/* ++ * soc-dapm.c -- ALSA SoC Dynamic Audio Power Management ++ * ++ * Copyright 2005 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * 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. ++ * ++ * Revision history ++ * 12th Aug 2005 Initial version. ++ * 25th Oct 2005 Implemented path power domain. ++ * 18th Dec 2005 Implemented machine and stream level power domain. ++ * ++ * Features: ++ * o Changes power status of internal codec blocks depending on the ++ * dynamic configuration of codec internal audio paths and active ++ * DAC's/ADC's. ++ * o Platform power domain - can support external components i.e. amps and ++ * mic/meadphone insertion events. ++ * o Automatic Mic Bias support ++ * o Jack insertion power event initiation - e.g. hp insertion will enable ++ * sinks, dacs, etc ++ * o Delayed powerdown of audio susbsytem to reduce pops between a quick ++ * device reopen. ++ * ++ * Todo: ++ * o DAPM power change sequencing - allow for configurable per ++ * codec sequences. ++ * o Support for analogue bias optimisation. ++ * o Support for reduced codec oversampling rates. ++ * o Support for reduced codec bias currents. ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/init.h> ++#include <linux/delay.h> ++#include <linux/pm.h> ++#include <linux/bitops.h> ++#include <linux/platform_device.h> ++#include <linux/jiffies.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/pcm_params.h> ++#include <sound/soc-dapm.h> ++#include <sound/initval.h> ++ ++/* debug */ ++#define DAPM_DEBUG 0 ++#if DAPM_DEBUG ++#define dump_dapm(codec, action) dbg_dump_dapm(codec, action) ++#define dbg(format, arg...) printk(format, ## arg) ++#else ++#define dump_dapm(codec, action) ++#define dbg(format, arg...) ++#endif ++ ++#define POP_DEBUG 0 ++#if POP_DEBUG ++#define POP_TIME 500 /* 500 msecs - change if pop debug is too fast */ ++#define pop_wait(time) schedule_timeout_interruptible(msecs_to_jiffies(time)) ++#define pop_dbg(format, arg...) printk(format, ## arg); pop_wait(POP_TIME) ++#else ++#define pop_dbg(format, arg...) ++#define pop_wait(time) ++#endif ++ ++/* dapm power sequences - make this per codec in the future */ ++static int dapm_up_seq[] = { ++ snd_soc_dapm_pre, snd_soc_dapm_micbias, snd_soc_dapm_mic, ++ snd_soc_dapm_mux, snd_soc_dapm_dac, snd_soc_dapm_mixer, snd_soc_dapm_pga, ++ snd_soc_dapm_adc, snd_soc_dapm_hp, snd_soc_dapm_spk, snd_soc_dapm_post ++}; ++static int dapm_down_seq[] = { ++ snd_soc_dapm_pre, snd_soc_dapm_adc, snd_soc_dapm_hp, snd_soc_dapm_spk, ++ snd_soc_dapm_pga, snd_soc_dapm_mixer, snd_soc_dapm_dac, snd_soc_dapm_mic, ++ snd_soc_dapm_micbias, snd_soc_dapm_mux, snd_soc_dapm_post ++}; ++ ++static int dapm_status = 1; ++module_param(dapm_status, int, 0); ++MODULE_PARM_DESC(dapm_status, "enable DPM sysfs entries"); ++ ++/* create a new dapm widget */ ++static struct snd_soc_dapm_widget *dapm_cnew_widget( ++ const struct snd_soc_dapm_widget *_widget) ++{ ++ struct snd_soc_dapm_widget* widget; ++ widget = kmalloc(sizeof(struct snd_soc_dapm_widget), GFP_KERNEL); ++ if (!widget) ++ return NULL; ++ ++ memcpy(widget, _widget, sizeof(struct snd_soc_dapm_widget)); ++ return widget; ++} ++ ++/* set up initial codec paths */ ++static void dapm_set_path_status(struct snd_soc_dapm_widget *w, ++ struct snd_soc_dapm_path *p, int i) ++{ ++ switch (w->id) { ++ case snd_soc_dapm_switch: ++ case snd_soc_dapm_mixer: { ++ int val; ++ int reg = w->kcontrols[i].private_value & 0xff; ++ int shift = (w->kcontrols[i].private_value >> 8) & 0x0f; ++ int mask = (w->kcontrols[i].private_value >> 16) & 0xff; ++ int invert = (w->kcontrols[i].private_value >> 24) & 0x01; ++ ++ val = snd_soc_read(w->codec, reg); ++ val = (val >> shift) & mask; ++ ++ if ((invert && !val) || (!invert && val)) ++ p->connect = 1; ++ else ++ p->connect = 0; ++ } ++ break; ++ case snd_soc_dapm_mux: { ++ struct soc_enum *e = (struct soc_enum *)w->kcontrols[i].private_value; ++ int val, item, bitmask; ++ ++ for (bitmask = 1; bitmask < e->mask; bitmask <<= 1) ++ ; ++ val = snd_soc_read(w->codec, e->reg); ++ item = (val >> e->shift_l) & (bitmask - 1); ++ ++ p->connect = 0; ++ for (i = 0; i < e->mask; i++) { ++ if (!(strcmp(p->name, e->texts[i])) && item == i) ++ p->connect = 1; ++ } ++ } ++ break; ++ /* does not effect routing - always connected */ ++ case snd_soc_dapm_pga: ++ case snd_soc_dapm_output: ++ case snd_soc_dapm_adc: ++ case snd_soc_dapm_input: ++ case snd_soc_dapm_dac: ++ case snd_soc_dapm_micbias: ++ case snd_soc_dapm_vmid: ++ p->connect = 1; ++ break; ++ /* does effect routing - dynamically connected */ ++ case snd_soc_dapm_hp: ++ case snd_soc_dapm_mic: ++ case snd_soc_dapm_spk: ++ case snd_soc_dapm_line: ++ case snd_soc_dapm_pre: ++ case snd_soc_dapm_post: ++ p->connect = 0; ++ break; ++ } ++} ++ ++/* connect mux widget to it's interconnecting audio paths */ ++static int dapm_connect_mux(struct snd_soc_codec *codec, ++ struct snd_soc_dapm_widget *src, struct snd_soc_dapm_widget *dest, ++ struct snd_soc_dapm_path *path, const char *control_name, ++ const struct snd_kcontrol_new *kcontrol) ++{ ++ struct soc_enum *e = (struct soc_enum *)kcontrol->private_value; ++ int i; ++ ++ for (i = 0; i < e->mask; i++) { ++ if (!(strcmp(control_name, e->texts[i]))) { ++ list_add(&path->list, &codec->dapm_paths); ++ list_add(&path->list_sink, &dest->sources); ++ list_add(&path->list_source, &src->sinks); ++ path->name = (char*)e->texts[i]; ++ dapm_set_path_status(dest, path, 0); ++ return 0; ++ } ++ } ++ ++ return -ENODEV; ++} ++ ++/* connect mixer widget to it's interconnecting audio paths */ ++static int dapm_connect_mixer(struct snd_soc_codec *codec, ++ struct snd_soc_dapm_widget *src, struct snd_soc_dapm_widget *dest, ++ struct snd_soc_dapm_path *path, const char *control_name) ++{ ++ int i; ++ ++ /* search for mixer kcontrol */ ++ for (i = 0; i < dest->num_kcontrols; i++) { ++ if (!strcmp(control_name, dest->kcontrols[i].name)) { ++ list_add(&path->list, &codec->dapm_paths); ++ list_add(&path->list_sink, &dest->sources); ++ list_add(&path->list_source, &src->sinks); ++ path->name = dest->kcontrols[i].name; ++ dapm_set_path_status(dest, path, i); ++ return 0; ++ } ++ } ++ return -ENODEV; ++} ++ ++/* update dapm codec register bits */ ++static int dapm_update_bits(struct snd_soc_dapm_widget *widget) ++{ ++ int change, power; ++ unsigned short old, new; ++ struct snd_soc_codec *codec = widget->codec; ++ ++ /* check for valid widgets */ ++ if (widget->reg < 0 || widget->id == snd_soc_dapm_input || ++ widget->id == snd_soc_dapm_output || ++ widget->id == snd_soc_dapm_hp || ++ widget->id == snd_soc_dapm_mic || ++ widget->id == snd_soc_dapm_line || ++ widget->id == snd_soc_dapm_spk) ++ return 0; ++ ++ power = widget->power; ++ if (widget->invert) ++ power = (power ? 0:1); ++ ++ old = snd_soc_read(codec, widget->reg); ++ new = (old & ~(0x1 << widget->shift)) | (power << widget->shift); ++ ++ change = old != new; ++ if (change) { ++ pop_dbg("pop test %s : %s in %d ms\n", widget->name, ++ widget->power ? "on" : "off", POP_TIME); ++ snd_soc_write(codec, widget->reg, new); ++ pop_wait(POP_TIME); ++ } ++ dbg("reg old %x new %x change %d\n", old, new, change); ++ return change; ++} ++ ++/* ramps the volume up or down to minimise pops before or after a ++ * DAPM power event */ ++static int dapm_set_pga(struct snd_soc_dapm_widget *widget, int power) ++{ ++ const struct snd_kcontrol_new *k = widget->kcontrols; ++ ++ if (widget->muted && !power) ++ return 0; ++ if (!widget->muted && power) ++ return 0; ++ ++ if (widget->num_kcontrols && k) { ++ int reg = k->private_value & 0xff; ++ int shift = (k->private_value >> 8) & 0x0f; ++ int mask = (k->private_value >> 16) & 0xff; ++ int invert = (k->private_value >> 24) & 0x01; ++ ++ if (power) { ++ int i; ++ /* power up has happended, increase volume to last level */ ++ if (invert) { ++ for (i = mask; i > widget->saved_value; i--) ++ snd_soc_update_bits(widget->codec, reg, mask, i); ++ } else { ++ for (i = 0; i < widget->saved_value; i++) ++ snd_soc_update_bits(widget->codec, reg, mask, i); ++ } ++ widget->muted = 0; ++ } else { ++ /* power down is about to occur, decrease volume to mute */ ++ int val = snd_soc_read(widget->codec, reg); ++ int i = widget->saved_value = (val >> shift) & mask; ++ if (invert) { ++ for (; i < mask; i++) ++ snd_soc_update_bits(widget->codec, reg, mask, i); ++ } else { ++ for (; i > 0; i--) ++ snd_soc_update_bits(widget->codec, reg, mask, i); ++ } ++ widget->muted = 1; ++ } ++ } ++ return 0; ++} ++ ++/* create new dapm mixer control */ ++static int dapm_new_mixer(struct snd_soc_codec *codec, ++ struct snd_soc_dapm_widget *w) ++{ ++ int i, ret = 0; ++ char name[32]; ++ struct snd_soc_dapm_path *path; ++ ++ /* add kcontrol */ ++ for (i = 0; i < w->num_kcontrols; i++) { ++ ++ /* match name */ ++ list_for_each_entry(path, &w->sources, list_sink) { ++ ++ /* mixer/mux paths name must match control name */ ++ if (path->name != (char*)w->kcontrols[i].name) ++ continue; ++ ++ /* add dapm control with long name */ ++ snprintf(name, 32, "%s %s", w->name, w->kcontrols[i].name); ++ path->long_name = kstrdup (name, GFP_KERNEL); ++ if (path->long_name == NULL) ++ return -ENOMEM; ++ ++ path->kcontrol = snd_soc_cnew(&w->kcontrols[i], w, ++ path->long_name); ++ ret = snd_ctl_add(codec->card, path->kcontrol); ++ if (ret < 0) { ++ printk(KERN_ERR "asoc: failed to add dapm kcontrol %s\n", ++ path->long_name); ++ kfree(path->long_name); ++ path->long_name = NULL; ++ return ret; ++ } ++ } ++ } ++ return ret; ++} ++ ++/* create new dapm mux control */ ++static int dapm_new_mux(struct snd_soc_codec *codec, ++ struct snd_soc_dapm_widget *w) ++{ ++ struct snd_soc_dapm_path *path = NULL; ++ struct snd_kcontrol *kcontrol; ++ int ret = 0; ++ ++ if (!w->num_kcontrols) { ++ printk(KERN_ERR "asoc: mux %s has no controls\n", w->name); ++ return -EINVAL; ++ } ++ ++ kcontrol = snd_soc_cnew(&w->kcontrols[0], w, w->name); ++ ret = snd_ctl_add(codec->card, kcontrol); ++ if (ret < 0) ++ goto err; ++ ++ list_for_each_entry(path, &w->sources, list_sink) ++ path->kcontrol = kcontrol; ++ ++ return ret; ++ ++err: ++ printk(KERN_ERR "asoc: failed to add kcontrol %s\n", w->name); ++ return ret; ++} ++ ++/* create new dapm volume control */ ++static int dapm_new_pga(struct snd_soc_codec *codec, ++ struct snd_soc_dapm_widget *w) ++{ ++ struct snd_kcontrol *kcontrol; ++ int ret = 0; ++ ++ if (!w->num_kcontrols) ++ return -EINVAL; ++ ++ kcontrol = snd_soc_cnew(&w->kcontrols[0], w, w->name); ++ ret = snd_ctl_add(codec->card, kcontrol); ++ if (ret < 0) { ++ printk(KERN_ERR "asoc: failed to add kcontrol %s\n", w->name); ++ return ret; ++ } ++ ++ return ret; ++} ++ ++/* reset 'walked' bit for each dapm path */ ++static inline void dapm_clear_walk(struct snd_soc_codec *codec) ++{ ++ struct snd_soc_dapm_path *p; ++ ++ list_for_each_entry(p, &codec->dapm_paths, list) ++ p->walked = 0; ++} ++ ++/* ++ * Recursively check for a completed path to an active or physically connected ++ * output widget. Returns number of complete paths. ++ */ ++static int is_connected_output_ep(struct snd_soc_dapm_widget *widget) ++{ ++ struct snd_soc_dapm_path *path; ++ int con = 0; ++ ++ if (widget->id == snd_soc_dapm_adc && widget->active) ++ return 1; ++ ++ if (widget->connected) { ++ /* connected pin ? */ ++ if (widget->id == snd_soc_dapm_output && !widget->ext) ++ return 1; ++ ++ /* connected jack or spk ? */ ++ if (widget->id == snd_soc_dapm_hp || widget->id == snd_soc_dapm_spk || ++ widget->id == snd_soc_dapm_line) ++ return 1; ++ } ++ ++ list_for_each_entry(path, &widget->sinks, list_source) { ++ if (path->walked) ++ continue; ++ ++ if (path->sink && path->connect) { ++ path->walked = 1; ++ con += is_connected_output_ep(path->sink); ++ } ++ } ++ ++ return con; ++} ++ ++/* ++ * Recursively check for a completed path to an active or physically connected ++ * input widget. Returns number of complete paths. ++ */ ++static int is_connected_input_ep(struct snd_soc_dapm_widget *widget) ++{ ++ struct snd_soc_dapm_path *path; ++ int con = 0; ++ ++ /* active stream ? */ ++ if (widget->id == snd_soc_dapm_dac && widget->active) ++ return 1; ++ ++ if (widget->connected) { ++ /* connected pin ? */ ++ if (widget->id == snd_soc_dapm_input && !widget->ext) ++ return 1; ++ ++ /* connected VMID/Bias for lower pops */ ++ if (widget->id == snd_soc_dapm_vmid) ++ return 1; ++ ++ /* connected jack ? */ ++ if (widget->id == snd_soc_dapm_mic || widget->id == snd_soc_dapm_line) ++ return 1; ++ } ++ ++ list_for_each_entry(path, &widget->sources, list_sink) { ++ if (path->walked) ++ continue; ++ ++ if (path->source && path->connect) { ++ path->walked = 1; ++ con += is_connected_input_ep(path->source); ++ } ++ } ++ ++ return con; ++} ++ ++/* ++ * Scan each dapm widget for complete audio path. ++ * A complete path is a route that has valid endpoints i.e.:- ++ * ++ * o DAC to output pin. ++ * o Input Pin to ADC. ++ * o Input pin to Output pin (bypass, sidetone) ++ * o DAC to ADC (loopback). ++ */ ++int dapm_power_widgets(struct snd_soc_codec *codec, int event) ++{ ++ struct snd_soc_dapm_widget *w; ++ int in, out, i, c = 1, *seq = NULL, ret = 0, power_change, power; ++ ++ /* do we have a sequenced stream event */ ++ if (event == SND_SOC_DAPM_STREAM_START) { ++ c = ARRAY_SIZE(dapm_up_seq); ++ seq = dapm_up_seq; ++ } else if (event == SND_SOC_DAPM_STREAM_STOP) { ++ c = ARRAY_SIZE(dapm_down_seq); ++ seq = dapm_down_seq; ++ } ++ ++ for(i = 0; i < c; i++) { ++ list_for_each_entry(w, &codec->dapm_widgets, list) { ++ ++ /* is widget in stream order */ ++ if (seq && seq[i] && w->id != seq[i]) ++ continue; ++ ++ /* vmid - no action */ ++ if (w->id == snd_soc_dapm_vmid) ++ continue; ++ ++ /* active ADC */ ++ if (w->id == snd_soc_dapm_adc && w->active) { ++ in = is_connected_input_ep(w); ++ dapm_clear_walk(w->codec); ++ w->power = (in != 0) ? 1 : 0; ++ dapm_update_bits(w); ++ continue; ++ } ++ ++ /* active DAC */ ++ if (w->id == snd_soc_dapm_dac && w->active) { ++ out = is_connected_output_ep(w); ++ dapm_clear_walk(w->codec); ++ w->power = (out != 0) ? 1 : 0; ++ dapm_update_bits(w); ++ continue; ++ } ++ ++ /* programmable gain/attenuation */ ++ if (w->id == snd_soc_dapm_pga) { ++ int on; ++ in = is_connected_input_ep(w); ++ dapm_clear_walk(w->codec); ++ out = is_connected_output_ep(w); ++ dapm_clear_walk(w->codec); ++ w->power = on = (out != 0 && in != 0) ? 1 : 0; ++ ++ if (!on) ++ dapm_set_pga(w, on); /* lower volume to reduce pops */ ++ dapm_update_bits(w); ++ if (on) ++ dapm_set_pga(w, on); /* restore volume from zero */ ++ ++ continue; ++ } ++ ++ /* pre and post event widgets */ ++ if (w->id == snd_soc_dapm_pre) { ++ if (!w->event) ++ continue; ++ ++ if (event == SND_SOC_DAPM_STREAM_START) { ++ ret = w->event(w, SND_SOC_DAPM_PRE_PMU); ++ if (ret < 0) ++ return ret; ++ } else if (event == SND_SOC_DAPM_STREAM_STOP) { ++ ret = w->event(w, SND_SOC_DAPM_PRE_PMD); ++ if (ret < 0) ++ return ret; ++ } ++ continue; ++ } ++ if (w->id == snd_soc_dapm_post) { ++ if (!w->event) ++ continue; ++ ++ if (event == SND_SOC_DAPM_STREAM_START) { ++ ret = w->event(w, SND_SOC_DAPM_POST_PMU); ++ if (ret < 0) ++ return ret; ++ } else if (event == SND_SOC_DAPM_STREAM_STOP) { ++ ret = w->event(w, SND_SOC_DAPM_POST_PMD); ++ if (ret < 0) ++ return ret; ++ } ++ continue; ++ } ++ ++ /* all other widgets */ ++ in = is_connected_input_ep(w); ++ dapm_clear_walk(w->codec); ++ out = is_connected_output_ep(w); ++ dapm_clear_walk(w->codec); ++ power = (out != 0 && in != 0) ? 1 : 0; ++ power_change = (w->power == power) ? 0: 1; ++ w->power = power; ++ ++ /* call any power change event handlers */ ++ if (power_change) { ++ if (w->event) { ++ dbg("power %s event for %s flags %x\n", ++ w->power ? "on" : "off", w->name, w->event_flags); ++ if (power) { ++ /* power up event */ ++ if (w->event_flags & SND_SOC_DAPM_PRE_PMU) { ++ ret = w->event(w, SND_SOC_DAPM_PRE_PMU); ++ if (ret < 0) ++ return ret; ++ } ++ dapm_update_bits(w); ++ if (w->event_flags & SND_SOC_DAPM_POST_PMU){ ++ ret = w->event(w, SND_SOC_DAPM_POST_PMU); ++ if (ret < 0) ++ return ret; ++ } ++ } else { ++ /* power down event */ ++ if (w->event_flags & SND_SOC_DAPM_PRE_PMD) { ++ ret = w->event(w, SND_SOC_DAPM_PRE_PMD); ++ if (ret < 0) ++ return ret; ++ } ++ dapm_update_bits(w); ++ if (w->event_flags & SND_SOC_DAPM_POST_PMD) { ++ ret = w->event(w, SND_SOC_DAPM_POST_PMD); ++ if (ret < 0) ++ return ret; ++ } ++ } ++ } else ++ /* no event handler */ ++ dapm_update_bits(w); ++ } ++ } ++ } ++ ++ return ret; ++} ++ ++#if DAPM_DEBUG ++static void dbg_dump_dapm(struct snd_soc_codec* codec, const char *action) ++{ ++ struct snd_soc_dapm_widget *w; ++ struct snd_soc_dapm_path *p = NULL; ++ int in, out; ++ ++ printk("DAPM %s %s\n", codec->name, action); ++ ++ list_for_each_entry(w, &codec->dapm_widgets, list) { ++ ++ /* only display widgets that effect routing */ ++ switch (w->id) { ++ case snd_soc_dapm_pre: ++ case snd_soc_dapm_post: ++ case snd_soc_dapm_vmid: ++ continue; ++ case snd_soc_dapm_mux: ++ case snd_soc_dapm_output: ++ case snd_soc_dapm_input: ++ case snd_soc_dapm_switch: ++ case snd_soc_dapm_hp: ++ case snd_soc_dapm_mic: ++ case snd_soc_dapm_spk: ++ case snd_soc_dapm_line: ++ case snd_soc_dapm_micbias: ++ case snd_soc_dapm_dac: ++ case snd_soc_dapm_adc: ++ case snd_soc_dapm_pga: ++ case snd_soc_dapm_mixer: ++ if (w->name) { ++ in = is_connected_input_ep(w); ++ dapm_clear_walk(w->codec); ++ out = is_connected_output_ep(w); ++ dapm_clear_walk(w->codec); ++ printk("%s: %s in %d out %d\n", w->name, ++ w->power ? "On":"Off",in, out); ++ ++ list_for_each_entry(p, &w->sources, list_sink) { ++ if (p->connect) ++ printk(" in %s %s\n", p->name ? p->name : "static", ++ p->source->name); ++ } ++ list_for_each_entry(p, &w->sinks, list_source) { ++ p = list_entry(lp, struct snd_soc_dapm_path, list_source); ++ if (p->connect) ++ printk(" out %s %s\n", p->name ? p->name : "static", ++ p->sink->name); ++ } ++ } ++ break; ++ } ++ } ++} ++#endif ++ ++/* test and update the power status of a mux widget */ ++int dapm_mux_update_power(struct snd_soc_dapm_widget *widget, ++ struct snd_kcontrol *kcontrol, int mask, int val, struct soc_enum* e) ++{ ++ struct snd_soc_dapm_path *path; ++ int found = 0; ++ ++ if (widget->id != snd_soc_dapm_mux) ++ return -ENODEV; ++ ++ if (!snd_soc_test_bits(widget->codec, e->reg, mask, val)) ++ return 0; ++ ++ /* find dapm widget path assoc with kcontrol */ ++ list_for_each_entry(path, &widget->codec->dapm_paths, list) { ++ if (path->kcontrol != kcontrol) ++ continue; ++ ++ if (!path->name || ! e->texts[val]) ++ continue; ++ ++ found = 1; ++ /* we now need to match the string in the enum to the path */ ++ if (!(strcmp(path->name, e->texts[val]))) ++ path->connect = 1; /* new connection */ ++ else ++ path->connect = 0; /* old connection must be powered down */ ++ } ++ ++ if (found) ++ dapm_power_widgets(widget->codec, SND_SOC_DAPM_STREAM_NOP); ++ ++ return 0; ++} ++EXPORT_SYMBOL_GPL(dapm_mux_update_power); ++ ++/* test and update the power status of a mixer widget */ ++int dapm_mixer_update_power(struct snd_soc_dapm_widget *widget, ++ struct snd_kcontrol *kcontrol, int reg, int val_mask, int val, int invert) ++{ ++ struct snd_soc_dapm_path *path; ++ int found = 0; ++ ++ if (widget->id != snd_soc_dapm_mixer) ++ return -ENODEV; ++ ++ if (!snd_soc_test_bits(widget->codec, reg, val_mask, val)) ++ return 0; ++ ++ /* find dapm widget path assoc with kcontrol */ ++ list_for_each_entry(path, &widget->codec->dapm_paths, list) { ++ if (path->kcontrol != kcontrol) ++ continue; ++ ++ /* found, now check type */ ++ found = 1; ++ if (val) ++ /* new connection */ ++ path->connect = invert ? 0:1; ++ else ++ /* old connection must be powered down */ ++ path->connect = invert ? 1:0; ++ break; ++ } ++ ++ if (found) ++ dapm_power_widgets(widget->codec, SND_SOC_DAPM_STREAM_NOP); ++ ++ return 0; ++} ++EXPORT_SYMBOL_GPL(dapm_mixer_update_power); ++ ++/* show dapm widget status in sys fs */ ++static ssize_t dapm_widget_show(struct device *dev, ++ struct device_attribute *attr, char *buf) ++{ ++ struct snd_soc_device *devdata = dev_get_drvdata(dev); ++ struct snd_soc_codec *codec = devdata->codec; ++ struct snd_soc_dapm_widget *w; ++ int count = 0; ++ char *state = "not set"; ++ ++ list_for_each_entry(w, &codec->dapm_widgets, list) { ++ ++ /* only display widgets that burnm power */ ++ switch (w->id) { ++ case snd_soc_dapm_hp: ++ case snd_soc_dapm_mic: ++ case snd_soc_dapm_spk: ++ case snd_soc_dapm_line: ++ case snd_soc_dapm_micbias: ++ case snd_soc_dapm_dac: ++ case snd_soc_dapm_adc: ++ case snd_soc_dapm_pga: ++ case snd_soc_dapm_mixer: ++ if (w->name) ++ count += sprintf(buf + count, "%s: %s\n", ++ w->name, w->power ? "On":"Off"); ++ break; ++ default: ++ break; ++ } ++ } ++ ++ switch(codec->dapm_state){ ++ case SNDRV_CTL_POWER_D0: ++ state = "D0"; ++ break; ++ case SNDRV_CTL_POWER_D1: ++ state = "D1"; ++ break; ++ case SNDRV_CTL_POWER_D2: ++ state = "D2"; ++ break; ++ case SNDRV_CTL_POWER_D3hot: ++ state = "D3hot"; ++ break; ++ case SNDRV_CTL_POWER_D3cold: ++ state = "D3cold"; ++ break; ++ } ++ count += sprintf(buf + count, "PM State: %s\n", state); ++ ++ return count; ++} ++ ++static DEVICE_ATTR(dapm_widget, 0444, dapm_widget_show, NULL); ++ ++int snd_soc_dapm_sys_add(struct device *dev) ++{ ++ int ret = 0; ++ ++ if (dapm_status) ++ ret = device_create_file(dev, &dev_attr_dapm_widget); ++ ++ return ret; ++} ++ ++static void snd_soc_dapm_sys_remove(struct device *dev) ++{ ++ if (dapm_status) ++ device_remove_file(dev, &dev_attr_dapm_widget); ++} ++ ++/* free all dapm widgets and resources */ ++void dapm_free_widgets(struct snd_soc_codec *codec) ++{ ++ struct snd_soc_dapm_widget *w, *next_w; ++ struct snd_soc_dapm_path *p, *next_p; ++ ++ list_for_each_entry_safe(w, next_w, &codec->dapm_widgets, list) { ++ list_del(&w->list); ++ kfree(w); ++ } ++ ++ list_for_each_entry_safe(p, next_p, &codec->dapm_paths, list) { ++ list_del(&p->list); ++ kfree(p->long_name); ++ kfree(p); ++ } ++} ++ ++/** ++ * snd_soc_dapm_sync_endpoints - scan and power dapm paths ++ * @codec: audio codec ++ * ++ * Walks all dapm audio paths and powers widgets according to their ++ * stream or path usage. ++ * ++ * Returns 0 for success. ++ */ ++int snd_soc_dapm_sync_endpoints(struct snd_soc_codec *codec) ++{ ++ return dapm_power_widgets(codec, SND_SOC_DAPM_STREAM_NOP); ++} ++EXPORT_SYMBOL_GPL(snd_soc_dapm_sync_endpoints); ++ ++/** ++ * snd_soc_dapm_connect_input - connect dapm widgets ++ * @codec: audio codec ++ * @sink: name of target widget ++ * @control: mixer control name ++ * @source: name of source name ++ * ++ * Connects 2 dapm widgets together via a named audio path. The sink is ++ * the widget receiving the audio signal, whilst the source is the sender ++ * of the audio signal. ++ * ++ * Returns 0 for success else error. ++ */ ++int snd_soc_dapm_connect_input(struct snd_soc_codec *codec, const char *sink, ++ const char * control, const char *source) ++{ ++ struct snd_soc_dapm_path *path; ++ struct snd_soc_dapm_widget *wsource = NULL, *wsink = NULL, *w; ++ int ret = 0; ++ ++ /* find src and dest widgets */ ++ list_for_each_entry(w, &codec->dapm_widgets, list) { ++ ++ if (!wsink && !(strcmp(w->name, sink))) { ++ wsink = w; ++ continue; ++ } ++ if (!wsource && !(strcmp(w->name, source))) { ++ wsource = w; ++ } ++ } ++ ++ if (wsource == NULL || wsink == NULL) ++ return -ENODEV; ++ ++ path = kzalloc(sizeof(struct snd_soc_dapm_path), GFP_KERNEL); ++ if (!path) ++ return -ENOMEM; ++ ++ path->source = wsource; ++ path->sink = wsink; ++ INIT_LIST_HEAD(&path->list); ++ INIT_LIST_HEAD(&path->list_source); ++ INIT_LIST_HEAD(&path->list_sink); ++ ++ /* check for external widgets */ ++ if (wsink->id == snd_soc_dapm_input) { ++ if (wsource->id == snd_soc_dapm_micbias || ++ wsource->id == snd_soc_dapm_mic || ++ wsink->id == snd_soc_dapm_line) ++ wsink->ext = 1; ++ } ++ if (wsource->id == snd_soc_dapm_output) { ++ if (wsink->id == snd_soc_dapm_spk || ++ wsink->id == snd_soc_dapm_hp || ++ wsink->id == snd_soc_dapm_line) ++ wsource->ext = 1; ++ } ++ ++ /* connect static paths */ ++ if (control == NULL) { ++ list_add(&path->list, &codec->dapm_paths); ++ list_add(&path->list_sink, &wsink->sources); ++ list_add(&path->list_source, &wsource->sinks); ++ path->connect = 1; ++ return 0; ++ } ++ ++ /* connect dynamic paths */ ++ switch(wsink->id) { ++ case snd_soc_dapm_adc: ++ case snd_soc_dapm_dac: ++ case snd_soc_dapm_pga: ++ case snd_soc_dapm_input: ++ case snd_soc_dapm_output: ++ case snd_soc_dapm_micbias: ++ case snd_soc_dapm_vmid: ++ case snd_soc_dapm_pre: ++ case snd_soc_dapm_post: ++ list_add(&path->list, &codec->dapm_paths); ++ list_add(&path->list_sink, &wsink->sources); ++ list_add(&path->list_source, &wsource->sinks); ++ path->connect = 1; ++ return 0; ++ case snd_soc_dapm_mux: ++ ret = dapm_connect_mux(codec, wsource, wsink, path, control, ++ &wsink->kcontrols[0]); ++ if (ret != 0) ++ goto err; ++ break; ++ case snd_soc_dapm_switch: ++ case snd_soc_dapm_mixer: ++ ret = dapm_connect_mixer(codec, wsource, wsink, path, control); ++ if (ret != 0) ++ goto err; ++ break; ++ case snd_soc_dapm_hp: ++ case snd_soc_dapm_mic: ++ case snd_soc_dapm_line: ++ case snd_soc_dapm_spk: ++ list_add(&path->list, &codec->dapm_paths); ++ list_add(&path->list_sink, &wsink->sources); ++ list_add(&path->list_source, &wsource->sinks); ++ path->connect = 0; ++ return 0; ++ } ++ return 0; ++ ++err: ++ printk(KERN_WARNING "asoc: no dapm match for %s --> %s --> %s\n", source, ++ control, sink); ++ kfree(path); ++ return ret; ++} ++EXPORT_SYMBOL_GPL(snd_soc_dapm_connect_input); ++ ++/** ++ * snd_soc_dapm_new_widgets - add new dapm widgets ++ * @codec: audio codec ++ * ++ * Checks the codec for any new dapm widgets and creates them if found. ++ * ++ * Returns 0 for success. ++ */ ++int snd_soc_dapm_new_widgets(struct snd_soc_codec *codec) ++{ ++ struct snd_soc_dapm_widget *w; ++ ++ mutex_lock(&codec->mutex); ++ list_for_each_entry(w, &codec->dapm_widgets, list) ++ { ++ if (w->new) ++ continue; ++ ++ switch(w->id) { ++ case snd_soc_dapm_switch: ++ case snd_soc_dapm_mixer: ++ dapm_new_mixer(codec, w); ++ break; ++ case snd_soc_dapm_mux: ++ dapm_new_mux(codec, w); ++ break; ++ case snd_soc_dapm_adc: ++ case snd_soc_dapm_dac: ++ case snd_soc_dapm_pga: ++ dapm_new_pga(codec, w); ++ break; ++ case snd_soc_dapm_input: ++ case snd_soc_dapm_output: ++ case snd_soc_dapm_micbias: ++ case snd_soc_dapm_spk: ++ case snd_soc_dapm_hp: ++ case snd_soc_dapm_mic: ++ case snd_soc_dapm_line: ++ case snd_soc_dapm_vmid: ++ case snd_soc_dapm_pre: ++ case snd_soc_dapm_post: ++ break; ++ } ++ w->new = 1; ++ } ++ ++ dapm_power_widgets(codec, SND_SOC_DAPM_STREAM_NOP); ++ mutex_unlock(&codec->mutex); ++ return 0; ++} ++EXPORT_SYMBOL_GPL(snd_soc_dapm_new_widgets); ++ ++/** ++ * snd_soc_dapm_get_volsw - dapm mixer get callback ++ * @kcontrol: mixer control ++ * @uinfo: control element information ++ * ++ * Callback to get the value of a dapm mixer control. ++ * ++ * Returns 0 for success. ++ */ ++int snd_soc_dapm_get_volsw(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_dapm_widget *widget = snd_kcontrol_chip(kcontrol); ++ int reg = kcontrol->private_value & 0xff; ++ int shift = (kcontrol->private_value >> 8) & 0x0f; ++ int rshift = (kcontrol->private_value >> 12) & 0x0f; ++ int mask = (kcontrol->private_value >> 16) & 0xff; ++ int invert = (kcontrol->private_value >> 24) & 0x01; ++ ++ /* return the saved value if we are powered down */ ++ if (widget->id == snd_soc_dapm_pga && !widget->power) { ++ ucontrol->value.integer.value[0] = widget->saved_value; ++ return 0; ++ } ++ ++ ucontrol->value.integer.value[0] = ++ (snd_soc_read(widget->codec, reg) >> shift) & mask; ++ if (shift != rshift) ++ ucontrol->value.integer.value[1] = ++ (snd_soc_read(widget->codec, reg) >> rshift) & mask; ++ if (invert) { ++ ucontrol->value.integer.value[0] = ++ mask - ucontrol->value.integer.value[0]; ++ if (shift != rshift) ++ ucontrol->value.integer.value[1] = ++ mask - ucontrol->value.integer.value[1]; ++ } ++ ++ return 0; ++} ++EXPORT_SYMBOL_GPL(snd_soc_dapm_get_volsw); ++ ++/** ++ * snd_soc_dapm_put_volsw - dapm mixer set callback ++ * @kcontrol: mixer control ++ * @uinfo: control element information ++ * ++ * Callback to set the value of a dapm mixer control. ++ * ++ * Returns 0 for success. ++ */ ++int snd_soc_dapm_put_volsw(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_dapm_widget *widget = snd_kcontrol_chip(kcontrol); ++ int reg = kcontrol->private_value & 0xff; ++ int shift = (kcontrol->private_value >> 8) & 0x0f; ++ int rshift = (kcontrol->private_value >> 12) & 0x0f; ++ int mask = (kcontrol->private_value >> 16) & 0xff; ++ int invert = (kcontrol->private_value >> 24) & 0x01; ++ unsigned short val, val2, val_mask; ++ int ret; ++ ++ val = (ucontrol->value.integer.value[0] & mask); ++ ++ if (invert) ++ val = mask - val; ++ val_mask = mask << shift; ++ val = val << shift; ++ if (shift != rshift) { ++ val2 = (ucontrol->value.integer.value[1] & mask); ++ if (invert) ++ val2 = mask - val2; ++ val_mask |= mask << rshift; ++ val |= val2 << rshift; ++ } ++ ++ mutex_lock(&widget->codec->mutex); ++ widget->value = val; ++ ++ /* save volume value if the widget is powered down */ ++ if (widget->id == snd_soc_dapm_pga && !widget->power) { ++ widget->saved_value = val; ++ mutex_unlock(&widget->codec->mutex); ++ return 1; ++ } ++ ++ dapm_mixer_update_power(widget, kcontrol, reg, val_mask, val, invert); ++ if (widget->event) { ++ if (widget->event_flags & SND_SOC_DAPM_PRE_REG) { ++ ret = widget->event(widget, SND_SOC_DAPM_PRE_REG); ++ if (ret < 0) ++ goto out; ++ } ++ ret = snd_soc_update_bits(widget->codec, reg, val_mask, val); ++ if (widget->event_flags & SND_SOC_DAPM_POST_REG) ++ ret = widget->event(widget, SND_SOC_DAPM_POST_REG); ++ } else ++ ret = snd_soc_update_bits(widget->codec, reg, val_mask, val); ++ ++out: ++ mutex_unlock(&widget->codec->mutex); ++ return ret; ++} ++EXPORT_SYMBOL_GPL(snd_soc_dapm_put_volsw); ++ ++/** ++ * snd_soc_dapm_get_enum_double - dapm enumerated double mixer get callback ++ * @kcontrol: mixer control ++ * @uinfo: control element information ++ * ++ * Callback to get the value of a dapm enumerated double mixer control. ++ * ++ * Returns 0 for success. ++ */ ++int snd_soc_dapm_get_enum_double(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_dapm_widget *widget = snd_kcontrol_chip(kcontrol); ++ struct soc_enum *e = (struct soc_enum *)kcontrol->private_value; ++ unsigned short val, bitmask; ++ ++ for (bitmask = 1; bitmask < e->mask; bitmask <<= 1) ++ ; ++ val = snd_soc_read(widget->codec, e->reg); ++ ucontrol->value.enumerated.item[0] = (val >> e->shift_l) & (bitmask - 1); ++ if (e->shift_l != e->shift_r) ++ ucontrol->value.enumerated.item[1] = ++ (val >> e->shift_r) & (bitmask - 1); ++ ++ return 0; ++} ++EXPORT_SYMBOL_GPL(snd_soc_dapm_get_enum_double); ++ ++/** ++ * snd_soc_dapm_put_enum_double - dapm enumerated double mixer set callback ++ * @kcontrol: mixer control ++ * @uinfo: control element information ++ * ++ * Callback to set the value of a dapm enumerated double mixer control. ++ * ++ * Returns 0 for success. ++ */ ++int snd_soc_dapm_put_enum_double(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_dapm_widget *widget = snd_kcontrol_chip(kcontrol); ++ struct soc_enum *e = (struct soc_enum *)kcontrol->private_value; ++ unsigned short val, mux; ++ unsigned short mask, bitmask; ++ int ret = 0; ++ ++ for (bitmask = 1; bitmask < e->mask; bitmask <<= 1) ++ ; ++ if (ucontrol->value.enumerated.item[0] > e->mask - 1) ++ return -EINVAL; ++ mux = ucontrol->value.enumerated.item[0]; ++ val = mux << e->shift_l; ++ mask = (bitmask - 1) << e->shift_l; ++ if (e->shift_l != e->shift_r) { ++ if (ucontrol->value.enumerated.item[1] > e->mask - 1) ++ return -EINVAL; ++ val |= ucontrol->value.enumerated.item[1] << e->shift_r; ++ mask |= (bitmask - 1) << e->shift_r; ++ } ++ ++ mutex_lock(&widget->codec->mutex); ++ widget->value = val; ++ dapm_mux_update_power(widget, kcontrol, mask, mux, e); ++ if (widget->event) { ++ if (widget->event_flags & SND_SOC_DAPM_PRE_REG) { ++ ret = widget->event(widget, SND_SOC_DAPM_PRE_REG); ++ if (ret < 0) ++ goto out; ++ } ++ ret = snd_soc_update_bits(widget->codec, e->reg, mask, val); ++ if (widget->event_flags & SND_SOC_DAPM_POST_REG) ++ ret = widget->event(widget, SND_SOC_DAPM_POST_REG); ++ } else ++ ret = snd_soc_update_bits(widget->codec, e->reg, mask, val); ++ ++out: ++ mutex_unlock(&widget->codec->mutex); ++ return ret; ++} ++EXPORT_SYMBOL_GPL(snd_soc_dapm_put_enum_double); ++ ++/** ++ * snd_soc_dapm_new_control - create new dapm control ++ * @codec: audio codec ++ * @widget: widget template ++ * ++ * Creates a new dapm control based upon the template. ++ * ++ * Returns 0 for success else error. ++ */ ++int snd_soc_dapm_new_control(struct snd_soc_codec *codec, ++ const struct snd_soc_dapm_widget *widget) ++{ ++ struct snd_soc_dapm_widget *w; ++ ++ if ((w = dapm_cnew_widget(widget)) == NULL) ++ return -ENOMEM; ++ ++ w->codec = codec; ++ INIT_LIST_HEAD(&w->sources); ++ INIT_LIST_HEAD(&w->sinks); ++ INIT_LIST_HEAD(&w->list); ++ list_add(&w->list, &codec->dapm_widgets); ++ ++ /* machine layer set ups unconnected pins and insertions */ ++ w->connected = 1; ++ return 0; ++} ++EXPORT_SYMBOL_GPL(snd_soc_dapm_new_control); ++ ++/** ++ * snd_soc_dapm_stream_event - send a stream event to the dapm core ++ * @codec: audio codec ++ * @stream: stream name ++ * @event: stream event ++ * ++ * Sends a stream event to the dapm core. The core then makes any ++ * necessary widget power changes. ++ * ++ * Returns 0 for success else error. ++ */ ++int snd_soc_dapm_stream_event(struct snd_soc_codec *codec, ++ char *stream, int event) ++{ ++ struct snd_soc_dapm_widget *w; ++ ++ mutex_lock(&codec->mutex); ++ list_for_each_entry(w, &codec->dapm_widgets, list) ++ { ++ if (!w->sname) ++ continue; ++ dbg("widget %s\n %s stream %s event %d\n", w->name, w->sname, ++ stream, event); ++ if (strstr(w->sname, stream)) { ++ switch(event) { ++ case SND_SOC_DAPM_STREAM_START: ++ w->active = 1; ++ break; ++ case SND_SOC_DAPM_STREAM_STOP: ++ w->active = 0; ++ break; ++ case SND_SOC_DAPM_STREAM_SUSPEND: ++ if (w->active) ++ w->suspend = 1; ++ w->active = 0; ++ break; ++ case SND_SOC_DAPM_STREAM_RESUME: ++ if (w->suspend) { ++ w->active = 1; ++ w->suspend = 0; ++ } ++ break; ++ case SND_SOC_DAPM_STREAM_PAUSE_PUSH: ++ break; ++ case SND_SOC_DAPM_STREAM_PAUSE_RELEASE: ++ break; ++ } ++ } ++ } ++ mutex_unlock(&codec->mutex); ++ ++ dapm_power_widgets(codec, event); ++ dump_dapm(codec, __FUNCTION__); ++ return 0; ++} ++EXPORT_SYMBOL_GPL(snd_soc_dapm_stream_event); ++ ++/** ++ * snd_soc_dapm_set_endpoint - set audio endpoint status ++ * @codec: audio codec ++ * @endpoint: audio signal endpoint (or start point) ++ * @status: point status ++ * ++ * Set audio endpoint status - connected or disconnected. ++ * ++ * Returns 0 for success else error. ++ */ ++int snd_soc_dapm_set_endpoint(struct snd_soc_codec *codec, ++ char *endpoint, int status) ++{ ++ struct snd_soc_dapm_widget *w; ++ ++ list_for_each_entry(w, &codec->dapm_widgets, list) { ++ if (!strcmp(w->name, endpoint)) { ++ w->connected = status; ++ } ++ } ++ ++ return 0; ++} ++EXPORT_SYMBOL_GPL(snd_soc_dapm_set_endpoint); ++ ++/** ++ * snd_soc_dapm_free - free dapm resources ++ * @socdev: SoC device ++ * ++ * Free all dapm widgets and resources. ++ */ ++void snd_soc_dapm_free(struct snd_soc_device *socdev) ++{ ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ snd_soc_dapm_sys_remove(socdev->dev); ++ dapm_free_widgets(codec); ++} ++EXPORT_SYMBOL_GPL(snd_soc_dapm_free); ++ ++/* Module information */ ++MODULE_AUTHOR("Liam Girdwood, liam.girdwood@wolfsonmicro.com, www.wolfsonmicro.com"); ++MODULE_DESCRIPTION("Dynamic Audio Power Management core for ALSA SoC"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/soc-core.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/soc-core.c +@@ -0,0 +1,2063 @@ ++/* ++ * soc-core.c -- ALSA SoC Audio Layer ++ * ++ * Copyright 2005 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * 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. ++ * ++ * Revision history ++ * 12th Aug 2005 Initial version. ++ * 25th Oct 2005 Working Codec, Interface and Platform registration. ++ * ++ * TODO: ++ * o Add hw rules to enforce rates, etc. ++ * o More testing with other codecs/machines. ++ * o Add more codecs and platforms to ensure good API coverage. ++ * o Support TDM on PCM and I2S ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/init.h> ++#include <linux/delay.h> ++#include <linux/pm.h> ++#include <linux/bitops.h> ++#include <linux/platform_device.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/pcm_params.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++#include <sound/initval.h> ++ ++/* debug */ ++#define SOC_DEBUG 1 ++#if SOC_DEBUG ++#define dbg(format, arg...) printk(format, ## arg) ++#else ++#define dbg(format, arg...) ++#endif ++/* debug DAI capabilities matching */ ++#define SOC_DEBUG_DAI 1 ++#if SOC_DEBUG_DAI ++#define dbgc(format, arg...) printk(format, ## arg) ++#else ++#define dbgc(format, arg...) ++#endif ++ ++#define CODEC_CPU(codec, cpu) ((codec << 4) | cpu) ++ ++static DEFINE_MUTEX(pcm_mutex); ++static DEFINE_MUTEX(io_mutex); ++static struct workqueue_struct *soc_workq; ++static struct work_struct soc_stream_work; ++static DECLARE_WAIT_QUEUE_HEAD(soc_pm_waitq); ++ ++/* supported sample rates */ ++/* ATTENTION: these values depend on the definition in pcm.h! */ ++static const unsigned int rates[] = { ++ 5512, 8000, 11025, 16000, 22050, 32000, 44100, ++ 48000, 64000, 88200, 96000, 176400, 192000 ++}; ++ ++/* ++ * This is a timeout to do a DAPM powerdown after a stream is closed(). ++ * It can be used to eliminate pops between different playback streams, e.g. ++ * between two audio tracks. ++ */ ++static int pmdown_time = 5000; ++module_param(pmdown_time, int, 0); ++MODULE_PARM_DESC(pmdown_time, "DAPM stream powerdown time (msecs)"); ++ ++#ifdef CONFIG_SND_SOC_AC97_BUS ++/* unregister ac97 codec */ ++static int soc_ac97_dev_unregister(struct snd_soc_codec *codec) ++{ ++ if (codec->ac97->dev.bus) ++ device_unregister(&codec->ac97->dev); ++ return 0; ++} ++ ++/* stop no dev release warning */ ++static void soc_ac97_device_release(struct device *dev){} ++ ++/* register ac97 codec to bus */ ++static int soc_ac97_dev_register(struct snd_soc_codec *codec) ++{ ++ int err; ++ ++ codec->ac97->dev.bus = &ac97_bus_type; ++ codec->ac97->dev.parent = NULL; ++ codec->ac97->dev.release = soc_ac97_device_release; ++ ++ snprintf(codec->ac97->dev.bus_id, BUS_ID_SIZE, "%d-%d:%s", ++ codec->card->number, 0, codec->name); ++ err = device_register(&codec->ac97->dev); ++ if (err < 0) { ++ snd_printk(KERN_ERR "Can't register ac97 bus\n"); ++ codec->ac97->dev.bus = NULL; ++ return err; ++ } ++ return 0; ++} ++#endif ++ ++static inline const char* get_dai_name(int type) ++{ ++ switch(type) { ++ case SND_SOC_DAI_AC97: ++ return "AC97"; ++ case SND_SOC_DAI_I2S: ++ return "I2S"; ++ case SND_SOC_DAI_PCM: ++ return "PCM"; ++ } ++ return NULL; ++} ++ ++/* get rate format from rate */ ++static inline int soc_get_rate_format(int rate) ++{ ++ int i; ++ ++ for (i = 0; i < ARRAY_SIZE(rates); i++) { ++ if (rates[i] == rate) ++ return 1 << i; ++ } ++ return 0; ++} ++ ++/* gets the audio system mclk/sysclk for the given parameters */ ++static unsigned inline int soc_get_mclk(struct snd_soc_pcm_runtime *rtd, ++ struct snd_soc_clock_info *info) ++{ ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_machine *machine = socdev->machine; ++ int i; ++ ++ /* find the matching machine config and get it's mclk for the given ++ * sample rate and hardware format */ ++ for(i = 0; i < machine->num_links; i++) { ++ if (machine->dai_link[i].cpu_dai == rtd->cpu_dai && ++ machine->dai_link[i].config_sysclk) ++ return machine->dai_link[i].config_sysclk(rtd, info); ++ } ++ return 0; ++} ++ ++/* changes a bitclk multiplier mask to a divider mask */ ++static u64 soc_bfs_rcw_to_div(u64 bfs, int rate, unsigned int mclk, ++ unsigned int pcmfmt, unsigned int chn) ++{ ++ int i, j; ++ u64 bfs_ = 0; ++ int size = snd_pcm_format_physical_width(pcmfmt), min = 0; ++ ++ if (size <= 0) ++ return 0; ++ ++ /* the minimum bit clock that has enough bandwidth */ ++ min = size * rate * chn; ++ dbgc("rcw --> div min bclk %d with mclk %d\n", min, mclk); ++ ++ for (i = 0; i < 64; i++) { ++ if ((bfs >> i) & 0x1) { ++ j = min * (i + 1); ++ bfs_ |= SND_SOC_FSBD(mclk/j); ++ dbgc("rcw --> div support mult %d\n", ++ SND_SOC_FSBD_REAL(1<<i)); ++ } ++ } ++ ++ return bfs_; ++} ++ ++/* changes a bitclk divider mask to a multiplier mask */ ++static u64 soc_bfs_div_to_rcw(u64 bfs, int rate, unsigned int mclk, ++ unsigned int pcmfmt, unsigned int chn) ++{ ++ int i, j; ++ u64 bfs_ = 0; ++ ++ int size = snd_pcm_format_physical_width(pcmfmt), min = 0; ++ ++ if (size <= 0) ++ return 0; ++ ++ /* the minimum bit clock that has enough bandwidth */ ++ min = size * rate * chn; ++ dbgc("div to rcw min bclk %d with mclk %d\n", min, mclk); ++ ++ for (i = 0; i < 64; i++) { ++ if ((bfs >> i) & 0x1) { ++ j = mclk / (i + 1); ++ if (j >= min) { ++ bfs_ |= SND_SOC_FSBW(j/min); ++ dbgc("div --> rcw support div %d\n", ++ SND_SOC_FSBW_REAL(1<<i)); ++ } ++ } ++ } ++ ++ return bfs_; ++} ++ ++/* changes a constant bitclk to a multiplier mask */ ++static u64 soc_bfs_rate_to_rcw(u64 bfs, int rate, unsigned int mclk, ++ unsigned int pcmfmt, unsigned int chn) ++{ ++ unsigned int bfs_ = rate * bfs; ++ int size = snd_pcm_format_physical_width(pcmfmt), min = 0; ++ ++ if (size <= 0) ++ return 0; ++ ++ /* the minimum bit clock that has enough bandwidth */ ++ min = size * rate * chn; ++ dbgc("rate --> rcw min bclk %d with mclk %d\n", min, mclk); ++ ++ if (bfs_ < min) ++ return 0; ++ else { ++ bfs_ = SND_SOC_FSBW(bfs_/min); ++ dbgc("rate --> rcw support div %d\n", SND_SOC_FSBW_REAL(bfs_)); ++ return bfs_; ++ } ++} ++ ++/* changes a bitclk multiplier mask to a divider mask */ ++static u64 soc_bfs_rate_to_div(u64 bfs, int rate, unsigned int mclk, ++ unsigned int pcmfmt, unsigned int chn) ++{ ++ unsigned int bfs_ = rate * bfs; ++ int size = snd_pcm_format_physical_width(pcmfmt), min = 0; ++ ++ if (size <= 0) ++ return 0; ++ ++ /* the minimum bit clock that has enough bandwidth */ ++ min = size * rate * chn; ++ dbgc("rate --> div min bclk %d with mclk %d\n", min, mclk); ++ ++ if (bfs_ < min) ++ return 0; ++ else { ++ bfs_ = SND_SOC_FSBW(mclk/bfs_); ++ dbgc("rate --> div support div %d\n", SND_SOC_FSBD_REAL(bfs_)); ++ return bfs_; ++ } ++} ++ ++/* Matches codec DAI and SoC CPU DAI hardware parameters */ ++static int soc_hw_match_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_dai_mode *codec_dai_mode = NULL; ++ struct snd_soc_dai_mode *cpu_dai_mode = NULL; ++ struct snd_soc_clock_info clk_info; ++ unsigned int fs, mclk, rate = params_rate(params), ++ chn, j, k, cpu_bclk, codec_bclk, pcmrate; ++ u16 fmt = 0; ++ u64 codec_bfs, cpu_bfs; ++ ++ dbg("asoc: match version %s\n", SND_SOC_VERSION); ++ clk_info.rate = rate; ++ pcmrate = soc_get_rate_format(rate); ++ ++ /* try and find a match from the codec and cpu DAI capabilities */ ++ for (j = 0; j < rtd->codec_dai->caps.num_modes; j++) { ++ for (k = 0; k < rtd->cpu_dai->caps.num_modes; k++) { ++ codec_dai_mode = &rtd->codec_dai->caps.mode[j]; ++ cpu_dai_mode = &rtd->cpu_dai->caps.mode[k]; ++ ++ if (!(codec_dai_mode->pcmrate & cpu_dai_mode->pcmrate & ++ pcmrate)) { ++ dbgc("asoc: DAI[%d:%d] failed to match rate\n", j, k); ++ continue; ++ } ++ ++ fmt = codec_dai_mode->fmt & cpu_dai_mode->fmt; ++ if (!(fmt & SND_SOC_DAIFMT_FORMAT_MASK)) { ++ dbgc("asoc: DAI[%d:%d] failed to match format\n", j, k); ++ continue; ++ } ++ ++ if (!(fmt & SND_SOC_DAIFMT_CLOCK_MASK)) { ++ dbgc("asoc: DAI[%d:%d] failed to match clock masters\n", ++ j, k); ++ continue; ++ } ++ ++ if (!(fmt & SND_SOC_DAIFMT_INV_MASK)) { ++ dbgc("asoc: DAI[%d:%d] failed to match invert\n", j, k); ++ continue; ++ } ++ ++ if (!(codec_dai_mode->pcmfmt & cpu_dai_mode->pcmfmt)) { ++ dbgc("asoc: DAI[%d:%d] failed to match pcm format\n", j, k); ++ continue; ++ } ++ ++ if (!(codec_dai_mode->pcmdir & cpu_dai_mode->pcmdir)) { ++ dbgc("asoc: DAI[%d:%d] failed to match direction\n", j, k); ++ continue; ++ } ++ ++ /* todo - still need to add tdm selection */ ++ rtd->cpu_dai->dai_runtime.fmt = ++ rtd->codec_dai->dai_runtime.fmt = ++ 1 << (ffs(fmt & SND_SOC_DAIFMT_FORMAT_MASK) -1) | ++ 1 << (ffs(fmt & SND_SOC_DAIFMT_CLOCK_MASK) - 1) | ++ 1 << (ffs(fmt & SND_SOC_DAIFMT_INV_MASK) - 1); ++ clk_info.bclk_master = ++ rtd->cpu_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK; ++ ++ /* make sure the ratio between rate and master ++ * clock is acceptable*/ ++ fs = (cpu_dai_mode->fs & codec_dai_mode->fs); ++ if (fs == 0) { ++ dbgc("asoc: DAI[%d:%d] failed to match FS\n", j, k); ++ continue; ++ } ++ clk_info.fs = rtd->cpu_dai->dai_runtime.fs = ++ rtd->codec_dai->dai_runtime.fs = fs; ++ ++ /* calculate audio system clocking using slowest clocks possible*/ ++ mclk = soc_get_mclk(rtd, &clk_info); ++ if (mclk == 0) { ++ dbgc("asoc: DAI[%d:%d] configuration not clockable\n", j, k); ++ dbgc("asoc: rate %d fs %d master %x\n", rate, fs, ++ clk_info.bclk_master); ++ continue; ++ } ++ ++ /* calculate word size (per channel) and frame size */ ++ rtd->codec_dai->dai_runtime.pcmfmt = ++ rtd->cpu_dai->dai_runtime.pcmfmt = ++ 1 << params_format(params); ++ ++ chn = params_channels(params); ++ /* i2s always has left and right */ ++ if (params_channels(params) == 1 && ++ rtd->cpu_dai->dai_runtime.fmt & (SND_SOC_DAIFMT_I2S | ++ SND_SOC_DAIFMT_RIGHT_J | SND_SOC_DAIFMT_LEFT_J)) ++ chn <<= 1; ++ ++ /* Calculate bfs - the ratio between bitclock and the sample rate ++ * We must take into consideration the dividers and multipliers ++ * used in the codec and cpu DAI modes. We always choose the ++ * lowest possible clocks to reduce power. ++ */ ++ switch (CODEC_CPU(codec_dai_mode->flags, cpu_dai_mode->flags)) { ++ case CODEC_CPU(SND_SOC_DAI_BFS_DIV, SND_SOC_DAI_BFS_DIV): ++ /* cpu & codec bfs dividers */ ++ rtd->cpu_dai->dai_runtime.bfs = ++ rtd->codec_dai->dai_runtime.bfs = ++ 1 << (fls(codec_dai_mode->bfs & cpu_dai_mode->bfs) - 1); ++ break; ++ case CODEC_CPU(SND_SOC_DAI_BFS_DIV, SND_SOC_DAI_BFS_RCW): ++ /* normalise bfs codec divider & cpu rcw mult */ ++ codec_bfs = soc_bfs_div_to_rcw(codec_dai_mode->bfs, rate, ++ mclk, rtd->codec_dai->dai_runtime.pcmfmt, chn); ++ rtd->cpu_dai->dai_runtime.bfs = ++ 1 << (ffs(codec_bfs & cpu_dai_mode->bfs) - 1); ++ cpu_bfs = soc_bfs_rcw_to_div(cpu_dai_mode->bfs, rate, mclk, ++ rtd->codec_dai->dai_runtime.pcmfmt, chn); ++ rtd->codec_dai->dai_runtime.bfs = ++ 1 << (fls(codec_dai_mode->bfs & cpu_bfs) - 1); ++ break; ++ case CODEC_CPU(SND_SOC_DAI_BFS_RCW, SND_SOC_DAI_BFS_DIV): ++ /* normalise bfs codec rcw mult & cpu divider */ ++ codec_bfs = soc_bfs_rcw_to_div(codec_dai_mode->bfs, rate, ++ mclk, rtd->codec_dai->dai_runtime.pcmfmt, chn); ++ rtd->cpu_dai->dai_runtime.bfs = ++ 1 << (fls(codec_bfs & cpu_dai_mode->bfs) -1); ++ cpu_bfs = soc_bfs_div_to_rcw(cpu_dai_mode->bfs, rate, mclk, ++ rtd->codec_dai->dai_runtime.pcmfmt, chn); ++ rtd->codec_dai->dai_runtime.bfs = ++ 1 << (ffs(codec_dai_mode->bfs & cpu_bfs) -1); ++ break; ++ case CODEC_CPU(SND_SOC_DAI_BFS_RCW, SND_SOC_DAI_BFS_RCW): ++ /* codec & cpu bfs rate rcw multipliers */ ++ rtd->cpu_dai->dai_runtime.bfs = ++ rtd->codec_dai->dai_runtime.bfs = ++ 1 << (ffs(codec_dai_mode->bfs & cpu_dai_mode->bfs) -1); ++ break; ++ case CODEC_CPU(SND_SOC_DAI_BFS_DIV, SND_SOC_DAI_BFS_RATE): ++ /* normalise cpu bfs rate const multiplier & codec div */ ++ cpu_bfs = soc_bfs_rate_to_div(cpu_dai_mode->bfs, rate, ++ mclk, rtd->codec_dai->dai_runtime.pcmfmt, chn); ++ if(codec_dai_mode->bfs & cpu_bfs) { ++ rtd->codec_dai->dai_runtime.bfs = cpu_bfs; ++ rtd->cpu_dai->dai_runtime.bfs = cpu_dai_mode->bfs; ++ } else ++ rtd->cpu_dai->dai_runtime.bfs = 0; ++ break; ++ case CODEC_CPU(SND_SOC_DAI_BFS_RCW, SND_SOC_DAI_BFS_RATE): ++ /* normalise cpu bfs rate const multiplier & codec rcw mult */ ++ cpu_bfs = soc_bfs_rate_to_rcw(cpu_dai_mode->bfs, rate, ++ mclk, rtd->codec_dai->dai_runtime.pcmfmt, chn); ++ if(codec_dai_mode->bfs & cpu_bfs) { ++ rtd->codec_dai->dai_runtime.bfs = cpu_bfs; ++ rtd->cpu_dai->dai_runtime.bfs = cpu_dai_mode->bfs; ++ } else ++ rtd->cpu_dai->dai_runtime.bfs = 0; ++ break; ++ case CODEC_CPU(SND_SOC_DAI_BFS_RATE, SND_SOC_DAI_BFS_RCW): ++ /* normalise cpu bfs rate rcw multiplier & codec const mult */ ++ codec_bfs = soc_bfs_rate_to_rcw(codec_dai_mode->bfs, rate, ++ mclk, rtd->codec_dai->dai_runtime.pcmfmt, chn); ++ if(cpu_dai_mode->bfs & codec_bfs) { ++ rtd->cpu_dai->dai_runtime.bfs = codec_bfs; ++ rtd->codec_dai->dai_runtime.bfs = codec_dai_mode->bfs; ++ } else ++ rtd->cpu_dai->dai_runtime.bfs = 0; ++ break; ++ case CODEC_CPU(SND_SOC_DAI_BFS_RATE, SND_SOC_DAI_BFS_DIV): ++ /* normalise cpu bfs div & codec const mult */ ++ codec_bfs = soc_bfs_rate_to_div(codec_dai_mode->bfs, rate, ++ mclk, rtd->codec_dai->dai_runtime.pcmfmt, chn); ++ if(codec_dai_mode->bfs & codec_bfs) { ++ rtd->cpu_dai->dai_runtime.bfs = codec_bfs; ++ rtd->codec_dai->dai_runtime.bfs = codec_dai_mode->bfs; ++ } else ++ rtd->cpu_dai->dai_runtime.bfs = 0; ++ break; ++ case CODEC_CPU(SND_SOC_DAI_BFS_RATE, SND_SOC_DAI_BFS_RATE): ++ /* cpu & codec constant mult */ ++ if(codec_dai_mode->bfs == cpu_dai_mode->bfs) ++ rtd->cpu_dai->dai_runtime.bfs = ++ rtd->codec_dai->dai_runtime.bfs = ++ codec_dai_mode->bfs; ++ else ++ rtd->cpu_dai->dai_runtime.bfs = ++ rtd->codec_dai->dai_runtime.bfs = 0; ++ break; ++ default: ++ if(codec_dai_mode->flags == 0) ++ printk(KERN_ERR "asoc: error missing codec DAI flags\n"); ++ else ++ printk(KERN_ERR "asoc: error missing CPU DAI flags\n"); ++ break; ++ } ++ ++ /* make sure the bit clock speed is acceptable */ ++ if (!rtd->cpu_dai->dai_runtime.bfs || ++ !rtd->codec_dai->dai_runtime.bfs) { ++ dbgc("asoc: DAI[%d:%d] failed to match BFS\n", j, k); ++ dbgc("asoc: cpu_dai %llu codec %llu\n", ++ rtd->cpu_dai->dai_runtime.bfs, ++ rtd->codec_dai->dai_runtime.bfs); ++ dbgc("asoc: mclk %d hwfmt %x\n", mclk, fmt); ++ continue; ++ } ++ ++ goto found; ++ } ++ } ++ printk(KERN_ERR "asoc: no matching DAI found between codec and CPU\n"); ++ return -EINVAL; ++ ++found: ++ /* we have matching DAI's, so complete the runtime info */ ++ rtd->codec_dai->dai_runtime.pcmrate = ++ rtd->cpu_dai->dai_runtime.pcmrate = ++ soc_get_rate_format(rate); ++ ++ rtd->codec_dai->dai_runtime.priv = codec_dai_mode->priv; ++ rtd->cpu_dai->dai_runtime.priv = cpu_dai_mode->priv; ++ rtd->codec_dai->dai_runtime.flags = codec_dai_mode->flags; ++ rtd->cpu_dai->dai_runtime.flags = cpu_dai_mode->flags; ++ ++ /* for debug atm */ ++ dbg("asoc: DAI[%d:%d] Match OK\n", j, k); ++ if (rtd->codec_dai->dai_runtime.flags == SND_SOC_DAI_BFS_DIV) { ++ codec_bclk = (rtd->codec_dai->dai_runtime.fs * params_rate(params)) / ++ SND_SOC_FSBD_REAL(rtd->codec_dai->dai_runtime.bfs); ++ dbg("asoc: codec fs %d mclk %d bfs div %d bclk %d\n", ++ rtd->codec_dai->dai_runtime.fs, mclk, ++ SND_SOC_FSBD_REAL(rtd->codec_dai->dai_runtime.bfs), codec_bclk); ++ } else if(rtd->codec_dai->dai_runtime.flags == SND_SOC_DAI_BFS_RATE) { ++ codec_bclk = params_rate(params) * rtd->codec_dai->dai_runtime.bfs; ++ dbg("asoc: codec fs %d mclk %d bfs rate mult %llu bclk %d\n", ++ rtd->codec_dai->dai_runtime.fs, mclk, ++ rtd->codec_dai->dai_runtime.bfs, codec_bclk); ++ } else if (rtd->cpu_dai->dai_runtime.flags == SND_SOC_DAI_BFS_RCW) { ++ codec_bclk = params_rate(params) * params_channels(params) * ++ snd_pcm_format_physical_width(rtd->codec_dai->dai_runtime.pcmfmt) * ++ SND_SOC_FSBW_REAL(rtd->codec_dai->dai_runtime.bfs); ++ dbg("asoc: codec fs %d mclk %d bfs rcw mult %d bclk %d\n", ++ rtd->codec_dai->dai_runtime.fs, mclk, ++ SND_SOC_FSBW_REAL(rtd->codec_dai->dai_runtime.bfs), codec_bclk); ++ } else ++ codec_bclk = 0; ++ ++ if (rtd->cpu_dai->dai_runtime.flags == SND_SOC_DAI_BFS_DIV) { ++ cpu_bclk = (rtd->cpu_dai->dai_runtime.fs * params_rate(params)) / ++ SND_SOC_FSBD_REAL(rtd->cpu_dai->dai_runtime.bfs); ++ dbg("asoc: cpu fs %d mclk %d bfs div %d bclk %d\n", ++ rtd->cpu_dai->dai_runtime.fs, mclk, ++ SND_SOC_FSBD_REAL(rtd->cpu_dai->dai_runtime.bfs), cpu_bclk); ++ } else if (rtd->cpu_dai->dai_runtime.flags == SND_SOC_DAI_BFS_RATE) { ++ cpu_bclk = params_rate(params) * rtd->cpu_dai->dai_runtime.bfs; ++ dbg("asoc: cpu fs %d mclk %d bfs rate mult %llu bclk %d\n", ++ rtd->cpu_dai->dai_runtime.fs, mclk, ++ rtd->cpu_dai->dai_runtime.bfs, cpu_bclk); ++ } else if (rtd->cpu_dai->dai_runtime.flags == SND_SOC_DAI_BFS_RCW) { ++ cpu_bclk = params_rate(params) * params_channels(params) * ++ snd_pcm_format_physical_width(rtd->cpu_dai->dai_runtime.pcmfmt) * ++ SND_SOC_FSBW_REAL(rtd->cpu_dai->dai_runtime.bfs); ++ dbg("asoc: cpu fs %d mclk %d bfs mult rcw %d bclk %d\n", ++ rtd->cpu_dai->dai_runtime.fs, mclk, ++ SND_SOC_FSBW_REAL(rtd->cpu_dai->dai_runtime.bfs), cpu_bclk); ++ } else ++ cpu_bclk = 0; ++ ++ /* ++ * Check we have matching bitclocks. If we don't then it means the ++ * sysclock returned by either the codec or cpu DAI (selected by the ++ * machine sysclock function) is wrong compared with the supported DAI ++ * modes for the codec or cpu DAI. ++ */ ++ if (cpu_bclk != codec_bclk && cpu_bclk){ ++ printk(KERN_ERR ++ "asoc: codec and cpu bitclocks differ, audio may be wrong speed\n" ++ ); ++ printk(KERN_ERR "asoc: codec %d != cpu %d\n", codec_bclk, cpu_bclk); ++ } ++ ++ switch(rtd->cpu_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK) { ++ case SND_SOC_DAIFMT_CBM_CFM: ++ dbg("asoc: DAI codec BCLK master, LRC master\n"); ++ break; ++ case SND_SOC_DAIFMT_CBS_CFM: ++ dbg("asoc: DAI codec BCLK slave, LRC master\n"); ++ break; ++ case SND_SOC_DAIFMT_CBM_CFS: ++ dbg("asoc: DAI codec BCLK master, LRC slave\n"); ++ break; ++ case SND_SOC_DAIFMT_CBS_CFS: ++ dbg("asoc: DAI codec BCLK slave, LRC slave\n"); ++ break; ++ } ++ dbg("asoc: mode %x, invert %x\n", ++ rtd->cpu_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK, ++ rtd->cpu_dai->dai_runtime.fmt & SND_SOC_DAIFMT_INV_MASK); ++ dbg("asoc: audio rate %d chn %d fmt %x\n", params_rate(params), ++ params_channels(params), params_format(params)); ++ ++ return 0; ++} ++ ++static inline u32 get_rates(struct snd_soc_dai_mode *modes, int nmodes) ++{ ++ int i; ++ u32 rates = 0; ++ ++ for(i = 0; i < nmodes; i++) ++ rates |= modes[i].pcmrate; ++ ++ return rates; ++} ++ ++static inline u64 get_formats(struct snd_soc_dai_mode *modes, int nmodes) ++{ ++ int i; ++ u64 formats = 0; ++ ++ for(i = 0; i < nmodes; i++) ++ formats |= modes[i].pcmfmt; ++ ++ return formats; ++} ++ ++/* ++ * Called by ALSA when a PCM substream is opened, the runtime->hw record is ++ * then initialized and any private data can be allocated. This also calls ++ * startup for the cpu DAI, platform, machine and codec DAI. ++ */ ++static int soc_pcm_open(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct snd_soc_machine *machine = socdev->machine; ++ struct snd_soc_platform *platform = socdev->platform; ++ struct snd_soc_codec_dai *codec_dai = rtd->codec_dai; ++ struct snd_soc_cpu_dai *cpu_dai = rtd->cpu_dai; ++ int ret = 0; ++ ++ mutex_lock(&pcm_mutex); ++ ++ /* startup the audio subsystem */ ++ if (rtd->cpu_dai->ops.startup) { ++ ret = rtd->cpu_dai->ops.startup(substream); ++ if (ret < 0) { ++ printk(KERN_ERR "asoc: can't open interface %s\n", ++ rtd->cpu_dai->name); ++ goto out; ++ } ++ } ++ ++ if (platform->pcm_ops->open) { ++ ret = platform->pcm_ops->open(substream); ++ if (ret < 0) { ++ printk(KERN_ERR "asoc: can't open platform %s\n", platform->name); ++ goto platform_err; ++ } ++ } ++ ++ if (machine->ops && machine->ops->startup) { ++ ret = machine->ops->startup(substream); ++ if (ret < 0) { ++ printk(KERN_ERR "asoc: %s startup failed\n", machine->name); ++ goto machine_err; ++ } ++ } ++ ++ if (rtd->codec_dai->ops.startup) { ++ ret = rtd->codec_dai->ops.startup(substream); ++ if (ret < 0) { ++ printk(KERN_ERR "asoc: can't open codec %s\n", ++ rtd->codec_dai->name); ++ goto codec_dai_err; ++ } ++ } ++ ++ /* create runtime params from DMA, codec and cpu DAI */ ++ if (runtime->hw.rates) ++ runtime->hw.rates &= ++ get_rates(codec_dai->caps.mode, codec_dai->caps.num_modes) & ++ get_rates(cpu_dai->caps.mode, cpu_dai->caps.num_modes); ++ else ++ runtime->hw.rates = ++ get_rates(codec_dai->caps.mode, codec_dai->caps.num_modes) & ++ get_rates(cpu_dai->caps.mode, cpu_dai->caps.num_modes); ++ if (runtime->hw.formats) ++ runtime->hw.formats &= ++ get_formats(codec_dai->caps.mode, codec_dai->caps.num_modes) & ++ get_formats(cpu_dai->caps.mode, cpu_dai->caps.num_modes); ++ else ++ runtime->hw.formats = ++ get_formats(codec_dai->caps.mode, codec_dai->caps.num_modes) & ++ get_formats(cpu_dai->caps.mode, cpu_dai->caps.num_modes); ++ ++ /* Check that the codec and cpu DAI's are compatible */ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { ++ runtime->hw.rate_min = ++ max(rtd->codec_dai->playback.rate_min, ++ rtd->cpu_dai->playback.rate_min); ++ runtime->hw.rate_max = ++ min(rtd->codec_dai->playback.rate_max, ++ rtd->cpu_dai->playback.rate_max); ++ runtime->hw.channels_min = ++ max(rtd->codec_dai->playback.channels_min, ++ rtd->cpu_dai->playback.channels_min); ++ runtime->hw.channels_max = ++ min(rtd->codec_dai->playback.channels_max, ++ rtd->cpu_dai->playback.channels_max); ++ } else { ++ runtime->hw.rate_min = ++ max(rtd->codec_dai->capture.rate_min, ++ rtd->cpu_dai->capture.rate_min); ++ runtime->hw.rate_max = ++ min(rtd->codec_dai->capture.rate_max, ++ rtd->cpu_dai->capture.rate_max); ++ runtime->hw.channels_min = ++ max(rtd->codec_dai->capture.channels_min, ++ rtd->cpu_dai->capture.channels_min); ++ runtime->hw.channels_max = ++ min(rtd->codec_dai->capture.channels_max, ++ rtd->cpu_dai->capture.channels_max); ++ } ++ ++ snd_pcm_limit_hw_rates(runtime); ++ if (!runtime->hw.rates) { ++ printk(KERN_ERR "asoc: %s <-> %s No matching rates\n", ++ rtd->codec_dai->name, rtd->cpu_dai->name); ++ goto codec_dai_err; ++ } ++ if (!runtime->hw.formats) { ++ printk(KERN_ERR "asoc: %s <-> %s No matching formats\n", ++ rtd->codec_dai->name, rtd->cpu_dai->name); ++ goto codec_dai_err; ++ } ++ if (!runtime->hw.channels_min || !runtime->hw.channels_max) { ++ printk(KERN_ERR "asoc: %s <-> %s No matching channels\n", ++ rtd->codec_dai->name, rtd->cpu_dai->name); ++ goto codec_dai_err; ++ } ++ ++ dbg("asoc: %s <-> %s info:\n", rtd->codec_dai->name, rtd->cpu_dai->name); ++ dbg("asoc: rate mask 0x%x\n", runtime->hw.rates); ++ dbg("asoc: min ch %d max ch %d\n", runtime->hw.channels_min, ++ runtime->hw.channels_max); ++ dbg("asoc: min rate %d max rate %d\n", runtime->hw.rate_min, ++ runtime->hw.rate_max); ++ ++ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ rtd->cpu_dai->playback.active = rtd->codec_dai->playback.active = 1; ++ else ++ rtd->cpu_dai->capture.active = rtd->codec_dai->capture.active = 1; ++ rtd->cpu_dai->active = rtd->codec_dai->active = 1; ++ rtd->cpu_dai->runtime = runtime; ++ socdev->codec->active++; ++ mutex_unlock(&pcm_mutex); ++ return 0; ++ ++codec_dai_err: ++ if (machine->ops && machine->ops->shutdown) ++ machine->ops->shutdown(substream); ++ ++machine_err: ++ if (platform->pcm_ops->close) ++ platform->pcm_ops->close(substream); ++ ++platform_err: ++ if (rtd->cpu_dai->ops.shutdown) ++ rtd->cpu_dai->ops.shutdown(substream); ++out: ++ mutex_unlock(&pcm_mutex); ++ return ret; ++} ++ ++/* ++ * Power down the audio subsytem pmdown_time msecs after close is called. ++ * This is to ensure there are no pops or clicks in between any music tracks ++ * due to DAPM power cycling. ++ */ ++static void close_delayed_work(void *data) ++{ ++ struct snd_soc_device *socdev = data; ++ struct snd_soc_codec *codec = socdev->codec; ++ struct snd_soc_codec_dai *codec_dai; ++ int i; ++ ++ mutex_lock(&pcm_mutex); ++ for(i = 0; i < codec->num_dai; i++) { ++ codec_dai = &codec->dai[i]; ++ ++ dbg("pop wq checking: %s status: %s waiting: %s\n", ++ codec_dai->playback.stream_name, ++ codec_dai->playback.active ? "active" : "inactive", ++ codec_dai->pop_wait ? "yes" : "no"); ++ ++ /* are we waiting on this codec DAI stream */ ++ if (codec_dai->pop_wait == 1) { ++ ++ codec_dai->pop_wait = 0; ++ snd_soc_dapm_stream_event(codec, codec_dai->playback.stream_name, ++ SND_SOC_DAPM_STREAM_STOP); ++ ++ /* power down the codec power domain if no longer active */ ++ if (codec->active == 0) { ++ dbg("pop wq D3 %s %s\n", codec->name, ++ codec_dai->playback.stream_name); ++ if (codec->dapm_event) ++ codec->dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ } ++ } ++ } ++ mutex_unlock(&pcm_mutex); ++} ++ ++/* ++ * Called by ALSA when a PCM substream is closed. Private data can be ++ * freed here. The cpu DAI, codec DAI, machine and platform are also ++ * shutdown. ++ */ ++static int soc_codec_close(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_machine *machine = socdev->machine; ++ struct snd_soc_platform *platform = socdev->platform; ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ mutex_lock(&pcm_mutex); ++ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ rtd->cpu_dai->playback.active = rtd->codec_dai->playback.active = 0; ++ else ++ rtd->cpu_dai->capture.active = rtd->codec_dai->capture.active = 0; ++ ++ if (rtd->codec_dai->playback.active == 0 && ++ rtd->codec_dai->capture.active == 0) { ++ rtd->cpu_dai->active = rtd->codec_dai->active = 0; ++ } ++ codec->active--; ++ ++ if (rtd->cpu_dai->ops.shutdown) ++ rtd->cpu_dai->ops.shutdown(substream); ++ ++ if (rtd->codec_dai->ops.shutdown) ++ rtd->codec_dai->ops.shutdown(substream); ++ ++ if (machine->ops && machine->ops->shutdown) ++ machine->ops->shutdown(substream); ++ ++ if (platform->pcm_ops->close) ++ platform->pcm_ops->close(substream); ++ rtd->cpu_dai->runtime = NULL; ++ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { ++ /* start delayed pop wq here for playback streams */ ++ rtd->codec_dai->pop_wait = 1; ++ queue_delayed_work(soc_workq, &soc_stream_work, ++ msecs_to_jiffies(pmdown_time)); ++ } else { ++ /* capture streams can be powered down now */ ++ snd_soc_dapm_stream_event(codec, rtd->codec_dai->capture.stream_name, ++ SND_SOC_DAPM_STREAM_STOP); ++ ++ if (codec->active == 0 && rtd->codec_dai->pop_wait == 0){ ++ if (codec->dapm_event) ++ codec->dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ } ++ } ++ ++ mutex_unlock(&pcm_mutex); ++ return 0; ++} ++ ++/* ++ * Called by ALSA when the PCM substream is prepared, can set format, sample ++ * rate, etc. This function is non atomic and can be called multiple times, ++ * it can refer to the runtime info. ++ */ ++static int soc_pcm_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_platform *platform = socdev->platform; ++ struct snd_soc_codec *codec = socdev->codec; ++ int ret = 0; ++ ++ mutex_lock(&pcm_mutex); ++ if (platform->pcm_ops->prepare) { ++ ret = platform->pcm_ops->prepare(substream); ++ if (ret < 0) { ++ printk(KERN_ERR "asoc: platform prepare error\n"); ++ goto out; ++ } ++ } ++ ++ if (rtd->codec_dai->ops.prepare) { ++ ret = rtd->codec_dai->ops.prepare(substream); ++ if (ret < 0) { ++ printk(KERN_ERR "asoc: codec DAI prepare error\n"); ++ goto out; ++ } ++ } ++ ++ if (rtd->cpu_dai->ops.prepare) ++ ret = rtd->cpu_dai->ops.prepare(substream); ++ ++ /* we only want to start a DAPM playback stream if we are not waiting ++ * on an existing one stopping */ ++ if (rtd->codec_dai->pop_wait) { ++ /* we are waiting for the delayed work to start */ ++ if (substream->stream == SNDRV_PCM_STREAM_CAPTURE) ++ snd_soc_dapm_stream_event(codec, ++ rtd->codec_dai->capture.stream_name, ++ SND_SOC_DAPM_STREAM_START); ++ else { ++ rtd->codec_dai->pop_wait = 0; ++ cancel_delayed_work(&soc_stream_work); ++ if (rtd->codec_dai->digital_mute) ++ rtd->codec_dai->digital_mute(codec, rtd->codec_dai, 0); ++ } ++ } else { ++ /* no delayed work - do we need to power up codec */ ++ if (codec->dapm_state != SNDRV_CTL_POWER_D0) { ++ ++ if (codec->dapm_event) ++ codec->dapm_event(codec, SNDRV_CTL_POWER_D1); ++ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ snd_soc_dapm_stream_event(codec, ++ rtd->codec_dai->playback.stream_name, ++ SND_SOC_DAPM_STREAM_START); ++ else ++ snd_soc_dapm_stream_event(codec, ++ rtd->codec_dai->capture.stream_name, ++ SND_SOC_DAPM_STREAM_START); ++ ++ if (codec->dapm_event) ++ codec->dapm_event(codec, SNDRV_CTL_POWER_D0); ++ if (rtd->codec_dai->digital_mute) ++ rtd->codec_dai->digital_mute(codec, rtd->codec_dai, 0); ++ ++ } else { ++ /* codec already powered - power on widgets */ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ snd_soc_dapm_stream_event(codec, ++ rtd->codec_dai->playback.stream_name, ++ SND_SOC_DAPM_STREAM_START); ++ else ++ snd_soc_dapm_stream_event(codec, ++ rtd->codec_dai->capture.stream_name, ++ SND_SOC_DAPM_STREAM_START); ++ if (rtd->codec_dai->digital_mute) ++ rtd->codec_dai->digital_mute(codec, rtd->codec_dai, 0); ++ } ++ } ++ ++out: ++ mutex_unlock(&pcm_mutex); ++ return ret; ++} ++ ++/* ++ * Called by ALSA when the hardware params are set by application. This ++ * function can also be called multiple times and can allocate buffers ++ * (using snd_pcm_lib_* ). It's non-atomic. ++ */ ++static int soc_pcm_hw_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_platform *platform = socdev->platform; ++ struct snd_soc_machine *machine = socdev->machine; ++ int ret = 0; ++ ++ mutex_lock(&pcm_mutex); ++ ++ /* we don't need to match any AC97 params */ ++ if (rtd->cpu_dai->type != SND_SOC_DAI_AC97) { ++ ret = soc_hw_match_params(substream, params); ++ if (ret < 0) ++ goto out; ++ } else { ++ struct snd_soc_clock_info clk_info; ++ clk_info.rate = params_rate(params); ++ ret = soc_get_mclk(rtd, &clk_info); ++ if (ret < 0) ++ goto out; ++ } ++ ++ if (rtd->codec_dai->ops.hw_params) { ++ ret = rtd->codec_dai->ops.hw_params(substream, params); ++ if (ret < 0) { ++ printk(KERN_ERR "asoc: can't set codec %s hw params\n", ++ rtd->codec_dai->name); ++ goto out; ++ } ++ } ++ ++ if (rtd->cpu_dai->ops.hw_params) { ++ ret = rtd->cpu_dai->ops.hw_params(substream, params); ++ if (ret < 0) { ++ printk(KERN_ERR "asoc: can't set interface %s hw params\n", ++ rtd->cpu_dai->name); ++ goto interface_err; ++ } ++ } ++ ++ if (platform->pcm_ops->hw_params) { ++ ret = platform->pcm_ops->hw_params(substream, params); ++ if (ret < 0) { ++ printk(KERN_ERR "asoc: can't set platform %s hw params\n", ++ platform->name); ++ goto platform_err; ++ } ++ } ++ ++ if (machine->ops && machine->ops->hw_params) { ++ ret = machine->ops->hw_params(substream, params); ++ if (ret < 0) { ++ printk(KERN_ERR "asoc: machine hw_params failed\n"); ++ goto machine_err; ++ } ++ } ++ ++out: ++ mutex_unlock(&pcm_mutex); ++ return ret; ++ ++machine_err: ++ if (platform->pcm_ops->hw_free) ++ platform->pcm_ops->hw_free(substream); ++ ++platform_err: ++ if (rtd->cpu_dai->ops.hw_free) ++ rtd->cpu_dai->ops.hw_free(substream); ++ ++interface_err: ++ if (rtd->codec_dai->ops.hw_free) ++ rtd->codec_dai->ops.hw_free(substream); ++ ++ mutex_unlock(&pcm_mutex); ++ return ret; ++} ++ ++/* ++ * Free's resources allocated by hw_params, can be called multiple times ++ */ ++static int soc_pcm_hw_free(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_platform *platform = socdev->platform; ++ struct snd_soc_codec *codec = socdev->codec; ++ struct snd_soc_machine *machine = socdev->machine; ++ ++ mutex_lock(&pcm_mutex); ++ ++ /* apply codec digital mute */ ++ if (!codec->active && rtd->codec_dai->digital_mute) ++ rtd->codec_dai->digital_mute(codec, rtd->codec_dai, 1); ++ ++ /* free any machine hw params */ ++ if (machine->ops && machine->ops->hw_free) ++ machine->ops->hw_free(substream); ++ ++ /* free any DMA resources */ ++ if (platform->pcm_ops->hw_free) ++ platform->pcm_ops->hw_free(substream); ++ ++ /* now free hw params for the DAI's */ ++ if (rtd->codec_dai->ops.hw_free) ++ rtd->codec_dai->ops.hw_free(substream); ++ ++ if (rtd->cpu_dai->ops.hw_free) ++ rtd->cpu_dai->ops.hw_free(substream); ++ ++ mutex_unlock(&pcm_mutex); ++ return 0; ++} ++ ++static int soc_pcm_trigger(struct snd_pcm_substream *substream, int cmd) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_platform *platform = socdev->platform; ++ int ret; ++ ++ if (rtd->codec_dai->ops.trigger) { ++ ret = rtd->codec_dai->ops.trigger(substream, cmd); ++ if (ret < 0) ++ return ret; ++ } ++ ++ if (platform->pcm_ops->trigger) { ++ ret = platform->pcm_ops->trigger(substream, cmd); ++ if (ret < 0) ++ return ret; ++ } ++ ++ if (rtd->cpu_dai->ops.trigger) { ++ ret = rtd->cpu_dai->ops.trigger(substream, cmd); ++ if (ret < 0) ++ return ret; ++ } ++ return 0; ++} ++ ++/* ASoC PCM operations */ ++static struct snd_pcm_ops soc_pcm_ops = { ++ .open = soc_pcm_open, ++ .close = soc_codec_close, ++ .hw_params = soc_pcm_hw_params, ++ .hw_free = soc_pcm_hw_free, ++ .prepare = soc_pcm_prepare, ++ .trigger = soc_pcm_trigger, ++}; ++ ++#ifdef CONFIG_PM ++/* powers down audio subsystem for suspend */ ++static int soc_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_machine *machine = socdev->machine; ++ struct snd_soc_platform *platform = socdev->platform; ++ struct snd_soc_codec_device *codec_dev = socdev->codec_dev; ++ struct snd_soc_codec *codec = socdev->codec; ++ int i; ++ ++ /* mute any active DAC's */ ++ for(i = 0; i < machine->num_links; i++) { ++ struct snd_soc_codec_dai *dai = machine->dai_link[i].codec_dai; ++ if (dai->digital_mute && dai->playback.active) ++ dai->digital_mute(codec, dai, 1); ++ } ++ ++ if (machine->suspend_pre) ++ machine->suspend_pre(pdev, state); ++ ++ for(i = 0; i < machine->num_links; i++) { ++ struct snd_soc_cpu_dai *cpu_dai = machine->dai_link[i].cpu_dai; ++ if (cpu_dai->suspend && cpu_dai->type != SND_SOC_DAI_AC97) ++ cpu_dai->suspend(pdev, cpu_dai); ++ if (platform->suspend) ++ platform->suspend(pdev, cpu_dai); ++ } ++ ++ /* close any waiting streams and save state */ ++ flush_workqueue(soc_workq); ++ codec->suspend_dapm_state = codec->dapm_state; ++ ++ for(i = 0; i < codec->num_dai; i++) { ++ char *stream = codec->dai[i].playback.stream_name; ++ if (stream != NULL) ++ snd_soc_dapm_stream_event(codec, stream, ++ SND_SOC_DAPM_STREAM_SUSPEND); ++ stream = codec->dai[i].capture.stream_name; ++ if (stream != NULL) ++ snd_soc_dapm_stream_event(codec, stream, ++ SND_SOC_DAPM_STREAM_SUSPEND); ++ } ++ ++ if (codec_dev->suspend) ++ codec_dev->suspend(pdev, state); ++ ++ for(i = 0; i < machine->num_links; i++) { ++ struct snd_soc_cpu_dai *cpu_dai = machine->dai_link[i].cpu_dai; ++ if (cpu_dai->suspend && cpu_dai->type == SND_SOC_DAI_AC97) ++ cpu_dai->suspend(pdev, cpu_dai); ++ } ++ ++ if (machine->suspend_post) ++ machine->suspend_post(pdev, state); ++ ++ return 0; ++} ++ ++/* powers up audio subsystem after a suspend */ ++static int soc_resume(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_machine *machine = socdev->machine; ++ struct snd_soc_platform *platform = socdev->platform; ++ struct snd_soc_codec_device *codec_dev = socdev->codec_dev; ++ struct snd_soc_codec *codec = socdev->codec; ++ int i; ++ ++ if (machine->resume_pre) ++ machine->resume_pre(pdev); ++ ++ for(i = 0; i < machine->num_links; i++) { ++ struct snd_soc_cpu_dai *cpu_dai = machine->dai_link[i].cpu_dai; ++ if (cpu_dai->resume && cpu_dai->type == SND_SOC_DAI_AC97) ++ cpu_dai->resume(pdev, cpu_dai); ++ } ++ ++ if (codec_dev->resume) ++ codec_dev->resume(pdev); ++ ++ for(i = 0; i < codec->num_dai; i++) { ++ char* stream = codec->dai[i].playback.stream_name; ++ if (stream != NULL) ++ snd_soc_dapm_stream_event(codec, stream, ++ SND_SOC_DAPM_STREAM_RESUME); ++ stream = codec->dai[i].capture.stream_name; ++ if (stream != NULL) ++ snd_soc_dapm_stream_event(codec, stream, ++ SND_SOC_DAPM_STREAM_RESUME); ++ } ++ ++ /* unmute any active DAC's */ ++ for(i = 0; i < machine->num_links; i++) { ++ struct snd_soc_codec_dai *dai = machine->dai_link[i].codec_dai; ++ if (dai->digital_mute && dai->playback.active) ++ dai->digital_mute(codec, dai, 0); ++ } ++ ++ for(i = 0; i < machine->num_links; i++) { ++ struct snd_soc_cpu_dai *cpu_dai = machine->dai_link[i].cpu_dai; ++ if (cpu_dai->resume && cpu_dai->type != SND_SOC_DAI_AC97) ++ cpu_dai->resume(pdev, cpu_dai); ++ if (platform->resume) ++ platform->resume(pdev, cpu_dai); ++ } ++ ++ if (machine->resume_post) ++ machine->resume_post(pdev); ++ ++ return 0; ++} ++ ++#else ++#define soc_suspend NULL ++#define soc_resume NULL ++#endif ++ ++/* probes a new socdev */ ++static int soc_probe(struct platform_device *pdev) ++{ ++ int ret = 0, i; ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_machine *machine = socdev->machine; ++ struct snd_soc_platform *platform = socdev->platform; ++ struct snd_soc_codec_device *codec_dev = socdev->codec_dev; ++ ++ if (machine->probe) { ++ ret = machine->probe(pdev); ++ if(ret < 0) ++ return ret; ++ } ++ ++ for (i = 0; i < machine->num_links; i++) { ++ struct snd_soc_cpu_dai *cpu_dai = machine->dai_link[i].cpu_dai; ++ if (cpu_dai->probe) { ++ ret = cpu_dai->probe(pdev); ++ if(ret < 0) ++ goto cpu_dai_err; ++ } ++ } ++ ++ if (codec_dev->probe) { ++ ret = codec_dev->probe(pdev); ++ if(ret < 0) ++ goto cpu_dai_err; ++ } ++ ++ if (platform->probe) { ++ ret = platform->probe(pdev); ++ if(ret < 0) ++ goto platform_err; ++ } ++ ++ /* DAPM stream work */ ++ soc_workq = create_workqueue("kdapm"); ++ if (soc_workq == NULL) ++ goto work_err; ++ INIT_WORK(&soc_stream_work, close_delayed_work, socdev); ++ return 0; ++ ++work_err: ++ if (platform->remove) ++ platform->remove(pdev); ++ ++platform_err: ++ if (codec_dev->remove) ++ codec_dev->remove(pdev); ++ ++cpu_dai_err: ++ for (i--; i > 0; i--) { ++ struct snd_soc_cpu_dai *cpu_dai = machine->dai_link[i].cpu_dai; ++ if (cpu_dai->remove) ++ cpu_dai->remove(pdev); ++ } ++ ++ if (machine->remove) ++ machine->remove(pdev); ++ ++ return ret; ++} ++ ++/* removes a socdev */ ++static int soc_remove(struct platform_device *pdev) ++{ ++ int i; ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_machine *machine = socdev->machine; ++ struct snd_soc_platform *platform = socdev->platform; ++ struct snd_soc_codec_device *codec_dev = socdev->codec_dev; ++ ++ if (soc_workq) ++ destroy_workqueue(soc_workq); ++ ++ if (platform->remove) ++ platform->remove(pdev); ++ ++ if (codec_dev->remove) ++ codec_dev->remove(pdev); ++ ++ for (i = 0; i < machine->num_links; i++) { ++ struct snd_soc_cpu_dai *cpu_dai = machine->dai_link[i].cpu_dai; ++ if (cpu_dai->remove) ++ cpu_dai->remove(pdev); ++ } ++ ++ if (machine->remove) ++ machine->remove(pdev); ++ ++ return 0; ++} ++ ++/* ASoC platform driver */ ++static struct platform_driver soc_driver = { ++ .driver = { ++ .name = "soc-audio", ++ }, ++ .probe = soc_probe, ++ .remove = soc_remove, ++ .suspend = soc_suspend, ++ .resume = soc_resume, ++}; ++ ++/* create a new pcm */ ++static int soc_new_pcm(struct snd_soc_device *socdev, ++ struct snd_soc_dai_link *dai_link, int num) ++{ ++ struct snd_soc_codec *codec = socdev->codec; ++ struct snd_soc_codec_dai *codec_dai = dai_link->codec_dai; ++ struct snd_soc_cpu_dai *cpu_dai = dai_link->cpu_dai; ++ struct snd_soc_pcm_runtime *rtd; ++ struct snd_pcm *pcm; ++ char new_name[64]; ++ int ret = 0, playback = 0, capture = 0; ++ ++ rtd = kzalloc(sizeof(struct snd_soc_pcm_runtime), GFP_KERNEL); ++ if (rtd == NULL) ++ return -ENOMEM; ++ rtd->cpu_dai = cpu_dai; ++ rtd->codec_dai = codec_dai; ++ rtd->socdev = socdev; ++ ++ /* check client and interface hw capabilities */ ++ sprintf(new_name, "%s %s-%s-%d",dai_link->stream_name, codec_dai->name, ++ get_dai_name(cpu_dai->type), num); ++ ++ if (codec_dai->playback.channels_min) ++ playback = 1; ++ if (codec_dai->capture.channels_min) ++ capture = 1; ++ ++ ret = snd_pcm_new(codec->card, new_name, codec->pcm_devs++, playback, ++ capture, &pcm); ++ if (ret < 0) { ++ printk(KERN_ERR "asoc: can't create pcm for codec %s\n", codec->name); ++ kfree(rtd); ++ return ret; ++ } ++ ++ pcm->private_data = rtd; ++ soc_pcm_ops.mmap = socdev->platform->pcm_ops->mmap; ++ soc_pcm_ops.pointer = socdev->platform->pcm_ops->pointer; ++ soc_pcm_ops.ioctl = socdev->platform->pcm_ops->ioctl; ++ soc_pcm_ops.copy = socdev->platform->pcm_ops->copy; ++ soc_pcm_ops.silence = socdev->platform->pcm_ops->silence; ++ soc_pcm_ops.ack = socdev->platform->pcm_ops->ack; ++ soc_pcm_ops.page = socdev->platform->pcm_ops->page; ++ ++ if (playback) ++ snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_PLAYBACK, &soc_pcm_ops); ++ ++ if (capture) ++ snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_CAPTURE, &soc_pcm_ops); ++ ++ ret = socdev->platform->pcm_new(codec->card, codec_dai, pcm); ++ if (ret < 0) { ++ printk(KERN_ERR "asoc: platform pcm constructor failed\n"); ++ kfree(rtd); ++ return ret; ++ } ++ ++ pcm->private_free = socdev->platform->pcm_free; ++ printk(KERN_INFO "asoc: %s <-> %s mapping ok\n", codec_dai->name, ++ cpu_dai->name); ++ return ret; ++} ++ ++/* codec register dump */ ++static ssize_t codec_reg_show(struct device *dev, ++ struct device_attribute *attr, char *buf) ++{ ++ struct snd_soc_device *devdata = dev_get_drvdata(dev); ++ struct snd_soc_codec *codec = devdata->codec; ++ int i, step = 1, count = 0; ++ ++ if (!codec->reg_cache_size) ++ return 0; ++ ++ if (codec->reg_cache_step) ++ step = codec->reg_cache_step; ++ ++ count += sprintf(buf, "%s registers\n", codec->name); ++ for(i = 0; i < codec->reg_cache_size; i += step) ++ count += sprintf(buf + count, "%2x: %4x\n", i, codec->read(codec, i)); ++ ++ return count; ++} ++static DEVICE_ATTR(codec_reg, 0444, codec_reg_show, NULL); ++ ++/** ++ * snd_soc_new_ac97_codec - initailise AC97 device ++ * @codec: audio codec ++ * @ops: AC97 bus operations ++ * @num: AC97 codec number ++ * ++ * Initialises AC97 codec resources for use by ad-hoc devices only. ++ */ ++int snd_soc_new_ac97_codec(struct snd_soc_codec *codec, ++ struct snd_ac97_bus_ops *ops, int num) ++{ ++ mutex_lock(&codec->mutex); ++ ++ codec->ac97 = kzalloc(sizeof(struct snd_ac97), GFP_KERNEL); ++ if (codec->ac97 == NULL) { ++ mutex_unlock(&codec->mutex); ++ return -ENOMEM; ++ } ++ ++ codec->ac97->bus = kzalloc(sizeof(struct snd_ac97_bus), GFP_KERNEL); ++ if (codec->ac97->bus == NULL) { ++ kfree(codec->ac97); ++ codec->ac97 = NULL; ++ mutex_unlock(&codec->mutex); ++ return -ENOMEM; ++ } ++ ++ codec->ac97->bus->ops = ops; ++ codec->ac97->num = num; ++ mutex_unlock(&codec->mutex); ++ return 0; ++} ++EXPORT_SYMBOL_GPL(snd_soc_new_ac97_codec); ++ ++/** ++ * snd_soc_free_ac97_codec - free AC97 codec device ++ * @codec: audio codec ++ * ++ * Frees AC97 codec device resources. ++ */ ++void snd_soc_free_ac97_codec(struct snd_soc_codec *codec) ++{ ++ mutex_lock(&codec->mutex); ++ kfree(codec->ac97->bus); ++ kfree(codec->ac97); ++ codec->ac97 = NULL; ++ mutex_unlock(&codec->mutex); ++} ++EXPORT_SYMBOL_GPL(snd_soc_free_ac97_codec); ++ ++/** ++ * snd_soc_update_bits - update codec register bits ++ * @codec: audio codec ++ * @reg: codec register ++ * @mask: register mask ++ * @value: new value ++ * ++ * Writes new register value. ++ * ++ * Returns 1 for change else 0. ++ */ ++int snd_soc_update_bits(struct snd_soc_codec *codec, unsigned short reg, ++ unsigned short mask, unsigned short value) ++{ ++ int change; ++ unsigned short old, new; ++ ++ mutex_lock(&io_mutex); ++ old = snd_soc_read(codec, reg); ++ new = (old & ~mask) | value; ++ change = old != new; ++ if (change) ++ snd_soc_write(codec, reg, new); ++ ++ mutex_unlock(&io_mutex); ++ return change; ++} ++EXPORT_SYMBOL_GPL(snd_soc_update_bits); ++ ++/** ++ * snd_soc_test_bits - test register for change ++ * @codec: audio codec ++ * @reg: codec register ++ * @mask: register mask ++ * @value: new value ++ * ++ * Tests a register with a new value and checks if the new value is ++ * different from the old value. ++ * ++ * Returns 1 for change else 0. ++ */ ++int snd_soc_test_bits(struct snd_soc_codec *codec, unsigned short reg, ++ unsigned short mask, unsigned short value) ++{ ++ int change; ++ unsigned short old, new; ++ ++ mutex_lock(&io_mutex); ++ old = snd_soc_read(codec, reg); ++ new = (old & ~mask) | value; ++ change = old != new; ++ mutex_unlock(&io_mutex); ++ ++ return change; ++} ++EXPORT_SYMBOL_GPL(snd_soc_test_bits); ++ ++/** ++ * snd_soc_get_rate - get int sample rate ++ * @hwpcmrate: the hardware pcm rate ++ * ++ * Returns the audio rate integaer value, else 0. ++ */ ++int snd_soc_get_rate(int hwpcmrate) ++{ ++ int rate = ffs(hwpcmrate) - 1; ++ ++ if (rate > ARRAY_SIZE(rates)) ++ return 0; ++ return rates[rate]; ++} ++EXPORT_SYMBOL_GPL(snd_soc_get_rate); ++ ++/** ++ * snd_soc_new_pcms - create new sound card and pcms ++ * @socdev: the SoC audio device ++ * ++ * Create a new sound card based upon the codec and interface pcms. ++ * ++ * Returns 0 for success, else error. ++ */ ++int snd_soc_new_pcms(struct snd_soc_device *socdev, int idx, const char * xid) ++{ ++ struct snd_soc_codec *codec = socdev->codec; ++ struct snd_soc_machine *machine = socdev->machine; ++ int ret = 0, i; ++ ++ mutex_lock(&codec->mutex); ++ ++ /* register a sound card */ ++ codec->card = snd_card_new(idx, xid, codec->owner, 0); ++ if (!codec->card) { ++ printk(KERN_ERR "asoc: can't create sound card for codec %s\n", ++ codec->name); ++ mutex_unlock(&codec->mutex); ++ return -ENODEV; ++ } ++ ++ codec->card->dev = socdev->dev; ++ codec->card->private_data = codec; ++ strncpy(codec->card->driver, codec->name, sizeof(codec->card->driver)); ++ ++ /* create the pcms */ ++ for(i = 0; i < machine->num_links; i++) { ++ ret = soc_new_pcm(socdev, &machine->dai_link[i], i); ++ if (ret < 0) { ++ printk(KERN_ERR "asoc: can't create pcm %s\n", ++ machine->dai_link[i].stream_name); ++ mutex_unlock(&codec->mutex); ++ return ret; ++ } ++ } ++ ++ mutex_unlock(&codec->mutex); ++ return ret; ++} ++EXPORT_SYMBOL_GPL(snd_soc_new_pcms); ++ ++/** ++ * snd_soc_register_card - register sound card ++ * @socdev: the SoC audio device ++ * ++ * Register a SoC sound card. Also registers an AC97 device if the ++ * codec is AC97 for ad hoc devices. ++ * ++ * Returns 0 for success, else error. ++ */ ++int snd_soc_register_card(struct snd_soc_device *socdev) ++{ ++ struct snd_soc_codec *codec = socdev->codec; ++ struct snd_soc_machine *machine = socdev->machine; ++ int ret = 0, i, ac97 = 0, err = 0; ++ ++ mutex_lock(&codec->mutex); ++ for(i = 0; i < machine->num_links; i++) { ++ if (socdev->machine->dai_link[i].init) { ++ err = socdev->machine->dai_link[i].init(codec); ++ if (err < 0) { ++ printk(KERN_ERR "asoc: failed to init %s\n", ++ socdev->machine->dai_link[i].stream_name); ++ continue; ++ } ++ } ++ if (socdev->machine->dai_link[i].cpu_dai->type == SND_SOC_DAI_AC97) ++ ac97 = 1; ++ } ++ snprintf(codec->card->shortname, sizeof(codec->card->shortname), ++ "%s", machine->name); ++ snprintf(codec->card->longname, sizeof(codec->card->longname), ++ "%s (%s)", machine->name, codec->name); ++ ++ ret = snd_card_register(codec->card); ++ if (ret < 0) { ++ printk(KERN_ERR "asoc: failed to register soundcard for codec %s\n", ++ codec->name); ++ goto out; ++ } ++ ++#ifdef CONFIG_SND_SOC_AC97_BUS ++ if (ac97) { ++ ret = soc_ac97_dev_register(codec); ++ if (ret < 0) { ++ printk(KERN_ERR "asoc: AC97 device register failed\n"); ++ snd_card_free(codec->card); ++ goto out; ++ } ++ } ++#endif ++ ++ err = snd_soc_dapm_sys_add(socdev->dev); ++ if (err < 0) ++ printk(KERN_WARNING "asoc: failed to add dapm sysfs entries\n"); ++ ++ err = device_create_file(socdev->dev, &dev_attr_codec_reg); ++ if (err < 0) ++ printk(KERN_WARNING "asoc: failed to add codec sysfs entries\n"); ++out: ++ mutex_unlock(&codec->mutex); ++ return ret; ++} ++EXPORT_SYMBOL_GPL(snd_soc_register_card); ++ ++/** ++ * snd_soc_free_pcms - free sound card and pcms ++ * @socdev: the SoC audio device ++ * ++ * Frees sound card and pcms associated with the socdev. ++ * Also unregister the codec if it is an AC97 device. ++ */ ++void snd_soc_free_pcms(struct snd_soc_device *socdev) ++{ ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ mutex_lock(&codec->mutex); ++#ifdef CONFIG_SND_SOC_AC97_BUS ++ if (codec->ac97) ++ soc_ac97_dev_unregister(codec); ++#endif ++ ++ if (codec->card) ++ snd_card_free(codec->card); ++ device_remove_file(socdev->dev, &dev_attr_codec_reg); ++ mutex_unlock(&codec->mutex); ++} ++EXPORT_SYMBOL_GPL(snd_soc_free_pcms); ++ ++/** ++ * snd_soc_set_runtime_hwparams - set the runtime hardware parameters ++ * @substream: the pcm substream ++ * @hw: the hardware parameters ++ * ++ * Sets the substream runtime hardware parameters. ++ */ ++int snd_soc_set_runtime_hwparams(struct snd_pcm_substream *substream, ++ const struct snd_pcm_hardware *hw) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ runtime->hw.info = hw->info; ++ runtime->hw.formats = hw->formats; ++ runtime->hw.period_bytes_min = hw->period_bytes_min; ++ runtime->hw.period_bytes_max = hw->period_bytes_max; ++ runtime->hw.periods_min = hw->periods_min; ++ runtime->hw.periods_max = hw->periods_max; ++ runtime->hw.buffer_bytes_max = hw->buffer_bytes_max; ++ runtime->hw.fifo_size = hw->fifo_size; ++ return 0; ++} ++EXPORT_SYMBOL_GPL(snd_soc_set_runtime_hwparams); ++ ++/** ++ * snd_soc_cnew - create new control ++ * @_template: control template ++ * @data: control private data ++ * @lnng_name: control long name ++ * ++ * Create a new mixer control from a template control. ++ * ++ * Returns 0 for success, else error. ++ */ ++struct snd_kcontrol *snd_soc_cnew(const struct snd_kcontrol_new *_template, ++ void *data, char *long_name) ++{ ++ struct snd_kcontrol_new template; ++ ++ memcpy(&template, _template, sizeof(template)); ++ if (long_name) ++ template.name = long_name; ++ template.access = SNDRV_CTL_ELEM_ACCESS_READWRITE; ++ template.index = 0; ++ ++ return snd_ctl_new1(&template, data); ++} ++EXPORT_SYMBOL_GPL(snd_soc_cnew); ++ ++/** ++ * snd_soc_info_enum_double - enumerated double mixer info callback ++ * @kcontrol: mixer control ++ * @uinfo: control element information ++ * ++ * Callback to provide information about a double enumerated ++ * mixer control. ++ * ++ * Returns 0 for success. ++ */ ++int snd_soc_info_enum_double(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_info *uinfo) ++{ ++ struct soc_enum *e = (struct soc_enum *)kcontrol->private_value; ++ ++ uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED; ++ uinfo->count = e->shift_l == e->shift_r ? 1 : 2; ++ uinfo->value.enumerated.items = e->mask; ++ ++ if (uinfo->value.enumerated.item > e->mask - 1) ++ uinfo->value.enumerated.item = e->mask - 1; ++ strcpy(uinfo->value.enumerated.name, ++ e->texts[uinfo->value.enumerated.item]); ++ return 0; ++} ++EXPORT_SYMBOL_GPL(snd_soc_info_enum_double); ++ ++/** ++ * snd_soc_get_enum_double - enumerated double mixer get callback ++ * @kcontrol: mixer control ++ * @uinfo: control element information ++ * ++ * Callback to get the value of a double enumerated mixer. ++ * ++ * Returns 0 for success. ++ */ ++int snd_soc_get_enum_double(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ struct soc_enum *e = (struct soc_enum *)kcontrol->private_value; ++ unsigned short val, bitmask; ++ ++ for (bitmask = 1; bitmask < e->mask; bitmask <<= 1) ++ ; ++ val = snd_soc_read(codec, e->reg); ++ ucontrol->value.enumerated.item[0] = (val >> e->shift_l) & (bitmask - 1); ++ if (e->shift_l != e->shift_r) ++ ucontrol->value.enumerated.item[1] = ++ (val >> e->shift_r) & (bitmask - 1); ++ ++ return 0; ++} ++EXPORT_SYMBOL_GPL(snd_soc_get_enum_double); ++ ++/** ++ * snd_soc_put_enum_double - enumerated double mixer put callback ++ * @kcontrol: mixer control ++ * @uinfo: control element information ++ * ++ * Callback to set the value of a double enumerated mixer. ++ * ++ * Returns 0 for success. ++ */ ++int snd_soc_put_enum_double(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ struct soc_enum *e = (struct soc_enum *)kcontrol->private_value; ++ unsigned short val; ++ unsigned short mask, bitmask; ++ ++ for (bitmask = 1; bitmask < e->mask; bitmask <<= 1) ++ ; ++ if (ucontrol->value.enumerated.item[0] > e->mask - 1) ++ return -EINVAL; ++ val = ucontrol->value.enumerated.item[0] << e->shift_l; ++ mask = (bitmask - 1) << e->shift_l; ++ if (e->shift_l != e->shift_r) { ++ if (ucontrol->value.enumerated.item[1] > e->mask - 1) ++ return -EINVAL; ++ val |= ucontrol->value.enumerated.item[1] << e->shift_r; ++ mask |= (bitmask - 1) << e->shift_r; ++ } ++ ++ return snd_soc_update_bits(codec, e->reg, mask, val); ++} ++EXPORT_SYMBOL_GPL(snd_soc_put_enum_double); ++ ++/** ++ * snd_soc_info_enum_ext - external enumerated single mixer info callback ++ * @kcontrol: mixer control ++ * @uinfo: control element information ++ * ++ * Callback to provide information about an external enumerated ++ * single mixer. ++ * ++ * Returns 0 for success. ++ */ ++int snd_soc_info_enum_ext(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_info *uinfo) ++{ ++ struct soc_enum *e = (struct soc_enum *)kcontrol->private_value; ++ ++ uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED; ++ uinfo->count = 1; ++ uinfo->value.enumerated.items = e->mask; ++ ++ if (uinfo->value.enumerated.item > e->mask - 1) ++ uinfo->value.enumerated.item = e->mask - 1; ++ strcpy(uinfo->value.enumerated.name, ++ e->texts[uinfo->value.enumerated.item]); ++ return 0; ++} ++EXPORT_SYMBOL_GPL(snd_soc_info_enum_ext); ++ ++/** ++ * snd_soc_info_volsw_ext - external single mixer info callback ++ * @kcontrol: mixer control ++ * @uinfo: control element information ++ * ++ * Callback to provide information about a single external mixer control. ++ * ++ * Returns 0 for success. ++ */ ++int snd_soc_info_volsw_ext(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_info *uinfo) ++{ ++ int mask = kcontrol->private_value; ++ ++ uinfo->type = ++ mask == 1 ? SNDRV_CTL_ELEM_TYPE_BOOLEAN : SNDRV_CTL_ELEM_TYPE_INTEGER; ++ uinfo->count = 1; ++ uinfo->value.integer.min = 0; ++ uinfo->value.integer.max = mask; ++ return 0; ++} ++EXPORT_SYMBOL_GPL(snd_soc_info_volsw_ext); ++ ++/** ++ * snd_soc_info_bool_ext - external single boolean mixer info callback ++ * @kcontrol: mixer control ++ * @uinfo: control element information ++ * ++ * Callback to provide information about a single boolean external mixer control. ++ * ++ * Returns 0 for success. ++ */ ++int snd_soc_info_bool_ext(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; ++} ++EXPORT_SYMBOL_GPL(snd_soc_info_bool_ext); ++ ++/** ++ * snd_soc_info_volsw - single mixer info callback ++ * @kcontrol: mixer control ++ * @uinfo: control element information ++ * ++ * Callback to provide information about a single mixer control. ++ * ++ * Returns 0 for success. ++ */ ++int snd_soc_info_volsw(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_info *uinfo) ++{ ++ int mask = (kcontrol->private_value >> 16) & 0xff; ++ int shift = (kcontrol->private_value >> 8) & 0x0f; ++ int rshift = (kcontrol->private_value >> 12) & 0x0f; ++ ++ uinfo->type = ++ mask == 1 ? SNDRV_CTL_ELEM_TYPE_BOOLEAN : SNDRV_CTL_ELEM_TYPE_INTEGER; ++ uinfo->count = shift == rshift ? 1 : 2; ++ uinfo->value.integer.min = 0; ++ uinfo->value.integer.max = mask; ++ return 0; ++} ++EXPORT_SYMBOL_GPL(snd_soc_info_volsw); ++ ++/** ++ * snd_soc_get_volsw - single mixer get callback ++ * @kcontrol: mixer control ++ * @uinfo: control element information ++ * ++ * Callback to get the value of a single mixer control. ++ * ++ * Returns 0 for success. ++ */ ++int snd_soc_get_volsw(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ int reg = kcontrol->private_value & 0xff; ++ int shift = (kcontrol->private_value >> 8) & 0x0f; ++ int rshift = (kcontrol->private_value >> 12) & 0x0f; ++ int mask = (kcontrol->private_value >> 16) & 0xff; ++ int invert = (kcontrol->private_value >> 24) & 0x01; ++ ++ ucontrol->value.integer.value[0] = ++ (snd_soc_read(codec, reg) >> shift) & mask; ++ if (shift != rshift) ++ ucontrol->value.integer.value[1] = ++ (snd_soc_read(codec, reg) >> rshift) & mask; ++ if (invert) { ++ ucontrol->value.integer.value[0] = ++ mask - ucontrol->value.integer.value[0]; ++ if (shift != rshift) ++ ucontrol->value.integer.value[1] = ++ mask - ucontrol->value.integer.value[1]; ++ } ++ ++ return 0; ++} ++EXPORT_SYMBOL_GPL(snd_soc_get_volsw); ++ ++/** ++ * snd_soc_put_volsw - single mixer put callback ++ * @kcontrol: mixer control ++ * @uinfo: control element information ++ * ++ * Callback to set the value of a single mixer control. ++ * ++ * Returns 0 for success. ++ */ ++int snd_soc_put_volsw(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ int reg = kcontrol->private_value & 0xff; ++ int shift = (kcontrol->private_value >> 8) & 0x0f; ++ int rshift = (kcontrol->private_value >> 12) & 0x0f; ++ int mask = (kcontrol->private_value >> 16) & 0xff; ++ int invert = (kcontrol->private_value >> 24) & 0x01; ++ int err; ++ unsigned short val, val2, val_mask; ++ ++ val = (ucontrol->value.integer.value[0] & mask); ++ if (invert) ++ val = mask - val; ++ val_mask = mask << shift; ++ val = val << shift; ++ if (shift != rshift) { ++ val2 = (ucontrol->value.integer.value[1] & mask); ++ if (invert) ++ val2 = mask - val2; ++ val_mask |= mask << rshift; ++ val |= val2 << rshift; ++ } ++ err = snd_soc_update_bits(codec, reg, val_mask, val); ++ return err; ++} ++EXPORT_SYMBOL_GPL(snd_soc_put_volsw); ++ ++/** ++ * snd_soc_info_volsw_2r - double mixer info callback ++ * @kcontrol: mixer control ++ * @uinfo: control element information ++ * ++ * Callback to provide information about a double mixer control that ++ * spans 2 codec registers. ++ * ++ * Returns 0 for success. ++ */ ++int snd_soc_info_volsw_2r(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_info *uinfo) ++{ ++ int mask = (kcontrol->private_value >> 12) & 0xff; ++ ++ uinfo->type = ++ mask == 1 ? SNDRV_CTL_ELEM_TYPE_BOOLEAN : SNDRV_CTL_ELEM_TYPE_INTEGER; ++ uinfo->count = 2; ++ uinfo->value.integer.min = 0; ++ uinfo->value.integer.max = mask; ++ return 0; ++} ++EXPORT_SYMBOL_GPL(snd_soc_info_volsw_2r); ++ ++/** ++ * snd_soc_get_volsw_2r - double mixer get callback ++ * @kcontrol: mixer control ++ * @uinfo: control element information ++ * ++ * Callback to get the value of a double mixer control that spans 2 registers. ++ * ++ * Returns 0 for success. ++ */ ++int snd_soc_get_volsw_2r(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ int reg = kcontrol->private_value & 0xff; ++ int reg2 = (kcontrol->private_value >> 24) & 0xff; ++ int shift = (kcontrol->private_value >> 8) & 0x0f; ++ int mask = (kcontrol->private_value >> 12) & 0xff; ++ int invert = (kcontrol->private_value >> 20) & 0x01; ++ ++ ucontrol->value.integer.value[0] = ++ (snd_soc_read(codec, reg) >> shift) & mask; ++ ucontrol->value.integer.value[1] = ++ (snd_soc_read(codec, reg2) >> shift) & mask; ++ if (invert) { ++ ucontrol->value.integer.value[0] = ++ mask - ucontrol->value.integer.value[0]; ++ ucontrol->value.integer.value[1] = ++ mask - ucontrol->value.integer.value[1]; ++ } ++ ++ return 0; ++} ++EXPORT_SYMBOL_GPL(snd_soc_get_volsw_2r); ++ ++/** ++ * snd_soc_put_volsw_2r - double mixer set callback ++ * @kcontrol: mixer control ++ * @uinfo: control element information ++ * ++ * Callback to set the value of a double mixer control that spans 2 registers. ++ * ++ * Returns 0 for success. ++ */ ++int snd_soc_put_volsw_2r(struct snd_kcontrol *kcontrol, ++ struct snd_ctl_elem_value *ucontrol) ++{ ++ struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); ++ int reg = kcontrol->private_value & 0xff; ++ int reg2 = (kcontrol->private_value >> 24) & 0xff; ++ int shift = (kcontrol->private_value >> 8) & 0x0f; ++ int mask = (kcontrol->private_value >> 12) & 0xff; ++ int invert = (kcontrol->private_value >> 20) & 0x01; ++ int err; ++ unsigned short val, val2, val_mask; ++ ++ val_mask = mask << shift; ++ val = (ucontrol->value.integer.value[0] & mask); ++ val2 = (ucontrol->value.integer.value[1] & mask); ++ ++ if (invert) { ++ val = mask - val; ++ val2 = mask - val2; ++ } ++ ++ val = val << shift; ++ val2 = val2 << shift; ++ ++ if ((err = snd_soc_update_bits(codec, reg, val_mask, val)) < 0) ++ return err; ++ ++ err = snd_soc_update_bits(codec, reg2, val_mask, val2); ++ return err; ++} ++EXPORT_SYMBOL_GPL(snd_soc_put_volsw_2r); ++ ++static int __devinit snd_soc_init(void) ++{ ++ printk(KERN_INFO "ASoC version %s\n", SND_SOC_VERSION); ++ return platform_driver_register(&soc_driver); ++} ++ ++static void snd_soc_exit(void) ++{ ++ platform_driver_unregister(&soc_driver); ++} ++ ++module_init(snd_soc_init); ++module_exit(snd_soc_exit); ++ ++/* Module information */ ++MODULE_AUTHOR("Liam Girdwood, liam.girdwood@wolfsonmicro.com, www.wolfsonmicro.com"); ++MODULE_DESCRIPTION("ALSA SoC Core"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/at91/Kconfig +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/at91/Kconfig +@@ -0,0 +1,24 @@ ++menu "SoC Audio for the Atmel AT91" ++ ++config SND_AT91_SOC ++ tristate "SoC Audio for the Atmel AT91 System-on-Chip" ++ depends on ARCH_AT91 && SND ++ select SND_PCM ++ help ++ Say Y or M if you want to add support for codecs attached to ++ the AT91 SSC interface. You will also need ++ to select the audio interfaces to support below. ++ ++config SND_AT91_SOC_I2S ++ tristate ++ ++config SND_AT91_SOC_ETI_B1_WM8731 ++ tristate "SoC I2S Audio support for Endrelia ETI-B1 board" ++ depends on SND_AT91_SOC && MACH_ETI_B1 ++ select SND_AT91_SOC_I2S ++ select SND_SOC_WM8731 ++ help ++ Say Y if you want to add support for SoC audio on Endrelia ++ ETI-B1 board. ++ ++endmenu +Index: linux-2.6-pxa-new/sound/soc/at91/Makefile +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/at91/Makefile +@@ -0,0 +1,11 @@ ++# AT91 Platform Support ++snd-soc-at91-objs := at91rm9200-pcm.o ++snd-soc-at91-i2s-objs := at91rm9200-i2s.o ++ ++obj-$(CONFIG_SND_AT91_SOC) += snd-soc-at91.o ++obj-$(CONFIG_SND_AT91_SOC_I2S) += snd-soc-at91-i2s.o ++ ++# AT91 Machine Support ++snd-soc-eti-b1-wm8731-objs := eti_b1_wm8731.o ++ ++obj-$(CONFIG_SND_AT91_SOC_ETI_B1_WM8731) += snd-soc-eti-b1-wm8731.o +Index: linux-2.6-pxa-new/sound/soc/at91/at91rm9200-i2s.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/at91/at91rm9200-i2s.c +@@ -0,0 +1,715 @@ ++/* ++ * at91rm9200-i2s.c -- ALSA Soc Audio Layer Platform driver and DMA engine ++ * ++ * Author: Frank Mandarino <fmandarino@endrelia.com> ++ * Endrelia Technologies Inc. ++ * ++ * Based on pxa2xx Platform drivers by ++ * Liam Girdwood <liam.girdwood@wolfsonmicro.com> ++ * ++ * 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. ++ * ++ * Revision history ++ * 3rd Mar 2006 Initial version. ++ */ ++ ++#include <linux/init.h> ++#include <linux/module.h> ++#include <linux/interrupt.h> ++#include <linux/device.h> ++#include <linux/delay.h> ++#include <linux/clk.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/initval.h> ++#include <sound/soc.h> ++ ++#include <asm/arch/at91rm9200.h> ++#include <asm/arch/at91rm9200_ssc.h> ++#include <asm/arch/at91rm9200_pdc.h> ++#include <asm/arch/hardware.h> ++ ++#include "at91rm9200-pcm.h" ++ ++#if 0 ++#define DBG(x...) printk(KERN_DEBUG "at91rm9200-i2s:" x) ++#else ++#define DBG(x...) ++#endif ++ ++#define AT91RM9200_I2S_DAIFMT \ ++ (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBS_CFS | SND_SOC_DAIFMT_NB_NF) ++ ++#define AT91RM9200_I2S_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++/* priv is (SSC_CMR.DIV << 16 | SSC_TCMR.PERIOD ) */ ++static struct snd_soc_dai_mode at91rm9200_i2s[] = { ++ ++ /* 8k: BCLK = (MCLK/10) = (60MHz/50) = 1.2MHz */ ++ { ++ .fmt = AT91RM9200_I2S_DAIFMT, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = AT91RM9200_I2S_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1500, ++ .bfs = SND_SOC_FSBD(10), ++ .priv = (25 << 16 | 74), ++ }, ++ ++ /* 16k: BCLK = (MCLK/3) ~= (60MHz/14) = 4.285714MHz */ ++ { ++ .fmt = AT91RM9200_I2S_DAIFMT, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_16000, ++ .pcmdir = AT91RM9200_I2S_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 750, ++ .bfs = SND_SOC_FSBD(3), ++ .priv = (7 << 16 | 133), ++ }, ++ ++ /* 32k: BCLK = (MCLK/3) ~= (60MHz/14) = 4.285714MHz */ ++ { ++ .fmt = AT91RM9200_I2S_DAIFMT, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = AT91RM9200_I2S_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 375, ++ .bfs = SND_SOC_FSBD(3), ++ .priv = (7 << 16 | 66), ++ }, ++ ++ /* 48k: BCLK = (MCLK/5) ~= (60MHz/26) = 2.3076923MHz */ ++ { ++ .fmt = AT91RM9200_I2S_DAIFMT, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_48000, ++ .pcmdir = AT91RM9200_I2S_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 250, ++ .bfs SND_SOC_FSBD(5), ++ .priv = (13 << 16 | 23), ++ }, ++}; ++ ++ ++/* ++ * SSC registers required by the PCM DMA engine. ++ */ ++static struct at91rm9200_ssc_regs ssc_reg[3] = { ++ { ++ .cr = (void __iomem *) (AT91_VA_BASE_SSC0 + AT91_SSC_CR), ++ .ier = (void __iomem *) (AT91_VA_BASE_SSC0 + AT91_SSC_IER), ++ .idr = (void __iomem *) (AT91_VA_BASE_SSC0 + AT91_SSC_IDR), ++ }, ++ { ++ .cr = (void __iomem *) (AT91_VA_BASE_SSC1 + AT91_SSC_CR), ++ .ier = (void __iomem *) (AT91_VA_BASE_SSC1 + AT91_SSC_IER), ++ .idr = (void __iomem *) (AT91_VA_BASE_SSC1 + AT91_SSC_IDR), ++ }, ++ { ++ .cr = (void __iomem *) (AT91_VA_BASE_SSC2 + AT91_SSC_CR), ++ .ier = (void __iomem *) (AT91_VA_BASE_SSC2 + AT91_SSC_IER), ++ .idr = (void __iomem *) (AT91_VA_BASE_SSC2 + AT91_SSC_IDR), ++ }, ++}; ++ ++static struct at91rm9200_pdc_regs pdc_tx_reg[3] = { ++ { ++ .xpr = (void __iomem *) (AT91_VA_BASE_SSC0 + AT91_PDC_TPR), ++ .xcr = (void __iomem *) (AT91_VA_BASE_SSC0 + AT91_PDC_TCR), ++ .xnpr = (void __iomem *) (AT91_VA_BASE_SSC0 + AT91_PDC_TNPR), ++ .xncr = (void __iomem *) (AT91_VA_BASE_SSC0 + AT91_PDC_TNCR), ++ .ptcr = (void __iomem *) (AT91_VA_BASE_SSC0 + AT91_PDC_PTCR), ++ }, ++ { ++ .xpr = (void __iomem *) (AT91_VA_BASE_SSC1 + AT91_PDC_TPR), ++ .xcr = (void __iomem *) (AT91_VA_BASE_SSC1 + AT91_PDC_TCR), ++ .xnpr = (void __iomem *) (AT91_VA_BASE_SSC1 + AT91_PDC_TNPR), ++ .xncr = (void __iomem *) (AT91_VA_BASE_SSC1 + AT91_PDC_TNCR), ++ .ptcr = (void __iomem *) (AT91_VA_BASE_SSC1 + AT91_PDC_PTCR), ++ }, ++ { ++ .xpr = (void __iomem *) (AT91_VA_BASE_SSC2 + AT91_PDC_TPR), ++ .xcr = (void __iomem *) (AT91_VA_BASE_SSC2 + AT91_PDC_TCR), ++ .xnpr = (void __iomem *) (AT91_VA_BASE_SSC2 + AT91_PDC_TNPR), ++ .xncr = (void __iomem *) (AT91_VA_BASE_SSC2 + AT91_PDC_TNCR), ++ .ptcr = (void __iomem *) (AT91_VA_BASE_SSC2 + AT91_PDC_PTCR), ++ }, ++}; ++ ++static struct at91rm9200_pdc_regs pdc_rx_reg[3] = { ++ { ++ .xpr = (void __iomem *) (AT91_VA_BASE_SSC0 + AT91_PDC_RPR), ++ .xcr = (void __iomem *) (AT91_VA_BASE_SSC0 + AT91_PDC_RCR), ++ .xnpr = (void __iomem *) (AT91_VA_BASE_SSC0 + AT91_PDC_RNPR), ++ .xncr = (void __iomem *) (AT91_VA_BASE_SSC0 + AT91_PDC_RNCR), ++ .ptcr = (void __iomem *) (AT91_VA_BASE_SSC0 + AT91_PDC_PTCR), ++ }, ++ { ++ .xpr = (void __iomem *) (AT91_VA_BASE_SSC1 + AT91_PDC_RPR), ++ .xcr = (void __iomem *) (AT91_VA_BASE_SSC1 + AT91_PDC_RCR), ++ .xnpr = (void __iomem *) (AT91_VA_BASE_SSC1 + AT91_PDC_RNPR), ++ .xncr = (void __iomem *) (AT91_VA_BASE_SSC1 + AT91_PDC_RNCR), ++ .ptcr = (void __iomem *) (AT91_VA_BASE_SSC1 + AT91_PDC_PTCR), ++ }, ++ { ++ .xpr = (void __iomem *) (AT91_VA_BASE_SSC2 + AT91_PDC_RPR), ++ .xcr = (void __iomem *) (AT91_VA_BASE_SSC2 + AT91_PDC_RCR), ++ .xnpr = (void __iomem *) (AT91_VA_BASE_SSC2 + AT91_PDC_RNPR), ++ .xncr = (void __iomem *) (AT91_VA_BASE_SSC2 + AT91_PDC_RNCR), ++ .ptcr = (void __iomem *) (AT91_VA_BASE_SSC2 + AT91_PDC_PTCR), ++ }, ++}; ++ ++/* ++ * SSC & PDC status bits for transmit and receive. ++ */ ++static struct at91rm9200_ssc_mask ssc_tx_mask = { ++ .ssc_enable = AT91_SSC_TXEN, ++ .ssc_disable = AT91_SSC_TXDIS, ++ .ssc_endx = AT91_SSC_ENDTX, ++ .ssc_endbuf = AT91_SSC_TXBUFE, ++ .pdc_enable = AT91_PDC_TXTEN, ++ .pdc_disable = AT91_PDC_TXTDIS, ++}; ++ ++static struct at91rm9200_ssc_mask ssc_rx_mask = { ++ .ssc_enable = AT91_SSC_RXEN, ++ .ssc_disable = AT91_SSC_RXDIS, ++ .ssc_endx = AT91_SSC_ENDRX, ++ .ssc_endbuf = AT91_SSC_RXBUFF, ++ .pdc_enable = AT91_PDC_RXTEN, ++ .pdc_disable = AT91_PDC_RXTDIS, ++}; ++ ++/* ++ * A MUTEX is used to protect an SSC initialzed flag which allows ++ * the substream hw_params() call to initialize the SSC only if ++ * there are no other substreams open. If there are other ++ * substreams open, the hw_param() call can only check that ++ * it is using the same format and rate. ++ */ ++static DECLARE_MUTEX(ssc0_mutex); ++static DECLARE_MUTEX(ssc1_mutex); ++static DECLARE_MUTEX(ssc2_mutex); ++ ++/* ++ * DMA parameters. ++ */ ++static at91rm9200_pcm_dma_params_t ssc_dma_params[3][2] = { ++ {{ ++ .name = "SSC0/I2S PCM Stereo out", ++ .ssc = &ssc_reg[0], ++ .pdc = &pdc_tx_reg[0], ++ .mask = &ssc_tx_mask, ++ }, ++ { ++ .name = "SSC0/I2S PCM Stereo in", ++ .ssc = &ssc_reg[0], ++ .pdc = &pdc_rx_reg[0], ++ .mask = &ssc_rx_mask, ++ }}, ++ {{ ++ .name = "SSC1/I2S PCM Stereo out", ++ .ssc = &ssc_reg[1], ++ .pdc = &pdc_tx_reg[1], ++ .mask = &ssc_tx_mask, ++ }, ++ { ++ .name = "SSC1/I2S PCM Stereo in", ++ .ssc = &ssc_reg[1], ++ .pdc = &pdc_rx_reg[1], ++ .mask = &ssc_rx_mask, ++ }}, ++ {{ ++ .name = "SSC2/I2S PCM Stereo out", ++ .ssc = &ssc_reg[2], ++ .pdc = &pdc_tx_reg[2], ++ .mask = &ssc_tx_mask, ++ }, ++ { ++ .name = "SSC1/I2S PCM Stereo in", ++ .ssc = &ssc_reg[2], ++ .pdc = &pdc_rx_reg[2], ++ .mask = &ssc_rx_mask, ++ }}, ++}; ++ ++ ++struct at91rm9200_ssc_state { ++ u32 ssc_cmr; ++ u32 ssc_rcmr; ++ u32 ssc_rfmr; ++ u32 ssc_tcmr; ++ u32 ssc_tfmr; ++ u32 ssc_sr; ++ u32 ssc_imr; ++}; ++ ++static struct at91rm9200_ssc_info { ++ char *name; ++ void __iomem *ssc_base; ++ u32 pid; ++ spinlock_t lock; /* lock for dir_mask */ ++ int dir_mask; /* 0=unused, 1=playback, 2=capture */ ++ struct semaphore *mutex; ++ int initialized; ++ int pcmfmt; ++ int rate; ++ at91rm9200_pcm_dma_params_t *dma_params[2]; ++ struct at91rm9200_ssc_state ssc_state; ++ ++} ssc_info[3] = { ++ { ++ .name = "ssc0", ++ .ssc_base = (void __iomem *) AT91_VA_BASE_SSC0, ++ .pid = AT91_ID_SSC0, ++ .lock = SPIN_LOCK_UNLOCKED, ++ .dir_mask = 0, ++ .mutex = &ssc0_mutex, ++ .initialized = 0, ++ }, ++ { ++ .name = "ssc1", ++ .ssc_base = (void __iomem *) AT91_VA_BASE_SSC1, ++ .pid = AT91_ID_SSC1, ++ .lock = SPIN_LOCK_UNLOCKED, ++ .dir_mask = 0, ++ .mutex = &ssc1_mutex, ++ .initialized = 0, ++ }, ++ { ++ .name = "ssc2", ++ .ssc_base = (void __iomem *) AT91_VA_BASE_SSC2, ++ .pid = AT91_ID_SSC2, ++ .lock = SPIN_LOCK_UNLOCKED, ++ .dir_mask = 0, ++ .mutex = &ssc2_mutex, ++ .initialized = 0, ++ }, ++}; ++ ++ ++static irqreturn_t at91rm9200_i2s_interrupt(int irq, void *dev_id) ++{ ++ struct at91rm9200_ssc_info *ssc_p = dev_id; ++ at91rm9200_pcm_dma_params_t *dma_params; ++ u32 ssc_sr; ++ int i; ++ ++ ssc_sr = at91_ssc_read(ssc_p->ssc_base + AT91_SSC_SR) ++ & at91_ssc_read(ssc_p->ssc_base + AT91_SSC_IMR); ++ ++ /* ++ * Loop through the substreams attached to this SSC. If ++ * a DMA-related interrupt occurred on that substream, call ++ * the DMA interrupt handler function, if one has been ++ * registered in the dma_params structure by the PCM driver. ++ */ ++ for (i = 0; i < ARRAY_SIZE(ssc_p->dma_params); i++) { ++ dma_params = ssc_p->dma_params[i]; ++ ++ if (dma_params != NULL && dma_params->dma_intr_handler != NULL && ++ (ssc_sr & ++ (dma_params->mask->ssc_endx | dma_params->mask->ssc_endbuf))) ++ ++ dma_params->dma_intr_handler(ssc_sr, dma_params->substream); ++ } ++ ++ return IRQ_HANDLED; ++} ++ ++static int at91rm9200_i2s_startup(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct at91rm9200_ssc_info *ssc_p = &ssc_info[rtd->cpu_dai->id]; ++ int dir_mask; ++ ++ DBG("i2s_startup: SSC_SR=0x%08lx\n", ++ at91_ssc_read(ssc_p->ssc_base + AT91_SSC_SR)); ++ dir_mask = substream->stream == SNDRV_PCM_STREAM_PLAYBACK ? 0x1 : 0x2; ++ ++ spin_lock_irq(&ssc_p->lock); ++ if (ssc_p->dir_mask & dir_mask) { ++ spin_unlock_irq(&ssc_p->lock); ++ return -EBUSY; ++ } ++ ssc_p->dir_mask |= dir_mask; ++ spin_unlock_irq(&ssc_p->lock); ++ ++ return 0; ++} ++ ++static void at91rm9200_i2s_shutdown(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct at91rm9200_ssc_info *ssc_p = &ssc_info[rtd->cpu_dai->id]; ++ at91rm9200_pcm_dma_params_t *dma_params = rtd->cpu_dai->dma_data; ++ int dir, dir_mask; ++ ++ dir = substream->stream == SNDRV_PCM_STREAM_PLAYBACK ? 0 : 1; ++ ++ if (dma_params != NULL) { ++ at91_ssc_write(dma_params->ssc->cr, dma_params->mask->ssc_disable); ++ DBG("%s disabled SSC_SR=0x%08lx\n", (dir ? "receive" : "transmit"), ++ at91_ssc_read(ssc_p->ssc_base + AT91_SSC_SR)); ++ ++ dma_params->substream = NULL; ++ ssc_p->dma_params[dir] = NULL; ++ } ++ ++ dir_mask = 1 << dir; ++ ++ spin_lock_irq(&ssc_p->lock); ++ ssc_p->dir_mask &= ~dir_mask; ++ if (!ssc_p->dir_mask) { ++ /* Shutdown the SSC clock. */ ++ DBG("Stopping pid %d clock\n", ssc_p->pid); ++ at91_sys_write(AT91_PMC_PCDR, 1<<ssc_p->pid); ++ ++ if (ssc_p->initialized) ++ free_irq(ssc_p->pid, ssc_p); ++ ++ /* Reset the SSC */ ++ at91_ssc_write(ssc_p->ssc_base + AT91_SSC_CR, AT91_SSC_SWRST); ++ ++ /* Force a re-init on the next hw_params() call. */ ++ ssc_p->initialized = 0; ++ } ++ spin_unlock_irq(&ssc_p->lock); ++} ++ ++#ifdef CONFIG_PM ++static int at91rm9200_i2s_suspend(struct platform_device *pdev, ++ struct snd_soc_cpu_dai *dai) ++{ ++ struct at91rm9200_ssc_info *ssc_p; ++ ++ if(!dai->active) ++ return 0; ++ ++ ssc_p = &ssc_info[dai->id]; ++ ++ /* Save the status register before disabling transmit and receive. */ ++ ssc_p->state->ssc_sr = at91_ssc_read(ssc_p->ssc_base + AT91_SSC_SR); ++ at91_ssc_write(ssc_p->ssc_base + ++ AT91_SSC_CR, AT91_SSC_TXDIS | AT91_SSC_RXDIS); ++ ++ /* Save the current interrupt mask, then disable unmasked interrupts. */ ++ ssc_p->state->ssc_imr = at91_ssc_read(ssc_p->ssc_base + AT91_SSC_IMR); ++ at91_ssc_write(ssc_p->ssc_base + AT91_SSC_IDR, ssc_p->state->ssc_imr); ++ ++ ssc_p->state->ssc_cmr = at91_ssc_read(ssc_p->ssc_base + AT91_SSC_CMR); ++ ssc_p->state->ssc_rcmr = at91_ssc_read(ssc_p->ssc_base + AT91_SSC_RCMR); ++ ssc_p->state->ssc_rfmr = at91_ssc_read(ssc_p->ssc_base + AT91_SSC_RCMR); ++ ssc_p->state->ssc_tcmr = at91_ssc_read(ssc_p->ssc_base + AT91_SSC_RCMR); ++ ssc_p->state->ssc_tfmr = at91_ssc_read(ssc_p->ssc_base + AT91_SSC_RCMR); ++ ++ return 0; ++} ++ ++static int at91rm9200_i2s_resume(struct platform_device *pdev, ++ struct snd_soc_cpu_dai *dai) ++{ ++ struct at91rm9200_ssc_info *ssc_p; ++ u32 cr_mask; ++ ++ if(!dai->active) ++ return 0; ++ ++ ssc_p = &ssc_info[dai->id]; ++ ++ at91_ssc_write(ssc_p->ssc_base + AT91_SSC_RCMR, ssc_p->state->ssc_tfmr); ++ at91_ssc_write(ssc_p->ssc_base + AT91_SSC_RCMR, ssc_p->state->ssc_tcmr); ++ at91_ssc_write(ssc_p->ssc_base + AT91_SSC_RCMR, ssc_p->state->ssc_rfmr); ++ at91_ssc_write(ssc_p->ssc_base + AT91_SSC_RCMR, ssc_p->state->ssc_rcmr); ++ at91_ssc_write(ssc_p->ssc_base + AT91_SSC_CMR, ssc_p->state->ssc_cmr); ++ ++ at91_ssc_write(ssc_p->ssc_base + AT91_SSC_IER, ssc_p->state->ssc_imr); ++ ++ at91_ssc_write(ssc_p->ssc_base + AT91_SSC_CR, ++ ((ssc_p->state->ssc_sr & AT91_SSC_RXENA) ? AT91_SSC_RXEN : 0) | ++ ((ssc_p->state->ssc_sr & AT91_SSC_TXENA) ? AT91_SSC_TXEN : 0)); ++ ++ return 0; ++} ++ ++#else ++#define at91rm9200_i2s_suspend NULL ++#define at91rm9200_i2s_resume NULL ++#endif ++ ++static unsigned int at91rm9200_i2s_config_sysclk( ++ struct snd_soc_cpu_dai *iface, struct snd_soc_clock_info *info, ++ unsigned int clk) ++{ ++ /* Currently, there is only support for USB (12Mhz) mode */ ++ if (clk != 12000000) ++ return 0; ++ return 12000000; ++} ++ ++static int at91rm9200_i2s_hw_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ int id = rtd->cpu_dai->id; ++ struct at91rm9200_ssc_info *ssc_p = &ssc_info[id]; ++ at91rm9200_pcm_dma_params_t *dma_params; ++ unsigned int pcmfmt, rate; ++ int dir, channels, bits; ++ struct clk *mck_clk; ++ unsigned long bclk; ++ u32 div, period, tfmr, rfmr, tcmr, rcmr; ++ int ret; ++ ++ /* ++ * Currently, there is only one set of dma params for ++ * each direction. If more are added, this code will ++ * have to be changed to select the proper set. ++ */ ++ dir = substream->stream == SNDRV_PCM_STREAM_PLAYBACK ? 0 : 1; ++ ++ dma_params = &ssc_dma_params[id][dir]; ++ dma_params->substream = substream; ++ ++ ssc_p->dma_params[dir] = dma_params; ++ rtd->cpu_dai->dma_data = dma_params; ++ ++ rate = params_rate(params); ++ channels = params_channels(params); ++ ++ pcmfmt = rtd->cpu_dai->dai_runtime.pcmfmt; ++ switch (pcmfmt) { ++ case SNDRV_PCM_FMTBIT_S16_LE: ++ /* likely this is all we'll ever support, but ... */ ++ bits = 16; ++ dma_params->pdc_xfer_size = 2; ++ break; ++ default: ++ printk(KERN_WARNING "at91rm9200-i2s: unsupported format %x\n", ++ pcmfmt); ++ return -EINVAL; ++ } ++ ++ /* Don't allow both SSC substreams to initialize at the same time. */ ++ down(ssc_p->mutex); ++ ++ /* ++ * If this SSC is alreadly initialized, then this substream must use ++ * the same format and rate. ++ */ ++ if (ssc_p->initialized) { ++ if (pcmfmt != ssc_p->pcmfmt || rate != ssc_p->rate) { ++ printk(KERN_WARNING "at91rm9200-i2s: " ++ "incompatible substream in other direction\n"); ++ up(ssc_p->mutex); ++ return -EINVAL; ++ } ++ } else { ++ /* Enable PMC peripheral clock for this SSC */ ++ DBG("Starting pid %d clock\n", ssc_p->pid); ++ at91_sys_write(AT91_PMC_PCER, 1<<ssc_p->pid); ++ ++ /* Reset the SSC */ ++ at91_ssc_write(ssc_p->ssc_base + AT91_SSC_CR, AT91_SSC_SWRST); ++ ++ at91_ssc_write(ssc_p->ssc_base + AT91_PDC_RPR, 0); ++ at91_ssc_write(ssc_p->ssc_base + AT91_PDC_RCR, 0); ++ at91_ssc_write(ssc_p->ssc_base + AT91_PDC_RNPR, 0); ++ at91_ssc_write(ssc_p->ssc_base + AT91_PDC_RNCR, 0); ++ at91_ssc_write(ssc_p->ssc_base + AT91_PDC_TPR, 0); ++ at91_ssc_write(ssc_p->ssc_base + AT91_PDC_TCR, 0); ++ at91_ssc_write(ssc_p->ssc_base + AT91_PDC_TNPR, 0); ++ at91_ssc_write(ssc_p->ssc_base + AT91_PDC_TNCR, 0); ++ ++ mck_clk = clk_get(NULL, "mck"); ++ ++ div = rtd->cpu_dai->dai_runtime.priv >> 16; ++ period = rtd->cpu_dai->dai_runtime.priv & 0xffff; ++ bclk = 60000000 / (2 * div); ++ ++ DBG("mck %ld fsbd %d bfs %d bfs_real %d bclk %ld div %d period %d\n", ++ clk_get_rate(mck_clk), ++ SND_SOC_FSBD(6), ++ rtd->cpu_dai->dai_runtime.bfs, ++ SND_SOC_FSBD_REAL(rtd->cpu_dai->dai_runtime.bfs), ++ bclk, ++ div, ++ period); ++ ++ clk_put(mck_clk); ++ ++ at91_ssc_write(ssc_p->ssc_base + AT91_SSC_CMR, div); ++ ++ /* ++ * Setup the TFMR and RFMR for the proper data format. ++ */ ++ tfmr = ++ (( AT91_SSC_FSEDGE_POSITIVE ) & AT91_SSC_FSEDGE) ++ | (( 0 << 23) & AT91_SSC_FSDEN) ++ | (( AT91_SSC_FSOS_NEGATIVE ) & AT91_SSC_FSOS) ++ | (((bits - 1) << 16) & AT91_SSC_FSLEN) ++ | (((channels - 1) << 8) & AT91_SSC_DATNB) ++ | (( 1 << 7) & AT91_SSC_MSBF) ++ | (( 0 << 5) & AT91_SSC_DATDEF) ++ | (((bits - 1) << 0) & AT91_SSC_DATALEN); ++ DBG("SSC_TFMR=0x%08x\n", tfmr); ++ at91_ssc_write(ssc_p->ssc_base + AT91_SSC_TFMR, tfmr); ++ ++ rfmr = ++ (( AT91_SSC_FSEDGE_POSITIVE ) & AT91_SSC_FSEDGE) ++ | (( AT91_SSC_FSOS_NONE ) & AT91_SSC_FSOS) ++ | (( 0 << 16) & AT91_SSC_FSLEN) ++ | (((channels - 1) << 8) & AT91_SSC_DATNB) ++ | (( 1 << 7) & AT91_SSC_MSBF) ++ | (( 0 << 5) & AT91_SSC_LOOP) ++ | (((bits - 1) << 0) & AT91_SSC_DATALEN); ++ ++ DBG("SSC_RFMR=0x%08x\n", rfmr); ++ at91_ssc_write(ssc_p->ssc_base + AT91_SSC_RFMR, rfmr); ++ ++ /* ++ * Setup the TCMR and RCMR to generate the proper BCLK ++ * and LRC signals. ++ */ ++ tcmr = ++ (( period << 24) & AT91_SSC_PERIOD) ++ | (( 1 << 16) & AT91_SSC_STTDLY) ++ | (( AT91_SSC_START_FALLING_RF ) & AT91_SSC_START) ++ | (( AT91_SSC_CKI_FALLING ) & AT91_SSC_CKI) ++ | (( AT91_SSC_CKO_CONTINUOUS ) & AT91_SSC_CKO) ++ | (( AT91_SSC_CKS_DIV ) & AT91_SSC_CKS); ++ ++ DBG("SSC_TCMR=0x%08x\n", tcmr); ++ at91_ssc_write(ssc_p->ssc_base + AT91_SSC_TCMR, tcmr); ++ ++ rcmr = ++ (( 0 << 24) & AT91_SSC_PERIOD) ++ | (( 1 << 16) & AT91_SSC_STTDLY) ++ | (( AT91_SSC_START_TX_RX ) & AT91_SSC_START) ++ | (( AT91_SSC_CK_RISING ) & AT91_SSC_CKI) ++ | (( AT91_SSC_CKO_NONE ) & AT91_SSC_CKO) ++ | (( AT91_SSC_CKS_CLOCK ) & AT91_SSC_CKS); ++ ++ DBG("SSC_RCMR=0x%08x\n", rcmr); ++ at91_ssc_write(ssc_p->ssc_base + AT91_SSC_RCMR, rcmr); ++ ++ if ((ret = request_irq(ssc_p->pid, at91rm9200_i2s_interrupt, ++ 0, ssc_p->name, ssc_p)) < 0) { ++ printk(KERN_WARNING "at91rm9200-i2s: request_irq failure\n"); ++ return ret; ++ } ++ ++ /* ++ * Save the current substream parameters in order to check ++ * that the substream in the opposite direction uses the ++ * same parameters. ++ */ ++ ssc_p->pcmfmt = pcmfmt; ++ ssc_p->rate = rate; ++ ssc_p->initialized = 1; ++ ++ DBG("hw_params: SSC initialized\n"); ++ } ++ ++ up(ssc_p->mutex); ++ ++ return 0; ++} ++ ++ ++static int at91rm9200_i2s_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ at91rm9200_pcm_dma_params_t *dma_params = rtd->cpu_dai->dma_data; ++ ++ at91_ssc_write(dma_params->ssc->cr, dma_params->mask->ssc_enable); ++ ++ DBG("%s enabled SSC_SR=0x%08lx\n", ++ substream->stream == SNDRV_PCM_STREAM_PLAYBACK ? "transmit" : "receive", ++ at91_ssc_read(ssc_info[rtd->cpu_dai->id].ssc_base + AT91_SSC_SR)); ++ return 0; ++} ++ ++ ++struct snd_soc_cpu_dai at91rm9200_i2s_dai[] = { ++ { .name = "at91rm9200-ssc0/i2s", ++ .id = 0, ++ .type = SND_SOC_DAI_I2S, ++ .suspend = at91rm9200_i2s_suspend, ++ .resume = at91rm9200_i2s_resume, ++ .config_sysclk = at91rm9200_i2s_config_sysclk, ++ .playback = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .capture = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .ops = { ++ .startup = at91rm9200_i2s_startup, ++ .shutdown = at91rm9200_i2s_shutdown, ++ .prepare = at91rm9200_i2s_prepare, ++ .hw_params = at91rm9200_i2s_hw_params,}, ++ .caps = { ++ .mode = &at91rm9200_i2s[0], ++ .num_modes = ARRAY_SIZE(at91rm9200_i2s),}, ++ }, ++ { .name = "at91rm9200-ssc1/i2s", ++ .id = 1, ++ .type = SND_SOC_DAI_I2S, ++ .suspend = at91rm9200_i2s_suspend, ++ .resume = at91rm9200_i2s_resume, ++ .config_sysclk = at91rm9200_i2s_config_sysclk, ++ .playback = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .capture = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .ops = { ++ .startup = at91rm9200_i2s_startup, ++ .shutdown = at91rm9200_i2s_shutdown, ++ .prepare = at91rm9200_i2s_prepare, ++ .hw_params = at91rm9200_i2s_hw_params,}, ++ .caps = { ++ .mode = &at91rm9200_i2s[0], ++ .num_modes = ARRAY_SIZE(at91rm9200_i2s),}, ++ }, ++ { .name = "at91rm9200-ssc2/i2s", ++ .id = 2, ++ .type = SND_SOC_DAI_I2S, ++ .suspend = at91rm9200_i2s_suspend, ++ .resume = at91rm9200_i2s_resume, ++ .config_sysclk = at91rm9200_i2s_config_sysclk, ++ .playback = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .capture = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .ops = { ++ .startup = at91rm9200_i2s_startup, ++ .shutdown = at91rm9200_i2s_shutdown, ++ .prepare = at91rm9200_i2s_prepare, ++ .hw_params = at91rm9200_i2s_hw_params,}, ++ .caps = { ++ .mode = &at91rm9200_i2s[0], ++ .num_modes = ARRAY_SIZE(at91rm9200_i2s),}, ++ }, ++}; ++ ++EXPORT_SYMBOL_GPL(at91rm9200_i2s_dai); ++ ++/* Module information */ ++MODULE_AUTHOR("Frank Mandarino, fmandarino@endrelia.com, www.endrelia.com"); ++MODULE_DESCRIPTION("AT91RM9200 I2S ASoC Interface"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/at91/at91rm9200-pcm.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/at91/at91rm9200-pcm.c +@@ -0,0 +1,428 @@ ++/* ++ * at91rm9200-pcm.c -- ALSA PCM interface for the Atmel AT91RM9200 chip. ++ * ++ * Author: Frank Mandarino <fmandarino@endrelia.com> ++ * Endrelia Technologies Inc. ++ * Created: Mar 3, 2006 ++ * ++ * Based on pxa2xx-pcm.c by: ++ * ++ * Author: Nicolas Pitre ++ * Created: Nov 30, 2004 ++ * Copyright: (C) 2004 MontaVista Software, Inc. ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#include <linux/module.h> ++#include <linux/init.h> ++#include <linux/platform_device.h> ++#include <linux/slab.h> ++#include <linux/dma-mapping.h> ++ ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/pcm_params.h> ++#include <sound/soc.h> ++ ++#include <asm/arch/at91rm9200.h> ++#include <asm/arch/at91rm9200_ssc.h> ++#include <asm/arch/at91rm9200_pdc.h> ++#include <asm/arch/hardware.h> ++ ++#include "at91rm9200-pcm.h" ++ ++#if 0 ++#define DBG(x...) printk(KERN_INFO "at91rm9200-pcm: " x) ++#else ++#define DBG(x...) ++#endif ++ ++static const snd_pcm_hardware_t at91rm9200_pcm_hardware = { ++ .info = SNDRV_PCM_INFO_MMAP | ++ SNDRV_PCM_INFO_MMAP_VALID | ++ SNDRV_PCM_INFO_INTERLEAVED | ++ SNDRV_PCM_INFO_PAUSE, ++ .formats = SNDRV_PCM_FMTBIT_S16_LE, ++ .period_bytes_min = 32, ++ .period_bytes_max = 8192, ++ .periods_min = 2, ++ .periods_max = 1024, ++ .buffer_bytes_max = 32 * 1024, ++}; ++ ++struct at91rm9200_runtime_data { ++ at91rm9200_pcm_dma_params_t *params; ++ dma_addr_t dma_buffer; /* physical address of dma buffer */ ++ dma_addr_t dma_buffer_end; /* first address beyond DMA buffer */ ++ size_t period_size; ++ dma_addr_t period_ptr; /* physical address of next period */ ++ u32 pdc_xpr_save; /* PDC register save */ ++ u32 pdc_xcr_save; ++ u32 pdc_xnpr_save; ++ u32 pdc_xncr_save; ++}; ++ ++static void at91rm9200_pcm_dma_irq(u32 ssc_sr, ++ struct snd_pcm_substream *substream) ++{ ++ struct at91rm9200_runtime_data *prtd = substream->runtime->private_data; ++ at91rm9200_pcm_dma_params_t *params = prtd->params; ++ static int count = 0; ++ ++ count++; ++ ++ if (ssc_sr & params->mask->ssc_endbuf) { ++ ++ printk(KERN_WARNING ++ "at91rm9200-pcm: buffer %s on %s (SSC_SR=%#x, count=%d)\n", ++ substream->stream == SNDRV_PCM_STREAM_PLAYBACK ++ ? "underrun" : "overrun", ++ params->name, ssc_sr, count); ++ ++ /* re-start the PDC */ ++ at91_ssc_write(params->pdc->ptcr, params->mask->pdc_disable); ++ ++ prtd->period_ptr += prtd->period_size; ++ if (prtd->period_ptr >= prtd->dma_buffer_end) { ++ prtd->period_ptr = prtd->dma_buffer; ++ } ++ ++ at91_ssc_write(params->pdc->xpr, prtd->period_ptr); ++ at91_ssc_write(params->pdc->xcr, ++ prtd->period_size / params->pdc_xfer_size); ++ ++ at91_ssc_write(params->pdc->ptcr, params->mask->pdc_enable); ++ } ++ ++ if (ssc_sr & params->mask->ssc_endx) { ++ ++ /* Load the PDC next pointer and counter registers */ ++ prtd->period_ptr += prtd->period_size; ++ if (prtd->period_ptr >= prtd->dma_buffer_end) { ++ prtd->period_ptr = prtd->dma_buffer; ++ } ++ at91_ssc_write(params->pdc->xnpr, prtd->period_ptr); ++ at91_ssc_write(params->pdc->xncr, ++ prtd->period_size / params->pdc_xfer_size); ++ } ++ ++ snd_pcm_period_elapsed(substream); ++} ++ ++static int at91rm9200_pcm_hw_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ snd_pcm_runtime_t *runtime = substream->runtime; ++ struct at91rm9200_runtime_data *prtd = runtime->private_data; ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ ++ /* this may get called several times by oss emulation ++ * with different params */ ++ ++ snd_pcm_set_runtime_buffer(substream, &substream->dma_buffer); ++ runtime->dma_bytes = params_buffer_bytes(params); ++ ++ prtd->params = rtd->cpu_dai->dma_data; ++ prtd->params->dma_intr_handler = at91rm9200_pcm_dma_irq; ++ ++ prtd->dma_buffer = runtime->dma_addr; ++ prtd->dma_buffer_end = runtime->dma_addr + runtime->dma_bytes; ++ prtd->period_size = params_period_bytes(params); ++ ++ DBG("hw_params: DMA for %s initialized (dma_bytes=%d, period_size=%d)\n", ++ prtd->params->name, runtime->dma_bytes, prtd->period_size); ++ return 0; ++} ++ ++static int at91rm9200_pcm_hw_free(struct snd_pcm_substream *substream) ++{ ++ struct at91rm9200_runtime_data *prtd = substream->runtime->private_data; ++ at91rm9200_pcm_dma_params_t *params = prtd->params; ++ ++ if (params != NULL) { ++ at91_ssc_write(params->pdc->ptcr, params->mask->pdc_disable); ++ prtd->params->dma_intr_handler = NULL; ++ } ++ ++ return 0; ++} ++ ++static int at91rm9200_pcm_prepare(struct snd_pcm_substream *substream) ++{ ++ struct at91rm9200_runtime_data *prtd = substream->runtime->private_data; ++ at91rm9200_pcm_dma_params_t *params = prtd->params; ++ ++ at91_ssc_write(params->ssc->idr, ++ params->mask->ssc_endx | params->mask->ssc_endbuf); ++ ++ at91_ssc_write(params->pdc->ptcr, params->mask->pdc_disable); ++ return 0; ++} ++ ++static int at91rm9200_pcm_trigger(struct snd_pcm_substream *substream, ++ int cmd) ++{ ++ struct at91rm9200_runtime_data *prtd = substream->runtime->private_data; ++ at91rm9200_pcm_dma_params_t *params = prtd->params; ++ int ret = 0; ++ ++ switch (cmd) { ++ case SNDRV_PCM_TRIGGER_START: ++ prtd->period_ptr = prtd->dma_buffer; ++ ++ at91_ssc_write(params->pdc->xpr, prtd->period_ptr); ++ at91_ssc_write(params->pdc->xcr, ++ prtd->period_size / params->pdc_xfer_size); ++ ++ prtd->period_ptr += prtd->period_size; ++ at91_ssc_write(params->pdc->xnpr, prtd->period_ptr); ++ at91_ssc_write(params->pdc->xncr, ++ prtd->period_size / params->pdc_xfer_size); ++ ++ DBG("trigger: period_ptr=%lx, xpr=%lx, xcr=%ld, xnpr=%lx, xncr=%ld\n", ++ (unsigned long) prtd->period_ptr, ++ at91_ssc_read(params->pdc->xpr), ++ at91_ssc_read(params->pdc->xcr), ++ at91_ssc_read(params->pdc->xnpr), ++ at91_ssc_read(params->pdc->xncr)); ++ ++ at91_ssc_write(params->ssc->ier, ++ params->mask->ssc_endx | params->mask->ssc_endbuf); ++ ++ at91_ssc_write(params->pdc->ptcr, params->mask->pdc_enable); ++ ++ DBG("sr=%lx imr=%lx\n", at91_ssc_read(params->ssc->ier - 4), ++ at91_ssc_read(params->ssc->ier + 8)); ++ break; ++ ++ case SNDRV_PCM_TRIGGER_STOP: ++ case SNDRV_PCM_TRIGGER_SUSPEND: ++ case SNDRV_PCM_TRIGGER_PAUSE_PUSH: ++ at91_ssc_write(params->pdc->ptcr, params->mask->pdc_disable); ++ break; ++ ++ case SNDRV_PCM_TRIGGER_RESUME: ++ case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: ++ at91_ssc_write(params->pdc->ptcr, params->mask->pdc_enable); ++ break; ++ ++ default: ++ ret = -EINVAL; ++ } ++ ++ return ret; ++} ++ ++static snd_pcm_uframes_t at91rm9200_pcm_pointer( ++ struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct at91rm9200_runtime_data *prtd = runtime->private_data; ++ at91rm9200_pcm_dma_params_t *params = prtd->params; ++ dma_addr_t ptr; ++ snd_pcm_uframes_t x; ++ ++ ptr = (dma_addr_t) at91_ssc_read(params->pdc->xpr); ++ x = bytes_to_frames(runtime, ptr - prtd->dma_buffer); ++ ++ if (x == runtime->buffer_size) ++ x = 0; ++ return x; ++} ++ ++static int at91rm9200_pcm_open(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct at91rm9200_runtime_data *prtd; ++ int ret = 0; ++ ++ snd_soc_set_runtime_hwparams(substream, &at91rm9200_pcm_hardware); ++ ++ /* ensure that buffer size is a multiple of period size */ ++ ret = snd_pcm_hw_constraint_integer(runtime, SNDRV_PCM_HW_PARAM_PERIODS); ++ if (ret < 0) ++ goto out; ++ ++ prtd = kzalloc(sizeof(struct at91rm9200_runtime_data), GFP_KERNEL); ++ if (prtd == NULL) { ++ ret = -ENOMEM; ++ goto out; ++ } ++ runtime->private_data = prtd; ++ ++ out: ++ return ret; ++} ++ ++static int at91rm9200_pcm_close(struct snd_pcm_substream *substream) ++{ ++ struct at91rm9200_runtime_data *prtd = substream->runtime->private_data; ++ ++ kfree(prtd); ++ return 0; ++} ++ ++static int at91rm9200_pcm_mmap(struct snd_pcm_substream *substream, ++ struct vm_area_struct *vma) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ ++ return dma_mmap_writecombine(substream->pcm->card->dev, vma, ++ runtime->dma_area, ++ runtime->dma_addr, ++ runtime->dma_bytes); ++} ++ ++struct snd_pcm_ops at91rm9200_pcm_ops = { ++ .open = at91rm9200_pcm_open, ++ .close = at91rm9200_pcm_close, ++ .ioctl = snd_pcm_lib_ioctl, ++ .hw_params = at91rm9200_pcm_hw_params, ++ .hw_free = at91rm9200_pcm_hw_free, ++ .prepare = at91rm9200_pcm_prepare, ++ .trigger = at91rm9200_pcm_trigger, ++ .pointer = at91rm9200_pcm_pointer, ++ .mmap = at91rm9200_pcm_mmap, ++}; ++ ++static int at91rm9200_pcm_preallocate_dma_buffer(struct snd_pcm *pcm, ++ int stream) ++{ ++ struct snd_pcm_substream *substream = pcm->streams[stream].substream; ++ struct snd_dma_buffer *buf = &substream->dma_buffer; ++ size_t size = at91rm9200_pcm_hardware.buffer_bytes_max; ++ ++ buf->dev.type = SNDRV_DMA_TYPE_DEV; ++ buf->dev.dev = pcm->card->dev; ++ buf->private_data = NULL; ++ buf->area = dma_alloc_writecombine(pcm->card->dev, size, ++ &buf->addr, GFP_KERNEL); ++ ++ DBG("preallocate_dma_buffer: area=%p, addr=%p, size=%d\n", ++ (void *) buf->area, ++ (void *) buf->addr, ++ size); ++ ++ if (!buf->area) ++ return -ENOMEM; ++ ++ buf->bytes = size; ++ return 0; ++} ++ ++static u64 at91rm9200_pcm_dmamask = 0xffffffff; ++ ++static int at91rm9200_pcm_new(struct snd_card *card, ++ struct snd_soc_codec_dai *dai, struct snd_pcm *pcm) ++{ ++ int ret = 0; ++ ++ if (!card->dev->dma_mask) ++ card->dev->dma_mask = &at91rm9200_pcm_dmamask; ++ if (!card->dev->coherent_dma_mask) ++ card->dev->coherent_dma_mask = 0xffffffff; ++ ++ if (dai->playback.channels_min) { ++ ret = at91rm9200_pcm_preallocate_dma_buffer(pcm, ++ SNDRV_PCM_STREAM_PLAYBACK); ++ if (ret) ++ goto out; ++ } ++ ++ if (dai->capture.channels_min) { ++ ret = at91rm9200_pcm_preallocate_dma_buffer(pcm, ++ SNDRV_PCM_STREAM_CAPTURE); ++ if (ret) ++ goto out; ++ } ++ out: ++ return ret; ++} ++ ++static void at91rm9200_pcm_free_dma_buffers(struct snd_pcm *pcm) ++{ ++ struct snd_pcm_substream *substream; ++ struct snd_dma_buffer *buf; ++ int stream; ++ ++ for (stream = 0; stream < 2; stream++) { ++ substream = pcm->streams[stream].substream; ++ if (!substream) ++ continue; ++ ++ buf = &substream->dma_buffer; ++ if (!buf->area) ++ continue; ++ ++ dma_free_writecombine(pcm->card->dev, buf->bytes, ++ buf->area, buf->addr); ++ buf->area = NULL; ++ } ++} ++ ++static int at91rm9200_pcm_suspend(struct platform_device *pdev, ++ struct snd_soc_cpu_dai *dai) ++{ ++ struct snd_pcm_runtime *runtime = dai->runtime; ++ struct at91rm9200_runtime_data *prtd; ++ at91rm9200_pcm_dma_params_t *params; ++ ++ if (!runtime) ++ return 0; ++ ++ prtd = runtime->private_data; ++ params = prtd->params; ++ ++ /* disable the PDC and save the PDC registers */ ++ ++ at91_ssc_write(params->pdc->ptcr, params->mask->pdc_disable); ++ ++ prtd->pdc_xpr_save = at91_ssc_read(params->pdc->xpr); ++ prtd->pdc_xcr_save = at91_ssc_read(params->pdc->xcr); ++ prtd->pdc_xnpr_save = at91_ssc_read(params->pdc->xnpr); ++ prtd->pdc_xncr_save = at91_ssc_read(params->pdc->xncr); ++ ++ return 0; ++} ++ ++static int at91rm9200_pcm_resume(struct platform_device *pdev, ++ struct snd_soc_cpu_dai *dai) ++{ ++ struct snd_pcm_runtime *runtime = dai->runtime; ++ struct at91rm9200_runtime_data *prtd; ++ at91rm9200_pcm_dma_params_t *params; ++ ++ if (!runtime) ++ return 0; ++ ++ prtd = runtime->private_data; ++ params = prtd->params; ++ ++ /* restore the PDC registers and enable the PDC */ ++ at91_ssc_write(params->pdc->xpr, prtd->pdc_xpr_save); ++ at91_ssc_write(params->pdc->xcr, prtd->pdc_xcr_save); ++ at91_ssc_write(params->pdc->xnpr, prtd->pdc_xnpr_save); ++ at91_ssc_write(params->pdc->xncr, prtd->pdc_xncr_save); ++ ++ at91_ssc_write(params->pdc->ptcr, params->mask->pdc_enable); ++ return 0; ++} ++ ++struct snd_soc_platform at91rm9200_soc_platform = { ++ .name = "at91rm9200-audio", ++ .pcm_ops = &at91rm9200_pcm_ops, ++ .pcm_new = at91rm9200_pcm_new, ++ .pcm_free = at91rm9200_pcm_free_dma_buffers, ++ .suspend = at91rm9200_pcm_suspend, ++ .resume = at91rm9200_pcm_resume, ++}; ++ ++EXPORT_SYMBOL_GPL(at91rm9200_soc_platform); ++ ++MODULE_AUTHOR("Frank Mandarino <fmandarino@endrelia.com>"); ++MODULE_DESCRIPTION("Atmel AT91RM9200 PCM module"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/at91/at91rm9200-pcm.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/at91/at91rm9200-pcm.h +@@ -0,0 +1,75 @@ ++/* ++ * at91rm9200-pcm.h - ALSA PCM interface for the Atmel AT91RM9200 chip ++ * ++ * Author: Frank Mandarino <fmandarino@endrelia.com> ++ * Endrelia Technologies Inc. ++ * Created: Mar 3, 2006 ++ * ++ * Based on pxa2xx-pcm.h by: ++ * ++ * Author: Nicolas Pitre ++ * Created: Nov 30, 2004 ++ * Copyright: MontaVista Software, Inc. ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++/* ++ * Registers and status bits that are required by the PCM driver. ++ */ ++struct at91rm9200_ssc_regs { ++ void __iomem *cr; /* SSC control */ ++ void __iomem *ier; /* SSC interrupt enable */ ++ void __iomem *idr; /* SSC interrupt disable */ ++}; ++ ++struct at91rm9200_pdc_regs { ++ void __iomem *xpr; /* PDC recv/trans pointer */ ++ void __iomem *xcr; /* PDC recv/trans counter */ ++ void __iomem *xnpr; /* PDC next recv/trans pointer */ ++ void __iomem *xncr; /* PDC next recv/trans counter */ ++ void __iomem *ptcr; /* PDC transfer control */ ++}; ++ ++struct at91rm9200_ssc_mask { ++ u32 ssc_enable; /* SSC recv/trans enable */ ++ u32 ssc_disable; /* SSC recv/trans disable */ ++ u32 ssc_endx; /* SSC ENDTX or ENDRX */ ++ u32 ssc_endbuf; /* SSC TXBUFE or RXBUFF */ ++ u32 pdc_enable; /* PDC recv/trans enable */ ++ u32 pdc_disable; /* PDC recv/trans disable */ ++}; ++ ++ ++/* ++ * This structure, shared between the PCM driver and the interface, ++ * contains all information required by the PCM driver to perform the ++ * PDC DMA operation. All fields except dma_intr_handler() are initialized ++ * by the interface. The dms_intr_handler() pointer is set by the PCM ++ * driver and called by the interface SSC interrupt handler if it is ++ * non-NULL. ++ */ ++typedef struct { ++ char *name; /* stream identifier */ ++ int pdc_xfer_size; /* PDC counter increment in bytes */ ++ struct at91rm9200_ssc_regs *ssc; /* SSC register addresses */ ++ struct at91rm9200_pdc_regs *pdc; /* PDC receive/transmit registers */ ++ struct at91rm9200_ssc_mask *mask;/* SSC & PDC status bits */ ++ snd_pcm_substream_t *substream; ++ void (*dma_intr_handler)(u32, snd_pcm_substream_t *); ++} at91rm9200_pcm_dma_params_t; ++ ++extern struct snd_soc_cpu_dai at91rm9200_i2s_dai[3]; ++extern struct snd_soc_platform at91rm9200_soc_platform; ++ ++ ++/* ++ * SSC I/O helpers. ++ * E.g., at91_ssc_write(AT91_SSC(1) + AT91_SSC_CR, AT91_SSC_RXEN); ++ */ ++#define AT91_SSC(x) (((x)==0) ? AT91_VA_BASE_SSC0 :\ ++ ((x)==1) ? AT91_VA_BASE_SSC1 : ((x)==2) ? AT91_VA_BASE_SSC2 : NULL) ++#define at91_ssc_read(a) ((unsigned long) __raw_readl(a)) ++#define at91_ssc_write(a,v) __raw_writel((v),(a)) +Index: linux-2.6-pxa-new/sound/soc/imx/imx-ssi.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/imx/imx-ssi.c +@@ -0,0 +1,452 @@ ++/* ++ * imx-ssi.c -- SSI driver for Freescale IMX ++ * ++ * Copyright 2006 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * Based on mxc-alsa-mc13783 (C) 2006 Freescale. ++ * ++ * 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. ++ * ++ * Revision history ++ * 29th Aug 2006 Initial version. ++ * ++ */ ++ ++#define IMX_DSP_DAIFMT \ ++ ( SND_SOC_DAIFMT_DSP__A |SND_SOC_DAIFMT_DSP_B | \ ++ SND_SOC_DAIFMT_CBS_CFS |SND_SOC_DAIFMT_CBM_CFS | \ ++ SND_SOC_DAIFMT_CBS_CFM |SND_SOC_DAIFMT_NB_NF |\ ++ SND_SOC_DAIFMT_NB_IF) ++ ++#define IMX_DSP_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define IMX_DSP_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | \ ++ SNDRV_PCM_RATE_16000 | SNDRV_PCM_RATE_22050 | \ ++ SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_88200 | \ ++ SNDRV_PCM_RATE_96000) ++ ++#define IMX_DSP_BITS \ ++ (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE | \ ++ SNDRV_PCM_FMTBIT_S24_LE) ++ ++static struct snd_soc_dai_mode imx_dsp_pcm_modes[] = { ++ ++ /* frame master and clock slave mode */ ++ {IMX_DSP_DAIFMT | SND_SOC_DAIFMT_CBM_CFS, ++ SND_SOC_DAITDM_LRDW(0,0), IMX_DSP_BITS, IMX_DSP_RATES, ++ IMX_DSP_DIR, 0, SND_SOC_FS_ALL, ++ SND_SOC_FSB(32) | SND_SOC_FSB(32) | SND_SOC_FSB(16)}, ++ ++}; ++ ++static imx_pcm_dma_params_t imx_ssi1_pcm_stereo_out = { ++ .name = "SSI1 PCM Stereo out", ++ .params = { ++ .bd_number = 1, ++ .transfer_type = emi_2_per, ++ .watermark_level = SDMA_TXFIFO_WATERMARK, ++ .word_size = TRANSFER_16BIT, // maybe add this in setup func ++ .per_address = SSI1_STX0, ++ .event_id = DMA_REQ_SSI1_TX1, ++ .peripheral_type = SSI, ++ }, ++}; ++ ++static imx_pcm_dma_params_t imx_ssi1_pcm_stereo_in = { ++ .name = "SSI1 PCM Stereo in", ++ .params = { ++ .bd_number = 1, ++ .transfer_type = per_2_emi, ++ .watermark_level = SDMA_RXFIFO_WATERMARK, ++ .word_size = TRANSFER_16BIT, // maybe add this in setup func ++ .per_address = SSI1_SRX0, ++ .event_id = DMA_REQ_SSI1_RX1, ++ .peripheral_type = SSI, ++ }, ++}; ++ ++static imx_pcm_dma_params_t imx_ssi2_pcm_stereo_out = { ++ .name = "SSI2 PCM Stereo out", ++ .params = { ++ .bd_number = 1, ++ .transfer_type = per_2_emi, ++ .watermark_level = SDMA_TXFIFO_WATERMARK, ++ .word_size = TRANSFER_16BIT, // maybe add this in setup func ++ .per_address = SSI2_STX0, ++ .event_id = DMA_REQ_SSI2_TX1, ++ .peripheral_type = SSI, ++ }, ++}; ++ ++static imx_pcm_dma_params_t imx_ssi2_pcm_stereo_in = { ++ .name = "SSI2 PCM Stereo in", ++ .params = { ++ .bd_number = 1, ++ .transfer_type = per_2_emi, ++ .watermark_level = SDMA_RXFIFO_WATERMARK, ++ .word_size = TRANSFER_16BIT, // maybe add this in setup func ++ .per_address = SSI2_SRX0, ++ .event_id = DMA_REQ_SSI2_RX1, ++ .peripheral_type = SSI, ++ }, ++}; ++ ++static int imx_dsp_startup(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ ++ if (!rtd->cpu_dai->active) { ++ ++ } ++ ++ return 0; ++} ++ ++static int imx_ssi1_hw_tx_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 bfs, div; ++ ++ bfs = SND_SOC_FSBD_REAL(rtd->cpu_dai->dai_runtime.bfs); ++ ++ SSI1_STCR = 0; ++ SSI1_STCCR = 0; ++ ++ /* DAI mode */ ++ switch(rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_DSP_B: ++ SSI1_STCR |= SSI_STCR_TEFS; // data 1 bit after sync ++ case SND_SOC_DAIFMT_DSP_A: ++ SSI1_STCR |= SSI_STCR_TFSL; // frame is 1 bclk long ++ break; ++ } ++ ++ /* DAI clock inversion */ ++ switch(rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_INV_MASK) { ++ case SND_SOC_DAIFMT_IB_IF: ++ SSI1_STCR |= SSI_STCR_TFSI | SSI_STCR_TSCKP; ++ break; ++ case SND_SOC_DAIFMT_IB_NF: ++ SSI1_STCR |= SSI_STCR_TSCKP; ++ break; ++ case SND_SOC_DAIFMT_NB_IF: ++ SSI1_STCR |= SSI_STCR_TFSI; ++ break; ++ } ++ ++ /* DAI data (word) size */ ++ switch(rtd->codec_dai->dai_runtime.pcmfmt) { ++ case SNDRV_PCM_FMTBIT_S16_LE: ++ SSI1_STCCR |= SSI_STCCR_WL(16); ++ break; ++ case SNDRV_PCM_FMTBIT_S20_3LE: ++ SSI1_STCCR |= SSI_STCCR_WL(20); ++ break; ++ case SNDRV_PCM_FMTBIT_S24_LE: ++ SSI1_STCCR |= SSI_STCCR_WL(24); ++ break; ++ } ++ ++ /* DAI clock master masks */ ++ switch(rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK){ ++ case SND_SOC_DAIFMT_CBM_CFM: ++ SSI1_STCR |= SSI_STCR_TFDIR | SSI_STCR_TXDIR; ++ break; ++ case SND_SOC_DAIFMT_CBS_CFM: ++ SSI1_STCR |= SSI_STCR_TFDIR; ++ break; ++ case SND_SOC_DAIFMT_CBM_CFS: ++ SSI1_STCR |= SSI_STCR_TXDIR; ++ break; ++ } ++ ++ /* DAI BCLK ratio to SYSCLK / MCLK */ ++ /* prescaler modulus - todo */ ++ switch (bfs) { ++ case 2: ++ break; ++ case 4: ++ break; ++ case 8: ++ break; ++ case 16: ++ break; ++ } ++ ++ /* TDM - todo, only fifo 0 atm */ ++ SSI1_STCR |= SSI_STCR_TFEN0; ++ SSI1_STCCR |= SSI_STCCR_DC(params_channels(params)); ++ ++ return 0; ++} ++ ++static int imx_ssi1_hw_rx_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 bfs, div; ++ ++ bfs = SND_SOC_FSBD_REAL(rtd->cpu_dai->dai_runtime.bfs); ++ ++ SSI1_SRCR = 0; ++ SSI1_SRCCR = 0; ++ ++ /* DAI mode */ ++ switch(rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_DSP_B: ++ SSI1_SRCR |= SSI_SRCR_REFS; // data 1 bit after sync ++ case SND_SOC_DAIFMT_DSP_A: ++ SSI1_SRCR |= SSI_SRCR_RFSL; // frame is 1 bclk long ++ break; ++ } ++ ++ /* DAI clock inversion */ ++ switch(rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_INV_MASK) { ++ case SND_SOC_DAIFMT_IB_IF: ++ SSI1_SRCR |= SSI_SRCR_TFSI | SSI_SRCR_TSCKP; ++ break; ++ case SND_SOC_DAIFMT_IB_NF: ++ SSI1_SRCR |= SSI_SRCR_RSCKP; ++ break; ++ case SND_SOC_DAIFMT_NB_IF: ++ SSI1_SRCR |= SSI_SRCR_RFSI; ++ break; ++ } ++ ++ /* DAI data (word) size */ ++ switch(rtd->codec_dai->dai_runtime.pcmfmt) { ++ case SNDRV_PCM_FMTBIT_S16_LE: ++ SSI1_SRCCR |= SSI_SRCCR_WL(16); ++ break; ++ case SNDRV_PCM_FMTBIT_S20_3LE: ++ SSI1_SRCCR |= SSI_SRCCR_WL(20); ++ break; ++ case SNDRV_PCM_FMTBIT_S24_LE: ++ SSI1_SRCCR |= SSI_SRCCR_WL(24); ++ break; ++ } ++ ++ /* DAI clock master masks */ ++ switch(rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK){ ++ case SND_SOC_DAIFMT_CBM_CFM: ++ SSI1_SRCR |= SSI_SRCR_RFDIR | SSI_SRCR_RXDIR; ++ break; ++ case SND_SOC_DAIFMT_CBS_CFM: ++ SSI1_SRCR |= SSI_SRCR_RFDIR; ++ break; ++ case SND_SOC_DAIFMT_CBM_CFS: ++ SSI1_SRCR |= SSI_SRCR_RXDIR; ++ break; ++ } ++ ++ /* DAI BCLK ratio to SYSCLK / MCLK */ ++ /* prescaler modulus - todo */ ++ switch (bfs) { ++ case 2: ++ break; ++ case 4: ++ break; ++ case 8: ++ break; ++ case 16: ++ break; ++ } ++ ++ /* TDM - todo, only fifo 0 atm */ ++ SSI1_SRCR |= SSI_SRCR_RFEN0; ++ SSI1_SRCCR |= SSI_SRCCR_DC(params_channels(params)); ++ ++ return 0; ++} ++ ++static int imx_ssi_dsp_hw_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ ++ /* clear register if not enabled */ ++ if(!(SSI1_SCR & SSI_SCR_SSIEN)) ++ SSI1_SCR = 0; ++ ++ /* async */ ++ if (rtd->cpu_dai->flags & SND_SOC_DAI_ASYNC) ++ SSI1_SCR |= SSI_SCR_SYN; ++ ++ /* DAI mode */ ++ switch(rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_I2S: ++ case SND_SOC_DAIFMT_LEFT_J: ++ SSI1_SCR |= SSI_SCR_NET; ++ break; ++ } ++ ++ /* TDM - to complete */ ++ ++ /* Tx/Rx config */ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { ++ return imx_ssi1_dsp_hw_tx_params(substream, params); ++ } else { ++ return imx_ssi1_dsp_hw_rx_params(substream, params); ++ } ++} ++ ++ ++ ++static int imx_ssi_dsp_trigger(struct snd_pcm_substream *substream, int cmd) ++{ ++ int ret = 0; ++ ++ switch (cmd) { ++ case SNDRV_PCM_TRIGGER_START: ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { ++ SSI1_SCR |= SSI_SCR_TE; ++ SSI1_SIER |= SSI_SIER_TDMAE; ++ } else { ++ SSI1_SCR |= SSI_SCR_RE; ++ SSI1_SIER |= SSI_SIER_RDMAE; ++ } ++ SSI1_SCR |= SSI_SCR_SSIEN; ++ ++ break; ++ case SNDRV_PCM_TRIGGER_RESUME: ++ break; ++ case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ SSI1_SCR |= SSI_SCR_TE; ++ else ++ SSI1_SCR |= SSI_SCR_RE; ++ break ++ case SNDRV_PCM_TRIGGER_STOP: ++ case SNDRV_PCM_TRIGGER_SUSPEND: ++ break; ++ case SNDRV_PCM_TRIGGER_PAUSE_PUSH: ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ SSI1_SCR &= ~SSI_SCR_TE; ++ else ++ SSI1_SCR &= ~SSI_SCR_RE; ++ break; ++ default: ++ ret = -EINVAL; ++ } ++ ++ return ret; ++} ++ ++static void imx_ssi_dsp_shutdown(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ ++ /* shutdown SSI */ ++ if (!rtd->cpu_dai->active) { ++ if(rtd->cpu_dai->id == 0) ++ SSI1_SCR &= ~SSI_SCR_SSIEN; ++ else ++ SSI2_SCR &= ~SSI_SCR_SSIEN; ++ } ++} ++ ++#ifdef CONFIG_PM ++static int imx_ssi_dsp_suspend(struct platform_device *dev, ++ struct snd_soc_cpu_dai *dai) ++{ ++ if(!dai->active) ++ return 0; ++ ++ if(rtd->cpu_dai->id == 0) ++ SSI1_SCR &= ~SSI_SCR_SSIEN; ++ else ++ SSI2_SCR &= ~SSI_SCR_SSIEN; ++ ++ return 0; ++} ++ ++static int imx_ssi_dsp_resume(struct platform_device *pdev, ++ struct snd_soc_cpu_dai *dai) ++{ ++ if(!dai->active) ++ return 0; ++ ++ if(rtd->cpu_dai->id == 0) ++ SSI1_SCR |= SSI_SCR_SSIEN; ++ else ++ SSI2_SCR |= SSI_SCR_SSIEN; ++ ++ return 0; ++} ++ ++#else ++#define imx_ssi_dsp_suspend NULL ++#define imx_ssi_dsp_resume NULL ++#endif ++ ++static unsigned int imx_ssi_config_dsp_sysclk(struct snd_soc_cpu_dai *iface, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ return clk; ++} ++ ++struct snd_soc_cpu_dai imx_ssi_dsp_dai = { ++ .name = "imx-dsp-1", ++ .id = 0, ++ .type = SND_SOC_DAI_PCM, ++ .suspend = imx_ssi_dsp_suspend, ++ .resume = imx_ssi_dsp_resume, ++ .config_sysclk = imx_ssi_config_dsp_sysclk, ++ .playback = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .capture = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .ops = { ++ .startup = imx_ssi_dsp_startup, ++ .shutdown = imx_ssi_dsp_shutdown, ++ .trigger = imx_ssi_trigger, ++ .hw_params = imx_ssi_dsp_hw_params,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(imx_dsp_modes), ++ .mode = imx_dsp_modes,}, ++}, ++{ ++ .name = "imx-dsp-2", ++ .id = 1, ++ .type = SND_SOC_DAI_PCM, ++ .suspend = imx_ssi_dsp_suspend, ++ .resume = imx_ssi_dsp_resume, ++ .config_sysclk = imx_ssi_config_dsp_sysclk, ++ .playback = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .capture = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .ops = { ++ .startup = imx_dsp_startup, ++ .shutdown = imx_dsp_shutdown, ++ .trigger = imx_ssi1_trigger, ++ .hw_params = imx_ssi1_pcm_hw_params,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(imx_dsp_modes), ++ .mode = imx_dsp_modes,}, ++}; ++ ++ ++EXPORT_SYMBOL_GPL(imx_ssi_dsp_dai); ++ ++/* Module information */ ++MODULE_AUTHOR("Liam Girdwood, liam.girdwood@wolfsonmicro.com, www.wolfsonmicro.com"); ++MODULE_DESCRIPTION("i.MX ASoC SSI driver"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/imx/Kconfig +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/imx/Kconfig +@@ -0,0 +1,31 @@ ++menu "SoC Audio for the Freescale i.MX" ++ ++config SND_MXC_SOC ++ tristate "SoC Audio for the Freescale i.MX CPU" ++ depends on ARCH_MXC && SND ++ select SND_PCM ++ help ++ Say Y or M if you want to add support for codecs attached to ++ the MXC AC97, I2S or SSP interface. You will also need ++ to select the audio interfaces to support below. ++ ++config SND_MXC_AC97 ++ tristate ++ select SND_AC97_CODEC ++ ++config SND_MXC_SOC_AC97 ++ tristate ++ select SND_AC97_BUS ++ ++config SND_MXC_SOC_SSI ++ tristate ++ ++config SND_MXC_SOC_MX3_WM8753 ++ tristate "SoC Audio support for MX31 - WM8753" ++ depends on SND_MXC_SOC && ARCH_MX3 ++ select SND_MXC_SOC_SSI ++ help ++ Say Y if you want to add support for SoC audio on MX31ADS ++ with the WM8753. ++ ++endmenu +Index: linux-2.6-pxa-new/sound/soc/imx/Makefile +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/imx/Makefile +@@ -0,0 +1,18 @@ ++# i.MX Platform Support ++snd-soc-imx21-objs := imx21-pcm.o ++snd-soc-imx31-objs := imx31-pcm.o ++snd-soc-imx-ac97-objs := imx-ac97.o ++snd-soc-imx-i2s-objs := imx-i2s.o ++ ++obj-$(CONFIG_SND_MXC_SOC) += snd-soc-imx.o ++obj-$(CONFIG_SND_MXC_SOC_AC97) += snd-soc-imx-ac97.o ++obj-$(CONFIG_SND_MXC_SOC_I2S) += snd-soc-imx-i2s.o ++ ++# i.MX Machine Support ++snd-soc-mx31ads-wm8753-objs := mx31ads_wm8753.o ++obj-$(CONFIG_SND_SOC_MX31ADS_WM8753) += snd-soc-mx31ads-wm8753.o ++snd-soc-mx21ads-wm8753-objs := mx21ads_wm8753.o ++obj-$(CONFIG_SND_SOC_MX21ADS_WM8753) += snd-soc-mx21ads-wm8753.o ++snd-soc-mx21ads-wm8731-objs := mx21ads_wm8731.o ++obj-$(CONFIG_SND_SOC_MX21ADS_WM8731) += snd-soc-mx21ads-wm8731.o ++ +Index: linux-2.6-pxa-new/sound/Makefile +=================================================================== +--- linux-2.6-pxa-new.orig/sound/Makefile ++++ linux-2.6-pxa-new/sound/Makefile +@@ -5,7 +5,7 @@ obj-$(CONFIG_SOUND) += soundcore.o + obj-$(CONFIG_SOUND_PRIME) += sound_firmware.o + obj-$(CONFIG_SOUND_PRIME) += oss/ + obj-$(CONFIG_DMASOUND) += oss/ +-obj-$(CONFIG_SND) += core/ i2c/ drivers/ isa/ pci/ ppc/ arm/ synth/ usb/ sparc/ parisc/ pcmcia/ mips/ ++obj-$(CONFIG_SND) += core/ i2c/ drivers/ isa/ pci/ ppc/ arm/ synth/ usb/ sparc/ parisc/ pcmcia/ mips/ soc/ + obj-$(CONFIG_SND_AOA) += aoa/ + + ifeq ($(CONFIG_SND),y) +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8711.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8711.c +@@ -0,0 +1,843 @@ ++/* ++ * wm8711.c -- WM8711 ALSA SoC Audio driver ++ * ++ * Copyright 2006 Wolfson Microelectronics ++ * ++ * Author: Mike Arthur <linux@wolfsonmicro.com> ++ * ++ * Based on wm8711.c by Richard Purdie ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/init.h> ++#include <linux/delay.h> ++#include <linux/pm.h> ++#include <linux/i2c.h> ++#include <linux/platform_device.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/pcm_params.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++#include <sound/initval.h> ++ ++#include "wm8711.h" ++ ++#define AUDIO_NAME "wm8711" ++#define WM8711_VERSION "0.2" ++ ++/* ++ * Debug ++ */ ++ ++#define WM8711_DEBUG 0 ++ ++#ifdef WM8711_DEBUG ++#define dbg(format, arg...) \ ++ printk(KERN_DEBUG AUDIO_NAME ": " format "\n" , ## arg) ++#else ++#define dbg(format, arg...) do {} while (0) ++#endif ++#define err(format, arg...) \ ++ printk(KERN_ERR AUDIO_NAME ": " format "\n" , ## arg) ++#define info(format, arg...) \ ++ printk(KERN_INFO AUDIO_NAME ": " format "\n" , ## arg) ++#define warn(format, arg...) \ ++ printk(KERN_WARNING AUDIO_NAME ": " format "\n" , ## arg) ++ ++struct snd_soc_codec_device soc_codec_dev_wm8711; ++ ++/* ++ * wm8711 register cache ++ * We can't read the WM8711 register space when we are ++ * using 2 wire for device control, so we cache them instead. ++ * There is no point in caching the reset register ++ */ ++static const u16 wm8711_reg[WM8711_CACHEREGNUM] = { ++ 0x0079, 0x0079, 0x000a, 0x0008, ++ 0x009f, 0x000a, 0x0000, 0x0000 ++}; ++ ++#define WM8711_DAIFMT \ ++ (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_LEFT_J | SND_SOC_DAIFMT_RIGHT_J | \ ++ SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_NB_IF | SND_SOC_DAIFMT_IB_NF | \ ++ SND_SOC_DAIFMT_IB_IF) ++ ++#define WM8711_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK) ++ ++#define WM8711_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000) ++ ++#define WM8711_HIFI_BITS \ ++ (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE | \ ++ SNDRV_PCM_FMTBIT_S24_3LE | SNDRV_PCM_FMTBIT_S24_LE | \ ++ SNDRV_PCM_FMTBIT_S32_LE) ++ ++static struct snd_soc_dai_mode wm8711_modes[] = { ++ /* codec frame and clock master modes */ ++ /* 8k */ ++ { ++ .fmt = WM8711_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8711_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8711_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 1536, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8711_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8711_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8711_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 2304, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8711_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8711_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8711_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 1408, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8711_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8711_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8711_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 2112, ++ .bfs = 64, ++ }, ++ ++ /* 32k */ ++ { ++ .fmt = WM8711_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8711_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = WM8711_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 384, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8711_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8711_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_32000, ++ .pcmdir = WM8711_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 576, ++ .bfs = 64, ++ }, ++ ++ /* 44.1k & 48k */ ++ { ++ .fmt = WM8711_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8711_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8711_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 256, ++ .bfs = 64, ++ }, ++ { ++ .fmt = WM8711_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8711_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8711_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 384, ++ .bfs = 64, ++ }, ++ ++ /* 88.2 & 96k */ ++ { ++ .fmt = WM8711_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8711_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000, ++ .pcmdir = WM8711_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 128, ++ .bfs = 64, ++ ++ }, ++ { ++ .fmt = WM8711_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8711_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000, ++ .pcmdir = WM8711_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 192, ++ .bfs = 64, ++ }, ++ ++ /* USB codec frame and clock master modes */ ++ /* 8k */ ++ { ++ .fmt = WM8711_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8711_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_8000, ++ .pcmdir = WM8711_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 1500, ++ .bfs = SND_SOC_FSBD(1), ++ }, ++ ++ /* 44.1k */ ++ { ++ .fmt = WM8711_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8711_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_44100, ++ .pcmdir = WM8711_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 272, ++ .bfs = SND_SOC_FSBD(1), ++ }, ++ ++ /* 48k */ ++ { ++ .fmt = WM8711_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8711_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_48000, ++ .pcmdir = WM8711_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 250, ++ .bfs = SND_SOC_FSBD(1), ++ }, ++ ++ /* 88.2k */ ++ { ++ .fmt = WM8711_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8711_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_88200, ++ .pcmdir = WM8711_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 136, ++ .bfs = SND_SOC_FSBD(1), ++ }, ++ ++ /* 96k */ ++ { ++ .fmt = WM8711_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8711_HIFI_BITS, ++ .pcmrate = SNDRV_PCM_RATE_96000, ++ .pcmdir = WM8711_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 125, ++ .bfs = SND_SOC_FSBD(1), ++ }, ++ ++ /* codec frame and clock slave modes */ ++ { ++ .fmt = WM8711_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = WM8711_HIFI_BITS, ++ .pcmrate = WM8711_RATES, ++ .pcmdir = WM8711_DIR, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = SND_SOC_FSB_ALL, ++ }, ++}; ++ ++/* ++ * read wm8711 register cache ++ */ ++static inline unsigned int wm8711_read_reg_cache(struct snd_soc_codec * codec, ++ unsigned int reg) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg == WM8711_RESET) ++ return 0; ++ if (reg >= WM8711_CACHEREGNUM) ++ return -1; ++ return cache[reg]; ++} ++ ++/* ++ * write wm8711 register cache ++ */ ++static inline void wm8711_write_reg_cache(struct snd_soc_codec *codec, ++ u16 reg, unsigned int value) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg >= WM8711_CACHEREGNUM) ++ return; ++ cache[reg] = value; ++} ++ ++/* ++ * write to the WM8711 register space ++ */ ++static int wm8711_write(struct snd_soc_codec * codec, unsigned int reg, ++ unsigned int value) ++{ ++ u8 data[2]; ++ ++ /* data is ++ * D15..D9 WM8753 register offset ++ * D8...D0 register data ++ */ ++ data[0] = (reg << 1) | ((value >> 8) & 0x0001); ++ data[1] = value & 0x00ff; ++ ++ wm8711_write_reg_cache (codec, reg, value); ++ if (codec->hw_write(codec->control_data, data, 2) == 2) ++ return 0; ++ else ++ return -EIO; ++} ++ ++#define wm8711_reset(c) wm8711_write(c, WM8711_RESET, 0) ++ ++static const struct snd_kcontrol_new wm8711_snd_controls[] = { ++ ++SOC_DOUBLE_R("Master Playback Volume", WM8711_LOUT1V, WM8711_ROUT1V, ++ 0, 127, 0), ++SOC_DOUBLE_R("Master Playback ZC Switch", WM8711_LOUT1V, WM8711_ROUT1V, ++ 7, 1, 0), ++ ++}; ++ ++/* add non dapm controls */ ++static int wm8711_add_controls(struct snd_soc_codec *codec) ++{ ++ int err, i; ++ ++ for (i = 0; i < ARRAY_SIZE(wm8711_snd_controls); i++) { ++ err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&wm8711_snd_controls[i],codec, NULL)); ++ if (err < 0) ++ return err; ++ } ++ ++ return 0; ++} ++ ++/* Output Mixer */ ++static const snd_kcontrol_new_t wm8711_output_mixer_controls[] = { ++SOC_DAPM_SINGLE("Line Bypass Switch", WM8711_APANA, 3, 1, 0), ++SOC_DAPM_SINGLE("HiFi Playback Switch", WM8711_APANA, 4, 1, 0), ++}; ++ ++static const struct snd_soc_dapm_widget wm8711_dapm_widgets[] = { ++SND_SOC_DAPM_MIXER("Output Mixer", WM8711_PWR, 4, 1, ++ &wm8711_output_mixer_controls[0], ++ ARRAY_SIZE(wm8711_output_mixer_controls)), ++SND_SOC_DAPM_DAC("DAC", "HiFi Playback", WM8711_PWR, 3, 1), ++SND_SOC_DAPM_OUTPUT("LOUT"), ++SND_SOC_DAPM_OUTPUT("LHPOUT"), ++SND_SOC_DAPM_OUTPUT("ROUT"), ++SND_SOC_DAPM_OUTPUT("RHPOUT"), ++}; ++ ++static const char *intercon[][3] = { ++ /* output mixer */ ++ {"Output Mixer", "Line Bypass Switch", "Line Input"}, ++ {"Output Mixer", "HiFi Playback Switch", "DAC"}, ++ ++ /* outputs */ ++ {"RHPOUT", NULL, "Output Mixer"}, ++ {"ROUT", NULL, "Output Mixer"}, ++ {"LHPOUT", NULL, "Output Mixer"}, ++ {"LOUT", NULL, "Output Mixer"}, ++ ++ /* terminator */ ++ {NULL, NULL, NULL}, ++}; ++ ++static int wm8711_add_widgets(struct snd_soc_codec *codec) ++{ ++ int i; ++ ++ for(i = 0; i < ARRAY_SIZE(wm8711_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &wm8711_dapm_widgets[i]); ++ } ++ ++ /* set up audio path interconnects */ ++ for(i = 0; intercon[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, intercon[i][0], intercon[i][1], ++ intercon[i][2]); ++ } ++ ++ snd_soc_dapm_new_widgets(codec); ++ return 0; ++} ++ ++struct _coeff_div { ++ u32 mclk; ++ u32 rate; ++ u16 fs; ++ u8 sr:4; ++ u8 bosr:1; ++ u8 usb:1; ++}; ++ ++/* codec mclk clock divider coefficients */ ++static const struct _coeff_div coeff_div[] = { ++ /* 48k */ ++ {12288000, 48000, 256, 0x0, 0x0, 0x0}, ++ {18432000, 48000, 384, 0x0, 0x1, 0x0}, ++ {12000000, 48000, 250, 0x0, 0x0, 0x1}, ++ ++ /* 32k */ ++ {12288000, 32000, 384, 0x6, 0x0, 0x0}, ++ {18432000, 32000, 576, 0x6, 0x1, 0x0}, ++ ++ /* 8k */ ++ {12288000, 8000, 1536, 0x3, 0x0, 0x0}, ++ {18432000, 8000, 2304, 0x3, 0x1, 0x0}, ++ {11289600, 8000, 1408, 0xb, 0x0, 0x0}, ++ {16934400, 8000, 2112, 0xb, 0x1, 0x0}, ++ {12000000, 8000, 1500, 0x3, 0x0, 0x1}, ++ ++ /* 96k */ ++ {12288000, 96000, 128, 0x7, 0x0, 0x0}, ++ {18432000, 96000, 192, 0x7, 0x1, 0x0}, ++ {12000000, 96000, 125, 0x7, 0x0, 0x1}, ++ ++ /* 44.1k */ ++ {11289600, 44100, 256, 0x8, 0x0, 0x0}, ++ {16934400, 44100, 384, 0x8, 0x1, 0x0}, ++ {12000000, 44100, 272, 0x8, 0x1, 0x1}, ++ ++ /* 88.2k */ ++ {11289600, 88200, 128, 0xf, 0x0, 0x0}, ++ {16934400, 88200, 192, 0xf, 0x1, 0x0}, ++ {12000000, 88200, 136, 0xf, 0x1, 0x1}, ++}; ++ ++static inline int get_coeff(int mclk, int rate) ++{ ++ int i; ++ ++ for (i = 0; i < ARRAY_SIZE(coeff_div); i++) { ++ if (coeff_div[i].rate == rate && coeff_div[i].mclk == mclk) ++ return i; ++ } ++ return 0; ++} ++ ++/* WM8711 supports numerous clocks per sample rate */ ++static unsigned int wm8711_config_sysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ dai->mclk = 0; ++ ++ /* check that the calculated FS and rate actually match a clock from ++ * the machine driver */ ++ if (info->fs * info->rate == clk) ++ dai->mclk = clk; ++ ++ return dai->mclk; ++} ++ ++static int wm8711_pcm_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 iface = 0, srate; ++ int i = get_coeff(rtd->codec_dai->mclk, ++ snd_soc_get_rate(rtd->codec_dai->dai_runtime.pcmrate)); ++ ++ /* set master/slave audio interface */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK) { ++ case SND_SOC_DAIFMT_CBM_CFM: ++ iface |= 0x0040; ++ break; ++ case SND_SOC_DAIFMT_CBS_CFS: ++ break; ++ } ++ srate = (coeff_div[i].sr << 2) | (coeff_div[i].bosr << 1) | ++ coeff_div[i].usb; ++ wm8711_write(codec, WM8711_SRATE, srate); ++ ++ /* interface format */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_I2S: ++ iface |= 0x0002; ++ break; ++ case SND_SOC_DAIFMT_RIGHT_J: ++ break; ++ case SND_SOC_DAIFMT_LEFT_J: ++ iface |= 0x0001; ++ break; ++ case SND_SOC_DAIFMT_DSP_A: ++ iface |= 0x0003; ++ break; ++ case SND_SOC_DAIFMT_DSP_B: ++ iface |= 0x0013; ++ break; ++ } ++ ++ /* bit size */ ++ switch (rtd->codec_dai->dai_runtime.pcmfmt) { ++ case SNDRV_PCM_FORMAT_S16_LE: ++ break; ++ case SNDRV_PCM_FORMAT_S20_3LE: ++ iface |= 0x0004; ++ break; ++ case SNDRV_PCM_FORMAT_S24_LE: ++ iface |= 0x0008; ++ break; ++ case SNDRV_PCM_FORMAT_S32_LE: ++ iface |= 0x000c; ++ break; ++ } ++ ++ /* clock inversion */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_INV_MASK) { ++ case SND_SOC_DAIFMT_NB_NF: ++ break; ++ case SND_SOC_DAIFMT_IB_IF: ++ iface |= 0x0090; ++ break; ++ case SND_SOC_DAIFMT_IB_NF: ++ iface |= 0x0080; ++ break; ++ case SND_SOC_DAIFMT_NB_IF: ++ iface |= 0x0010; ++ break; ++ } ++ ++ /* set iface */ ++ wm8711_write(codec, WM8711_IFACE, iface); ++ ++ /* set active */ ++ wm8711_write(codec, WM8711_ACTIVE, 0x0001); ++ return 0; ++} ++ ++static void wm8711_shutdown(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ /* deactivate */ ++ if (!codec->active) { ++ udelay(50); ++ wm8711_write(codec, WM8711_ACTIVE, 0x0); ++ } ++} ++ ++static int wm8711_mute(struct snd_soc_codec *codec, ++ struct snd_soc_codec_dai *dai, int mute) ++{ ++ u16 mute_reg = wm8711_read_reg_cache(codec, WM8711_APDIGI) & 0xfff7; ++ if (mute) ++ wm8711_write(codec, WM8711_APDIGI, mute_reg | 0x8); ++ else ++ wm8711_write(codec, WM8711_APDIGI, mute_reg); ++ ++ return 0; ++} ++ ++static int wm8711_dapm_event(struct snd_soc_codec *codec, int event) ++{ ++ u16 reg = wm8711_read_reg_cache(codec, WM8711_PWR) & 0xff7f; ++ ++ switch (event) { ++ case SNDRV_CTL_POWER_D0: /* full On */ ++ /* vref/mid, osc on, dac unmute */ ++ wm8711_write(codec, WM8711_PWR, reg); ++ break; ++ case SNDRV_CTL_POWER_D1: /* partial On */ ++ case SNDRV_CTL_POWER_D2: /* partial On */ ++ break; ++ case SNDRV_CTL_POWER_D3hot: /* Off, with power */ ++ /* everything off except vref/vmid, */ ++ wm8711_write(codec, WM8711_PWR, reg | 0x0040); ++ break; ++ case SNDRV_CTL_POWER_D3cold: /* Off, without power */ ++ /* everything off, dac mute, inactive */ ++ wm8711_write(codec, WM8711_ACTIVE, 0x0); ++ wm8711_write(codec, WM8711_PWR, 0xffff); ++ break; ++ } ++ codec->dapm_state = event; ++ return 0; ++} ++ ++struct snd_soc_codec_dai wm8711_dai = { ++ .name = "WM8711", ++ .playback = { ++ .stream_name = "Playback", ++ .channels_min = 1, ++ .channels_max = 2, ++ }, ++ .config_sysclk = wm8711_config_sysclk, ++ .digital_mute = wm8711_mute, ++ .ops = { ++ .prepare = wm8711_pcm_prepare, ++ .shutdown = wm8711_shutdown, ++ }, ++ .caps = { ++ .num_modes = ARRAY_SIZE(wm8711_modes), ++ .mode = wm8711_modes, ++ }, ++}; ++EXPORT_SYMBOL_GPL(wm8711_dai); ++ ++static int wm8711_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ wm8711_write(codec, WM8711_ACTIVE, 0x0); ++ wm8711_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ return 0; ++} ++ ++static int wm8711_resume(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ int i; ++ u8 data[2]; ++ u16 *cache = codec->reg_cache; ++ ++ /* Sync reg_cache with the hardware */ ++ for (i = 0; i < ARRAY_SIZE(wm8711_reg); i++) { ++ data[0] = (i << 1) | ((cache[i] >> 8) & 0x0001); ++ data[1] = cache[i] & 0x00ff; ++ codec->hw_write(codec->control_data, data, 2); ++ } ++ wm8711_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ wm8711_dapm_event(codec, codec->suspend_dapm_state); ++ return 0; ++} ++ ++/* ++ * initialise the WM8711 driver ++ * register the mixer and dsp interfaces with the kernel ++ */ ++static int wm8711_init(struct snd_soc_device* socdev) ++{ ++ struct snd_soc_codec* codec = socdev->codec; ++ int reg, ret = 0; ++ ++ codec->name = "WM8711"; ++ codec->owner = THIS_MODULE; ++ codec->read = wm8711_read_reg_cache; ++ codec->write = wm8711_write; ++ codec->dapm_event = wm8711_dapm_event; ++ codec->dai = &wm8711_dai; ++ codec->num_dai = 1; ++ codec->reg_cache_size = ARRAY_SIZE(wm8711_reg); ++ codec->reg_cache = ++ kzalloc(sizeof(u16) * ARRAY_SIZE(wm8711_reg), GFP_KERNEL); ++ if (codec->reg_cache == NULL) ++ return -ENOMEM; ++ memcpy(codec->reg_cache, wm8711_reg, ++ sizeof(u16) * ARRAY_SIZE(wm8711_reg)); ++ codec->reg_cache_size = sizeof(u16) * ARRAY_SIZE(wm8711_reg); ++ ++ wm8711_reset(codec); ++ ++ /* register pcms */ ++ ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); ++ if (ret < 0) { ++ kfree(codec->reg_cache); ++ return ret; ++ } ++ ++ /* power on device */ ++ wm8711_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ ++ /* set the update bits */ ++ reg = wm8711_read_reg_cache(codec, WM8711_LOUT1V); ++ wm8711_write(codec, WM8711_LOUT1V, reg | 0x0100); ++ reg = wm8711_read_reg_cache(codec, WM8711_ROUT1V); ++ wm8711_write(codec, WM8711_ROUT1V, reg | 0x0100); ++ ++ wm8711_add_controls(codec); ++ wm8711_add_widgets(codec); ++ ret = snd_soc_register_card(socdev); ++ if (ret < 0) { ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++ } ++ ++ return ret; ++} ++ ++static struct snd_soc_device *wm8711_socdev; ++ ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ ++/* ++ * WM8711 2 wire address is determined by GPIO5 ++ * state during powerup. ++ * low = 0x1a ++ * high = 0x1b ++ */ ++#define I2C_DRIVERID_WM8711 0xfefe /* liam - need a proper id */ ++ ++static unsigned short normal_i2c[] = { 0, I2C_CLIENT_END }; ++ ++/* Magic definition of all other variables and things */ ++I2C_CLIENT_INSMOD; ++ ++static struct i2c_driver wm8711_i2c_driver; ++static struct i2c_client client_template; ++ ++/* If the i2c layer weren't so broken, we could pass this kind of data ++ around */ ++ ++static int wm8711_codec_probe(struct i2c_adapter *adap, int addr, int kind) ++{ ++ struct snd_soc_device *socdev = wm8711_socdev; ++ struct wm8711_setup_data *setup = socdev->codec_data; ++ struct snd_soc_codec *codec = socdev->codec; ++ struct i2c_client *i2c; ++ int ret; ++ ++ if (addr != setup->i2c_address) ++ return -ENODEV; ++ ++ client_template.adapter = adap; ++ client_template.addr = addr; ++ ++ i2c = kzalloc(sizeof(struct i2c_client), GFP_KERNEL); ++ if (i2c == NULL){ ++ kfree(codec); ++ return -ENOMEM; ++ } ++ memcpy(i2c, &client_template, sizeof(struct i2c_client)); ++ ++ i2c_set_clientdata(i2c, codec); ++ ++ codec->control_data = i2c; ++ ++ ret = i2c_attach_client(i2c); ++ if (ret < 0) { ++ err("failed to attach codec at addr %x\n", addr); ++ goto err; ++ } ++ ++ ret = wm8711_init(socdev); ++ if (ret < 0) { ++ err("failed to initialise WM8711\n"); ++ goto err; ++ } ++ return ret; ++ ++err: ++ kfree(codec); ++ kfree(i2c); ++ return ret; ++ ++} ++ ++static int wm8711_i2c_detach(struct i2c_client *client) ++{ ++ struct snd_soc_codec* codec = i2c_get_clientdata(client); ++ ++ i2c_detach_client(client); ++ ++ kfree(codec->reg_cache); ++ kfree(client); ++ ++ return 0; ++} ++ ++static int wm8711_i2c_attach(struct i2c_adapter *adap) ++{ ++ return i2c_probe(adap, &addr_data, wm8711_codec_probe); ++} ++ ++/* corgi i2c codec control layer */ ++static struct i2c_driver wm8711_i2c_driver = { ++ .driver = { ++ .name = "WM8711 I2C Codec", ++ .owner = THIS_MODULE, ++ }, ++ .id = I2C_DRIVERID_WM8711, ++ .attach_adapter = wm8711_i2c_attach, ++ .detach_client = wm8711_i2c_detach, ++ .command = NULL, ++}; ++ ++static struct i2c_client client_template = { ++ .name = "WM8711", ++ .driver = &wm8711_i2c_driver, ++}; ++#endif ++ ++static int wm8711_probe(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct wm8711_setup_data *setup; ++ struct snd_soc_codec* codec; ++ int ret = 0; ++ ++ info("WM8711 Audio Codec %s", WM8711_VERSION); ++ ++ setup = socdev->codec_data; ++ codec = kzalloc(sizeof(struct snd_soc_codec), GFP_KERNEL); ++ if (codec == NULL) ++ return -ENOMEM; ++ ++ socdev->codec = codec; ++ mutex_init(&codec->mutex); ++ INIT_LIST_HEAD(&codec->dapm_widgets); ++ INIT_LIST_HEAD(&codec->dapm_paths); ++ ++ wm8711_socdev = socdev; ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ if (setup->i2c_address) { ++ normal_i2c[0] = setup->i2c_address; ++ codec->hw_write = (hw_write_t)i2c_master_send; ++ ret = i2c_add_driver(&wm8711_i2c_driver); ++ if (ret != 0) ++ printk(KERN_ERR "can't add i2c driver"); ++ } ++#else ++ /* Add other interfaces here */ ++#endif ++ return ret; ++} ++ ++/* power down chip */ ++static int wm8711_remove(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ if (codec->control_data) ++ wm8711_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ i2c_del_driver(&wm8711_i2c_driver); ++#endif ++ kfree(codec); ++ ++ return 0; ++} ++ ++struct snd_soc_codec_device soc_codec_dev_wm8711 = { ++ .probe = wm8711_probe, ++ .remove = wm8711_remove, ++ .suspend = wm8711_suspend, ++ .resume = wm8711_resume, ++}; ++ ++EXPORT_SYMBOL_GPL(soc_codec_dev_wm8711); ++ ++MODULE_DESCRIPTION("ASoC WM8711 driver"); ++MODULE_AUTHOR("Mike Arthur"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8711.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8711.h +@@ -0,0 +1,39 @@ ++/* ++ * wm8711.h -- WM8711 Soc Audio driver ++ * ++ * Copyright 2006 Wolfson Microelectronics ++ * ++ * Author: Mike Arthur <linux@wolfsonmicro.com> ++ * ++ * Based on wm8731.h ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#ifndef _WM8711_H ++#define _WM8711_H ++ ++/* WM8711 register space */ ++ ++#define WM8711_LOUT1V 0x02 ++#define WM8711_ROUT1V 0x03 ++#define WM8711_APANA 0x04 ++#define WM8711_APDIGI 0x05 ++#define WM8711_PWR 0x06 ++#define WM8711_IFACE 0x07 ++#define WM8711_SRATE 0x08 ++#define WM8711_ACTIVE 0x09 ++#define WM8711_RESET 0x0f ++ ++#define WM8711_CACHEREGNUM 8 ++ ++struct wm8711_setup_data { ++ unsigned short i2c_address; ++}; ++ ++extern struct snd_soc_codec_dai wm8711_dai; ++extern struct snd_soc_codec_device soc_codec_dev_wm8711; ++ ++#endif +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8980.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8980.c +@@ -0,0 +1,991 @@ ++/* ++ * wm8980.c -- WM8980 ALSA Soc Audio driver ++ * ++ * Copyright 2006 Wolfson Microelectronics PLC. ++ * ++ * Authors: ++ * Mike Arthur <linux@wolfsonmicro.com> ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/version.h> ++#include <linux/kernel.h> ++#include <linux/init.h> ++#include <linux/delay.h> ++#include <linux/pm.h> ++#include <linux/i2c.h> ++#include <linux/platform_device.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/pcm_params.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++#include <sound/initval.h> ++ ++#include "wm8980.h" ++ ++#define AUDIO_NAME "wm8980" ++#define WM8980_VERSION "0.2" ++ ++/* ++ * Debug ++ */ ++ ++#define WM8980_DEBUG 0 ++ ++#ifdef WM8980_DEBUG ++#define dbg(format, arg...) \ ++ printk(KERN_DEBUG AUDIO_NAME ": " format "\n" , ## arg) ++#else ++#define dbg(format, arg...) do {} while (0) ++#endif ++#define err(format, arg...) \ ++ printk(KERN_ERR AUDIO_NAME ": " format "\n" , ## arg) ++#define info(format, arg...) \ ++ printk(KERN_INFO AUDIO_NAME ": " format "\n" , ## arg) ++#define warn(format, arg...) \ ++ printk(KERN_WARNING AUDIO_NAME ": " format "\n" , ## arg) ++ ++struct snd_soc_codec_device soc_codec_dev_wm8980; ++ ++/* ++ * wm8980 register cache ++ * We can't read the WM8980 register space when we are ++ * using 2 wire for device control, so we cache them instead. ++ */ ++static const u16 wm8980_reg[WM8980_CACHEREGNUM] = { ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0050, 0x0000, 0x0140, 0x0000, ++ 0x0000, 0x0000, 0x0000, 0x00ff, ++ 0x00ff, 0x0000, 0x0100, 0x00ff, ++ 0x00ff, 0x0000, 0x012c, 0x002c, ++ 0x002c, 0x002c, 0x002c, 0x0000, ++ 0x0032, 0x0000, 0x0000, 0x0000, ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0038, 0x000b, 0x0032, 0x0000, ++ 0x0008, 0x000c, 0x0093, 0x00e9, ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0033, 0x0010, 0x0010, 0x0100, ++ 0x0100, 0x0002, 0x0001, 0x0001, ++ 0x0039, 0x0039, 0x0039, 0x0039, ++ 0x0001, 0x0001, ++}; ++ ++#define WM8980_DAIFMT \ ++ (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_LEFT_J | SND_SOC_DAIFMT_RIGHT_J | \ ++ SND_SOC_DAIFMT_DSP_A | SND_SOC_DAIFMT_DSP_B | SND_SOC_DAIFMT_NB_NF | \ ++ SND_SOC_DAIFMT_NB_IF | SND_SOC_DAIFMT_IB_NF | SND_SOC_DAIFMT_IB_IF) ++ ++#define WM8980_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define WM8980_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000) ++ ++#define WM8980_PCM_FORMATS \ ++ (SNDRV_PCM_FORMAT_S16_LE | SNDRV_PCM_FORMAT_S20_3LE | \ ++ SNDRV_PCM_FORMAT_S24_3LE | SNDRV_PCM_FORMAT_S24_LE | \ ++ SNDRV_PCM_FORMAT_S32_LE) ++ ++#define WM8980_BCLK \ ++ (SND_SOC_FSBD(1) | SND_SOC_FSBD(2) | SND_SOC_FSBD(4) | SND_SOC_FSBD(8) |\ ++ SND_SOC_FSBD(16) | SND_SOC_FSBD(32)) ++ ++static struct snd_soc_dai_mode wm8980_modes[] = { ++ /* codec frame and clock master modes */ ++ { ++ .fmt = WM8980_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8980_PCM_FORMATS, ++ .pcmrate = WM8980_RATES, ++ .pcmdir = WM8980_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = WM8980_BCLK, ++ }, ++ ++ /* codec frame and clock slave modes */ ++ { ++ .fmt = WM8980_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = WM8980_PCM_FORMATS, ++ .pcmrate = WM8980_RATES, ++ .pcmdir = WM8980_DIR, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = SND_SOC_FSB_ALL, ++ }, ++}; ++ ++/* ++ * read wm8980 register cache ++ */ ++static inline unsigned int wm8980_read_reg_cache(struct snd_soc_codec *codec, ++ unsigned int reg) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg == WM8980_RESET) ++ return 0; ++ if (reg >= WM8980_CACHEREGNUM) ++ return -1; ++ return cache[reg]; ++} ++ ++/* ++ * write wm8980 register cache ++ */ ++static inline void wm8980_write_reg_cache(struct snd_soc_codec *codec, ++ u16 reg, unsigned int value) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg >= WM8980_CACHEREGNUM) ++ return; ++ cache[reg] = value; ++} ++ ++/* ++ * write to the WM8980 register space ++ */ ++static int wm8980_write(struct snd_soc_codec *codec, unsigned int reg, ++ unsigned int value) ++{ ++ u8 data[2]; ++ ++ /* data is ++ * D15..D9 WM8980 register offset ++ * D8...D0 register data ++ */ ++ data[0] = (reg << 1) | ((value >> 8) & 0x0001); ++ data[1] = value & 0x00ff; ++ ++ wm8980_write_reg_cache (codec, reg, value); ++ if (codec->hw_write(codec->control_data, data, 2) == 2) ++ return 0; ++ else ++ return -1; ++} ++ ++#define wm8980_reset(c) wm8980_write(c, WM8980_RESET, 0) ++ ++static const char *wm8980_companding[] = {"Off", "NC", "u-law", "A-law" }; ++static const char *wm8980_deemp[] = {"None", "32kHz", "44.1kHz", "48kHz" }; ++static const char *wm8980_eqmode[] = {"Capture", "Playback" }; ++static const char *wm8980_bw[] = {"Narrow", "Wide" }; ++static const char *wm8980_eq1[] = {"80Hz", "105Hz", "135Hz", "175Hz" }; ++static const char *wm8980_eq2[] = {"230Hz", "300Hz", "385Hz", "500Hz" }; ++static const char *wm8980_eq3[] = {"650Hz", "850Hz", "1.1kHz", "1.4kHz" }; ++static const char *wm8980_eq4[] = {"1.8kHz", "2.4kHz", "3.2kHz", "4.1kHz" }; ++static const char *wm8980_eq5[] = {"5.3kHz", "6.9kHz", "9kHz", "11.7kHz" }; ++static const char *wm8980_alc[] = ++ {"ALC both on", "ALC left only", "ALC right only", "Limiter" }; ++ ++static const struct soc_enum wm8980_enum[] = { ++ SOC_ENUM_SINGLE(WM8980_COMP, 1, 4, wm8980_companding), /* adc */ ++ SOC_ENUM_SINGLE(WM8980_COMP, 3, 4, wm8980_companding), /* dac */ ++ SOC_ENUM_SINGLE(WM8980_DAC, 4, 4, wm8980_deemp), ++ SOC_ENUM_SINGLE(WM8980_EQ1, 8, 2, wm8980_eqmode), ++ ++ SOC_ENUM_SINGLE(WM8980_EQ1, 5, 4, wm8980_eq1), ++ SOC_ENUM_SINGLE(WM8980_EQ2, 8, 2, wm8980_bw), ++ SOC_ENUM_SINGLE(WM8980_EQ2, 5, 4, wm8980_eq2), ++ SOC_ENUM_SINGLE(WM8980_EQ3, 8, 2, wm8980_bw), ++ ++ SOC_ENUM_SINGLE(WM8980_EQ3, 5, 4, wm8980_eq3), ++ SOC_ENUM_SINGLE(WM8980_EQ4, 8, 2, wm8980_bw), ++ SOC_ENUM_SINGLE(WM8980_EQ4, 5, 4, wm8980_eq4), ++ SOC_ENUM_SINGLE(WM8980_EQ5, 8, 2, wm8980_bw), ++ ++ SOC_ENUM_SINGLE(WM8980_EQ5, 5, 4, wm8980_eq5), ++ SOC_ENUM_SINGLE(WM8980_ALC3, 8, 2, wm8980_alc), ++}; ++ ++static const struct snd_kcontrol_new wm8980_snd_controls[] = { ++SOC_SINGLE("Digital Loopback Switch", WM8980_COMP, 0, 1, 0), ++ ++SOC_ENUM("ADC Companding", wm8980_enum[0]), ++SOC_ENUM("DAC Companding", wm8980_enum[1]), ++ ++SOC_SINGLE("Jack Detection Enable", WM8980_JACK1, 6, 1, 0), ++ ++SOC_SINGLE("DAC Right Inversion Switch", WM8980_DAC, 1, 1, 0), ++SOC_SINGLE("DAC Left Inversion Switch", WM8980_DAC, 0, 1, 0), ++ ++SOC_SINGLE("Left Playback Volume", WM8980_DACVOLL, 0, 127, 0), ++SOC_SINGLE("Right Playback Volume", WM8980_DACVOLR, 0, 127, 0), ++ ++SOC_SINGLE("High Pass Filter Switch", WM8980_ADC, 8, 1, 0), ++SOC_SINGLE("High Pass Filter Switch", WM8980_ADC, 8, 1, 0), ++SOC_SINGLE("High Pass Cut Off", WM8980_ADC, 4, 7, 0), ++SOC_SINGLE("Right ADC Inversion Switch", WM8980_ADC, 1, 1, 0), ++SOC_SINGLE("Left ADC Inversion Switch", WM8980_ADC, 0, 1, 0), ++ ++SOC_SINGLE("Left Capture Volume", WM8980_ADCVOLL, 0, 127, 0), ++SOC_SINGLE("Right Capture Volume", WM8980_ADCVOLR, 0, 127, 0), ++ ++SOC_ENUM("Equaliser Function", wm8980_enum[3]), ++SOC_ENUM("EQ1 Cut Off", wm8980_enum[4]), ++SOC_SINGLE("EQ1 Volume", WM8980_EQ1, 0, 31, 1), ++ ++SOC_ENUM("Equaliser EQ2 Bandwith", wm8980_enum[5]), ++SOC_ENUM("EQ2 Cut Off", wm8980_enum[6]), ++SOC_SINGLE("EQ2 Volume", WM8980_EQ2, 0, 31, 1), ++ ++SOC_ENUM("Equaliser EQ3 Bandwith", wm8980_enum[7]), ++SOC_ENUM("EQ3 Cut Off", wm8980_enum[8]), ++SOC_SINGLE("EQ3 Volume", WM8980_EQ3, 0, 31, 1), ++ ++SOC_ENUM("Equaliser EQ4 Bandwith", wm8980_enum[9]), ++SOC_ENUM("EQ4 Cut Off", wm8980_enum[10]), ++SOC_SINGLE("EQ4 Volume", WM8980_EQ4, 0, 31, 1), ++ ++SOC_ENUM("Equaliser EQ5 Bandwith", wm8980_enum[11]), ++SOC_ENUM("EQ5 Cut Off", wm8980_enum[12]), ++SOC_SINGLE("EQ5 Volume", WM8980_EQ5, 0, 31, 1), ++ ++SOC_SINGLE("DAC Playback Limiter Switch", WM8980_DACLIM1, 8, 1, 0), ++SOC_SINGLE("DAC Playback Limiter Decay", WM8980_DACLIM1, 4, 15, 0), ++SOC_SINGLE("DAC Playback Limiter Attack", WM8980_DACLIM1, 0, 15, 0), ++ ++SOC_SINGLE("DAC Playback Limiter Threshold", WM8980_DACLIM2, 4, 7, 0), ++SOC_SINGLE("DAC Playback Limiter Boost", WM8980_DACLIM2, 0, 15, 0), ++ ++SOC_SINGLE("ALC Enable Switch", WM8980_ALC1, 8, 1, 0), ++SOC_SINGLE("ALC Capture Max Gain", WM8980_ALC1, 3, 7, 0), ++SOC_SINGLE("ALC Capture Min Gain", WM8980_ALC1, 0, 7, 0), ++ ++SOC_SINGLE("ALC Capture ZC Switch", WM8980_ALC2, 8, 1, 0), ++SOC_SINGLE("ALC Capture Hold", WM8980_ALC2, 4, 7, 0), ++SOC_SINGLE("ALC Capture Target", WM8980_ALC2, 0, 15, 0), ++ ++SOC_ENUM("ALC Capture Mode", wm8980_enum[13]), ++SOC_SINGLE("ALC Capture Decay", WM8980_ALC3, 4, 15, 0), ++SOC_SINGLE("ALC Capture Attack", WM8980_ALC3, 0, 15, 0), ++ ++SOC_SINGLE("ALC Capture Noise Gate Switch", WM8980_NGATE, 3, 1, 0), ++SOC_SINGLE("ALC Capture Noise Gate Threshold", WM8980_NGATE, 0, 7, 0), ++ ++SOC_SINGLE("Left Capture PGA ZC Switch", WM8980_INPPGAL, 7, 1, 0), ++SOC_SINGLE("Left Capture PGA Volume", WM8980_INPPGAL, 0, 63, 0), ++ ++SOC_SINGLE("Right Capture PGA ZC Switch", WM8980_INPPGAR, 7, 1, 0), ++SOC_SINGLE("Right Capture PGA Volume", WM8980_INPPGAR, 0, 63, 0), ++ ++SOC_SINGLE("Left Headphone Playback ZC Switch", WM8980_HPVOLL, 7, 1, 0), ++SOC_SINGLE("Left Headphone Playback Switch", WM8980_HPVOLL, 6, 1, 1), ++SOC_SINGLE("Left Headphone Playback Volume", WM8980_HPVOLL, 0, 63, 0), ++ ++SOC_SINGLE("Right Headphone Playback ZC Switch", WM8980_HPVOLR, 7, 1, 0), ++SOC_SINGLE("Right Headphone Playback Switch", WM8980_HPVOLR, 6, 1, 1), ++SOC_SINGLE("Right Headphone Playback Volume", WM8980_HPVOLR, 0, 63, 0), ++ ++SOC_SINGLE("Left Speaker Playback ZC Switch", WM8980_SPKVOLL, 7, 1, 0), ++SOC_SINGLE("Left Speaker Playback Switch", WM8980_SPKVOLL, 6, 1, 1), ++SOC_SINGLE("Left Speaker Playback Volume", WM8980_SPKVOLL, 0, 63, 0), ++ ++SOC_SINGLE("Right Speaker Playback ZC Switch", WM8980_SPKVOLR, 7, 1, 0), ++SOC_SINGLE("Right Speaker Playback Switch", WM8980_SPKVOLR, 6, 1, 1), ++SOC_SINGLE("Right Speaker Playback Volume", WM8980_SPKVOLR, 0, 63, 0), ++ ++SOC_DOUBLE_R("Capture Boost(+20dB)", WM8980_ADCBOOSTL, WM8980_ADCBOOSTR, ++ 8, 1, 0), ++}; ++ ++/* add non dapm controls */ ++static int wm8980_add_controls(struct snd_soc_codec *codec) ++{ ++ int err, i; ++ ++ for (i = 0; i < ARRAY_SIZE(wm8980_snd_controls); i++) { ++ err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&wm8980_snd_controls[i],codec, NULL)); ++ if (err < 0) ++ return err; ++ } ++ ++ return 0; ++} ++ ++/* Left Output Mixer */ ++static const snd_kcontrol_new_t wm8980_left_mixer_controls[] = { ++SOC_DAPM_SINGLE("Right PCM Playback Switch", WM8980_OUTPUT, 6, 1, 1), ++SOC_DAPM_SINGLE("Left PCM Playback Switch", WM8980_MIXL, 0, 1, 1), ++SOC_DAPM_SINGLE("Line Bypass Switch", WM8980_MIXL, 1, 1, 0), ++SOC_DAPM_SINGLE("Aux Playback Switch", WM8980_MIXL, 5, 1, 0), ++}; ++ ++/* Right Output Mixer */ ++static const snd_kcontrol_new_t wm8980_right_mixer_controls[] = { ++SOC_DAPM_SINGLE("Left PCM Playback Switch", WM8980_OUTPUT, 5, 1, 1), ++SOC_DAPM_SINGLE("Right PCM Playback Switch", WM8980_MIXR, 0, 1, 1), ++SOC_DAPM_SINGLE("Line Bypass Switch", WM8980_MIXR, 1, 1, 0), ++SOC_DAPM_SINGLE("Aux Playback Switch", WM8980_MIXR, 5, 1, 0), ++}; ++ ++/* Left AUX Input boost vol */ ++static const snd_kcontrol_new_t wm8980_laux_boost_controls = ++SOC_DAPM_SINGLE("Left Aux Volume", WM8980_ADCBOOSTL, 0, 3, 0); ++ ++/* Right AUX Input boost vol */ ++static const snd_kcontrol_new_t wm8980_raux_boost_controls = ++SOC_DAPM_SINGLE("Right Aux Volume", WM8980_ADCBOOSTR, 0, 3, 0); ++ ++/* Left Input boost vol */ ++static const snd_kcontrol_new_t wm8980_lmic_boost_controls = ++SOC_DAPM_SINGLE("Left Input Volume", WM8980_ADCBOOSTL, 4, 3, 0); ++ ++/* Right Input boost vol */ ++static const snd_kcontrol_new_t wm8980_rmic_boost_controls = ++SOC_DAPM_SINGLE("Right Input Volume", WM8980_ADCBOOSTR, 4, 3, 0); ++ ++/* Left Aux In to PGA */ ++static const snd_kcontrol_new_t wm8980_laux_capture_boost_controls = ++SOC_DAPM_SINGLE("Left Capture Switch", WM8980_ADCBOOSTL, 8, 1, 0); ++ ++/* Right Aux In to PGA */ ++static const snd_kcontrol_new_t wm8980_raux_capture_boost_controls = ++SOC_DAPM_SINGLE("Right Capture Switch", WM8980_ADCBOOSTR, 8, 1, 0); ++ ++/* Left Input P In to PGA */ ++static const snd_kcontrol_new_t wm8980_lmicp_capture_boost_controls = ++SOC_DAPM_SINGLE("Left Input P Capture Boost Switch", WM8980_INPUT, 0, 1, 0); ++ ++/* Right Input P In to PGA */ ++static const snd_kcontrol_new_t wm8980_rmicp_capture_boost_controls = ++SOC_DAPM_SINGLE("Right Input P Capture Boost Switch", WM8980_INPUT, 4, 1, 0); ++ ++/* Left Input N In to PGA */ ++static const snd_kcontrol_new_t wm8980_lmicn_capture_boost_controls = ++SOC_DAPM_SINGLE("Left Input N Capture Boost Switch", WM8980_INPUT, 1, 1, 0); ++ ++/* Right Input N In to PGA */ ++static const snd_kcontrol_new_t wm8980_rmicn_capture_boost_controls = ++SOC_DAPM_SINGLE("Right Input N Capture Boost Switch", WM8980_INPUT, 5, 1, 0); ++ ++// TODO Widgets ++static const struct snd_soc_dapm_widget wm8980_dapm_widgets[] = { ++#if 0 ++//SND_SOC_DAPM_MUTE("Mono Mute", WM8980_MONOMIX, 6, 0), ++//SND_SOC_DAPM_MUTE("Speaker Mute", WM8980_SPKMIX, 6, 0), ++ ++SND_SOC_DAPM_MIXER("Speaker Mixer", WM8980_POWER3, 2, 0, ++ &wm8980_speaker_mixer_controls[0], ++ ARRAY_SIZE(wm8980_speaker_mixer_controls)), ++SND_SOC_DAPM_MIXER("Mono Mixer", WM8980_POWER3, 3, 0, ++ &wm8980_mono_mixer_controls[0], ++ ARRAY_SIZE(wm8980_mono_mixer_controls)), ++SND_SOC_DAPM_DAC("DAC", "HiFi Playback", WM8980_POWER3, 0, 0), ++SND_SOC_DAPM_ADC("ADC", "HiFi Capture", WM8980_POWER3, 0, 0), ++SND_SOC_DAPM_PGA("Aux Input", WM8980_POWER1, 6, 0, NULL, 0), ++SND_SOC_DAPM_PGA("SpkN Out", WM8980_POWER3, 5, 0, NULL, 0), ++SND_SOC_DAPM_PGA("SpkP Out", WM8980_POWER3, 6, 0, NULL, 0), ++SND_SOC_DAPM_PGA("Mono Out", WM8980_POWER3, 7, 0, NULL, 0), ++SND_SOC_DAPM_PGA("Mic PGA", WM8980_POWER2, 2, 0, NULL, 0), ++ ++SND_SOC_DAPM_PGA("Aux Boost", SND_SOC_NOPM, 0, 0, ++ &wm8980_aux_boost_controls, 1), ++SND_SOC_DAPM_PGA("Mic Boost", SND_SOC_NOPM, 0, 0, ++ &wm8980_mic_boost_controls, 1), ++SND_SOC_DAPM_SWITCH("Capture Boost", SND_SOC_NOPM, 0, 0, ++ &wm8980_capture_boost_controls), ++ ++SND_SOC_DAPM_MIXER("Boost Mixer", WM8980_POWER2, 4, 0, NULL, 0), ++ ++SND_SOC_DAPM_MICBIAS("Mic Bias", WM8980_POWER1, 4, 0), ++ ++SND_SOC_DAPM_INPUT("MICN"), ++SND_SOC_DAPM_INPUT("MICP"), ++SND_SOC_DAPM_INPUT("AUX"), ++SND_SOC_DAPM_OUTPUT("MONOOUT"), ++SND_SOC_DAPM_OUTPUT("SPKOUTP"), ++SND_SOC_DAPM_OUTPUT("SPKOUTN"), ++#endif ++}; ++ ++static const char *audio_map[][3] = { ++ /* Mono output mixer */ ++ {"Mono Mixer", "PCM Playback Switch", "DAC"}, ++ {"Mono Mixer", "Aux Playback Switch", "Aux Input"}, ++ {"Mono Mixer", "Line Bypass Switch", "Boost Mixer"}, ++ ++ /* Speaker output mixer */ ++ {"Speaker Mixer", "PCM Playback Switch", "DAC"}, ++ {"Speaker Mixer", "Aux Playback Switch", "Aux Input"}, ++ {"Speaker Mixer", "Line Bypass Switch", "Boost Mixer"}, ++ ++ /* Outputs */ ++ {"Mono Out", NULL, "Mono Mixer"}, ++ {"MONOOUT", NULL, "Mono Out"}, ++ {"SpkN Out", NULL, "Speaker Mixer"}, ++ {"SpkP Out", NULL, "Speaker Mixer"}, ++ {"SPKOUTN", NULL, "SpkN Out"}, ++ {"SPKOUTP", NULL, "SpkP Out"}, ++ ++ /* Boost Mixer */ ++ {"Boost Mixer", NULL, "ADC"}, ++ {"Capture Boost Switch", "Aux Capture Boost Switch", "AUX"}, ++ {"Aux Boost", "Aux Volume", "Boost Mixer"}, ++ {"Capture Boost", "Capture Switch", "Boost Mixer"}, ++ {"Mic Boost", "Mic Volume", "Boost Mixer"}, ++ ++ /* Inputs */ ++ {"MICP", NULL, "Mic Boost"}, ++ {"MICN", NULL, "Mic PGA"}, ++ {"Mic PGA", NULL, "Capture Boost"}, ++ {"AUX", NULL, "Aux Input"}, ++ ++ /* */ ++ ++ /* terminator */ ++ {NULL, NULL, NULL}, ++}; ++ ++static int wm8980_add_widgets(struct snd_soc_codec *codec) ++{ ++ int i; ++ ++ for(i = 0; i < ARRAY_SIZE(wm8980_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &wm8980_dapm_widgets[i]); ++ } ++ ++ /* set up audio path map */ ++ for(i = 0; audio_map[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, audio_map[i][0], audio_map[i][1], ++ audio_map[i][2]); ++ } ++ ++ snd_soc_dapm_new_widgets(codec); ++ return 0; ++} ++ ++struct pll_ { ++ unsigned int in_hz, out_hz; ++ unsigned int pre:4; /* prescale - 1 */ ++ unsigned int n:4; ++ unsigned int k; ++}; ++ ++struct pll_ pll[] = { ++ {12000000, 11289600, 0, 7, 0x86c220}, ++ {12000000, 12288000, 0, 8, 0x3126e8}, ++ {13000000, 11289600, 0, 6, 0xf28bd4}, ++ {13000000, 12288000, 0, 7, 0x8fd525}, ++ {12288000, 11289600, 0, 7, 0x59999a}, ++ {11289600, 12288000, 0, 8, 0x80dee9}, ++ /* TODO: liam - add more entries */ ++}; ++ ++static int set_pll(struct snd_soc_codec *codec, unsigned int in, ++ unsigned int out) ++{ ++ int i; ++ u16 reg; ++ ++ if(out == 0) { ++ reg = wm8980_read_reg_cache(codec, WM8980_POWER1); ++ wm8980_write(codec, WM8980_POWER1, reg & 0x1df); ++ return 0; ++ } ++ ++ for(i = 0; i < ARRAY_SIZE(pll); i++) { ++ if (in == pll[i].in_hz && out == pll[i].out_hz) { ++ wm8980_write(codec, WM8980_PLLN, (pll[i].pre << 4) | pll[i].n); ++ wm8980_write(codec, WM8980_PLLK1, pll[i].k >> 18); ++ wm8980_write(codec, WM8980_PLLK1, (pll[i].k >> 9) && 0x1ff); ++ wm8980_write(codec, WM8980_PLLK1, pll[i].k && 0x1ff); ++ reg = wm8980_read_reg_cache(codec, WM8980_POWER1); ++ wm8980_write(codec, WM8980_POWER1, reg | 0x020); ++ return 0; ++ } ++ } ++ return -EINVAL; ++} ++ ++/* mclk dividers * 2 */ ++static unsigned char mclk_div[] = {2, 3, 4, 6, 8, 12, 16, 24}; ++ ++/* we need 256FS to drive the DAC's and ADC's */ ++static unsigned int wm8980_config_sysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ int i, j, best_clk = info->fs * info->rate; ++ ++ /* can we run at this clk without the PLL ? */ ++ for (i = 0; i < ARRAY_SIZE(mclk_div); i++) { ++ if ((best_clk >> 1) * mclk_div[i] == clk) { ++ dai->pll_in = 0; ++ dai->clk_div = mclk_div[i]; ++ dai->mclk = best_clk; ++ return dai->mclk; ++ } ++ } ++ ++ /* now check for PLL support */ ++ for (i = 0; i < ARRAY_SIZE(pll); i++) { ++ if (pll[i].in_hz == clk) { ++ for (j = 0; j < ARRAY_SIZE(mclk_div); j++) { ++ if (pll[i].out_hz == mclk_div[j] * (best_clk >> 1)) { ++ dai->pll_in = clk; ++ dai->pll_out = pll[i].out_hz; ++ dai->clk_div = mclk_div[j]; ++ dai->mclk = best_clk; ++ return dai->mclk; ++ } ++ } ++ } ++ } ++ ++ /* this clk is not supported */ ++ return 0; ++} ++ ++static int wm8980_pcm_prepare(snd_pcm_substream_t *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ struct snd_soc_codec_dai *dai = rtd->codec_dai; ++ u16 iface = 0, bfs, clk = 0, adn; ++ int fs = 48000 << 7, i; ++ ++ bfs = SND_SOC_FSBD_REAL(rtd->codec_dai->dai_runtime.bfs); ++ switch (bfs) { ++ case 2: ++ clk |= 0x1 << 2; ++ break; ++ case 4: ++ clk |= 0x2 << 2; ++ break; ++ case 8: ++ clk |= 0x3 << 2; ++ break; ++ case 16: ++ clk |= 0x4 << 2; ++ break; ++ case 32: ++ clk |= 0x5 << 2; ++ break; ++ } ++ ++ /* set master/slave audio interface */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK) { ++ case SND_SOC_DAIFMT_CBM_CFM: ++ clk |= 0x0001; ++ break; ++ case SND_SOC_DAIFMT_CBS_CFS: ++ break; ++ } ++ ++ /* interface format */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_I2S: ++ iface |= 0x0010; ++ break; ++ case SND_SOC_DAIFMT_RIGHT_J: ++ break; ++ case SND_SOC_DAIFMT_LEFT_J: ++ iface |= 0x0008; ++ break; ++ case SND_SOC_DAIFMT_DSP_A: ++ iface |= 0x00018; ++ break; ++ } ++ ++ /* bit size */ ++ switch (rtd->codec_dai->dai_runtime.pcmfmt) { ++ case SNDRV_PCM_FMTBIT_S16_LE: ++ break; ++ case SNDRV_PCM_FMTBIT_S20_3LE: ++ iface |= 0x0020; ++ break; ++ case SNDRV_PCM_FMTBIT_S24_LE: ++ iface |= 0x0040; ++ break; ++ case SNDRV_PCM_FMTBIT_S32_LE: ++ iface |= 0x0060; ++ break; ++ } ++ ++ /* clock inversion */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_INV_MASK) { ++ case SND_SOC_DAIFMT_NB_NF: ++ break; ++ case SND_SOC_DAIFMT_IB_IF: ++ iface |= 0x0180; ++ break; ++ case SND_SOC_DAIFMT_IB_NF: ++ iface |= 0x0100; ++ break; ++ case SND_SOC_DAIFMT_NB_IF: ++ iface |= 0x0080; ++ break; ++ } ++ ++ /* filter coefficient */ ++ adn = wm8980_read_reg_cache(codec, WM8980_ADD) & 0x1f1; ++ switch (rtd->codec_dai->dai_runtime.pcmrate) { ++ case SNDRV_PCM_RATE_8000: ++ adn |= 0x5 << 1; ++ fs = 8000 << 7; ++ break; ++ case SNDRV_PCM_RATE_11025: ++ adn |= 0x4 << 1; ++ fs = 11025 << 7; ++ break; ++ case SNDRV_PCM_RATE_16000: ++ adn |= 0x3 << 1; ++ fs = 16000 << 7; ++ break; ++ case SNDRV_PCM_RATE_22050: ++ adn |= 0x2 << 1; ++ fs = 22050 << 7; ++ break; ++ case SNDRV_PCM_RATE_32000: ++ adn |= 0x1 << 1; ++ fs = 32000 << 7; ++ break; ++ case SNDRV_PCM_RATE_44100: ++ fs = 44100 << 7; ++ break; ++ } ++ ++ /* do we need to enable the PLL */ ++ if(dai->pll_in) ++ set_pll(codec, dai->pll_in, dai->pll_out); ++ ++ /* divide the clock to 256 fs */ ++ for(i = 0; i < ARRAY_SIZE(mclk_div); i++) { ++ if (dai->clk_div == mclk_div[i]) { ++ clk |= i << 5; ++ clk &= 0xff; ++ goto set; ++ } ++ } ++ ++set: ++ /* set iface */ ++ wm8980_write(codec, WM8980_IFACE, iface); ++ wm8980_write(codec, WM8980_CLOCK, clk); ++ ++ return 0; ++} ++ ++static int wm8980_hw_free(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ set_pll(codec, 0, 0); ++ return 0; ++} ++ ++static int wm8980_mute(struct snd_soc_codec *codec, ++ struct snd_soc_codec_dai *dai, int mute) ++{ ++ u16 mute_reg = wm8980_read_reg_cache(codec, WM8980_DAC) & 0xffbf; ++ if(mute) ++ wm8980_write(codec, WM8980_DAC, mute_reg | 0x40); ++ else ++ wm8980_write(codec, WM8980_DAC, mute_reg); ++ ++ return 0; ++} ++ ++/* TODO: liam need to make this lower power with dapm */ ++static int wm8980_dapm_event(struct snd_soc_codec *codec, int event) ++{ ++ ++ switch (event) { ++ case SNDRV_CTL_POWER_D0: /* full On */ ++ /* vref/mid, clk and osc on, dac unmute, active */ ++ wm8980_write(codec, WM8980_POWER1, 0x1ff); ++ wm8980_write(codec, WM8980_POWER2, 0x1ff); ++ wm8980_write(codec, WM8980_POWER3, 0x1ff); ++ break; ++ case SNDRV_CTL_POWER_D1: /* partial On */ ++ case SNDRV_CTL_POWER_D2: /* partial On */ ++ break; ++ case SNDRV_CTL_POWER_D3hot: /* Off, with power */ ++ /* everything off except vref/vmid, dac mute, inactive */ ++ ++ break; ++ case SNDRV_CTL_POWER_D3cold: /* Off, without power */ ++ /* everything off, dac mute, inactive */ ++ wm8980_write(codec, WM8980_POWER1, 0x0); ++ wm8980_write(codec, WM8980_POWER2, 0x0); ++ wm8980_write(codec, WM8980_POWER3, 0x0); ++ break; ++ } ++ codec->dapm_state = event; ++ return 0; ++} ++ ++struct snd_soc_codec_dai wm8980_dai = { ++ .name = "WM8980 HiFi", ++ .playback = { ++ .stream_name = "Playback", ++ .channels_min = 1, ++ .channels_max = 2, ++ }, ++ .capture = { ++ .stream_name = "Capture", ++ .channels_min = 1, ++ .channels_max = 2, ++ }, ++ .config_sysclk = wm8980_config_sysclk, ++ .digital_mute = wm8980_mute, ++ .ops = { ++ .prepare = wm8980_pcm_prepare, ++ .hw_free = wm8980_hw_free, ++ }, ++ .caps = { ++ .num_modes = ARRAY_SIZE(wm8980_modes), ++ .mode = wm8980_modes, ++ }, ++}; ++EXPORT_SYMBOL_GPL(wm8980_dai); ++ ++static int wm8980_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ wm8980_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ return 0; ++} ++ ++static int wm8980_resume(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ int i; ++ u8 data[2]; ++ u16 *cache = codec->reg_cache; ++ ++ /* Sync reg_cache with the hardware */ ++ for (i = 0; i < ARRAY_SIZE(wm8980_reg); i++) { ++ data[0] = (i << 1) | ((cache[i] >> 8) & 0x0001); ++ data[1] = cache[i] & 0x00ff; ++ codec->hw_write(codec->control_data, data, 2); ++ } ++ wm8980_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ wm8980_dapm_event(codec, codec->suspend_dapm_state); ++ return 0; ++} ++ ++/* ++ * initialise the WM8980 driver ++ * register the mixer and dsp interfaces with the kernel ++ */ ++static int wm8980_init(struct snd_soc_device* socdev) ++{ ++ struct snd_soc_codec *codec = socdev->codec; ++ int ret = 0; ++ ++ codec->name = "WM8980"; ++ codec->owner = THIS_MODULE; ++ codec->read = wm8980_read_reg_cache; ++ codec->write = wm8980_write; ++ codec->dapm_event = wm8980_dapm_event; ++ codec->dai = &wm8980_dai; ++ codec->num_dai = 1; ++ codec->reg_cache_size = ARRAY_SIZE(wm8980_reg); ++ codec->reg_cache = ++ kzalloc(sizeof(u16) * ARRAY_SIZE(wm8980_reg), GFP_KERNEL); ++ if (codec->reg_cache == NULL) ++ return -ENOMEM; ++ memcpy(codec->reg_cache, wm8980_reg, ++ sizeof(u16) * ARRAY_SIZE(wm8980_reg)); ++ codec->reg_cache_size = sizeof(u16) * ARRAY_SIZE(wm8980_reg); ++ ++ wm8980_reset(codec); ++ ++ /* register pcms */ ++ ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); ++ if(ret < 0) { ++ kfree(codec->reg_cache); ++ return ret; ++ } ++ ++ /* power on device */ ++ wm8980_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ wm8980_add_controls(codec); ++ wm8980_add_widgets(codec); ++ ret = snd_soc_register_card(socdev); ++ if(ret < 0) { ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++ } ++ ++ return ret; ++} ++ ++static struct snd_soc_device *wm8980_socdev; ++ ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ ++/* ++ * WM8980 2 wire address is 0x1a ++ */ ++#define I2C_DRIVERID_WM8980 0xfefe /* liam - need a proper id */ ++ ++static unsigned short normal_i2c[] = { 0, I2C_CLIENT_END }; ++ ++/* Magic definition of all other variables and things */ ++I2C_CLIENT_INSMOD; ++ ++static struct i2c_driver wm8980_i2c_driver; ++static struct i2c_client client_template; ++ ++/* If the i2c layer weren't so broken, we could pass this kind of data ++ around */ ++ ++static int wm8980_codec_probe(struct i2c_adapter *adap, int addr, int kind) ++{ ++ struct snd_soc_device *socdev = wm8980_socdev; ++ struct wm8980_setup_data *setup = socdev->codec_data; ++ struct snd_soc_codec *codec = socdev->codec; ++ struct i2c_client *i2c; ++ int ret; ++ ++ if (addr != setup->i2c_address) ++ return -ENODEV; ++ ++ client_template.adapter = adap; ++ client_template.addr = addr; ++ ++ i2c = kzalloc(sizeof(struct i2c_client), GFP_KERNEL); ++ if (i2c == NULL){ ++ kfree(codec); ++ return -ENOMEM; ++ } ++ memcpy(i2c, &client_template, sizeof(struct i2c_client)); ++ ++ i2c_set_clientdata(i2c, codec); ++ ++ codec->control_data = i2c; ++ ++ ret = i2c_attach_client(i2c); ++ if(ret < 0) { ++ err("failed to attach codec at addr %x\n", addr); ++ goto err; ++ } ++ ++ ret = wm8980_init(socdev); ++ if(ret < 0) { ++ err("failed to initialise WM8980\n"); ++ goto err; ++ } ++ return ret; ++ ++err: ++ kfree(codec); ++ kfree(i2c); ++ return ret; ++ ++} ++ ++static int wm8980_i2c_detach(struct i2c_client *client) ++{ ++ struct snd_soc_codec *codec = i2c_get_clientdata(client); ++ ++ i2c_detach_client(client); ++ ++ kfree(codec->reg_cache); ++ kfree(client); ++ ++ return 0; ++} ++ ++static int wm8980_i2c_attach(struct i2c_adapter *adap) ++{ ++ return i2c_probe(adap, &addr_data, wm8980_codec_probe); ++} ++ ++/* corgi i2c codec control layer */ ++static struct i2c_driver wm8980_i2c_driver = { ++ .driver = { ++ .name = "WM8980 I2C Codec", ++ .owner = THIS_MODULE, ++ }, ++ .id = I2C_DRIVERID_WM8980, ++ .attach_adapter = wm8980_i2c_attach, ++ .detach_client = wm8980_i2c_detach, ++ .command = NULL, ++}; ++ ++static struct i2c_client client_template = { ++ .name = "WM8980", ++ .driver = &wm8980_i2c_driver, ++}; ++#endif ++ ++static int wm8980_probe(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct wm8980_setup_data *setup; ++ struct snd_soc_codec *codec; ++ int ret = 0; ++ ++ info("WM8980 Audio Codec %s", WM8980_VERSION); ++ ++ setup = socdev->codec_data; ++ codec = kzalloc(sizeof(struct snd_soc_codec), GFP_KERNEL); ++ if (codec == NULL) ++ return -ENOMEM; ++ ++ socdev->codec = codec; ++ mutex_init(&codec->mutex); ++ INIT_LIST_HEAD(&codec->dapm_widgets); ++ INIT_LIST_HEAD(&codec->dapm_paths); ++ ++ wm8980_socdev = socdev; ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ if (setup->i2c_address) { ++ normal_i2c[0] = setup->i2c_address; ++ codec->hw_write = (hw_write_t)i2c_master_send; ++ ret = i2c_add_driver(&wm8980_i2c_driver); ++ if (ret != 0) ++ printk(KERN_ERR "can't add i2c driver"); ++ } ++#else ++ /* Add other interfaces here */ ++#endif ++ return ret; ++} ++ ++/* power down chip */ ++static int wm8980_remove(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ if (codec->control_data) ++ wm8980_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ i2c_del_driver(&wm8980_i2c_driver); ++#endif ++ kfree(codec); ++ ++ return 0; ++} ++ ++struct snd_soc_codec_device soc_codec_dev_wm8980 = { ++ .probe = wm8980_probe, ++ .remove = wm8980_remove, ++ .suspend = wm8980_suspend, ++ .resume = wm8980_resume, ++}; ++ ++EXPORT_SYMBOL_GPL(soc_codec_dev_wm8980); ++ ++MODULE_DESCRIPTION("ASoC WM8980 driver"); ++MODULE_AUTHOR("Mike Arthur"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8980.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8980.h +@@ -0,0 +1,77 @@ ++/* ++ * wm8980.h -- WM8980 Soc Audio driver ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#ifndef _WM8980_H ++#define _WM8980_H ++ ++/* WM8980 register space */ ++ ++#define WM8980_RESET 0x0 ++#define WM8980_POWER1 0x1 ++#define WM8980_POWER2 0x2 ++#define WM8980_POWER3 0x3 ++#define WM8980_IFACE 0x4 ++#define WM8980_COMP 0x5 ++#define WM8980_CLOCK 0x6 ++#define WM8980_ADD 0x7 ++#define WM8980_GPIO 0x8 ++#define WM8980_JACK1 0x9 ++#define WM8980_DAC 0xa ++#define WM8980_DACVOLL 0xb ++#define WM8980_DACVOLR 0xc ++#define WM8980_JACK2 0xd ++#define WM8980_ADC 0xe ++#define WM8980_ADCVOLL 0xf ++#define WM8980_ADCVOLR 0x10 ++#define WM8980_EQ1 0x12 ++#define WM8980_EQ2 0x13 ++#define WM8980_EQ3 0x14 ++#define WM8980_EQ4 0x15 ++#define WM8980_EQ5 0x16 ++#define WM8980_DACLIM1 0x18 ++#define WM8980_DACLIM2 0x19 ++#define WM8980_NOTCH1 0x1b ++#define WM8980_NOTCH2 0x1c ++#define WM8980_NOTCH3 0x1d ++#define WM8980_NOTCH4 0x1e ++#define WM8980_ALC1 0x20 ++#define WM8980_ALC2 0x21 ++#define WM8980_ALC3 0x22 ++#define WM8980_NGATE 0x23 ++#define WM8980_PLLN 0x24 ++#define WM8980_PLLK1 0x25 ++#define WM8980_PLLK2 0x26 ++#define WM8980_PLLK3 0x27 ++#define WM8980_VIDEO 0x28 ++#define WM8980_3D 0x29 ++#define WM8980_BEEP 0x2b ++#define WM8980_INPUT 0x2c ++#define WM8980_INPPGAL 0x2d ++#define WM8980_INPPGAR 0x2e ++#define WM8980_ADCBOOSTL 0x2f ++#define WM8980_ADCBOOSTR 0x30 ++#define WM8980_OUTPUT 0x31 ++#define WM8980_MIXL 0x32 ++#define WM8980_MIXR 0x33 ++#define WM8980_HPVOLL 0x34 ++#define WM8980_HPVOLR 0x35 ++#define WM8980_SPKVOLL 0x36 ++#define WM8980_SPKVOLR 0x37 ++#define WM8980_OUT3MIX 0x38 ++#define WM8980_MONOMIX 0x39 ++ ++#define WM8980_CACHEREGNUM 58 ++ ++struct wm8980_setup_data { ++ unsigned short i2c_address; ++}; ++ ++extern struct snd_soc_codec_dai wm8980_dai; ++extern struct snd_soc_codec_device soc_codec_dev_wm8980; ++ ++#endif +Index: linux-2.6-pxa-new/sound/soc/at91/eti_b1_wm8731.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/at91/eti_b1_wm8731.c +@@ -0,0 +1,230 @@ ++/* ++ * eti_b1_wm8731 -- SoC audio for Endrelia ETI_B1. ++ * ++ * Author: Frank Mandarino <fmandarino@endrelia.com> ++ * Endrelia Technologies Inc. ++ * Created: Mar 29, 2006 ++ * ++ * Based on corgi.c by: ++ * ++ * Copyright 2005 Wolfson Microelectronics PLC. ++ * Copyright 2005 Openedhand Ltd. ++ * ++ * Authors: Liam Girdwood <liam.girdwood@wolfsonmicro.com> ++ * Richard Purdie <richard@openedhand.com> ++ * ++ * 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. ++ * ++ * Revision history ++ * 30th Nov 2005 Initial version. ++ * ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/version.h> ++#include <linux/kernel.h> ++#include <linux/clk.h> ++#include <linux/timer.h> ++#include <linux/interrupt.h> ++#include <linux/platform_device.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++ ++#include <asm/arch/at91rm9200.h> ++#include <asm/arch/gpio.h> ++#include <asm/arch/hardware.h> ++ ++#include "../codecs/wm8731.h" ++#include "at91rm9200-pcm.h" ++ ++#if 0 ++#define DBG(x...) printk(KERN_INFO "eti_b1_wm8731:" x) ++#else ++#define DBG(x...) ++#endif ++ ++static struct clk *pck1_clk; ++static struct clk *pllb_clk; ++ ++static int eti_b1_startup(snd_pcm_substream_t *substream) ++{ ++ /* Start PCK1 clock. */ ++ clk_enable(pck1_clk); ++ DBG("pck1 started\n"); ++ ++ return 0; ++} ++ ++static void eti_b1_shutdown(snd_pcm_substream_t *substream) ++{ ++ /* Stop PCK1 clock. */ ++ clk_disable(pck1_clk); ++ DBG("pck1 stopped\n"); ++} ++ ++static struct snd_soc_ops eti_b1_ops = { ++ .startup = eti_b1_startup, ++ .shutdown = eti_b1_shutdown, ++}; ++ ++ ++static const struct snd_soc_dapm_widget eti_b1_dapm_widgets[] = { ++ SND_SOC_DAPM_MIC("Int Mic", NULL), ++ SND_SOC_DAPM_SPK("Ext Spk", NULL), ++}; ++ ++static const char *intercon[][3] = { ++ ++ /* speaker connected to LHPOUT */ ++ {"Ext Spk", NULL, "LHPOUT"}, ++ ++ /* mic is connected to Mic Jack, with WM8731 Mic Bias */ ++ {"MICIN", NULL, "Mic Bias"}, ++ {"Mic Bias", NULL, "Int Mic"}, ++ ++ /* terminator */ ++ {NULL, NULL, NULL}, ++}; ++ ++/* ++ * Logic for a wm8731 as connected on a Endrelia ETI-B1 board. ++ */ ++static int eti_b1_wm8731_init(struct snd_soc_codec *codec) ++{ ++ int i; ++ ++ DBG("eti_b1_wm8731_init() called\n"); ++ ++ /* Add specific widgets */ ++ for(i = 0; i < ARRAY_SIZE(eti_b1_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &eti_b1_dapm_widgets[i]); ++ } ++ ++ /* Set up specific audio path interconnects */ ++ for(i = 0; intercon[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, intercon[i][0], ++ intercon[i][1], intercon[i][2]); ++ } ++ ++ /* not connected */ ++ snd_soc_dapm_set_endpoint(codec, "RLINEIN", 0); ++ snd_soc_dapm_set_endpoint(codec, "LLINEIN", 0); ++ ++ /* always connected */ ++ snd_soc_dapm_set_endpoint(codec, "Int Mic", 1); ++ snd_soc_dapm_set_endpoint(codec, "Ext Spk", 1); ++ ++ snd_soc_dapm_sync_endpoints(codec); ++ ++ return 0; ++} ++ ++unsigned int eti_b1_config_sysclk(struct snd_soc_pcm_runtime *rtd, ++ struct snd_soc_clock_info *info) ++{ ++ if(info->bclk_master & SND_SOC_DAIFMT_CBS_CFS) { ++ return rtd->codec_dai->config_sysclk(rtd->codec_dai, info, 12000000); ++ } ++ return 0; ++} ++ ++static struct snd_soc_dai_link eti_b1_dai = { ++ .name = "WM8731", ++ .stream_name = "WM8731", ++ .cpu_dai = &at91rm9200_i2s_dai[1], ++ .codec_dai = &wm8731_dai, ++ .init = eti_b1_wm8731_init, ++ .config_sysclk = eti_b1_config_sysclk, ++}; ++ ++static struct snd_soc_machine snd_soc_machine_eti_b1 = { ++ .name = "ETI_B1", ++ .dai_link = &eti_b1_dai, ++ .num_links = 1, ++ .ops = &eti_b1_ops, ++}; ++ ++static struct wm8731_setup_data eti_b1_wm8731_setup = { ++ .i2c_address = 0x1a, ++}; ++ ++static struct snd_soc_device eti_b1_snd_devdata = { ++ .machine = &snd_soc_machine_eti_b1, ++ .platform = &at91rm9200_soc_platform, ++ .codec_dev = &soc_codec_dev_wm8731, ++ .codec_data = &eti_b1_wm8731_setup, ++}; ++ ++static struct platform_device *eti_b1_snd_device; ++ ++static int __init eti_b1_init(void) ++{ ++ int ret; ++ u32 ssc_pio_lines; ++ ++ eti_b1_snd_device = platform_device_alloc("soc-audio", -1); ++ if (!eti_b1_snd_device) ++ return -ENOMEM; ++ ++ platform_set_drvdata(eti_b1_snd_device, &eti_b1_snd_devdata); ++ eti_b1_snd_devdata.dev = &eti_b1_snd_device->dev; ++ ++ ret = platform_device_add(eti_b1_snd_device); ++ if (ret) { ++ platform_device_put(eti_b1_snd_device); ++ return ret; ++ } ++ ++ ssc_pio_lines = AT91_PB6_TF1 | AT91_PB7_TK1 | AT91_PB8_TD1 ++ | AT91_PB9_RD1 /* | AT91_PB10_RK1 | AT91_PB11_RF1 */; ++ ++ /* Reset all PIO registers and assign lines to peripheral A */ ++ at91_sys_write(AT91_PIOB + PIO_PDR, ssc_pio_lines); ++ at91_sys_write(AT91_PIOB + PIO_ODR, ssc_pio_lines); ++ at91_sys_write(AT91_PIOB + PIO_IFDR, ssc_pio_lines); ++ at91_sys_write(AT91_PIOB + PIO_CODR, ssc_pio_lines); ++ at91_sys_write(AT91_PIOB + PIO_IDR, ssc_pio_lines); ++ at91_sys_write(AT91_PIOB + PIO_MDDR, ssc_pio_lines); ++ at91_sys_write(AT91_PIOB + PIO_PUDR, ssc_pio_lines); ++ at91_sys_write(AT91_PIOB + PIO_ASR, ssc_pio_lines); ++ at91_sys_write(AT91_PIOB + PIO_OWDR, ssc_pio_lines); ++ ++ /* ++ * Set PCK1 parent to PLLB and its rate to 12 Mhz. ++ */ ++ pllb_clk = clk_get(NULL, "pllb"); ++ pck1_clk = clk_get(NULL, "pck1"); ++ ++ clk_set_parent(pck1_clk, pllb_clk); ++ clk_set_rate(pck1_clk, 12000000); ++ ++ DBG("MCLK rate %luHz\n", clk_get_rate(pck1_clk)); ++ ++ /* assign the GPIO pin to PCK1 */ ++ at91_set_B_periph(AT91_PIN_PA24, 0); ++ ++ return ret; ++} ++ ++static void __exit eti_b1_exit(void) ++{ ++ clk_put(pck1_clk); ++ clk_put(pllb_clk); ++ ++ platform_device_unregister(eti_b1_snd_device); ++} ++ ++module_init(eti_b1_init); ++module_exit(eti_b1_exit); ++ ++/* Module information */ ++MODULE_AUTHOR("Frank Mandarino <fmandarino@endrelia.com>"); ++MODULE_DESCRIPTION("ALSA SoC ETI-B1-WM8731"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8510.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8510.c +@@ -0,0 +1,895 @@ ++/* ++ * wm8510.c -- WM8510 ALSA Soc Audio driver ++ * ++ * Copyright 2006 Wolfson Microelectronics PLC. ++ * ++ * Author: Liam Girdwood <liam.girdwood@wolfsonmicro.com> ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/version.h> ++#include <linux/kernel.h> ++#include <linux/init.h> ++#include <linux/delay.h> ++#include <linux/pm.h> ++#include <linux/i2c.h> ++#include <linux/platform_device.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/pcm_params.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++#include <sound/initval.h> ++ ++#include "wm8510.h" ++ ++#define AUDIO_NAME "wm8510" ++#define WM8510_VERSION "0.5" ++ ++/* ++ * Debug ++ */ ++ ++#define WM8510_DEBUG 0 ++ ++#ifdef WM8510_DEBUG ++#define dbg(format, arg...) \ ++ printk(KERN_DEBUG AUDIO_NAME ": " format "\n" , ## arg) ++#else ++#define dbg(format, arg...) do {} while (0) ++#endif ++#define err(format, arg...) \ ++ printk(KERN_ERR AUDIO_NAME ": " format "\n" , ## arg) ++#define info(format, arg...) \ ++ printk(KERN_INFO AUDIO_NAME ": " format "\n" , ## arg) ++#define warn(format, arg...) \ ++ printk(KERN_WARNING AUDIO_NAME ": " format "\n" , ## arg) ++ ++struct snd_soc_codec_device soc_codec_dev_wm8510; ++ ++/* ++ * wm8510 register cache ++ * We can't read the WM8510 register space when we are ++ * using 2 wire for device control, so we cache them instead. ++ */ ++static const u16 wm8510_reg[WM8510_CACHEREGNUM] = { ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0050, 0x0000, 0x0140, 0x0000, ++ 0x0000, 0x0000, 0x0000, 0x00ff, ++ 0x0000, 0x0000, 0x0100, 0x00ff, ++ 0x0000, 0x0000, 0x012c, 0x002c, ++ 0x002c, 0x002c, 0x002c, 0x0000, ++ 0x0032, 0x0000, 0x0000, 0x0000, ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0038, 0x000b, 0x0032, 0x0000, ++ 0x0008, 0x000c, 0x0093, 0x00e9, ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0003, 0x0010, 0x0000, 0x0000, ++ 0x0000, 0x0002, 0x0000, 0x0000, ++ 0x0000, 0x0000, 0x0039, 0x0000, ++ 0x0000, ++}; ++ ++#define WM8510_DAIFMT \ ++ (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_LEFT_J | SND_SOC_DAIFMT_RIGHT_J | \ ++ SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_NB_IF | SND_SOC_DAIFMT_IB_NF | \ ++ SND_SOC_DAIFMT_IB_IF) ++ ++#define WM8510_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define WM8510_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000) ++ ++#define WM8794_BCLK \ ++ (SND_SOC_FSBD(1) | SND_SOC_FSBD(2) | SND_SOC_FSBD(4) | SND_SOC_FSBD(8) |\ ++ SND_SOC_FSBD(16) | SND_SOC_FSBD(32)) ++ ++#define WM8794_HIFI_BITS \ ++ (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE | \ ++ SNDRV_PCM_FMTBIT_S24_LE | SNDRV_PCM_FMTBIT_S32_LE) ++ ++static struct snd_soc_dai_mode wm8510_modes[] = { ++ /* codec frame and clock master modes */ ++ { ++ .fmt = WM8510_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8794_HIFI_BITS, ++ .pcmrate = WM8510_RATES, ++ .pcmdir = WM8510_DIR, ++ .fs = 256, ++ .bfs = WM8794_BCLK, ++ }, ++ ++ /* codec frame and clock slave modes */ ++ { ++ .fmt = WM8510_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = WM8794_HIFI_BITS, ++ .pcmrate = WM8510_RATES, ++ .pcmdir = WM8510_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = SND_SOC_FSB_ALL, ++ }, ++}; ++ ++/* ++ * read wm8510 register cache ++ */ ++static inline unsigned int wm8510_read_reg_cache(struct snd_soc_codec * codec, ++ unsigned int reg) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg == WM8510_RESET) ++ return 0; ++ if (reg >= WM8510_CACHEREGNUM) ++ return -1; ++ return cache[reg]; ++} ++ ++/* ++ * write wm8510 register cache ++ */ ++static inline void wm8510_write_reg_cache(struct snd_soc_codec *codec, ++ u16 reg, unsigned int value) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg >= WM8510_CACHEREGNUM) ++ return; ++ cache[reg] = value; ++} ++ ++/* ++ * write to the WM8510 register space ++ */ ++static int wm8510_write(struct snd_soc_codec *codec, unsigned int reg, ++ unsigned int value) ++{ ++ u8 data[2]; ++ ++ /* data is ++ * D15..D9 WM8510 register offset ++ * D8...D0 register data ++ */ ++ data[0] = (reg << 1) | ((value >> 8) & 0x0001); ++ data[1] = value & 0x00ff; ++ ++ wm8510_write_reg_cache (codec, reg, value); ++ if (codec->hw_write(codec->control_data, data, 2) == 2) ++ return 0; ++ else ++ return -EIO; ++} ++ ++#define wm8510_reset(c) wm8510_write(c, WM8510_RESET, 0) ++ ++static const char *wm8510_companding[] = {"Off", "NC", "u-law", "A-law" }; ++static const char *wm8510_deemp[] = {"None", "32kHz", "44.1kHz", "48kHz" }; ++static const char *wm8510_alc[] = {"ALC", "Limiter" }; ++ ++static const struct soc_enum wm8510_enum[] = { ++ SOC_ENUM_SINGLE(WM8510_COMP, 1, 4, wm8510_companding), /* adc */ ++ SOC_ENUM_SINGLE(WM8510_COMP, 3, 4, wm8510_companding), /* dac */ ++ SOC_ENUM_SINGLE(WM8510_DAC, 4, 4, wm8510_deemp), ++ SOC_ENUM_SINGLE(WM8510_ALC3, 8, 2, wm8510_alc), ++}; ++ ++static const struct snd_kcontrol_new wm8510_snd_controls[] = { ++ ++SOC_SINGLE("Digital Loopback Switch", WM8510_COMP, 0, 1, 0), ++ ++SOC_ENUM("DAC Companding", wm8510_enum[1]), ++SOC_ENUM("ADC Companding", wm8510_enum[0]), ++ ++SOC_ENUM("Playback De-emphasis", wm8510_enum[2]), ++SOC_SINGLE("DAC Inversion Switch", WM8510_DAC, 0, 1, 0), ++ ++SOC_SINGLE("Master Playback Volume", WM8510_DACVOL, 0, 127, 0), ++ ++SOC_SINGLE("High Pass Filter Switch", WM8510_ADC, 8, 1, 0), ++SOC_SINGLE("High Pass Cut Off", WM8510_ADC, 4, 7, 0), ++SOC_SINGLE("ADC Inversion Switch", WM8510_COMP, 0, 1, 0), ++ ++SOC_SINGLE("Capture Volume", WM8510_ADCVOL, 0, 127, 0), ++ ++SOC_SINGLE("DAC Playback Limiter Switch", WM8510_DACLIM1, 8, 1, 0), ++SOC_SINGLE("DAC Playback Limiter Decay", WM8510_DACLIM1, 4, 15, 0), ++SOC_SINGLE("DAC Playback Limiter Attack", WM8510_DACLIM1, 0, 15, 0), ++ ++SOC_SINGLE("DAC Playback Limiter Threshold", WM8510_DACLIM2, 4, 7, 0), ++SOC_SINGLE("DAC Playback Limiter Boost", WM8510_DACLIM2, 0, 15, 0), ++ ++SOC_SINGLE("ALC Enable Switch", WM8510_ALC1, 8, 1, 0), ++SOC_SINGLE("ALC Capture Max Gain", WM8510_ALC1, 3, 7, 0), ++SOC_SINGLE("ALC Capture Min Gain", WM8510_ALC1, 0, 7, 0), ++ ++SOC_SINGLE("ALC Capture ZC Switch", WM8510_ALC2, 8, 1, 0), ++SOC_SINGLE("ALC Capture Hold", WM8510_ALC2, 4, 7, 0), ++SOC_SINGLE("ALC Capture Target", WM8510_ALC2, 0, 15, 0), ++ ++SOC_ENUM("ALC Capture Mode", wm8510_enum[3]), ++SOC_SINGLE("ALC Capture Decay", WM8510_ALC3, 4, 15, 0), ++SOC_SINGLE("ALC Capture Attack", WM8510_ALC3, 0, 15, 0), ++ ++SOC_SINGLE("ALC Capture Noise Gate Switch", WM8510_NGATE, 3, 1, 0), ++SOC_SINGLE("ALC Capture Noise Gate Threshold", WM8510_NGATE, 0, 7, 0), ++ ++SOC_SINGLE("Capture PGA ZC Switch", WM8510_INPPGA, 7, 1, 0), ++SOC_SINGLE("Capture PGA Volume", WM8510_INPPGA, 0, 63, 0), ++ ++SOC_SINGLE("Speaker Playback ZC Switch", WM8510_SPKVOL, 7, 1, 0), ++SOC_SINGLE("Speaker Playback Switch", WM8510_SPKVOL, 6, 1, 1), ++SOC_SINGLE("Speaker Playback Volume", WM8510_SPKVOL, 0, 63, 0), ++ ++SOC_SINGLE("Capture Boost(+20dB)", WM8510_ADCBOOST, 8, 1, 0), ++SOC_SINGLE("Mono Playback Switch", WM8510_MONOMIX, 6, 1, 0), ++}; ++ ++/* add non dapm controls */ ++static int wm8510_add_controls(struct snd_soc_codec *codec) ++{ ++ int err, i; ++ ++ for (i = 0; i < ARRAY_SIZE(wm8510_snd_controls); i++) { ++ err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&wm8510_snd_controls[i],codec, NULL)); ++ if (err < 0) ++ return err; ++ } ++ ++ return 0; ++} ++ ++/* Speaker Output Mixer */ ++static const struct snd_kcontrol_new wm8510_speaker_mixer_controls[] = { ++SOC_DAPM_SINGLE("Line Bypass Switch", WM8510_SPKMIX, 1, 1, 0), ++SOC_DAPM_SINGLE("Aux Playback Switch", WM8510_SPKMIX, 5, 1, 0), ++SOC_DAPM_SINGLE("PCM Playback Switch", WM8510_SPKMIX, 0, 1, 1), ++}; ++ ++/* Mono Output Mixer */ ++static const struct snd_kcontrol_new wm8510_mono_mixer_controls[] = { ++SOC_DAPM_SINGLE("Line Bypass Switch", WM8510_MONOMIX, 1, 1, 0), ++SOC_DAPM_SINGLE("Aux Playback Switch", WM8510_MONOMIX, 2, 1, 0), ++SOC_DAPM_SINGLE("PCM Playback Switch", WM8510_MONOMIX, 0, 1, 1), ++}; ++ ++/* AUX Input boost vol */ ++static const struct snd_kcontrol_new wm8510_aux_boost_controls = ++SOC_DAPM_SINGLE("Aux Volume", WM8510_ADCBOOST, 0, 7, 0); ++ ++/* Mic Input boost vol */ ++static const struct snd_kcontrol_new wm8510_mic_boost_controls = ++SOC_DAPM_SINGLE("Mic Volume", WM8510_ADCBOOST, 4, 7, 0); ++ ++/* Capture boost switch */ ++static const struct snd_kcontrol_new wm8510_capture_boost_controls = ++SOC_DAPM_SINGLE("Capture Boost Switch", WM8510_INPPGA, 6, 1, 0); ++ ++/* Aux In to PGA */ ++static const struct snd_kcontrol_new wm8510_aux_capture_boost_controls = ++SOC_DAPM_SINGLE("Aux Capture Boost Switch", WM8510_INPPGA, 2, 1, 0); ++ ++/* Mic P In to PGA */ ++static const struct snd_kcontrol_new wm8510_micp_capture_boost_controls = ++SOC_DAPM_SINGLE("Mic P Capture Boost Switch", WM8510_INPPGA, 0, 1, 0); ++ ++/* Mic N In to PGA */ ++static const struct snd_kcontrol_new wm8510_micn_capture_boost_controls = ++SOC_DAPM_SINGLE("Mic N Capture Boost Switch", WM8510_INPPGA, 1, 1, 0); ++ ++static const struct snd_soc_dapm_widget wm8510_dapm_widgets[] = { ++SND_SOC_DAPM_MIXER("Speaker Mixer", WM8510_POWER3, 2, 0, ++ &wm8510_speaker_mixer_controls[0], ++ ARRAY_SIZE(wm8510_speaker_mixer_controls)), ++SND_SOC_DAPM_MIXER("Mono Mixer", WM8510_POWER3, 3, 0, ++ &wm8510_mono_mixer_controls[0], ++ ARRAY_SIZE(wm8510_mono_mixer_controls)), ++SND_SOC_DAPM_DAC("DAC", "HiFi Playback", WM8510_POWER3, 0, 0), ++SND_SOC_DAPM_ADC("ADC", "HiFi Capture", WM8510_POWER3, 0, 0), ++SND_SOC_DAPM_PGA("Aux Input", WM8510_POWER1, 6, 0, NULL, 0), ++SND_SOC_DAPM_PGA("SpkN Out", WM8510_POWER3, 5, 0, NULL, 0), ++SND_SOC_DAPM_PGA("SpkP Out", WM8510_POWER3, 6, 0, NULL, 0), ++SND_SOC_DAPM_PGA("Mono Out", WM8510_POWER3, 7, 0, NULL, 0), ++SND_SOC_DAPM_PGA("Mic PGA", WM8510_POWER2, 2, 0, NULL, 0), ++ ++SND_SOC_DAPM_PGA("Aux Boost", SND_SOC_NOPM, 0, 0, ++ &wm8510_aux_boost_controls, 1), ++SND_SOC_DAPM_PGA("Mic Boost", SND_SOC_NOPM, 0, 0, ++ &wm8510_mic_boost_controls, 1), ++SND_SOC_DAPM_SWITCH("Capture Boost", SND_SOC_NOPM, 0, 0, ++ &wm8510_capture_boost_controls), ++ ++SND_SOC_DAPM_MIXER("Boost Mixer", WM8510_POWER2, 4, 0, NULL, 0), ++ ++SND_SOC_DAPM_MICBIAS("Mic Bias", WM8510_POWER1, 4, 0), ++ ++SND_SOC_DAPM_INPUT("MICN"), ++SND_SOC_DAPM_INPUT("MICP"), ++SND_SOC_DAPM_INPUT("AUX"), ++SND_SOC_DAPM_OUTPUT("MONOOUT"), ++SND_SOC_DAPM_OUTPUT("SPKOUTP"), ++SND_SOC_DAPM_OUTPUT("SPKOUTN"), ++}; ++ ++static const char *audio_map[][3] = { ++ /* Mono output mixer */ ++ {"Mono Mixer", "PCM Playback Switch", "DAC"}, ++ {"Mono Mixer", "Aux Playback Switch", "Aux Input"}, ++ {"Mono Mixer", "Line Bypass Switch", "Boost Mixer"}, ++ ++ /* Speaker output mixer */ ++ {"Speaker Mixer", "PCM Playback Switch", "DAC"}, ++ {"Speaker Mixer", "Aux Playback Switch", "Aux Input"}, ++ {"Speaker Mixer", "Line Bypass Switch", "Boost Mixer"}, ++ ++ /* Outputs */ ++ {"Mono Out", NULL, "Mono Mixer"}, ++ {"MONOOUT", NULL, "Mono Out"}, ++ {"SpkN Out", NULL, "Speaker Mixer"}, ++ {"SpkP Out", NULL, "Speaker Mixer"}, ++ {"SPKOUTN", NULL, "SpkN Out"}, ++ {"SPKOUTP", NULL, "SpkP Out"}, ++ ++ /* Boost Mixer */ ++ {"Boost Mixer", NULL, "ADC"}, ++ {"Capture Boost Switch", "Aux Capture Boost Switch", "AUX"}, ++ {"Aux Boost", "Aux Volume", "Boost Mixer"}, ++ {"Capture Boost", "Capture Switch", "Boost Mixer"}, ++ {"Mic Boost", "Mic Volume", "Boost Mixer"}, ++ ++ /* Inputs */ ++ {"MICP", NULL, "Mic Boost"}, ++ {"MICN", NULL, "Mic PGA"}, ++ {"Mic PGA", NULL, "Capture Boost"}, ++ {"AUX", NULL, "Aux Input"}, ++ ++ /* terminator */ ++ {NULL, NULL, NULL}, ++}; ++ ++static int wm8510_add_widgets(struct snd_soc_codec *codec) ++{ ++ int i; ++ ++ for(i = 0; i < ARRAY_SIZE(wm8510_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &wm8510_dapm_widgets[i]); ++ } ++ ++ /* set up audio path audio_mapnects */ ++ for(i = 0; audio_map[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, audio_map[i][0], ++ audio_map[i][1], audio_map[i][2]); ++ } ++ ++ snd_soc_dapm_new_widgets(codec); ++ return 0; ++} ++ ++struct pll_ { ++ unsigned int in_hz, out_hz; ++ unsigned int pre:4; /* prescale - 1 */ ++ unsigned int n:4; ++ unsigned int k; ++}; ++ ++struct pll_ pll[] = { ++ {12000000, 11289600, 0, 7, 0x86c220}, ++ {12000000, 12288000, 0, 8, 0x3126e8}, ++ {13000000, 11289600, 0, 6, 0xf28bd4}, ++ {13000000, 12288000, 0, 7, 0x8fd525}, ++ {12288000, 11289600, 0, 7, 0x59999a}, ++ {11289600, 12288000, 0, 8, 0x80dee9}, ++ /* liam - add more entries */ ++}; ++ ++static int set_pll(struct snd_soc_codec *codec, unsigned int in, ++ unsigned int out) ++{ ++ int i; ++ u16 reg; ++ ++ if(out == 0) { ++ reg = wm8510_read_reg_cache(codec, WM8510_POWER1); ++ wm8510_write(codec, WM8510_POWER1, reg & 0x1df); ++ return 0; ++ } ++ ++ for(i = 0; i < ARRAY_SIZE(pll); i++) { ++ if (in == pll[i].in_hz && out == pll[i].out_hz) { ++ wm8510_write(codec, WM8510_PLLN, (pll[i].pre << 4) | pll[i].n); ++ wm8510_write(codec, WM8510_PLLK1, pll[i].k >> 18); ++ wm8510_write(codec, WM8510_PLLK1, (pll[i].k >> 9) && 0x1ff); ++ wm8510_write(codec, WM8510_PLLK1, pll[i].k && 0x1ff); ++ reg = wm8510_read_reg_cache(codec, WM8510_POWER1); ++ wm8510_write(codec, WM8510_POWER1, reg | 0x020); ++ return 0; ++ } ++ } ++ return -EINVAL; ++} ++ ++/* mclk dividers * 2 */ ++static unsigned char mclk_div[] = {2, 3, 4, 6, 8, 12, 16, 24}; ++ ++/* we need 256FS to drive the DAC's and ADC's */ ++static unsigned int wm8510_config_sysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ int i, j, best_clk = info->fs * info->rate; ++ ++ /* can we run at this clk without the PLL ? */ ++ for (i = 0; i < ARRAY_SIZE(mclk_div); i++) { ++ if ((best_clk >> 1) * mclk_div[i] == clk) { ++ dai->pll_in = 0; ++ dai->clk_div = mclk_div[i]; ++ dai->mclk = best_clk; ++ return dai->mclk; ++ } ++ } ++ ++ /* now check for PLL support */ ++ for (i = 0; i < ARRAY_SIZE(pll); i++) { ++ if (pll[i].in_hz == clk) { ++ for (j = 0; j < ARRAY_SIZE(mclk_div); j++) { ++ if (pll[i].out_hz == mclk_div[j] * (best_clk >> 1)) { ++ dai->pll_in = clk; ++ dai->pll_out = pll[i].out_hz; ++ dai->clk_div = mclk_div[j]; ++ dai->mclk = best_clk; ++ return dai->mclk; ++ } ++ } ++ } ++ } ++ ++ /* this clk is not supported */ ++ return 0; ++} ++ ++static int wm8510_pcm_prepare(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ struct snd_soc_codec_dai *dai = rtd->codec_dai; ++ u16 iface = 0, bfs, clk = 0, adn; ++ int fs = 48000 << 7, i; ++ ++ bfs = SND_SOC_FSBD_REAL(rtd->codec_dai->dai_runtime.bfs); ++ switch (bfs) { ++ case 2: ++ clk |= 0x1 << 2; ++ break; ++ case 4: ++ clk |= 0x2 << 2; ++ break; ++ case 8: ++ clk |= 0x3 << 2; ++ break; ++ case 16: ++ clk |= 0x4 << 2; ++ break; ++ case 32: ++ clk |= 0x5 << 2; ++ break; ++ } ++ ++ /* set master/slave audio interface */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK) { ++ case SND_SOC_DAIFMT_CBM_CFM: ++ clk |= 0x0001; ++ break; ++ case SND_SOC_DAIFMT_CBS_CFS: ++ break; ++ } ++ ++ /* interface format */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_I2S: ++ iface |= 0x0010; ++ break; ++ case SND_SOC_DAIFMT_RIGHT_J: ++ break; ++ case SND_SOC_DAIFMT_LEFT_J: ++ iface |= 0x0008; ++ break; ++ case SND_SOC_DAIFMT_DSP_A: ++ iface |= 0x00018; ++ break; ++ } ++ ++ /* bit size */ ++ switch (rtd->codec_dai->dai_runtime.pcmfmt) { ++ case SNDRV_PCM_FMTBIT_S16_LE: ++ break; ++ case SNDRV_PCM_FMTBIT_S20_3LE: ++ iface |= 0x0020; ++ break; ++ case SNDRV_PCM_FMTBIT_S24_LE: ++ iface |= 0x0040; ++ break; ++ case SNDRV_PCM_FMTBIT_S32_LE: ++ iface |= 0x0060; ++ break; ++ } ++ ++ /* clock inversion */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_INV_MASK) { ++ case SND_SOC_DAIFMT_NB_NF: ++ break; ++ case SND_SOC_DAIFMT_IB_IF: ++ iface |= 0x0180; ++ break; ++ case SND_SOC_DAIFMT_IB_NF: ++ iface |= 0x0100; ++ break; ++ case SND_SOC_DAIFMT_NB_IF: ++ iface |= 0x0080; ++ break; ++ } ++ ++ /* filter coefficient */ ++ adn = wm8510_read_reg_cache(codec, WM8510_ADD) & 0x1f1; ++ switch (rtd->codec_dai->dai_runtime.pcmrate) { ++ case SNDRV_PCM_RATE_8000: ++ adn |= 0x5 << 1; ++ fs = 8000 << 7; ++ break; ++ case SNDRV_PCM_RATE_11025: ++ adn |= 0x4 << 1; ++ fs = 11025 << 7; ++ break; ++ case SNDRV_PCM_RATE_16000: ++ adn |= 0x3 << 1; ++ fs = 16000 << 7; ++ break; ++ case SNDRV_PCM_RATE_22050: ++ adn |= 0x2 << 1; ++ fs = 22050 << 7; ++ break; ++ case SNDRV_PCM_RATE_32000: ++ adn |= 0x1 << 1; ++ fs = 32000 << 7; ++ break; ++ case SNDRV_PCM_RATE_44100: ++ fs = 44100 << 7; ++ break; ++ } ++ ++ /* do we need to enable the PLL */ ++ if(dai->pll_in) ++ set_pll(codec, dai->pll_in, dai->pll_out); ++ ++ /* divide the clock to 256 fs */ ++ for(i = 0; i < ARRAY_SIZE(mclk_div); i++) { ++ if (dai->clk_div == mclk_div[i]) { ++ clk |= i << 5; ++ clk &= 0xff; ++ goto set; ++ } ++ } ++ ++set: ++ /* set iface */ ++ wm8510_write(codec, WM8510_IFACE, iface); ++ wm8510_write(codec, WM8510_CLOCK, clk); ++ ++ return 0; ++} ++ ++static int wm8510_hw_free(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ set_pll(codec, 0, 0); ++ return 0; ++} ++ ++static int wm8510_mute(struct snd_soc_codec *codec, ++ struct snd_soc_codec_dai *dai, int mute) ++{ ++ u16 mute_reg = wm8510_read_reg_cache(codec, WM8510_DAC) & 0xffbf; ++ if(mute) ++ wm8510_write(codec, WM8510_DAC, mute_reg | 0x40); ++ else ++ wm8510_write(codec, WM8510_DAC, mute_reg); ++ return 0; ++} ++ ++/* liam need to make this lower power with dapm */ ++static int wm8510_dapm_event(struct snd_soc_codec *codec, int event) ++{ ++ ++ switch (event) { ++ case SNDRV_CTL_POWER_D0: /* full On */ ++ /* vref/mid, clk and osc on, dac unmute, active */ ++ wm8510_write(codec, WM8510_POWER1, 0x1ff); ++ wm8510_write(codec, WM8510_POWER2, 0x1ff); ++ wm8510_write(codec, WM8510_POWER3, 0x1ff); ++ break; ++ case SNDRV_CTL_POWER_D1: /* partial On */ ++ case SNDRV_CTL_POWER_D2: /* partial On */ ++ break; ++ case SNDRV_CTL_POWER_D3hot: /* Off, with power */ ++ /* everything off except vref/vmid, dac mute, inactive */ ++ ++ break; ++ case SNDRV_CTL_POWER_D3cold: /* Off, without power */ ++ /* everything off, dac mute, inactive */ ++ wm8510_write(codec, WM8510_POWER1, 0x0); ++ wm8510_write(codec, WM8510_POWER2, 0x0); ++ wm8510_write(codec, WM8510_POWER3, 0x0); ++ break; ++ } ++ codec->dapm_state = event; ++ return 0; ++} ++ ++struct snd_soc_codec_dai wm8510_dai = { ++ .name = "WM8510 HiFi", ++ .playback = { ++ .stream_name = "Playback", ++ .channels_min = 1, ++ .channels_max = 1, ++ }, ++ .capture = { ++ .stream_name = "Capture", ++ .channels_min = 1, ++ .channels_max = 1, ++ }, ++ .config_sysclk = wm8510_config_sysclk, ++ .digital_mute = wm8510_mute, ++ .ops = { ++ .prepare = wm8510_pcm_prepare, ++ .hw_free = wm8510_hw_free, ++ }, ++ .caps = { ++ .num_modes = ARRAY_SIZE(wm8510_modes), ++ .mode = wm8510_modes, ++ }, ++}; ++EXPORT_SYMBOL_GPL(wm8510_dai); ++ ++static int wm8510_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ wm8510_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ return 0; ++} ++ ++static int wm8510_resume(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ int i; ++ u8 data[2]; ++ u16 *cache = codec->reg_cache; ++ ++ /* Sync reg_cache with the hardware */ ++ for (i = 0; i < ARRAY_SIZE(wm8510_reg); i++) { ++ data[0] = (i << 1) | ((cache[i] >> 8) & 0x0001); ++ data[1] = cache[i] & 0x00ff; ++ codec->hw_write(codec->control_data, data, 2); ++ } ++ wm8510_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ wm8510_dapm_event(codec, codec->suspend_dapm_state); ++ return 0; ++} ++ ++/* ++ * initialise the WM8510 driver ++ * register the mixer and dsp interfaces with the kernel ++ */ ++static int wm8510_init(struct snd_soc_device *socdev) ++{ ++ struct snd_soc_codec *codec = socdev->codec; ++ int ret = 0; ++ ++ codec->name = "WM8510"; ++ codec->owner = THIS_MODULE; ++ codec->read = wm8510_read_reg_cache; ++ codec->write = wm8510_write; ++ codec->dapm_event = wm8510_dapm_event; ++ codec->dai = &wm8510_dai; ++ codec->num_dai = 1; ++ codec->reg_cache_size = ARRAY_SIZE(wm8510_reg); ++ codec->reg_cache = ++ kzalloc(sizeof(u16) * ARRAY_SIZE(wm8510_reg), GFP_KERNEL); ++ if (codec->reg_cache == NULL) ++ return -ENOMEM; ++ memcpy(codec->reg_cache, wm8510_reg, ++ sizeof(u16) * ARRAY_SIZE(wm8510_reg)); ++ codec->reg_cache_size = sizeof(u16) * ARRAY_SIZE(wm8510_reg); ++ ++ wm8510_reset(codec); ++ ++ /* register pcms */ ++ ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); ++ if(ret < 0) { ++ kfree(codec->reg_cache); ++ return ret; ++ } ++ ++ /* power on device */ ++ wm8510_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ wm8510_add_controls(codec); ++ wm8510_add_widgets(codec); ++ ret = snd_soc_register_card(socdev); ++ if(ret < 0) { ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++ } ++ ++ return ret; ++} ++ ++static struct snd_soc_device *wm8510_socdev; ++ ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ ++/* ++ * WM8510 2 wire address is 0x1a ++ */ ++#define I2C_DRIVERID_WM8510 0xfefe /* liam - need a proper id */ ++ ++static unsigned short normal_i2c[] = { 0, I2C_CLIENT_END }; ++ ++/* Magic definition of all other variables and things */ ++I2C_CLIENT_INSMOD; ++ ++static struct i2c_driver wm8510_i2c_driver; ++static struct i2c_client client_template; ++ ++/* If the i2c layer weren't so broken, we could pass this kind of data ++ around */ ++ ++static int wm8510_codec_probe(struct i2c_adapter *adap, int addr, int kind) ++{ ++ struct snd_soc_device *socdev = wm8510_socdev; ++ struct wm8510_setup_data *setup = socdev->codec_data; ++ struct snd_soc_codec *codec = socdev->codec; ++ struct i2c_client *i2c; ++ int ret; ++ ++ if (addr != setup->i2c_address) ++ return -ENODEV; ++ ++ client_template.adapter = adap; ++ client_template.addr = addr; ++ ++ i2c = kzalloc(sizeof(struct i2c_client), GFP_KERNEL); ++ if (i2c == NULL){ ++ kfree(codec); ++ return -ENOMEM; ++ } ++ memcpy(i2c, &client_template, sizeof(struct i2c_client)); ++ i2c_set_clientdata(i2c, codec); ++ codec->control_data = i2c; ++ ++ ret = i2c_attach_client(i2c); ++ if(ret < 0) { ++ err("failed to attach codec at addr %x\n", addr); ++ goto err; ++ } ++ ++ ret = wm8510_init(socdev); ++ if(ret < 0) { ++ err("failed to initialise WM8510\n"); ++ goto err; ++ } ++ return ret; ++ ++err: ++ kfree(codec); ++ kfree(i2c); ++ return ret; ++} ++ ++static int wm8510_i2c_detach(struct i2c_client *client) ++{ ++ struct snd_soc_codec *codec = i2c_get_clientdata(client); ++ i2c_detach_client(client); ++ kfree(codec->reg_cache); ++ kfree(client); ++ return 0; ++} ++ ++static int wm8510_i2c_attach(struct i2c_adapter *adap) ++{ ++ return i2c_probe(adap, &addr_data, wm8510_codec_probe); ++} ++ ++/* corgi i2c codec control layer */ ++static struct i2c_driver wm8510_i2c_driver = { ++ .driver = { ++ .name = "WM8510 I2C Codec", ++ .owner = THIS_MODULE, ++ }, ++ .id = I2C_DRIVERID_WM8510, ++ .attach_adapter = wm8510_i2c_attach, ++ .detach_client = wm8510_i2c_detach, ++ .command = NULL, ++}; ++ ++static struct i2c_client client_template = { ++ .name = "WM8510", ++ .driver = &wm8510_i2c_driver, ++}; ++#endif ++ ++static int wm8510_probe(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct wm8510_setup_data *setup; ++ struct snd_soc_codec *codec; ++ int ret = 0; ++ ++ info("WM8510 Audio Codec %s", WM8510_VERSION); ++ ++ setup = socdev->codec_data; ++ codec = kzalloc(sizeof(struct snd_soc_codec), GFP_KERNEL); ++ if (codec == NULL) ++ return -ENOMEM; ++ ++ socdev->codec = codec; ++ mutex_init(&codec->mutex); ++ INIT_LIST_HEAD(&codec->dapm_widgets); ++ INIT_LIST_HEAD(&codec->dapm_paths); ++ ++ wm8510_socdev = socdev; ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ if (setup->i2c_address) { ++ normal_i2c[0] = setup->i2c_address; ++ codec->hw_write = (hw_write_t)i2c_master_send; ++ ret = i2c_add_driver(&wm8510_i2c_driver); ++ if (ret != 0) ++ printk(KERN_ERR "can't add i2c driver"); ++ } ++#else ++ /* Add other interfaces here */ ++#endif ++ return ret; ++} ++ ++/* power down chip */ ++static int wm8510_remove(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ if (codec->control_data) ++ wm8510_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ i2c_del_driver(&wm8510_i2c_driver); ++#endif ++ kfree(codec); ++ ++ return 0; ++} ++ ++struct snd_soc_codec_device soc_codec_dev_wm8510 = { ++ .probe = wm8510_probe, ++ .remove = wm8510_remove, ++ .suspend = wm8510_suspend, ++ .resume = wm8510_resume, ++}; ++ ++EXPORT_SYMBOL_GPL(soc_codec_dev_wm8510); ++ ++MODULE_DESCRIPTION("ASoC WM8510 driver"); ++MODULE_AUTHOR("Liam Girdwood"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8510.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8510.h +@@ -0,0 +1,64 @@ ++/* ++ * wm8510.h -- WM8510 Soc Audio driver ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#ifndef _WM8510_H ++#define _WM8510_H ++ ++/* WM8510 register space */ ++ ++#define WM8510_RESET 0x0 ++#define WM8510_POWER1 0x1 ++#define WM8510_POWER2 0x2 ++#define WM8510_POWER3 0x3 ++#define WM8510_IFACE 0x4 ++#define WM8510_COMP 0x5 ++#define WM8510_CLOCK 0x6 ++#define WM8510_ADD 0x7 ++#define WM8510_GPIO 0x8 ++#define WM8510_DAC 0xa ++#define WM8510_DACVOL 0xb ++#define WM8510_ADC 0xe ++#define WM8510_ADCVOL 0xf ++#define WM8510_EQ1 0x12 ++#define WM8510_EQ2 0x13 ++#define WM8510_EQ3 0x14 ++#define WM8510_EQ4 0x15 ++#define WM8510_EQ5 0x16 ++#define WM8510_DACLIM1 0x18 ++#define WM8510_DACLIM2 0x19 ++#define WM8510_NOTCH1 0x1b ++#define WM8510_NOTCH2 0x1c ++#define WM8510_NOTCH3 0x1d ++#define WM8510_NOTCH4 0x1e ++#define WM8510_ALC1 0x20 ++#define WM8510_ALC2 0x21 ++#define WM8510_ALC3 0x22 ++#define WM8510_NGATE 0x23 ++#define WM8510_PLLN 0x24 ++#define WM8510_PLLK1 0x25 ++#define WM8510_PLLK2 0x26 ++#define WM8510_PLLK3 0x27 ++#define WM8510_ATTEN 0x28 ++#define WM8510_INPUT 0x2c ++#define WM8510_INPPGA 0x2d ++#define WM8510_ADCBOOST 0x2f ++#define WM8510_OUTPUT 0x31 ++#define WM8510_SPKMIX 0x32 ++#define WM8510_SPKVOL 0x36 ++#define WM8510_MONOMIX 0x38 ++ ++#define WM8510_CACHEREGNUM 57 ++ ++struct wm8510_setup_data { ++ unsigned short i2c_address; ++}; ++ ++extern struct snd_soc_codec_dai wm8510_dai; ++extern struct snd_soc_codec_device soc_codec_dev_wm8510; ++ ++#endif +Index: linux-2.6-pxa-new/sound/soc/imx/imx-ac97.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/imx/imx-ac97.c +@@ -0,0 +1,281 @@ ++/* ++ * imx-ssi.c -- SSI driver for Freescale IMX ++ * ++ * Copyright 2006 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * Based on mxc-alsa-mc13783 (C) 2006 Freescale. ++ * ++ * 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. ++ * ++ * Revision history ++ * 29th Aug 2006 Initial version. ++ * ++ */ ++ ++#define IMX_AC97_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 |\ ++ SNDRV_PCM_RATE_16000 | SNDRV_PCM_RATE_22050 |\ ++ SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000) ++ ++/* may need to expand this */ ++static struct snd_soc_dai_mode imx_ssi_ac97_modes[] = { ++ {0, 0, SNDRV_PCM_FMTBIT_S16_LE, IMX_AC97_RATES}, ++ {0, 0, SNDRV_PCM_FMTBIT_S18_3LE, IMX_AC97_RATES}, ++ {0, 0, SNDRV_PCM_FMTBIT_S20_3LE, IMX_AC97_RATES}, ++}; ++ ++static imx_pcm_dma_params_t imx_ssi1_pcm_stereo_out = { ++ .name = "SSI1 PCM Stereo out", ++ .params = { ++ .bd_number = 1, ++ .transfer_type = emi_2_per, ++ .watermark_level = SDMA_TXFIFO_WATERMARK, ++ .word_size = TRANSFER_16BIT, // maybe add this in setup func ++ .per_address = SSI1_STX0, ++ .event_id = DMA_REQ_SSI1_TX1, ++ .peripheral_type = SSI, ++ }, ++}; ++ ++static imx_pcm_dma_params_t imx_ssi1_pcm_stereo_in = { ++ .name = "SSI1 PCM Stereo in", ++ .params = { ++ .bd_number = 1, ++ .transfer_type = per_2_emi, ++ .watermark_level = SDMA_RXFIFO_WATERMARK, ++ .word_size = TRANSFER_16BIT, // maybe add this in setup func ++ .per_address = SSI1_SRX0, ++ .event_id = DMA_REQ_SSI1_RX1, ++ .peripheral_type = SSI, ++ }, ++}; ++ ++static imx_pcm_dma_params_t imx_ssi2_pcm_stereo_out = { ++ .name = "SSI2 PCM Stereo out", ++ .params = { ++ .bd_number = 1, ++ .transfer_type = per_2_emi, ++ .watermark_level = SDMA_TXFIFO_WATERMARK, ++ .word_size = TRANSFER_16BIT, // maybe add this in setup func ++ .per_address = SSI2_STX0, ++ .event_id = DMA_REQ_SSI2_TX1, ++ .peripheral_type = SSI, ++ }, ++}; ++ ++static imx_pcm_dma_params_t imx_ssi2_pcm_stereo_in = { ++ .name = "SSI2 PCM Stereo in", ++ .params = { ++ .bd_number = 1, ++ .transfer_type = per_2_emi, ++ .watermark_level = SDMA_RXFIFO_WATERMARK, ++ .word_size = TRANSFER_16BIT, // maybe add this in setup func ++ .per_address = SSI2_SRX0, ++ .event_id = DMA_REQ_SSI2_RX1, ++ .peripheral_type = SSI, ++ }, ++}; ++ ++static unsigned short imx_ssi_ac97_read(struct snd_ac97 *ac97, unsigned short reg) ++{ ++} ++ ++static void imx_ssi_ac97_write(struct snd_ac97 *ac97, unsigned short reg, unsigned short val) ++{ ++} ++ ++static void imx_ssi_ac97_warm_reset(struct snd_ac97 *ac97) ++{ ++} ++ ++static void imx_ssi_ac97_cold_reset(struct snd_ac97 *ac97) ++{ ++} ++ ++struct snd_ac97_bus_ops soc_ac97_ops = { ++ .read = imx_ssi_ac97_read, ++ .write = imx_ssi_ac97_write, ++ .warm_reset = imx_ssi_ac97_warm_reset, ++ .reset = imx_ssi_ac97_cold_reset, ++}; ++ ++ ++static intimx_ssi1_ac97_probe(struct platform_device *pdev) ++{ ++ int ret; ++ ++ ++ return ret; ++} ++ ++static void imx_ssi1_ac97_remove(struct platform_device *pdev) ++{ ++ /* shutdown SSI */ ++ if(rtd->cpu_dai->id == 0) ++ SSI1_SCR &= ~SSI_SCR_SSIEN; ++ else ++ SSI2_SCR &= ~SSI_SCR_SSIEN; ++ } ++ ++} ++ ++static int imx_ssi1_ac97_prepare(struct snd_pcm_substream *substream) ++{ ++ // set vra ++} ++ ++static int imx_ssi_startup(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ ++ if (!rtd->cpu_dai->active) { ++ ++ } ++ ++ return 0; ++} ++ ++static int imx_ssi1_trigger(struct snd_pcm_substream *substream, int cmd) ++{ ++ int ret = 0; ++ ++ switch (cmd) { ++ case SNDRV_PCM_TRIGGER_START: ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { ++ SSI1_SCR |= SSI_SCR_TE; ++ SSI1_SIER |= SSI_SIER_TDMAE; ++ } else { ++ SSI1_SCR |= SSI_SCR_RE; ++ SSI1_SIER |= SSI_SIER_RDMAE; ++ } ++ SSI1_SCR |= SSI_SCR_SSIEN; ++ ++ break; ++ case SNDRV_PCM_TRIGGER_RESUME: ++ break; ++ case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ SSI1_SCR |= SSI_SCR_TE; ++ else ++ SSI1_SCR |= SSI_SCR_RE; ++ break ++ case SNDRV_PCM_TRIGGER_STOP: ++ case SNDRV_PCM_TRIGGER_SUSPEND: ++ break; ++ case SNDRV_PCM_TRIGGER_PAUSE_PUSH: ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ SSI1_SCR &= ~SSI_SCR_TE; ++ else ++ SSI1_SCR &= ~SSI_SCR_RE; ++ break; ++ default: ++ ret = -EINVAL; ++ } ++ ++ return ret; ++} ++ ++static void imx_ssi_shutdown(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ ++ ++} ++ ++#ifdef CONFIG_PM ++static int imx_ssi_suspend(struct platform_device *dev, ++ struct snd_soc_cpu_dai *dai) ++{ ++ if(!dai->active) ++ return 0; ++ ++ if(rtd->cpu_dai->id == 0) ++ SSI1_SCR &= ~SSI_SCR_SSIEN; ++ else ++ SSI2_SCR &= ~SSI_SCR_SSIEN; ++ ++ return 0; ++} ++ ++static int imx_ssi_resume(struct platform_device *pdev, ++ struct snd_soc_cpu_dai *dai) ++{ ++ if(!dai->active) ++ return 0; ++ ++ if(rtd->cpu_dai->id == 0) ++ SSI1_SCR |= SSI_SCR_SSIEN; ++ else ++ SSI2_SCR |= SSI_SCR_SSIEN; ++ ++ return 0; ++} ++ ++#else ++#define imx_ssi_suspend NULL ++#define imx_ssi_resume NULL ++#endif ++ ++static unsigned int imx_ssi_config_ac97_sysclk(struct snd_soc_cpu_dai *iface, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ return clk; ++} ++ ++struct snd_soc_cpu_dai imx_ssi_ac97_dai = { ++ .name = "imx-ac97-1", ++ .id = 0, ++ .type = SND_SOC_DAI_AC97, ++ .suspend = imx_ssi_suspend, ++ .resume = imx_ssi_resume, ++ .config_sysclk = imx_ssi_ac97_config_sysclk, ++ .playback = { ++ .channels_min = 2, ++ .channels_max = 2,}, ++ .capture = { ++ .channels_min = 2, ++ .channels_max = 2,}, ++ .ops = { ++ .probe = imx_ac97_probe, ++ .remove = imx_ac97_shutdown, ++ .trigger = imx_ssi1_trigger, ++ .prepare = imx_ssi_ac97_prepare,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(imx_ssi_ac97_modes), ++ .mode = imx_ssi_ac97_modes,}, ++}, ++{ ++ .name = "imx-ac97-2", ++ .id = 1, ++ .type = SND_SOC_DAI_AC97, ++ .suspend = imx_ssi_suspend, ++ .resume = imx_ssi_resume, ++ .config_sysclk = imx_ssi_ac97_config_sysclk, ++ .playback = { ++ .channels_min = 2, ++ .channels_max = 2,}, ++ .capture = { ++ .channels_min = 2, ++ .channels_max = 2,}, ++ .ops = { ++ .probe = imx_ac97_probe, ++ .remove = imx_ac97_shutdown, ++ .trigger = imx_ssi1_trigger, ++ .prepare = imx_ssi_ac97_prepare,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(imx_ssi_ac97_modes), ++ .mode = imx_ssi_ac97_modes,}, ++}; ++ ++EXPORT_SYMBOL_GPL(imx_ssi_ac97_dai); ++ ++/* Module information */ ++MODULE_AUTHOR("Liam Girdwood, liam.girdwood@wolfsonmicro.com, www.wolfsonmicro.com"); ++MODULE_DESCRIPTION("i.MX ASoC AC97 driver"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/imx/imx-i2s.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/imx/imx-i2s.c +@@ -0,0 +1,473 @@ ++/* ++ * imx-ssi.c -- SSI driver for Freescale IMX ++ * ++ * Copyright 2006 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * Based on mxc-alsa-mc13783 (C) 2006 Freescale. ++ * ++ * 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. ++ * ++ * Revision history ++ * 29th Aug 2006 Initial version. ++ * ++ */ ++ ++#define IMX_SSI_DAIFMT \ ++ (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_LEFT_J |\ ++ SND_SOC_DAIFMT_RIGHT_J | SND_SOC_DAIFMT_DSP__A |\ ++ SND_SOC_DAIFMT_DSP_B | SND_SOC_DAIFMT_CBS_CFS |\ ++ SND_SOC_DAIFMT_CBM_CFS | SND_SOC_DAIFMT_CBS_CFM |\ ++ SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_NB_IF) ++ ++#define IMX_SSI_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define IMX_SSI_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | \ ++ SNDRV_PCM_RATE_16000 | SNDRV_PCM_RATE_22050 | \ ++ SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_88200 | \ ++ SNDRV_PCM_RATE_96000) ++ ++#define IMX_SSI_BITS \ ++ (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE | \ ++ SNDRV_PCM_FMTBIT_S24_LE) ++ ++static struct snd_soc_dai_mode imx_ssi_pcm_modes[] = { ++ ++ /* frame master and clock slave mode */ ++ { ++ .fmt = IMX_SSI_DAIFMT | SND_SOC_DAIFMT_CBM_CFS, ++ .tdm = SND_SOC_DAITDM_LRDW(0,0), ++ .pcmfmt = IMX_SSI_BITS, ++ .pcmrate = IMX_SSI_RATES, ++ .pcmdir = IMX_SSI_DIR, ++ .flags = SND_SOC_DAI_BFS_RCW, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = SND_SOC_FSBW(1) | SND_SOC_FSBW(2), ++ }, ++}; ++ ++static imx_pcm_dma_params_t imx_ssi1_pcm_stereo_out = { ++ .name = "SSI1 PCM Stereo out", ++ .params = { ++ .bd_number = 1, ++ .transfer_type = emi_2_per, ++ .watermark_level = SDMA_TXFIFO_WATERMARK, ++ .word_size = TRANSFER_16BIT, // maybe add this in setup func ++ .per_address = SSI1_STX0, ++ .event_id = DMA_REQ_SSI1_TX1, ++ .peripheral_type = SSI, ++ }, ++}; ++ ++static imx_pcm_dma_params_t imx_ssi1_pcm_stereo_in = { ++ .name = "SSI1 PCM Stereo in", ++ .params = { ++ .bd_number = 1, ++ .transfer_type = per_2_emi, ++ .watermark_level = SDMA_RXFIFO_WATERMARK, ++ .word_size = TRANSFER_16BIT, // maybe add this in setup func ++ .per_address = SSI1_SRX0, ++ .event_id = DMA_REQ_SSI1_RX1, ++ .peripheral_type = SSI, ++ }, ++}; ++ ++static imx_pcm_dma_params_t imx_ssi2_pcm_stereo_out = { ++ .name = "SSI2 PCM Stereo out", ++ .params = { ++ .bd_number = 1, ++ .transfer_type = per_2_emi, ++ .watermark_level = SDMA_TXFIFO_WATERMARK, ++ .word_size = TRANSFER_16BIT, // maybe add this in setup func ++ .per_address = SSI2_STX0, ++ .event_id = DMA_REQ_SSI2_TX1, ++ .peripheral_type = SSI, ++ }, ++}; ++ ++static imx_pcm_dma_params_t imx_ssi2_pcm_stereo_in = { ++ .name = "SSI2 PCM Stereo in", ++ .params = { ++ .bd_number = 1, ++ .transfer_type = per_2_emi, ++ .watermark_level = SDMA_RXFIFO_WATERMARK, ++ .word_size = TRANSFER_16BIT, // maybe add this in setup func ++ .per_address = SSI2_SRX0, ++ .event_id = DMA_REQ_SSI2_RX1, ++ .peripheral_type = SSI, ++ }, ++}; ++ ++ ++static int imx_ssi_startup(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ ++ if (!rtd->cpu_dai->active) { ++ ++ } ++ ++ return 0; ++} ++ ++static int imx_ssi1_hw_tx_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 bfs, div; ++ ++ bfs = SND_SOC_FSBD_REAL(rtd->cpu_dai->dai_runtime.bfs); ++ ++ SSI1_STCR = 0; ++ SSI1_STCCR = 0; ++ ++ /* DAI mode */ ++ switch(rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_I2S: ++ SSI1_STCR |= SSI_STCR_TSCKP | SSI_STCR_TFSI | ++ SSI_STCR_TEFS | SSI_STCR_TXBIT0; ++ break; ++ case SND_SOC_DAIFMT_LEFT_J: ++ SSI1_STCR |= SSI_STCR_TSCKP | SSI_STCR_TFSI | SSI_STCR_TXBIT0; ++ break; ++ case SND_SOC_DAIFMT_DSP_B: ++ SSI1_STCR |= SSI_STCR_TEFS; // data 1 bit after sync ++ case SND_SOC_DAIFMT_DSP_A: ++ SSI1_STCR |= SSI_STCR_TFSL; // frame is 1 bclk long ++ ++ /* DAI clock inversion */ ++ switch(rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_INV_MASK) { ++ case SND_SOC_DAIFMT_IB_IF: ++ SSI1_STCR |= SSI_STCR_TFSI | SSI_STCR_TSCKP; ++ break; ++ case SND_SOC_DAIFMT_IB_NF: ++ SSI1_STCR |= SSI_STCR_TSCKP; ++ break; ++ case SND_SOC_DAIFMT_NB_IF: ++ SSI1_STCR |= SSI_STCR_TFSI; ++ break; ++ } ++ break; ++ } ++ ++ /* DAI data (word) size */ ++ switch(rtd->codec_dai->dai_runtime.pcmfmt) { ++ case SNDRV_PCM_FMTBIT_S16_LE: ++ SSI1_STCCR |= SSI_STCCR_WL(16); ++ break; ++ case SNDRV_PCM_FMTBIT_S20_3LE: ++ SSI1_STCCR |= SSI_STCCR_WL(20); ++ break; ++ case SNDRV_PCM_FMTBIT_S24_LE: ++ SSI1_STCCR |= SSI_STCCR_WL(24); ++ break; ++ } ++ ++ /* DAI clock master masks */ ++ switch(rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK){ ++ case SND_SOC_DAIFMT_CBM_CFM: ++ SSI1_STCR |= SSI_STCR_TFDIR | SSI_STCR_TXDIR; ++ break; ++ case SND_SOC_DAIFMT_CBS_CFM: ++ SSI1_STCR |= SSI_STCR_TFDIR; ++ break; ++ case SND_SOC_DAIFMT_CBM_CFS: ++ SSI1_STCR |= SSI_STCR_TXDIR; ++ break; ++ } ++ ++ /* DAI BCLK ratio to SYSCLK / MCLK */ ++ /* prescaler modulus - todo */ ++ switch (bfs) { ++ case 2: ++ break; ++ case 4: ++ break; ++ case 8: ++ break; ++ case 16: ++ break; ++ } ++ ++ /* TDM - todo, only fifo 0 atm */ ++ SSI1_STCR |= SSI_STCR_TFEN0; ++ SSI1_STCCR |= SSI_STCCR_DC(params_channels(params)); ++ ++ return 0; ++} ++ ++static int imx_ssi1_hw_rx_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ u16 bfs, div; ++ ++ bfs = SND_SOC_FSBD_REAL(rtd->cpu_dai->dai_runtime.bfs); ++ ++ SSI1_SRCR = 0; ++ SSI1_SRCCR = 0; ++ ++ /* DAI mode */ ++ switch(rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_I2S: ++ SSI1_SRCR |= SSI_SRCR_RSCKP | SSI_SRCR_RFSI | ++ SSI_STCR_REFS | SSI_STCR_RXBIT0; ++ break; ++ case SND_SOC_DAIFMT_LEFT_J: ++ SSI1_SRCR |= SSI_SRCR_RSCKP | SSI_SRCR_RFSI | SSI_SRCR_RXBIT0; ++ break; ++ case SND_SOC_DAIFMT_DSP_B: ++ SSI1_SRCR |= SSI_SRCR_REFS; // data 1 bit after sync ++ case SND_SOC_DAIFMT_DSP_A: ++ SSI1_SRCR |= SSI_SRCR_RFSL; // frame is 1 bclk long ++ ++ /* DAI clock inversion */ ++ switch(rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_INV_MASK) { ++ case SND_SOC_DAIFMT_IB_IF: ++ SSI1_SRCR |= SSI_SRCR_TFSI | SSI_SRCR_TSCKP; ++ break; ++ case SND_SOC_DAIFMT_IB_NF: ++ SSI1_SRCR |= SSI_SRCR_RSCKP; ++ break; ++ case SND_SOC_DAIFMT_NB_IF: ++ SSI1_SRCR |= SSI_SRCR_RFSI; ++ break; ++ } ++ break; ++ } ++ ++ /* DAI data (word) size */ ++ switch(rtd->codec_dai->dai_runtime.pcmfmt) { ++ case SNDRV_PCM_FMTBIT_S16_LE: ++ SSI1_SRCCR |= SSI_SRCCR_WL(16); ++ break; ++ case SNDRV_PCM_FMTBIT_S20_3LE: ++ SSI1_SRCCR |= SSI_SRCCR_WL(20); ++ break; ++ case SNDRV_PCM_FMTBIT_S24_LE: ++ SSI1_SRCCR |= SSI_SRCCR_WL(24); ++ break; ++ } ++ ++ /* DAI clock master masks */ ++ switch(rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK){ ++ case SND_SOC_DAIFMT_CBM_CFM: ++ SSI1_SRCR |= SSI_SRCR_RFDIR | SSI_SRCR_RXDIR; ++ break; ++ case SND_SOC_DAIFMT_CBS_CFM: ++ SSI1_SRCR |= SSI_SRCR_RFDIR; ++ break; ++ case SND_SOC_DAIFMT_CBM_CFS: ++ SSI1_SRCR |= SSI_SRCR_RXDIR; ++ break; ++ } ++ ++ /* DAI BCLK ratio to SYSCLK / MCLK */ ++ /* prescaler modulus - todo */ ++ switch (bfs) { ++ case 2: ++ break; ++ case 4: ++ break; ++ case 8: ++ break; ++ case 16: ++ break; ++ } ++ ++ /* TDM - todo, only fifo 0 atm */ ++ SSI1_SRCR |= SSI_SRCR_RFEN0; ++ SSI1_SRCCR |= SSI_SRCCR_DC(params_channels(params)); ++ ++ return 0; ++} ++ ++static int imx_ssi1_hw_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ ++ /* clear register if not enabled */ ++ if(!(SSI1_SCR & SSI_SCR_SSIEN)) ++ SSI1_SCR = 0; ++ ++ /* async */ ++ if (rtd->cpu_dai->flags & SND_SOC_DAI_ASYNC) ++ SSI1_SCR |= SSI_SCR_SYN; ++ ++ /* DAI mode */ ++ switch(rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_I2S: ++ case SND_SOC_DAIFMT_LEFT_J: ++ SSI1_SCR |= SSI_SCR_NET; ++ break; ++ } ++ ++ /* TDM - to complete */ ++ ++ /* Tx/Rx config */ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { ++ return imx_ssi1_hw_tx_params(substream, params); ++ } else { ++ return imx_ssi1_hw_rx_params(substream, params); ++ } ++} ++ ++ ++ ++static int imx_ssi1_trigger(struct snd_pcm_substream *substream, int cmd) ++{ ++ int ret = 0; ++ ++ switch (cmd) { ++ case SNDRV_PCM_TRIGGER_START: ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { ++ SSI1_SCR |= SSI_SCR_TE; ++ SSI1_SIER |= SSI_SIER_TDMAE; ++ } else { ++ SSI1_SCR |= SSI_SCR_RE; ++ SSI1_SIER |= SSI_SIER_RDMAE; ++ } ++ SSI1_SCR |= SSI_SCR_SSIEN; ++ ++ break; ++ case SNDRV_PCM_TRIGGER_RESUME: ++ break; ++ case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ SSI1_SCR |= SSI_SCR_TE; ++ else ++ SSI1_SCR |= SSI_SCR_RE; ++ break ++ case SNDRV_PCM_TRIGGER_STOP: ++ case SNDRV_PCM_TRIGGER_SUSPEND: ++ break; ++ case SNDRV_PCM_TRIGGER_PAUSE_PUSH: ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ SSI1_SCR &= ~SSI_SCR_TE; ++ else ++ SSI1_SCR &= ~SSI_SCR_RE; ++ break; ++ default: ++ ret = -EINVAL; ++ } ++ ++ return ret; ++} ++ ++static void imx_ssi_shutdown(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ ++ /* shutdown SSI */ ++ if (!rtd->cpu_dai->active) { ++ if(rtd->cpu_dai->id == 0) ++ SSI1_SCR &= ~SSI_SCR_SSIEN; ++ else ++ SSI2_SCR &= ~SSI_SCR_SSIEN; ++ } ++} ++ ++#ifdef CONFIG_PM ++static int imx_ssi_suspend(struct platform_device *dev, ++ struct snd_soc_cpu_dai *dai) ++{ ++ if(!dai->active) ++ return 0; ++ ++ if(rtd->cpu_dai->id == 0) ++ SSI1_SCR &= ~SSI_SCR_SSIEN; ++ else ++ SSI2_SCR &= ~SSI_SCR_SSIEN; ++ ++ return 0; ++} ++ ++static int imx_ssi_resume(struct platform_device *pdev, ++ struct snd_soc_cpu_dai *dai) ++{ ++ if(!dai->active) ++ return 0; ++ ++ if(rtd->cpu_dai->id == 0) ++ SSI1_SCR |= SSI_SCR_SSIEN; ++ else ++ SSI2_SCR |= SSI_SCR_SSIEN; ++ ++ return 0; ++} ++ ++#else ++#define imx_ssi_suspend NULL ++#define imx_ssi_resume NULL ++#endif ++ ++static unsigned int imx_ssi_config_pcm_sysclk(struct snd_soc_cpu_dai *iface, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ return clk; ++} ++ ++struct snd_soc_cpu_dai imx_ssi_pcm_dai = { ++ .name = "imx-i2s-1", ++ .id = 0, ++ .type = SND_SOC_DAI_I2S, ++ .suspend = imx_ssi_suspend, ++ .resume = imx_ssi_resume, ++ .config_sysclk = imx_ssi_config_pcm_sysclk, ++ .playback = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .capture = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .ops = { ++ .startup = imx_ssi_startup, ++ .shutdown = imx_ssi_shutdown, ++ .trigger = imx_ssi1_trigger, ++ .hw_params = imx_ssi1_pcm_hw_params,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(imx_ssi_modes), ++ .mode = imx_ssi_modes,}, ++}, ++{ ++ .name = "imx-i2s-2", ++ .id = 1, ++ .type = SND_SOC_DAI_I2S, ++ .suspend = imx_ssi_suspend, ++ .resume = imx_ssi_resume, ++ .config_sysclk = imx_ssi_config_pcm_sysclk, ++ .playback = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .capture = { ++ .channels_min = 1, ++ .channels_max = 2,}, ++ .ops = { ++ .startup = imx_ssi_startup, ++ .shutdown = imx_ssi_shutdown, ++ .trigger = imx_ssi1_trigger, ++ .hw_params = imx_ssi1_pcm_hw_params,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(imx_ssi_modes), ++ .mode = imx_ssi_modes,}, ++}; ++ ++ ++EXPORT_SYMBOL_GPL(imx_ssi_i2s_dai); ++ ++/* Module information */ ++MODULE_AUTHOR("Liam Girdwood, liam.girdwood@wolfsonmicro.com, www.wolfsonmicro.com"); ++MODULE_DESCRIPTION("i.MX ASoC I2S driver"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/include/linux/i2c-id.h +=================================================================== +--- linux-2.6-pxa-new.orig/include/linux/i2c-id.h ++++ linux-2.6-pxa-new/include/linux/i2c-id.h +@@ -116,6 +116,8 @@ + #define I2C_DRIVERID_KS0127 86 /* Samsung ks0127 video decoder */ + #define I2C_DRIVERID_TLV320AIC23B 87 /* TI TLV320AIC23B audio codec */ + #define I2C_DRIVERID_ISL1208 88 /* Intersil ISL1208 RTC */ ++#define I2C_DRIVERID_WM8731 89 /* Wolfson WM8731 audio codec */ ++#define I2C_DRIVERID_WM8750 90 /* Wolfson WM8750 audio codec */ + + #define I2C_DRIVERID_I2CDEV 900 + #define I2C_DRIVERID_ARP 902 /* SMBus ARP Client */ +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8976.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8976.c +@@ -0,0 +1,953 @@ ++/* ++ * wm8976.c -- WM8976 ALSA Soc Audio driver ++ * ++ * Copyright 2006 Wolfson Microelectronics PLC. ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/version.h> ++#include <linux/kernel.h> ++#include <linux/init.h> ++#include <linux/delay.h> ++#include <linux/pm.h> ++#include <linux/i2c.h> ++#include <linux/platform_device.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/pcm_params.h> ++#include <sound/soc.h> ++#include <sound/soc-dapm.h> ++#include <sound/initval.h> ++ ++#include "wm8976.h" ++ ++#define AUDIO_NAME "wm8976" ++#define WM8976_VERSION "0.2" ++ ++/* ++ * Debug ++ */ ++ ++#define WM8976_DEBUG 0 ++ ++#ifdef WM8976_DEBUG ++#define dbg(format, arg...) \ ++ printk(KERN_DEBUG AUDIO_NAME ": " format "\n" , ## arg) ++#else ++#define dbg(format, arg...) do {} while (0) ++#endif ++#define err(format, arg...) \ ++ printk(KERN_ERR AUDIO_NAME ": " format "\n" , ## arg) ++#define info(format, arg...) \ ++ printk(KERN_INFO AUDIO_NAME ": " format "\n" , ## arg) ++#define warn(format, arg...) \ ++ printk(KERN_WARNING AUDIO_NAME ": " format "\n" , ## arg) ++ ++struct snd_soc_codec_device soc_codec_dev_wm8976; ++ ++/* ++ * wm8976 register cache ++ * We can't read the WM8976 register space when we are ++ * using 2 wire for device control, so we cache them instead. ++ */ ++static const u16 wm8976_reg[WM8976_CACHEREGNUM] = { ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0050, 0x0000, 0x0140, 0x0000, ++ 0x0000, 0x0000, 0x0000, 0x00ff, ++ 0x00ff, 0x0000, 0x0100, 0x00ff, ++ 0x00ff, 0x0000, 0x012c, 0x002c, ++ 0x002c, 0x002c, 0x002c, 0x0000, ++ 0x0032, 0x0000, 0x0000, 0x0000, ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0038, 0x000b, 0x0032, 0x0000, ++ 0x0008, 0x000c, 0x0093, 0x00e9, ++ 0x0000, 0x0000, 0x0000, 0x0000, ++ 0x0033, 0x0010, 0x0010, 0x0100, ++ 0x0100, 0x0002, 0x0001, 0x0001, ++ 0x0039, 0x0039, 0x0039, 0x0039, ++ 0x0001, 0x0001, ++}; ++ ++#define WM8976_DAIFMT \ ++ (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_LEFT_J | SND_SOC_DAIFMT_RIGHT_J | \ ++ SND_SOC_DAIFMT_DSP_A | SND_SOC_DAIFMT_DSP_B | SND_SOC_DAIFMT_NB_NF | \ ++ SND_SOC_DAIFMT_NB_IF | SND_SOC_DAIFMT_IB_NF | SND_SOC_DAIFMT_IB_IF) ++ ++#define WM8976_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define WM8976_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000) ++ ++#define WM8976_PCM_FORMATS \ ++ (SNDRV_PCM_FORMAT_S16_LE | SNDRV_PCM_FORMAT_S20_3LE | \ ++ SNDRV_PCM_FORMAT_S24_3LE | SNDRV_PCM_FORMAT_S24_LE | \ ++ SNDRV_PCM_FORMAT_S32_LE) ++ ++#define WM8976_BCLK \ ++ (SND_SOC_FSBD(1) | SND_SOC_FSBD(2) | SND_SOC_FSBD(4) | SND_SOC_FSBD(8) |\ ++ SND_SOC_FSBD(16) | SND_SOC_FSBD(32)) ++ ++static struct snd_soc_dai_mode wm8976_modes[] = { ++ /* codec frame and clock master modes */ ++ { ++ .fmt = WM8976_DAIFMT | SND_SOC_DAIFMT_CBM_CFM, ++ .pcmfmt = WM8976_PCM_FORMATS, ++ .pcmrate = WM8976_RATES, ++ .pcmdir = WM8976_DIR, ++ .flags = SND_SOC_DAI_BFS_DIV, ++ .fs = 256, ++ .bfs = WM8976_BCLK, ++ }, ++ ++ /* codec frame and clock slave modes */ ++ { ++ .fmt = WM8976_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = WM8976_PCM_FORMATS, ++ .pcmrate = WM8976_RATES, ++ .pcmdir = WM8976_DIR, ++ .fs = SND_SOC_FS_ALL, ++ .bfs = SND_SOC_FSB_ALL, ++ }, ++}; ++ ++/* ++ * read wm8976 register cache ++ */ ++static inline unsigned int wm8976_read_reg_cache(struct snd_soc_codec *codec, ++ unsigned int reg) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg == WM8976_RESET) ++ return 0; ++ if (reg >= WM8976_CACHEREGNUM) ++ return -1; ++ return cache[reg]; ++} ++ ++/* ++ * write wm8976 register cache ++ */ ++static inline void wm8976_write_reg_cache(struct snd_soc_codec *codec, ++ u16 reg, unsigned int value) ++{ ++ u16 *cache = codec->reg_cache; ++ if (reg >= WM8976_CACHEREGNUM) ++ return; ++ cache[reg] = value; ++} ++ ++/* ++ * write to the WM8976 register space ++ */ ++static int wm8976_write(struct snd_soc_codec *codec, unsigned int reg, ++ unsigned int value) ++{ ++ u8 data[2]; ++ ++ /* data is ++ * D15..D9 WM8976 register offset ++ * D8...D0 register data ++ */ ++ data[0] = (reg << 1) | ((value >> 8) & 0x0001); ++ data[1] = value & 0x00ff; ++ ++ wm8976_write_reg_cache (codec, reg, value); ++ if (codec->hw_write(codec->control_data, data, 2) == 2) ++ return 0; ++ else ++ return -1; ++} ++ ++#define wm8976_reset(c) wm8976_write(c, WM8976_RESET, 0) ++ ++static const char *wm8976_companding[] = {"Off", "NC", "u-law", "A-law" }; ++static const char *wm8976_deemp[] = {"None", "32kHz", "44.1kHz", "48kHz" }; ++static const char *wm8976_eqmode[] = {"Capture", "Playback" }; ++static const char *wm8976_bw[] = {"Narrow", "Wide" }; ++static const char *wm8976_eq1[] = {"80Hz", "105Hz", "135Hz", "175Hz" }; ++static const char *wm8976_eq2[] = {"230Hz", "300Hz", "385Hz", "500Hz" }; ++static const char *wm8976_eq3[] = {"650Hz", "850Hz", "1.1kHz", "1.4kHz" }; ++static const char *wm8976_eq4[] = {"1.8kHz", "2.4kHz", "3.2kHz", "4.1kHz" }; ++static const char *wm8976_eq5[] = {"5.3kHz", "6.9kHz", "9kHz", "11.7kHz" }; ++static const char *wm8976_alc[] = ++ {"ALC both on", "ALC left only", "ALC right only", "Limiter" }; ++ ++static const struct soc_enum wm8976_enum[] = { ++ SOC_ENUM_SINGLE(WM8976_COMP, 1, 4, wm8976_companding), /* adc */ ++ SOC_ENUM_SINGLE(WM8976_COMP, 3, 4, wm8976_companding), /* dac */ ++ SOC_ENUM_SINGLE(WM8976_DAC, 4, 4, wm8976_deemp), ++ SOC_ENUM_SINGLE(WM8976_EQ1, 8, 2, wm8976_eqmode), ++ ++ SOC_ENUM_SINGLE(WM8976_EQ1, 5, 4, wm8976_eq1), ++ SOC_ENUM_SINGLE(WM8976_EQ2, 8, 2, wm8976_bw), ++ SOC_ENUM_SINGLE(WM8976_EQ2, 5, 4, wm8976_eq2), ++ SOC_ENUM_SINGLE(WM8976_EQ3, 8, 2, wm8976_bw), ++ ++ SOC_ENUM_SINGLE(WM8976_EQ3, 5, 4, wm8976_eq3), ++ SOC_ENUM_SINGLE(WM8976_EQ4, 8, 2, wm8976_bw), ++ SOC_ENUM_SINGLE(WM8976_EQ4, 5, 4, wm8976_eq4), ++ SOC_ENUM_SINGLE(WM8976_EQ5, 8, 2, wm8976_bw), ++ ++ SOC_ENUM_SINGLE(WM8976_EQ5, 5, 4, wm8976_eq5), ++ SOC_ENUM_SINGLE(WM8976_ALC3, 8, 2, wm8976_alc), ++}; ++ ++static const struct snd_kcontrol_new wm8976_snd_controls[] = { ++SOC_SINGLE("Digital Loopback Switch", WM8976_COMP, 0, 1, 0), ++ ++SOC_ENUM("ADC Companding", wm8976_enum[0]), ++SOC_ENUM("DAC Companding", wm8976_enum[1]), ++ ++SOC_SINGLE("Jack Detection Enable", WM8976_JACK1, 6, 1, 0), ++ ++SOC_DOUBLE("DAC Inversion Switch", WM8976_DAC, 0, 1, 1, 0), ++ ++SOC_DOUBLE_R("Headphone Playback Volume", WM8976_DACVOLL, WM8976_DACVOLR, 0, 127, 0), ++ ++SOC_SINGLE("High Pass Filter Switch", WM8976_ADC, 8, 1, 0), ++SOC_SINGLE("High Pass Filter Switch", WM8976_ADC, 8, 1, 0), ++SOC_SINGLE("High Pass Cut Off", WM8976_ADC, 4, 7, 0), ++ ++SOC_DOUBLE("ADC Inversion Switch", WM8976_ADC, 0, 1, 1, 0), ++ ++SOC_SINGLE("Capture Volume", WM8976_ADCVOL, 0, 127, 0), ++ ++SOC_ENUM("Equaliser Function", wm8976_enum[3]), ++SOC_ENUM("EQ1 Cut Off", wm8976_enum[4]), ++SOC_SINGLE("EQ1 Volume", WM8976_EQ1, 0, 31, 1), ++ ++SOC_ENUM("Equaliser EQ2 Bandwith", wm8976_enum[5]), ++SOC_ENUM("EQ2 Cut Off", wm8976_enum[6]), ++SOC_SINGLE("EQ2 Volume", WM8976_EQ2, 0, 31, 1), ++ ++SOC_ENUM("Equaliser EQ3 Bandwith", wm8976_enum[7]), ++SOC_ENUM("EQ3 Cut Off", wm8976_enum[8]), ++SOC_SINGLE("EQ3 Volume", WM8976_EQ3, 0, 31, 1), ++ ++SOC_ENUM("Equaliser EQ4 Bandwith", wm8976_enum[9]), ++SOC_ENUM("EQ4 Cut Off", wm8976_enum[10]), ++SOC_SINGLE("EQ4 Volume", WM8976_EQ4, 0, 31, 1), ++ ++SOC_ENUM("Equaliser EQ5 Bandwith", wm8976_enum[11]), ++SOC_ENUM("EQ5 Cut Off", wm8976_enum[12]), ++SOC_SINGLE("EQ5 Volume", WM8976_EQ5, 0, 31, 1), ++ ++SOC_SINGLE("DAC Playback Limiter Switch", WM8976_DACLIM1, 8, 1, 0), ++SOC_SINGLE("DAC Playback Limiter Decay", WM8976_DACLIM1, 4, 15, 0), ++SOC_SINGLE("DAC Playback Limiter Attack", WM8976_DACLIM1, 0, 15, 0), ++ ++SOC_SINGLE("DAC Playback Limiter Threshold", WM8976_DACLIM2, 4, 7, 0), ++SOC_SINGLE("DAC Playback Limiter Boost", WM8976_DACLIM2, 0, 15, 0), ++ ++SOC_SINGLE("ALC Enable Switch", WM8976_ALC1, 8, 1, 0), ++SOC_SINGLE("ALC Capture Max Gain", WM8976_ALC1, 3, 7, 0), ++SOC_SINGLE("ALC Capture Min Gain", WM8976_ALC1, 0, 7, 0), ++ ++SOC_SINGLE("ALC Capture ZC Switch", WM8976_ALC2, 8, 1, 0), ++SOC_SINGLE("ALC Capture Hold", WM8976_ALC2, 4, 7, 0), ++SOC_SINGLE("ALC Capture Target", WM8976_ALC2, 0, 15, 0), ++ ++SOC_ENUM("ALC Capture Mode", wm8976_enum[13]), ++SOC_SINGLE("ALC Capture Decay", WM8976_ALC3, 4, 15, 0), ++SOC_SINGLE("ALC Capture Attack", WM8976_ALC3, 0, 15, 0), ++ ++SOC_SINGLE("ALC Capture Noise Gate Switch", WM8976_NGATE, 3, 1, 0), ++SOC_SINGLE("ALC Capture Noise Gate Threshold", WM8976_NGATE, 0, 7, 0), ++ ++SOC_SINGLE("Capture PGA ZC Switch", WM8976_INPPGA, 7, 1, 0), ++SOC_SINGLE("Capture PGA Volume", WM8976_INPPGA, 0, 63, 0), ++ ++SOC_DOUBLE_R("Headphone Playback ZC Switch", WM8976_HPVOLL, WM8976_HPVOLR, 7, 1, 0), ++SOC_DOUBLE_R("Headphone Playback Switch", WM8976_HPVOLL, WM8976_HPVOLR, 6, 1, 1), ++SOC_DOUBLE_R("Headphone Playback Volume", WM8976_HPVOLL, WM8976_HPVOLR, 0, 63, 0), ++ ++SOC_DOUBLE_R("Speaker Playback ZC Switch", WM8976_SPKVOLL, WM8976_SPKVOLR, 7, 1, 0), ++SOC_DOUBLE_R("Speaker Playback Switch", WM8976_SPKVOLL, WM8976_SPKVOLR, 6, 1, 1), ++SOC_DOUBLE_R("Speaker Playback Volume", WM8976_SPKVOLL, WM8976_SPKVOLR, 0, 63, 0), ++ ++SOC_SINGLE("Capture Boost(+20dB)", WM8976_ADCBOOST, 8, 1, 0), ++}; ++ ++/* add non dapm controls */ ++static int wm8976_add_controls(struct snd_soc_codec *codec) ++{ ++ int err, i; ++ ++ for (i = 0; i < ARRAY_SIZE(wm8976_snd_controls); i++) { ++ err = snd_ctl_add(codec->card, ++ snd_soc_cnew(&wm8976_snd_controls[i],codec, NULL)); ++ if (err < 0) ++ return err; ++ } ++ ++ return 0; ++} ++ ++/* Left Output Mixer */ ++static const snd_kcontrol_new_t wm8976_left_mixer_controls[] = { ++SOC_DAPM_SINGLE("Right PCM Playback Switch", WM8976_OUTPUT, 6, 1, 1), ++SOC_DAPM_SINGLE("Left PCM Playback Switch", WM8976_MIXL, 0, 1, 1), ++SOC_DAPM_SINGLE("Line Bypass Switch", WM8976_MIXL, 1, 1, 0), ++SOC_DAPM_SINGLE("Aux Playback Switch", WM8976_MIXL, 5, 1, 0), ++}; ++ ++/* Right Output Mixer */ ++static const snd_kcontrol_new_t wm8976_right_mixer_controls[] = { ++SOC_DAPM_SINGLE("Left PCM Playback Switch", WM8976_OUTPUT, 5, 1, 1), ++SOC_DAPM_SINGLE("Right PCM Playback Switch", WM8976_MIXR, 0, 1, 1), ++SOC_DAPM_SINGLE("Line Bypass Switch", WM8976_MIXR, 1, 1, 0), ++SOC_DAPM_SINGLE("Aux Playback Switch", WM8976_MIXR, 5, 1, 0), ++}; ++ ++/* Left AUX Input boost vol */ ++static const snd_kcontrol_new_t wm8976_laux_boost_controls = ++SOC_DAPM_SINGLE("Aux Volume", WM8976_ADCBOOST, 0, 3, 0); ++ ++/* Left Input boost vol */ ++static const snd_kcontrol_new_t wm8976_lmic_boost_controls = ++SOC_DAPM_SINGLE("Input Volume", WM8976_ADCBOOST, 4, 3, 0); ++ ++/* Left Aux In to PGA */ ++static const snd_kcontrol_new_t wm8976_laux_capture_boost_controls = ++SOC_DAPM_SINGLE("Capture Switch", WM8976_ADCBOOST, 8, 1, 0); ++ ++/* Left Input P In to PGA */ ++static const snd_kcontrol_new_t wm8976_lmicp_capture_boost_controls = ++SOC_DAPM_SINGLE("Input P Capture Boost Switch", WM8976_INPUT, 0, 1, 0); ++ ++/* Left Input N In to PGA */ ++static const snd_kcontrol_new_t wm8976_lmicn_capture_boost_controls = ++SOC_DAPM_SINGLE("Input N Capture Boost Switch", WM8976_INPUT, 1, 1, 0); ++ ++// TODO Widgets ++static const struct snd_soc_dapm_widget wm8976_dapm_widgets[] = { ++#if 0 ++//SND_SOC_DAPM_MUTE("Mono Mute", WM8976_MONOMIX, 6, 0), ++//SND_SOC_DAPM_MUTE("Speaker Mute", WM8976_SPKMIX, 6, 0), ++ ++SND_SOC_DAPM_MIXER("Speaker Mixer", WM8976_POWER3, 2, 0, ++ &wm8976_speaker_mixer_controls[0], ++ ARRAY_SIZE(wm8976_speaker_mixer_controls)), ++SND_SOC_DAPM_MIXER("Mono Mixer", WM8976_POWER3, 3, 0, ++ &wm8976_mono_mixer_controls[0], ++ ARRAY_SIZE(wm8976_mono_mixer_controls)), ++SND_SOC_DAPM_DAC("DAC", "HiFi Playback", WM8976_POWER3, 0, 0), ++SND_SOC_DAPM_ADC("ADC", "HiFi Capture", WM8976_POWER3, 0, 0), ++SND_SOC_DAPM_PGA("Aux Input", WM8976_POWER1, 6, 0, NULL, 0), ++SND_SOC_DAPM_PGA("SpkN Out", WM8976_POWER3, 5, 0, NULL, 0), ++SND_SOC_DAPM_PGA("SpkP Out", WM8976_POWER3, 6, 0, NULL, 0), ++SND_SOC_DAPM_PGA("Mono Out", WM8976_POWER3, 7, 0, NULL, 0), ++SND_SOC_DAPM_PGA("Mic PGA", WM8976_POWER2, 2, 0, NULL, 0), ++ ++SND_SOC_DAPM_PGA("Aux Boost", SND_SOC_NOPM, 0, 0, ++ &wm8976_aux_boost_controls, 1), ++SND_SOC_DAPM_PGA("Mic Boost", SND_SOC_NOPM, 0, 0, ++ &wm8976_mic_boost_controls, 1), ++SND_SOC_DAPM_SWITCH("Capture Boost", SND_SOC_NOPM, 0, 0, ++ &wm8976_capture_boost_controls), ++ ++SND_SOC_DAPM_MIXER("Boost Mixer", WM8976_POWER2, 4, 0, NULL, 0), ++ ++SND_SOC_DAPM_MICBIAS("Mic Bias", WM8976_POWER1, 4, 0), ++ ++SND_SOC_DAPM_INPUT("MICN"), ++SND_SOC_DAPM_INPUT("MICP"), ++SND_SOC_DAPM_INPUT("AUX"), ++SND_SOC_DAPM_OUTPUT("MONOOUT"), ++SND_SOC_DAPM_OUTPUT("SPKOUTP"), ++SND_SOC_DAPM_OUTPUT("SPKOUTN"), ++#endif ++}; ++ ++static const char *audio_map[][3] = { ++ /* Mono output mixer */ ++ {"Mono Mixer", "PCM Playback Switch", "DAC"}, ++ {"Mono Mixer", "Aux Playback Switch", "Aux Input"}, ++ {"Mono Mixer", "Line Bypass Switch", "Boost Mixer"}, ++ ++ /* Speaker output mixer */ ++ {"Speaker Mixer", "PCM Playback Switch", "DAC"}, ++ {"Speaker Mixer", "Aux Playback Switch", "Aux Input"}, ++ {"Speaker Mixer", "Line Bypass Switch", "Boost Mixer"}, ++ ++ /* Outputs */ ++ {"Mono Out", NULL, "Mono Mixer"}, ++ {"MONOOUT", NULL, "Mono Out"}, ++ {"SpkN Out", NULL, "Speaker Mixer"}, ++ {"SpkP Out", NULL, "Speaker Mixer"}, ++ {"SPKOUTN", NULL, "SpkN Out"}, ++ {"SPKOUTP", NULL, "SpkP Out"}, ++ ++ /* Boost Mixer */ ++ {"Boost Mixer", NULL, "ADC"}, ++ {"Capture Boost Switch", "Aux Capture Boost Switch", "AUX"}, ++ {"Aux Boost", "Aux Volume", "Boost Mixer"}, ++ {"Capture Boost", "Capture Switch", "Boost Mixer"}, ++ {"Mic Boost", "Mic Volume", "Boost Mixer"}, ++ ++ /* Inputs */ ++ {"MICP", NULL, "Mic Boost"}, ++ {"MICN", NULL, "Mic PGA"}, ++ {"Mic PGA", NULL, "Capture Boost"}, ++ {"AUX", NULL, "Aux Input"}, ++ ++ /* */ ++ ++ /* terminator */ ++ {NULL, NULL, NULL}, ++}; ++ ++static int wm8976_add_widgets(struct snd_soc_codec *codec) ++{ ++ int i; ++ ++ for(i = 0; i < ARRAY_SIZE(wm8976_dapm_widgets); i++) { ++ snd_soc_dapm_new_control(codec, &wm8976_dapm_widgets[i]); ++ } ++ ++ /* set up audio path map */ ++ for(i = 0; audio_map[i][0] != NULL; i++) { ++ snd_soc_dapm_connect_input(codec, audio_map[i][0], audio_map[i][1], ++ audio_map[i][2]); ++ } ++ ++ snd_soc_dapm_new_widgets(codec); ++ return 0; ++} ++ ++struct pll_ { ++ unsigned int in_hz, out_hz; ++ unsigned int pre:4; /* prescale - 1 */ ++ unsigned int n:4; ++ unsigned int k; ++}; ++ ++struct pll_ pll[] = { ++ {12000000, 11289600, 0, 7, 0x86c220}, ++ {12000000, 12288000, 0, 8, 0x3126e8}, ++ {13000000, 11289600, 0, 6, 0xf28bd4}, ++ {13000000, 12288000, 0, 7, 0x8fd525}, ++ {12288000, 11289600, 0, 7, 0x59999a}, ++ {11289600, 12288000, 0, 8, 0x80dee9}, ++ /* TODO: liam - add more entries */ ++}; ++ ++static int set_pll(struct snd_soc_codec *codec, unsigned int in, ++ unsigned int out) ++{ ++ int i; ++ u16 reg; ++ ++ if(out == 0) { ++ reg = wm8976_read_reg_cache(codec, WM8976_POWER1); ++ wm8976_write(codec, WM8976_POWER1, reg & 0x1df); ++ return 0; ++ } ++ ++ for(i = 0; i < ARRAY_SIZE(pll); i++) { ++ if (in == pll[i].in_hz && out == pll[i].out_hz) { ++ wm8976_write(codec, WM8976_PLLN, (pll[i].pre << 4) | pll[i].n); ++ wm8976_write(codec, WM8976_PLLK1, pll[i].k >> 18); ++ wm8976_write(codec, WM8976_PLLK1, (pll[i].k >> 9) && 0x1ff); ++ wm8976_write(codec, WM8976_PLLK1, pll[i].k && 0x1ff); ++ reg = wm8976_read_reg_cache(codec, WM8976_POWER1); ++ wm8976_write(codec, WM8976_POWER1, reg | 0x020); ++ return 0; ++ } ++ } ++ return -EINVAL; ++} ++ ++/* mclk dividers * 2 */ ++static unsigned char mclk_div[] = {2, 3, 4, 6, 8, 12, 16, 24}; ++ ++/* we need 256FS to drive the DAC's and ADC's */ ++static unsigned int wm8976_config_sysclk(struct snd_soc_codec_dai *dai, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ int i, j, best_clk = info->fs * info->rate; ++ ++ /* can we run at this clk without the PLL ? */ ++ for (i = 0; i < ARRAY_SIZE(mclk_div); i++) { ++ if ((best_clk >> 1) * mclk_div[i] == clk) { ++ dai->pll_in = 0; ++ dai->clk_div = mclk_div[i]; ++ dai->mclk = best_clk; ++ return dai->mclk; ++ } ++ } ++ ++ /* now check for PLL support */ ++ for (i = 0; i < ARRAY_SIZE(pll); i++) { ++ if (pll[i].in_hz == clk) { ++ for (j = 0; j < ARRAY_SIZE(mclk_div); j++) { ++ if (pll[i].out_hz == mclk_div[j] * (best_clk >> 1)) { ++ dai->pll_in = clk; ++ dai->pll_out = pll[i].out_hz; ++ dai->clk_div = mclk_div[j]; ++ dai->mclk = best_clk; ++ return dai->mclk; ++ } ++ } ++ } ++ } ++ ++ /* this clk is not supported */ ++ return 0; ++} ++ ++static int wm8976_pcm_prepare(snd_pcm_substream_t *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ struct snd_soc_codec_dai *dai = rtd->codec_dai; ++ u16 iface = 0, bfs, clk = 0, adn; ++ int fs = 48000 << 7, i; ++ ++ bfs = SND_SOC_FSBD_REAL(rtd->codec_dai->dai_runtime.bfs); ++ switch (bfs) { ++ case 2: ++ clk |= 0x1 << 2; ++ break; ++ case 4: ++ clk |= 0x2 << 2; ++ break; ++ case 8: ++ clk |= 0x3 << 2; ++ break; ++ case 16: ++ clk |= 0x4 << 2; ++ break; ++ case 32: ++ clk |= 0x5 << 2; ++ break; ++ } ++ ++ /* set master/slave audio interface */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CLOCK_MASK) { ++ case SND_SOC_DAIFMT_CBM_CFM: ++ clk |= 0x0001; ++ break; ++ case SND_SOC_DAIFMT_CBS_CFS: ++ break; ++ } ++ ++ /* interface format */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_FORMAT_MASK) { ++ case SND_SOC_DAIFMT_I2S: ++ iface |= 0x0010; ++ break; ++ case SND_SOC_DAIFMT_RIGHT_J: ++ break; ++ case SND_SOC_DAIFMT_LEFT_J: ++ iface |= 0x0008; ++ break; ++ case SND_SOC_DAIFMT_DSP_A: ++ iface |= 0x00018; ++ break; ++ } ++ ++ /* bit size */ ++ switch (rtd->codec_dai->dai_runtime.pcmfmt) { ++ case SNDRV_PCM_FMTBIT_S16_LE: ++ break; ++ case SNDRV_PCM_FMTBIT_S20_3LE: ++ iface |= 0x0020; ++ break; ++ case SNDRV_PCM_FMTBIT_S24_LE: ++ iface |= 0x0040; ++ break; ++ case SNDRV_PCM_FMTBIT_S32_LE: ++ iface |= 0x0060; ++ break; ++ } ++ ++ /* clock inversion */ ++ switch (rtd->codec_dai->dai_runtime.fmt & SND_SOC_DAIFMT_INV_MASK) { ++ case SND_SOC_DAIFMT_NB_NF: ++ break; ++ case SND_SOC_DAIFMT_IB_IF: ++ iface |= 0x0180; ++ break; ++ case SND_SOC_DAIFMT_IB_NF: ++ iface |= 0x0100; ++ break; ++ case SND_SOC_DAIFMT_NB_IF: ++ iface |= 0x0080; ++ break; ++ } ++ ++ /* filter coefficient */ ++ adn = wm8976_read_reg_cache(codec, WM8976_ADD) & 0x1f1; ++ switch (rtd->codec_dai->dai_runtime.pcmrate) { ++ case SNDRV_PCM_RATE_8000: ++ adn |= 0x5 << 1; ++ fs = 8000 << 7; ++ break; ++ case SNDRV_PCM_RATE_11025: ++ adn |= 0x4 << 1; ++ fs = 11025 << 7; ++ break; ++ case SNDRV_PCM_RATE_16000: ++ adn |= 0x3 << 1; ++ fs = 16000 << 7; ++ break; ++ case SNDRV_PCM_RATE_22050: ++ adn |= 0x2 << 1; ++ fs = 22050 << 7; ++ break; ++ case SNDRV_PCM_RATE_32000: ++ adn |= 0x1 << 1; ++ fs = 32000 << 7; ++ break; ++ case SNDRV_PCM_RATE_44100: ++ fs = 44100 << 7; ++ break; ++ } ++ ++ /* do we need to enable the PLL */ ++ if(dai->pll_in) ++ set_pll(codec, dai->pll_in, dai->pll_out); ++ ++ /* divide the clock to 256 fs */ ++ for(i = 0; i < ARRAY_SIZE(mclk_div); i++) { ++ if (dai->clk_div == mclk_div[i]) { ++ clk |= i << 5; ++ clk &= 0xff; ++ goto set; ++ } ++ } ++ ++set: ++ /* set iface */ ++ wm8976_write(codec, WM8976_IFACE, iface); ++ wm8976_write(codec, WM8976_CLOCK, clk); ++ ++ return 0; ++} ++ ++static int wm8976_hw_free(struct snd_pcm_substream *substream) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ struct snd_soc_device *socdev = rtd->socdev; ++ struct snd_soc_codec *codec = socdev->codec; ++ set_pll(codec, 0, 0); ++ return 0; ++} ++ ++static int wm8976_mute(struct snd_soc_codec *codec, ++ struct snd_soc_codec_dai *dai, int mute) ++{ ++ u16 mute_reg = wm8976_read_reg_cache(codec, WM8976_DAC) & 0xffbf; ++ if(mute) ++ wm8976_write(codec, WM8976_DAC, mute_reg | 0x40); ++ else ++ wm8976_write(codec, WM8976_DAC, mute_reg); ++ ++ return 0; ++} ++ ++/* TODO: liam need to make this lower power with dapm */ ++static int wm8976_dapm_event(struct snd_soc_codec *codec, int event) ++{ ++ ++ switch (event) { ++ case SNDRV_CTL_POWER_D0: /* full On */ ++ /* vref/mid, clk and osc on, dac unmute, active */ ++ wm8976_write(codec, WM8976_POWER1, 0x1ff); ++ wm8976_write(codec, WM8976_POWER2, 0x1ff); ++ wm8976_write(codec, WM8976_POWER3, 0x1ff); ++ break; ++ case SNDRV_CTL_POWER_D1: /* partial On */ ++ case SNDRV_CTL_POWER_D2: /* partial On */ ++ break; ++ case SNDRV_CTL_POWER_D3hot: /* Off, with power */ ++ /* everything off except vref/vmid, dac mute, inactive */ ++ ++ break; ++ case SNDRV_CTL_POWER_D3cold: /* Off, without power */ ++ /* everything off, dac mute, inactive */ ++ wm8976_write(codec, WM8976_POWER1, 0x0); ++ wm8976_write(codec, WM8976_POWER2, 0x0); ++ wm8976_write(codec, WM8976_POWER3, 0x0); ++ break; ++ } ++ codec->dapm_state = event; ++ return 0; ++} ++ ++struct snd_soc_codec_dai wm8976_dai = { ++ .name = "WM8976 HiFi", ++ .playback = { ++ .stream_name = "Playback", ++ .channels_min = 1, ++ .channels_max = 2, ++ }, ++ .capture = { ++ .stream_name = "Capture", ++ .channels_min = 1, ++ .channels_max = 1, ++ }, ++ .config_sysclk = wm8976_config_sysclk, ++ .digital_mute = wm8976_mute, ++ .ops = { ++ .prepare = wm8976_pcm_prepare, ++ .hw_free = wm8976_hw_free, ++ }, ++ .caps = { ++ .num_modes = ARRAY_SIZE(wm8976_modes), ++ .mode = wm8976_modes, ++ }, ++}; ++EXPORT_SYMBOL_GPL(wm8976_dai); ++ ++static int wm8976_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ wm8976_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ return 0; ++} ++ ++static int wm8976_resume(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ int i; ++ u8 data[2]; ++ u16 *cache = codec->reg_cache; ++ ++ /* Sync reg_cache with the hardware */ ++ for (i = 0; i < ARRAY_SIZE(wm8976_reg); i++) { ++ data[0] = (i << 1) | ((cache[i] >> 8) & 0x0001); ++ data[1] = cache[i] & 0x00ff; ++ codec->hw_write(codec->control_data, data, 2); ++ } ++ wm8976_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ wm8976_dapm_event(codec, codec->suspend_dapm_state); ++ return 0; ++} ++ ++/* ++ * initialise the WM8976 driver ++ * register the mixer and dsp interfaces with the kernel ++ */ ++static int wm8976_init(struct snd_soc_device* socdev) ++{ ++ struct snd_soc_codec *codec = socdev->codec; ++ int ret = 0; ++ ++ codec->name = "WM8976"; ++ codec->owner = THIS_MODULE; ++ codec->read = wm8976_read_reg_cache; ++ codec->write = wm8976_write; ++ codec->dapm_event = wm8976_dapm_event; ++ codec->dai = &wm8976_dai; ++ codec->num_dai = 1; ++ codec->reg_cache_size = ARRAY_SIZE(wm8976_reg); ++ codec->reg_cache = ++ kzalloc(sizeof(u16) * ARRAY_SIZE(wm8976_reg), GFP_KERNEL); ++ if (codec->reg_cache == NULL) ++ return -ENOMEM; ++ memcpy(codec->reg_cache, wm8976_reg, ++ sizeof(u16) * ARRAY_SIZE(wm8976_reg)); ++ codec->reg_cache_size = sizeof(u16) * ARRAY_SIZE(wm8976_reg); ++ ++ wm8976_reset(codec); ++ ++ /* register pcms */ ++ ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); ++ if(ret < 0) { ++ kfree(codec->reg_cache); ++ return ret; ++ } ++ ++ /* power on device */ ++ wm8976_dapm_event(codec, SNDRV_CTL_POWER_D3hot); ++ wm8976_add_controls(codec); ++ wm8976_add_widgets(codec); ++ ret = snd_soc_register_card(socdev); ++ if(ret < 0) { ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++ } ++ ++ return ret; ++} ++ ++static struct snd_soc_device *wm8976_socdev; ++ ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ ++/* ++ * WM8976 2 wire address is 0x1a ++ */ ++#define I2C_DRIVERID_WM8976 0xfefe /* liam - need a proper id */ ++ ++static unsigned short normal_i2c[] = { 0, I2C_CLIENT_END }; ++ ++/* Magic definition of all other variables and things */ ++I2C_CLIENT_INSMOD; ++ ++static struct i2c_driver wm8976_i2c_driver; ++static struct i2c_client client_template; ++ ++/* If the i2c layer weren't so broken, we could pass this kind of data ++ around */ ++ ++static int wm8976_codec_probe(struct i2c_adapter *adap, int addr, int kind) ++{ ++ struct snd_soc_device *socdev = wm8976_socdev; ++ struct wm8976_setup_data *setup = socdev->codec_data; ++ struct snd_soc_codec *codec = socdev->codec; ++ struct i2c_client *i2c; ++ int ret; ++ ++ if (addr != setup->i2c_address) ++ return -ENODEV; ++ ++ client_template.adapter = adap; ++ client_template.addr = addr; ++ ++ i2c = kzalloc(sizeof(struct i2c_client), GFP_KERNEL); ++ if (i2c == NULL){ ++ kfree(codec); ++ return -ENOMEM; ++ } ++ memcpy(i2c, &client_template, sizeof(struct i2c_client)); ++ ++ i2c_set_clientdata(i2c, codec); ++ ++ codec->control_data = i2c; ++ ++ ret = i2c_attach_client(i2c); ++ if(ret < 0) { ++ err("failed to attach codec at addr %x\n", addr); ++ goto err; ++ } ++ ++ ret = wm8976_init(socdev); ++ if(ret < 0) { ++ err("failed to initialise WM8976\n"); ++ goto err; ++ } ++ return ret; ++ ++err: ++ kfree(codec); ++ kfree(i2c); ++ return ret; ++ ++} ++ ++static int wm8976_i2c_detach(struct i2c_client *client) ++{ ++ struct snd_soc_codec *codec = i2c_get_clientdata(client); ++ ++ i2c_detach_client(client); ++ ++ kfree(codec->reg_cache); ++ kfree(client); ++ ++ return 0; ++} ++ ++static int wm8976_i2c_attach(struct i2c_adapter *adap) ++{ ++ return i2c_probe(adap, &addr_data, wm8976_codec_probe); ++} ++ ++/* corgi i2c codec control layer */ ++static struct i2c_driver wm8976_i2c_driver = { ++ .driver = { ++ .name = "WM8976 I2C Codec", ++ .owner = THIS_MODULE, ++ }, ++ .id = I2C_DRIVERID_WM8976, ++ .attach_adapter = wm8976_i2c_attach, ++ .detach_client = wm8976_i2c_detach, ++ .command = NULL, ++}; ++ ++static struct i2c_client client_template = { ++ .name = "WM8976", ++ .driver = &wm8976_i2c_driver, ++}; ++#endif ++ ++static int wm8976_probe(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct wm8976_setup_data *setup; ++ struct snd_soc_codec *codec; ++ int ret = 0; ++ ++ info("WM8976 Audio Codec %s", WM8976_VERSION); ++ ++ setup = socdev->codec_data; ++ codec = kzalloc(sizeof(struct snd_soc_codec), GFP_KERNEL); ++ if (codec == NULL) ++ return -ENOMEM; ++ ++ socdev->codec = codec; ++ mutex_init(&codec->mutex); ++ INIT_LIST_HEAD(&codec->dapm_widgets); ++ INIT_LIST_HEAD(&codec->dapm_paths); ++ ++ wm8976_socdev = socdev; ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ if (setup->i2c_address) { ++ normal_i2c[0] = setup->i2c_address; ++ codec->hw_write = (hw_write_t)i2c_master_send; ++ ret = i2c_add_driver(&wm8976_i2c_driver); ++ if (ret != 0) ++ printk(KERN_ERR "can't add i2c driver"); ++ } ++#else ++ /* Add other interfaces here */ ++#endif ++ return ret; ++} ++ ++/* power down chip */ ++static int wm8976_remove(struct platform_device *pdev) ++{ ++ struct snd_soc_device *socdev = platform_get_drvdata(pdev); ++ struct snd_soc_codec *codec = socdev->codec; ++ ++ if (codec->control_data) ++ wm8976_dapm_event(codec, SNDRV_CTL_POWER_D3cold); ++ ++ snd_soc_free_pcms(socdev); ++ snd_soc_dapm_free(socdev); ++#if defined (CONFIG_I2C) || defined (CONFIG_I2C_MODULE) ++ i2c_del_driver(&wm8976_i2c_driver); ++#endif ++ kfree(codec); ++ ++ return 0; ++} ++ ++struct snd_soc_codec_device soc_codec_dev_wm8976 = { ++ .probe = wm8976_probe, ++ .remove = wm8976_remove, ++ .suspend = wm8976_suspend, ++ .resume = wm8976_resume, ++}; ++ ++EXPORT_SYMBOL_GPL(soc_codec_dev_wm8976); ++ ++MODULE_DESCRIPTION("ASoC WM8976 driver"); ++MODULE_AUTHOR("Graeme Gregory"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/codecs/wm8976.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/codecs/wm8976.h +@@ -0,0 +1,73 @@ ++/* ++ * wm8976.h -- WM8976 Soc Audio driver ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#ifndef _WM8976_H ++#define _WM8976_H ++ ++/* WM8976 register space */ ++ ++#define WM8976_RESET 0x0 ++#define WM8976_POWER1 0x1 ++#define WM8976_POWER2 0x2 ++#define WM8976_POWER3 0x3 ++#define WM8976_IFACE 0x4 ++#define WM8976_COMP 0x5 ++#define WM8976_CLOCK 0x6 ++#define WM8976_ADD 0x7 ++#define WM8976_GPIO 0x8 ++#define WM8976_JACK1 0x9 ++#define WM8976_DAC 0xa ++#define WM8976_DACVOLL 0xb ++#define WM8976_DACVOLR 0xc ++#define WM8976_JACK2 0xd ++#define WM8976_ADC 0xe ++#define WM8976_ADCVOL 0xf ++#define WM8976_EQ1 0x12 ++#define WM8976_EQ2 0x13 ++#define WM8976_EQ3 0x14 ++#define WM8976_EQ4 0x15 ++#define WM8976_EQ5 0x16 ++#define WM8976_DACLIM1 0x18 ++#define WM8976_DACLIM2 0x19 ++#define WM8976_NOTCH1 0x1b ++#define WM8976_NOTCH2 0x1c ++#define WM8976_NOTCH3 0x1d ++#define WM8976_NOTCH4 0x1e ++#define WM8976_ALC1 0x20 ++#define WM8976_ALC2 0x21 ++#define WM8976_ALC3 0x22 ++#define WM8976_NGATE 0x23 ++#define WM8976_PLLN 0x24 ++#define WM8976_PLLK1 0x25 ++#define WM8976_PLLK2 0x26 ++#define WM8976_PLLK3 0x27 ++#define WM8976_3D 0x29 ++#define WM8976_BEEP 0x2b ++#define WM8976_INPUT 0x2c ++#define WM8976_INPPGA 0x2d ++#define WM8976_ADCBOOST 0x2f ++#define WM8976_OUTPUT 0x31 ++#define WM8976_MIXL 0x32 ++#define WM8976_MIXR 0x33 ++#define WM8976_HPVOLL 0x34 ++#define WM8976_HPVOLR 0x35 ++#define WM8976_SPKVOLL 0x36 ++#define WM8976_SPKVOLR 0x37 ++#define WM8976_OUT3MIX 0x38 ++#define WM8976_MONOMIX 0x39 ++ ++#define WM8976_CACHEREGNUM 58 ++ ++struct wm8976_setup_data { ++ unsigned short i2c_address; ++}; ++ ++extern struct snd_soc_codec_dai wm8976_dai; ++extern struct snd_soc_codec_device soc_codec_dev_wm8976; ++ ++#endif +Index: linux-2.6-pxa-new/sound/soc/imx/imx21-pcm.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/imx/imx21-pcm.c +@@ -0,0 +1,454 @@ ++/* ++ * linux/sound/arm/mxc-pcm.c -- ALSA SoC interface for the Freescale i.MX CPU's ++ * ++ * Copyright 2006 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * Based on pxa2xx-pcm.c by Nicolas Pitre, (C) 2004 MontaVista Software, Inc. ++ * and on mxc-alsa-mc13783 (C) 2006 Freescale. ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ * ++ * Revision history ++ * 29th Aug 2006 Initial version. ++ * ++ */ ++ ++#include <linux/module.h> ++#include <linux/init.h> ++#include <linux/platform_device.h> ++#include <linux/slab.h> ++#include <linux/dma-mapping.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/pcm_params.h> ++#include <sound/soc.h> ++#include <asm/dma.h> ++#include <asm/hardware.h> ++ ++#include "imx-pcm.h" ++ ++/* debug */ ++#define IMX_DEBUG 0 ++#if IMX_DEBUG ++#define dbg(format, arg...) printk(format, ## arg) ++#else ++#define dbg(format, arg...) ++#endif ++ ++static const struct snd_pcm_hardware mxc_pcm_hardware = { ++ .info = (SNDRV_PCM_INFO_INTERLEAVED | ++ SNDRV_PCM_INFO_BLOCK_TRANSFER | ++ SNDRV_PCM_INFO_MMAP | ++ SNDRV_PCM_INFO_MMAP_VALID | ++ SNDRV_PCM_INFO_PAUSE | ++ SNDRV_PCM_INFO_RESUME), ++ .formats = SNDRV_PCM_FMTBIT_S16_LE | ++ SNDRV_PCM_FMTBIT_S24_LE, ++ .buffer_bytes_max = 32 * 1024, ++ .period_bytes_min = 64, ++ .period_bytes_max = 8 * 1024, ++ .periods_min = 2, ++ .periods_max = 255, ++ .fifo_size = 0, ++}; ++ ++struct mxc_runtime_data { ++ int dma_ch; ++ struct mxc_pcm_dma_param *dma_params; ++}; ++ ++/*! ++ * This function stops the current dma transfert for playback ++ * and clears the dma pointers. ++ * ++ * @param substream pointer to the structure of the current stream. ++ * ++ */ ++static void audio_stop_dma(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct mxc_runtime_data *prtd = runtime->private_data; ++ unsigned int dma_size = frames_to_bytes(runtime, runtime->period_size); ++ unsigned int offset dma_size * s->periods; ++ unsigned long flags; ++ ++ spin_lock_irqsave(&prtd->dma_lock, flags); ++ ++ dbg("MXC : audio_stop_dma active = 0\n"); ++ prtd->active = 0; ++ prtd->period = 0; ++ prtd->periods = 0; ++ ++ /* this stops the dma channel and clears the buffer ptrs */ ++ mxc_dma_stop(prtd->dma_wchannel); ++ if(substream == SNDRV_PCM_STREAM_PLAYBACK) ++ dma_unmap_single(NULL, runtime->dma_addr + offset, dma_size, ++ DMA_TO_DEVICE); ++ else ++ dma_unmap_single(NULL, runtime->dma_addr + offset, dma_size, ++ DMA_FROM_DEVICE); ++ ++ spin_unlock_irqrestore(&prtd->dma_lock, flags); ++} ++ ++/*! ++ * This function is called whenever a new audio block needs to be ++ * transferred to mc13783. The function receives the address and the size ++ * of the new block and start a new DMA transfer. ++ * ++ * @param substream pointer to the structure of the current stream. ++ * ++ */ ++static int dma_new_period(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct mxc_runtime_data *prtd = runtime->private_data; ++ unsigned int dma_size; ++ unsigned int offset; ++ int ret=0; ++ dma_request_t sdma_request; ++ ++ if (prtd->active){ ++ memset(&sdma_request, 0, sizeof(dma_request_t)); ++ dma_size = frames_to_bytes(runtime, runtime->period_size); ++ dbg("s->period (%x) runtime->periods (%d)\n", ++ s->period,runtime->periods); ++ dbg("runtime->period_size (%d) dma_size (%d)\n", ++ (unsigned int)runtime->period_size, ++ runtime->dma_bytes); ++ ++ offset = dma_size * prtd->period; ++ snd_assert(dma_size <= DMA_BUF_SIZE, ); ++ if(substream == SNDRV_PCM_STREAM_PLAYBACK) ++ sdma_request.sourceAddr = (char*)(dma_map_single(NULL, ++ runtime->dma_area + offset, dma_size, DMA_TO_DEVICE)); ++ else ++ sdma_request.destAddr = (char*)(dma_map_single(NULL, ++ runtime->dma_area + offset, dma_size, DMA_FROM_DEVICE)); ++ sdma_request.count = dma_size; ++ ++ dbg("MXC: Start DMA offset (%d) size (%d)\n", offset, ++ runtime->dma_bytes); ++ ++ mxc_dma_set_config(prtd->dma_wchannel, &sdma_request, 0); ++ if((ret = mxc_dma_start(prtd->dma_wchannel)) < 0) { ++ dbg("audio_process_dma: cannot queue DMA buffer\ ++ (%i)\n", ret); ++ return err; ++ } ++ prtd->tx_spin = 1; /* FGA little trick to retrieve DMA pos */ ++ prtd->period++; ++ prtd->period %= runtime->periods; ++ } ++ return ret; ++} ++ ++ ++/*! ++ * This is a callback which will be called ++ * when a TX transfer finishes. The call occurs ++ * in interrupt context. ++ * ++ * @param dat pointer to the structure of the current stream. ++ * ++ */ ++static void audio_dma_irq(void *data) ++{ ++ struct snd_pcm_substream *substream; ++ struct snd_pcm_runtime *runtime; ++ struct mxc_runtime_data *prtd; ++ unsigned int dma_size; ++ unsigned int previous_period; ++ unsigned int offset; ++ ++ substream = data; ++ runtime = substream->runtime; ++ prtd = runtime->private_data; ++ previous_period = prtd->periods; ++ dma_size = frames_to_bytes(runtime, runtime->period_size); ++ offset = dma_size * previous_period; ++ ++ prtd->tx_spin = 0; ++ prtd->periods++; ++ prtd->periods %= runtime->periods; ++ ++ /* ++ * Give back to the CPU the access to the non cached memory ++ */ ++ if(substream == SNDRV_PCM_STREAM_PLAYBACK) ++ dma_unmap_single(NULL, runtime->dma_addr + offset, dma_size, ++ DMA_TO_DEVICE); ++ else ++ dma_unmap_single(NULL, runtime->dma_addr + offset, dma_size, ++ DMA_FROM_DEVICE); ++ /* ++ * If we are getting a callback for an active stream then we inform ++ * the PCM middle layer we've finished a period ++ */ ++ if (prtd->active) ++ snd_pcm_period_elapsed(substream); ++ ++ /* ++ * Trig next DMA transfer ++ */ ++ dma_new_period(substream); ++} ++ ++/*! ++ * This function configures the hardware to allow audio ++ * playback operations. It is called by ALSA framework. ++ * ++ * @param substream pointer to the structure of the current stream. ++ * ++ * @return 0 on success, -1 otherwise. ++ */ ++static int ++snd_mxc_prepare(struct snd_pcm_substream *substream) ++{ ++ struct mxc_runtime_data *prtd = runtime->private_data; ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ int ret = 0; ++ prtd->period = 0; ++ prtd->periods = 0; ++ ++ dma_channel_params params; ++ int channel = 0; // passed in ? ++ ++ if ((ret = mxc_request_dma(&channel, "ALSA TX SDMA") < 0)){ ++ dbg("error requesting a write dma channel\n"); ++ return ret; ++ } ++ ++ /* configure DMA params */ ++ memset(¶ms, 0, sizeof(dma_channel_params)); ++ params.bd_number = 1; ++ params.arg = s; ++ params.callback = callback; ++ params.transfer_type = emi_2_per; ++ params.watermark_level = SDMA_TXFIFO_WATERMARK; ++ params.word_size = TRANSFER_16BIT; ++ //dbg(KERN_ERR "activating connection SSI1 - SDMA\n"); ++ params.per_address = SSI1_BASE_ADDR; ++ params.event_id = DMA_REQ_SSI1_TX1; ++ params.peripheral_type = SSI; ++ ++ /* set up chn with params */ ++ mxc_dma_setup_channel(channel, ¶ms); ++ s->dma_wchannel = channel; ++ ++ return ret; ++} ++ ++static int mxc_pcm_hw_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ int ret; ++ ++ if((ret=snd_pcm_lib_malloc_pages(substream, params_buffer_bytes(hw_params))) < 0) ++ return ret; ++ runtime->dma_addr = virt_to_phys(runtime->dma_area); ++ ++ return ret; ++} ++ ++static int mxc_pcm_hw_free(struct snd_pcm_substream *substream) ++{ ++ return snd_pcm_lib_free_pages(substream); ++} ++ ++static int mxc_pcm_trigger(struct snd_pcm_substream *substream, int cmd) ++{ ++ struct mxc_runtime_data *prtd = substream->runtime->private_data; ++ int ret = 0; ++ ++ switch (cmd) { ++ case SNDRV_PCM_TRIGGER_START: ++ prtd->tx_spin = 0; ++ /* requested stream startup */ ++ prtd->active = 1; ++ ret = dma_new_period(substream); ++ break; ++ case SNDRV_PCM_TRIGGER_STOP: ++ /* requested stream shutdown */ ++ ret = audio_stop_dma(substream); ++ break; ++ case SNDRV_PCM_TRIGGER_SUSPEND: ++ prtd->active = 0; ++ prtd->periods = 0; ++ break; ++ case SNDRV_PCM_TRIGGER_RESUME: ++ prtd->active = 1; ++ prtd->tx_spin = 0; ++ ret = dma_new_period(substream); ++ break; ++ case SNDRV_PCM_TRIGGER_PAUSE_PUSH: ++ prtd->active = 0; ++ break; ++ case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: ++ prtd->active = 1; ++ if (prtd->old_offset) { ++ prtd->tx_spin = 0; ++ ret = dma_new_period(substream); ++ } ++ break; ++ default: ++ ret = -EINVAL; ++ break; ++ } ++ ++ return ret; ++} ++ ++static snd_pcm_uframes_t mxc_pcm_pointer(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct mxc_runtime_data *prtd = runtime->private_data; ++ unsigned int offset = 0; ++ ++ /* tx_spin value is used here to check if a transfert is active */ ++ if (prtd->tx_spin){ ++ offset = (runtime->period_size * (prtd->periods)) + ++ (runtime->period_size >> 1); ++ if (offset >= runtime->buffer_size) ++ offset = runtime->period_size >> 1; ++ } else { ++ offset = (runtime->period_size * (s->periods)); ++ if (offset >= runtime->buffer_size) ++ offset = 0; ++ } ++ ++ return offset; ++} ++ ++ ++static int mxc_pcm_open(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct mxc_runtime_data *prtd; ++ int ret; ++ ++ snd_soc_set_runtime_hwparams(substream, &mxc_pcm_hardware); ++ ++ if ((err = snd_pcm_hw_constraint_integer(runtime, ++ SNDRV_PCM_HW_PARAM_PERIODS)) < 0) ++ return err; ++ if ((err = snd_pcm_hw_constraint_list(runtime, 0, ++ SNDRV_PCM_HW_PARAM_RATE, &hw_playback_rates)) < 0) ++ return err; ++ msleep(10); // liam - why ++ ++ /* setup DMA controller for playback */ ++ if((err = configure_write_channel(&mxc_mc13783->s[SNDRV_PCM_STREAM_PLAYBACK], ++ audio_dma_irq)) < 0 ) ++ return err; ++ ++ if((prtd = kzalloc(sizeof(struct mxc_runtime_data), GFP_KERNEL)) == NULL) { ++ ret = -ENOMEM; ++ goto out; ++ } ++ ++ runtime->private_data = prtd; ++ return 0; ++ ++ err1: ++ kfree(prtd); ++ out: ++ return ret; ++} ++ ++static int mxc_pcm_close(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct mxc_runtime_data *prtd = runtime->private_data; ++ ++// mxc_mc13783_t *chip; ++ audio_stream_t *s; ++ device_data_t* device; ++ int ssi; ++ ++ //chip = snd_pcm_substream_chip(substream); ++ s = &chip->s[substream->pstr->stream]; ++ device = &s->stream_device; ++ ssi = device->ssi; ++ ++ //disable_stereodac(); ++ ++ ssi_transmit_enable(ssi, false); ++ ssi_interrupt_disable(ssi, ssi_tx_dma_interrupt_enable); ++ ssi_tx_fifo_enable(ssi, ssi_fifo_0, false); ++ ssi_enable(ssi, false); ++ ++ chip->s[substream->pstr->stream].stream = NULL; ++ ++ return 0; ++} ++ ++static int ++mxc_pcm_mmap(struct snd_pcm_substream *substream, struct vm_area_struct *vma) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ return dma_mmap_writecombine(substream->pcm->card->dev, vma, ++ runtime->dma_area, ++ runtime->dma_addr, ++ runtime->dma_bytes); ++} ++ ++struct snd_pcm_ops mxc_pcm_ops = { ++ .open = mxc_pcm_open, ++ .close = mxc_pcm_close, ++ .ioctl = snd_pcm_lib_ioctl, ++ .hw_params = mxc_pcm_hw_params, ++ .hw_free = mxc_pcm_hw_free, ++ .prepare = mxc_pcm_prepare, ++ .trigger = mxc_pcm_trigger, ++ .pointer = mxc_pcm_pointer, ++ .mmap = mxc_pcm_mmap, ++}; ++ ++static u64 mxc_pcm_dmamask = 0xffffffff; ++ ++int mxc_pcm_new(struct snd_card *card, struct snd_soc_codec_dai *dai, ++ struct snd_pcm *pcm) ++{ ++ int ret = 0; ++ ++ if (!card->dev->dma_mask) ++ card->dev->dma_mask = &mxc_pcm_dmamask; ++ if (!card->dev->coherent_dma_mask) ++ card->dev->coherent_dma_mask = 0xffffffff; ++ ++ if (dai->playback.channels_min) { ++ ret = mxc_pcm_preallocate_dma_buffer(pcm, ++ SNDRV_PCM_STREAM_PLAYBACK); ++ if (ret) ++ goto out; ++ } ++ ++ if (dai->capture.channels_min) { ++ ret = mxc_pcm_preallocate_dma_buffer(pcm, ++ SNDRV_PCM_STREAM_CAPTURE); ++ if (ret) ++ goto out; ++ } ++ out: ++ return ret; ++} ++ ++struct snd_soc_platform mxc_soc_platform = { ++ .name = "mxc-audio", ++ .pcm_ops = &mxc_pcm_ops, ++ .pcm_new = mxc_pcm_new, ++ .pcm_free = mxc_pcm_free_dma_buffers, ++}; ++ ++EXPORT_SYMBOL_GPL(mxc_soc_platform); ++ ++MODULE_AUTHOR("Liam Girdwood"); ++MODULE_DESCRIPTION("Freescale i.MX PCM DMA module"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/imx/imx21-pcm.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/imx/imx21-pcm.h +@@ -0,0 +1,237 @@ ++/* ++ * mxc-pcm.h :- ASoC platform header for Freescale i.MX ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#ifndef _MXC_PCM_H ++#define _MXC_PCM_H ++ ++struct { ++ char *name; /* stream identifier */ ++ dma_channel_params dma_params; ++} mxc_pcm_dma_param; ++ ++extern struct snd_soc_cpu_dai mxc_ssi_dai[3]; ++ ++/* platform data */ ++extern struct snd_soc_platform mxc_soc_platform; ++extern struct snd_ac97_bus_ops mxc_ac97_ops; ++ ++/* temp until imx-regs.h is up2date */ ++#define SSI1_STX0 __REG(IMX_SSI1_BASE + 0x00) ++#define SSI1_STX0_PHYS __PHYS_REG(IMX_SSI1_BASE + 0x00) ++#define SSI1_STX1 __REG(IMX_SSI1_BASE + 0x04) ++#define SSI1_STX1_PHYS __PHYS_REG(IMX_SSI1_BASE + 0x04) ++#define SSI1_SRX0 __REG(IMX_SSI1_BASE + 0x08) ++#define SSI1_SRX0_PHYS __PHYS_REG(IMX_SSI1_BASE + 0x08) ++#define SSI1_SRX1 __REG(IMX_SSI1_BASE + 0x0c) ++#define SSI1_SRX1_PHYS __PHYS_REG(IMX_SSI1_BASE + 0x0c) ++#define SSI1_SCR __REG(IMX_SSI1_BASE + 0x10) ++#define SSI1_SISR __REG(IMX_SSI1_BASE + 0x14) ++#define SSI1_SIER __REG(IMX_SSI1_BASE + 0x18) ++#define SSI1_STCR __REG(IMX_SSI1_BASE + 0x1c) ++#define SSI1_SRCR __REG(IMX_SSI1_BASE + 0x20) ++#define SSI1_STCCR __REG(IMX_SSI1_BASE + 0x24) ++#define SSI1_SRCCR __REG(IMX_SSI1_BASE + 0x28) ++#define SSI1_SFCSR __REG(IMX_SSI1_BASE + 0x2c) ++#define SSI1_STR __REG(IMX_SSI1_BASE + 0x30) ++#define SSI1_SOR __REG(IMX_SSI1_BASE + 0x34) ++#define SSI1_SACNT __REG(IMX_SSI1_BASE + 0x38) ++#define SSI1_SACADD __REG(IMX_SSI1_BASE + 0x3c) ++#define SSI1_SACDAT __REG(IMX_SSI1_BASE + 0x40) ++#define SSI1_SATAG __REG(IMX_SSI1_BASE + 0x44) ++#define SSI1_STMSK __REG(IMX_SSI1_BASE + 0x48) ++#define SSI1_SRMSK __REG(IMX_SSI1_BASE + 0x4c) ++ ++#define SSI2_STX0 __REG(IMX_SSI2_BASE + 0x00) ++#define SSI2_STX0_PHYS __PHYS_REG(IMX_SSI2_BASE + 0x00) ++#define SSI2_STX1 __REG(IMX_SSI2_BASE + 0x04) ++#define SSI2_STX1_PHYS __PHYS_REG(IMX_SSI2_BASE + 0x04) ++#define SSI2_SRX0 __REG(IMX_SSI2_BASE + 0x08) ++#define SSI2_SRX0_PHYS __PHYS_REG(IMX_SSI2_BASE + 0x08) ++#define SSI2_SRX1 __REG(IMX_SSI2_BASE + 0x0c) ++#define SSI2_SRX1_PHYS __PHYS_REG(IMX_SSI2_BASE + 0x0c) ++#define SSI2_SCR __REG(IMX_SSI2_BASE + 0x10) ++#define SSI2_SISR __REG(IMX_SSI2_BASE + 0x14) ++#define SSI2_SIER __REG(IMX_SSI2_BASE + 0x18) ++#define SSI2_STCR __REG(IMX_SSI2_BASE + 0x1c) ++#define SSI2_SRCR __REG(IMX_SSI2_BASE + 0x20) ++#define SSI2_STCCR __REG(IMX_SSI2_BASE + 0x24) ++#define SSI2_SRCCR __REG(IMX_SSI2_BASE + 0x28) ++#define SSI2_SFCSR __REG(IMX_SSI2_BASE + 0x2c) ++#define SSI2_STR __REG(IMX_SSI2_BASE + 0x30) ++#define SSI2_SOR __REG(IMX_SSI2_BASE + 0x34) ++#define SSI2_SACNT __REG(IMX_SSI2_BASE + 0x38) ++#define SSI2_SACADD __REG(IMX_SSI2_BASE + 0x3c) ++#define SSI2_SACDAT __REG(IMX_SSI2_BASE + 0x40) ++#define SSI2_SATAG __REG(IMX_SSI2_BASE + 0x44) ++#define SSI2_STMSK __REG(IMX_SSI2_BASE + 0x48) ++#define SSI2_SRMSK __REG(IMX_SSI2_BASE + 0x4c) ++ ++#define SSI_SCR_CLK_IST (1 << 9) ++#define SSI_SCR_TCH_EN (1 << 8) ++#define SSI_SCR_SYS_CLK_EN (1 << 7) ++#define SSI_SCR_I2S_MODE_NORM (0 << 5) ++#define SSI_SCR_I2S_MODE_MSTR (1 << 5) ++#define SSI_SCR_I2S_MODE_SLAVE (2 << 5) ++#define SSI_SCR_SYN (1 << 4) ++#define SSI_SCR_NET (1 << 3) ++#define SSI_SCR_RE (1 << 2) ++#define SSI_SCR_TE (1 << 1) ++#define SSI_SCR_SSIEN (1 << 0) ++ ++#define SSI_SISR_CMDAU (1 << 18) ++#define SSI_SISR_CMDDU (1 << 17) ++#define SSI_SISR_RXT (1 << 16) ++#define SSI_SISR_RDR1 (1 << 15) ++#define SSI_SISR_RDR0 (1 << 14) ++#define SSI_SISR_TDE1 (1 << 13) ++#define SSI_SISR_TDE0 (1 << 12) ++#define SSI_SISR_ROE1 (1 << 11) ++#define SSI_SISR_ROE0 (1 << 10) ++#define SSI_SISR_TUE1 (1 << 9) ++#define SSI_SISR_TUE0 (1 << 8) ++#define SSI_SISR_TFS (1 << 7) ++#define SSI_SISR_RFS (1 << 6) ++#define SSI_SISR_TLS (1 << 5) ++#define SSI_SISR_RLS (1 << 4) ++#define SSI_SISR_RFF1 (1 << 3) ++#define SSI_SISR_RFF0 (1 << 2) ++#define SSI_SISR_TFE1 (1 << 1) ++#define SSI_SISR_TFE0 (1 << 0) ++ ++#define SSI_SIER_RDMAE (1 << 22) ++#define SSI_SIER_RIE (1 << 21) ++#define SSI_SIER_TDMAE (1 << 20) ++#define SSI_SIER_TIE (1 << 19) ++#define SSI_SIER_CMDAU_EN (1 << 18) ++#define SSI_SIER_CMDDU_EN (1 << 17) ++#define SSI_SIER_RXT_EN (1 << 16) ++#define SSI_SIER_RDR1_EN (1 << 15) ++#define SSI_SIER_RDR0_EN (1 << 14) ++#define SSI_SIER_TDE1_EN (1 << 13) ++#define SSI_SIER_TDE0_EN (1 << 12) ++#define SSI_SIER_ROE1_EN (1 << 11) ++#define SSI_SIER_ROE0_EN (1 << 10) ++#define SSI_SIER_TUE1_EN (1 << 9) ++#define SSI_SIER_TUE0_EN (1 << 8) ++#define SSI_SIER_TFS_EN (1 << 7) ++#define SSI_SIER_RFS_EN (1 << 6) ++#define SSI_SIER_TLS_EN (1 << 5) ++#define SSI_SIER_RLS_EN (1 << 4) ++#define SSI_SIER_RFF1_EN (1 << 3) ++#define SSI_SIER_RFF0_EN (1 << 2) ++#define SSI_SIER_TFE1_EN (1 << 1) ++#define SSI_SIER_TFE0_EN (1 << 0) ++ ++#define SSI_STCR_TXBIT0 (1 << 9) ++#define SSI_STCR_TFEN1 (1 << 8) ++#define SSI_STCR_TFEN0 (1 << 7) ++#define SSI_STCR_TFDIR (1 << 6) ++#define SSI_STCR_TXDIR (1 << 5) ++#define SSI_STCR_TSHFD (1 << 4) ++#define SSI_STCR_TSCKP (1 << 3) ++#define SSI_STCR_TFSI (1 << 2) ++#define SSI_STCR_TFSL (1 << 1) ++#define SSI_STCR_TEFS (1 << 0) ++ ++#define SSI_SRCR_RXBIT0 (1 << 9) ++#define SSI_SRCR_RFEN1 (1 << 8) ++#define SSI_SRCR_RFEN0 (1 << 7) ++#define SSI_SRCR_RFDIR (1 << 6) ++#define SSI_SRCR_RXDIR (1 << 5) ++#define SSI_SRCR_RSHFD (1 << 4) ++#define SSI_SRCR_RSCKP (1 << 3) ++#define SSI_SRCR_RFSI (1 << 2) ++#define SSI_SRCR_RFSL (1 << 1) ++#define SSI_SRCR_REFS (1 << 0) ++ ++#define SSI_STCCR_DIV2 (1 << 18) ++#define SSI_STCCR_PSR (1 << 15) ++#define SSI_STCCR_WL(x) ((((x) - 2) >> 1) << 13) ++#define SSI_STCCR_DC(x) (((x) & 0x1f) << 8) ++#define SSI_STCCR_PM(x) (((x) & 0xff) << 0) ++ ++#define SSI_SRCCR_DIV2 (1 << 18) ++#define SSI_SRCCR_PSR (1 << 15) ++#define SSI_SRCCR_WL(x) ((((x) - 2) >> 1) << 13) ++#define SSI_SRCCR_DC(x) (((x) & 0x1f) << 8) ++#define SSI_SRCCR_PM(x) (((x) & 0xff) << 0) ++ ++ ++#define SSI_SFCSR_RFCNT1(x) (((x) & 0xf) << 28) ++#define SSI_SFCSR_TFCNT1(x) (((x) & 0xf) << 24) ++#define SSI_SFCSR_RFWM1(x) (((x) & 0xf) << 20) ++#define SSI_SFCSR_TFWM1(x) (((x) & 0xf) << 16) ++#define SSI_SFCSR_RFCNT0(x) (((x) & 0xf) << 12) ++#define SSI_SFCSR_TFCNT0(x) (((x) & 0xf) << 8) ++#define SSI_SFCSR_RFWM0(x) (((x) & 0xf) << 4) ++#define SSI_SFCSR_TFWM0(x) (((x) & 0xf) << 0) ++ ++#define SSI_STR_TEST (1 << 15) ++#define SSI_STR_RCK2TCK (1 << 14) ++#define SSI_STR_RFS2TFS (1 << 13) ++#define SSI_STR_RXSTATE(x) (((x) & 0xf) << 8) ++#define SSI_STR_TXD2RXD (1 << 7) ++#define SSI_STR_TCK2RCK (1 << 6) ++#define SSI_STR_TFS2RFS (1 << 5) ++#define SSI_STR_TXSTATE(x) (((x) & 0xf) << 0) ++ ++#define SSI_SOR_CLKOFF (1 << 6) ++#define SSI_SOR_RX_CLR (1 << 5) ++#define SSI_SOR_TX_CLR (1 << 4) ++#define SSI_SOR_INIT (1 << 3) ++#define SSI_SOR_WAIT(x) (((x) & 0x3) << 1) ++#define SSI_SOR_SYNRST (1 << 0) ++ ++#define SSI_SACNT_FRDIV(x) (((x) & 0x3f) << 5) ++#define SSI_SACNT_WR (x << 4) ++#define SSI_SACNT_RD (x << 3) ++#define SSI_SACNT_TIF (x << 2) ++#define SSI_SACNT_FV (x << 1) ++#define SSI_SACNT_A97EN (x << 0) ++ ++ ++/* AUDMUX registers */ ++#define AUDMUX_HPCR1 __REG(IMX_AUDMUX_BASE + 0x00) ++#define AUDMUX_HPCR2 __REG(IMX_AUDMUX_BASE + 0x04) ++#define AUDMUX_HPCR3 __REG(IMX_AUDMUX_BASE + 0x08) ++#define AUDMUX_PPCR1 __REG(IMX_AUDMUX_BASE + 0x10) ++#define AUDMUX_PPCR2 __REG(IMX_AUDMUX_BASE + 0x14) ++#define AUDMUX_PPCR3 __REG(IMX_AUDMUX_BASE + 0x18) ++ ++#define AUDMUX_HPCR_TFSDIR (1 << 31) ++#define AUDMUX_HPCR_TCLKDIR (1 << 30) ++#define AUDMUX_HPCR_TFCSEL_TX (0 << 26) ++#define AUDMUX_HPCR_TFCSEL_RX (8 << 26) ++#define AUDMUX_HPCR_TFCSEL(x) (((x) & 0x7) << 26) ++#define AUDMUX_HPCR_RFSDIR (1 << 25) ++#define AUDMUX_HPCR_RCLKDIR (1 << 24) ++#define AUDMUX_HPCR_RFCSEL_TX (0 << 20) ++#define AUDMUX_HPCR_RFCSEL_RX (8 << 20) ++#define AUDMUX_HPCR_RFCSEL(x) (((x) & 0x7) << 20) ++#define AUDMUX_HPCR_RXDSEL(x) (((x) & 0x7) << 13) ++#define AUDMUX_HPCR_SYN (1 << 12) ++#define AUDMUX_HPCR_TXRXEN (1 << 10) ++#define AUDMUX_HPCR_INMEN (1 << 8) ++#define AUDMUX_HPCR_INMMASK(x) (((x) & 0xff) << 0) ++ ++#define AUDMUX_PPCR_TFSDIR (1 << 31) ++#define AUDMUX_PPCR_TCLKDIR (1 << 30) ++#define AUDMUX_PPCR_TFCSEL_TX (0 << 26) ++#define AUDMUX_PPCR_TFCSEL_RX (8 << 26) ++#define AUDMUX_PPCR_TFCSEL(x) (((x) & 0x7) << 26) ++#define AUDMUX_PPCR_RFSDIR (1 << 25) ++#define AUDMUX_PPCR_RCLKDIR (1 << 24) ++#define AUDMUX_PPCR_RFCSEL_TX (0 << 20) ++#define AUDMUX_PPCR_RFCSEL_RX (8 << 20) ++#define AUDMUX_PPCR_RFCSEL(x) (((x) & 0x7) << 20) ++#define AUDMUX_PPCR_RXDSEL(x) (((x) & 0x7) << 13) ++#define AUDMUX_PPCR_SYN (1 << 12) ++#define AUDMUX_PPCR_TXRXEN (1 << 10) ++ ++ ++#endif +Index: linux-2.6-pxa-new/sound/soc/imx/imx31-pcm.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/imx/imx31-pcm.c +@@ -0,0 +1,454 @@ ++/* ++ * linux/sound/arm/mxc-pcm.c -- ALSA SoC interface for the Freescale i.MX CPU's ++ * ++ * Copyright 2006 Wolfson Microelectronics PLC. ++ * Author: Liam Girdwood ++ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * Based on pxa2xx-pcm.c by Nicolas Pitre, (C) 2004 MontaVista Software, Inc. ++ * and on mxc-alsa-mc13783 (C) 2006 Freescale. ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ * ++ * Revision history ++ * 29th Aug 2006 Initial version. ++ * ++ */ ++ ++#include <linux/module.h> ++#include <linux/init.h> ++#include <linux/platform_device.h> ++#include <linux/slab.h> ++#include <linux/dma-mapping.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/pcm_params.h> ++#include <sound/soc.h> ++#include <asm/dma.h> ++#include <asm/hardware.h> ++ ++#include "imx-pcm.h" ++ ++/* debug */ ++#define IMX_DEBUG 0 ++#if IMX_DEBUG ++#define dbg(format, arg...) printk(format, ## arg) ++#else ++#define dbg(format, arg...) ++#endif ++ ++static const struct snd_pcm_hardware mxc_pcm_hardware = { ++ .info = (SNDRV_PCM_INFO_INTERLEAVED | ++ SNDRV_PCM_INFO_BLOCK_TRANSFER | ++ SNDRV_PCM_INFO_MMAP | ++ SNDRV_PCM_INFO_MMAP_VALID | ++ SNDRV_PCM_INFO_PAUSE | ++ SNDRV_PCM_INFO_RESUME), ++ .formats = SNDRV_PCM_FMTBIT_S16_LE | ++ SNDRV_PCM_FMTBIT_S24_LE, ++ .buffer_bytes_max = 32 * 1024, ++ .period_bytes_min = 64, ++ .period_bytes_max = 8 * 1024, ++ .periods_min = 2, ++ .periods_max = 255, ++ .fifo_size = 0, ++}; ++ ++struct mxc_runtime_data { ++ int dma_ch; ++ struct mxc_pcm_dma_param *dma_params; ++}; ++ ++/*! ++ * This function stops the current dma transfert for playback ++ * and clears the dma pointers. ++ * ++ * @param substream pointer to the structure of the current stream. ++ * ++ */ ++static void audio_stop_dma(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct mxc_runtime_data *prtd = runtime->private_data; ++ unsigned int dma_size = frames_to_bytes(runtime, runtime->period_size); ++ unsigned int offset dma_size * s->periods; ++ unsigned long flags; ++ ++ spin_lock_irqsave(&prtd->dma_lock, flags); ++ ++ dbg("MXC : audio_stop_dma active = 0\n"); ++ prtd->active = 0; ++ prtd->period = 0; ++ prtd->periods = 0; ++ ++ /* this stops the dma channel and clears the buffer ptrs */ ++ mxc_dma_stop(prtd->dma_wchannel); ++ if(substream == SNDRV_PCM_STREAM_PLAYBACK) ++ dma_unmap_single(NULL, runtime->dma_addr + offset, dma_size, ++ DMA_TO_DEVICE); ++ else ++ dma_unmap_single(NULL, runtime->dma_addr + offset, dma_size, ++ DMA_FROM_DEVICE); ++ ++ spin_unlock_irqrestore(&prtd->dma_lock, flags); ++} ++ ++/*! ++ * This function is called whenever a new audio block needs to be ++ * transferred to mc13783. The function receives the address and the size ++ * of the new block and start a new DMA transfer. ++ * ++ * @param substream pointer to the structure of the current stream. ++ * ++ */ ++static int dma_new_period(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct mxc_runtime_data *prtd = runtime->private_data; ++ unsigned int dma_size; ++ unsigned int offset; ++ int ret=0; ++ dma_request_t sdma_request; ++ ++ if (prtd->active){ ++ memset(&sdma_request, 0, sizeof(dma_request_t)); ++ dma_size = frames_to_bytes(runtime, runtime->period_size); ++ dbg("s->period (%x) runtime->periods (%d)\n", ++ s->period,runtime->periods); ++ dbg("runtime->period_size (%d) dma_size (%d)\n", ++ (unsigned int)runtime->period_size, ++ runtime->dma_bytes); ++ ++ offset = dma_size * prtd->period; ++ snd_assert(dma_size <= DMA_BUF_SIZE, ); ++ if(substream == SNDRV_PCM_STREAM_PLAYBACK) ++ sdma_request.sourceAddr = (char*)(dma_map_single(NULL, ++ runtime->dma_area + offset, dma_size, DMA_TO_DEVICE)); ++ else ++ sdma_request.destAddr = (char*)(dma_map_single(NULL, ++ runtime->dma_area + offset, dma_size, DMA_FROM_DEVICE)); ++ sdma_request.count = dma_size; ++ ++ dbg("MXC: Start DMA offset (%d) size (%d)\n", offset, ++ runtime->dma_bytes); ++ ++ mxc_dma_set_config(prtd->dma_wchannel, &sdma_request, 0); ++ if((ret = mxc_dma_start(prtd->dma_wchannel)) < 0) { ++ dbg("audio_process_dma: cannot queue DMA buffer\ ++ (%i)\n", ret); ++ return err; ++ } ++ prtd->tx_spin = 1; /* FGA little trick to retrieve DMA pos */ ++ prtd->period++; ++ prtd->period %= runtime->periods; ++ } ++ return ret; ++} ++ ++ ++/*! ++ * This is a callback which will be called ++ * when a TX transfer finishes. The call occurs ++ * in interrupt context. ++ * ++ * @param dat pointer to the structure of the current stream. ++ * ++ */ ++static void audio_dma_irq(void *data) ++{ ++ struct snd_pcm_substream *substream; ++ struct snd_pcm_runtime *runtime; ++ struct mxc_runtime_data *prtd; ++ unsigned int dma_size; ++ unsigned int previous_period; ++ unsigned int offset; ++ ++ substream = data; ++ runtime = substream->runtime; ++ prtd = runtime->private_data; ++ previous_period = prtd->periods; ++ dma_size = frames_to_bytes(runtime, runtime->period_size); ++ offset = dma_size * previous_period; ++ ++ prtd->tx_spin = 0; ++ prtd->periods++; ++ prtd->periods %= runtime->periods; ++ ++ /* ++ * Give back to the CPU the access to the non cached memory ++ */ ++ if(substream == SNDRV_PCM_STREAM_PLAYBACK) ++ dma_unmap_single(NULL, runtime->dma_addr + offset, dma_size, ++ DMA_TO_DEVICE); ++ else ++ dma_unmap_single(NULL, runtime->dma_addr + offset, dma_size, ++ DMA_FROM_DEVICE); ++ /* ++ * If we are getting a callback for an active stream then we inform ++ * the PCM middle layer we've finished a period ++ */ ++ if (prtd->active) ++ snd_pcm_period_elapsed(substream); ++ ++ /* ++ * Trig next DMA transfer ++ */ ++ dma_new_period(substream); ++} ++ ++/*! ++ * This function configures the hardware to allow audio ++ * playback operations. It is called by ALSA framework. ++ * ++ * @param substream pointer to the structure of the current stream. ++ * ++ * @return 0 on success, -1 otherwise. ++ */ ++static int ++snd_mxc_prepare(struct snd_pcm_substream *substream) ++{ ++ struct mxc_runtime_data *prtd = runtime->private_data; ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ int ret = 0; ++ prtd->period = 0; ++ prtd->periods = 0; ++ ++ dma_channel_params params; ++ int channel = 0; // passed in ? ++ ++ if ((ret = mxc_request_dma(&channel, "ALSA TX SDMA") < 0)){ ++ dbg("error requesting a write dma channel\n"); ++ return ret; ++ } ++ ++ /* configure DMA params */ ++ memset(¶ms, 0, sizeof(dma_channel_params)); ++ params.bd_number = 1; ++ params.arg = s; ++ params.callback = callback; ++ params.transfer_type = emi_2_per; ++ params.watermark_level = SDMA_TXFIFO_WATERMARK; ++ params.word_size = TRANSFER_16BIT; ++ //dbg(KERN_ERR "activating connection SSI1 - SDMA\n"); ++ params.per_address = SSI1_BASE_ADDR; ++ params.event_id = DMA_REQ_SSI1_TX1; ++ params.peripheral_type = SSI; ++ ++ /* set up chn with params */ ++ mxc_dma_setup_channel(channel, ¶ms); ++ s->dma_wchannel = channel; ++ ++ return ret; ++} ++ ++static int mxc_pcm_hw_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ int ret; ++ ++ if((ret=snd_pcm_lib_malloc_pages(substream, params_buffer_bytes(hw_params))) < 0) ++ return ret; ++ runtime->dma_addr = virt_to_phys(runtime->dma_area); ++ ++ return ret; ++} ++ ++static int mxc_pcm_hw_free(struct snd_pcm_substream *substream) ++{ ++ return snd_pcm_lib_free_pages(substream); ++} ++ ++static int mxc_pcm_trigger(struct snd_pcm_substream *substream, int cmd) ++{ ++ struct mxc_runtime_data *prtd = substream->runtime->private_data; ++ int ret = 0; ++ ++ switch (cmd) { ++ case SNDRV_PCM_TRIGGER_START: ++ prtd->tx_spin = 0; ++ /* requested stream startup */ ++ prtd->active = 1; ++ ret = dma_new_period(substream); ++ break; ++ case SNDRV_PCM_TRIGGER_STOP: ++ /* requested stream shutdown */ ++ ret = audio_stop_dma(substream); ++ break; ++ case SNDRV_PCM_TRIGGER_SUSPEND: ++ prtd->active = 0; ++ prtd->periods = 0; ++ break; ++ case SNDRV_PCM_TRIGGER_RESUME: ++ prtd->active = 1; ++ prtd->tx_spin = 0; ++ ret = dma_new_period(substream); ++ break; ++ case SNDRV_PCM_TRIGGER_PAUSE_PUSH: ++ prtd->active = 0; ++ break; ++ case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: ++ prtd->active = 1; ++ if (prtd->old_offset) { ++ prtd->tx_spin = 0; ++ ret = dma_new_period(substream); ++ } ++ break; ++ default: ++ ret = -EINVAL; ++ break; ++ } ++ ++ return ret; ++} ++ ++static snd_pcm_uframes_t mxc_pcm_pointer(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct mxc_runtime_data *prtd = runtime->private_data; ++ unsigned int offset = 0; ++ ++ /* tx_spin value is used here to check if a transfert is active */ ++ if (prtd->tx_spin){ ++ offset = (runtime->period_size * (prtd->periods)) + ++ (runtime->period_size >> 1); ++ if (offset >= runtime->buffer_size) ++ offset = runtime->period_size >> 1; ++ } else { ++ offset = (runtime->period_size * (s->periods)); ++ if (offset >= runtime->buffer_size) ++ offset = 0; ++ } ++ ++ return offset; ++} ++ ++ ++static int mxc_pcm_open(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct mxc_runtime_data *prtd; ++ int ret; ++ ++ snd_soc_set_runtime_hwparams(substream, &mxc_pcm_hardware); ++ ++ if ((err = snd_pcm_hw_constraint_integer(runtime, ++ SNDRV_PCM_HW_PARAM_PERIODS)) < 0) ++ return err; ++ if ((err = snd_pcm_hw_constraint_list(runtime, 0, ++ SNDRV_PCM_HW_PARAM_RATE, &hw_playback_rates)) < 0) ++ return err; ++ msleep(10); // liam - why ++ ++ /* setup DMA controller for playback */ ++ if((err = configure_write_channel(&mxc_mc13783->s[SNDRV_PCM_STREAM_PLAYBACK], ++ audio_dma_irq)) < 0 ) ++ return err; ++ ++ if((prtd = kzalloc(sizeof(struct mxc_runtime_data), GFP_KERNEL)) == NULL) { ++ ret = -ENOMEM; ++ goto out; ++ } ++ ++ runtime->private_data = prtd; ++ return 0; ++ ++ err1: ++ kfree(prtd); ++ out: ++ return ret; ++} ++ ++static int mxc_pcm_close(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct mxc_runtime_data *prtd = runtime->private_data; ++ ++// mxc_mc13783_t *chip; ++ audio_stream_t *s; ++ device_data_t* device; ++ int ssi; ++ ++ //chip = snd_pcm_substream_chip(substream); ++ s = &chip->s[substream->pstr->stream]; ++ device = &s->stream_device; ++ ssi = device->ssi; ++ ++ //disable_stereodac(); ++ ++ ssi_transmit_enable(ssi, false); ++ ssi_interrupt_disable(ssi, ssi_tx_dma_interrupt_enable); ++ ssi_tx_fifo_enable(ssi, ssi_fifo_0, false); ++ ssi_enable(ssi, false); ++ ++ chip->s[substream->pstr->stream].stream = NULL; ++ ++ return 0; ++} ++ ++static int ++mxc_pcm_mmap(struct snd_pcm_substream *substream, struct vm_area_struct *vma) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ return dma_mmap_writecombine(substream->pcm->card->dev, vma, ++ runtime->dma_area, ++ runtime->dma_addr, ++ runtime->dma_bytes); ++} ++ ++struct snd_pcm_ops mxc_pcm_ops = { ++ .open = mxc_pcm_open, ++ .close = mxc_pcm_close, ++ .ioctl = snd_pcm_lib_ioctl, ++ .hw_params = mxc_pcm_hw_params, ++ .hw_free = mxc_pcm_hw_free, ++ .prepare = mxc_pcm_prepare, ++ .trigger = mxc_pcm_trigger, ++ .pointer = mxc_pcm_pointer, ++ .mmap = mxc_pcm_mmap, ++}; ++ ++static u64 mxc_pcm_dmamask = 0xffffffff; ++ ++int mxc_pcm_new(struct snd_card *card, struct snd_soc_codec_dai *dai, ++ struct snd_pcm *pcm) ++{ ++ int ret = 0; ++ ++ if (!card->dev->dma_mask) ++ card->dev->dma_mask = &mxc_pcm_dmamask; ++ if (!card->dev->coherent_dma_mask) ++ card->dev->coherent_dma_mask = 0xffffffff; ++ ++ if (dai->playback.channels_min) { ++ ret = mxc_pcm_preallocate_dma_buffer(pcm, ++ SNDRV_PCM_STREAM_PLAYBACK); ++ if (ret) ++ goto out; ++ } ++ ++ if (dai->capture.channels_min) { ++ ret = mxc_pcm_preallocate_dma_buffer(pcm, ++ SNDRV_PCM_STREAM_CAPTURE); ++ if (ret) ++ goto out; ++ } ++ out: ++ return ret; ++} ++ ++struct snd_soc_platform mxc_soc_platform = { ++ .name = "mxc-audio", ++ .pcm_ops = &mxc_pcm_ops, ++ .pcm_new = mxc_pcm_new, ++ .pcm_free = mxc_pcm_free_dma_buffers, ++}; ++ ++EXPORT_SYMBOL_GPL(mxc_soc_platform); ++ ++MODULE_AUTHOR("Liam Girdwood"); ++MODULE_DESCRIPTION("Freescale i.MX PCM DMA module"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/imx/imx31-pcm.h +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/imx/imx31-pcm.h +@@ -0,0 +1,237 @@ ++/* ++ * mxc-pcm.h :- ASoC platform header for Freescale i.MX ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#ifndef _MXC_PCM_H ++#define _MXC_PCM_H ++ ++struct { ++ char *name; /* stream identifier */ ++ dma_channel_params dma_params; ++} mxc_pcm_dma_param; ++ ++extern struct snd_soc_cpu_dai mxc_ssi_dai[3]; ++ ++/* platform data */ ++extern struct snd_soc_platform mxc_soc_platform; ++extern struct snd_ac97_bus_ops mxc_ac97_ops; ++ ++/* temp until imx-regs.h is up2date */ ++#define SSI1_STX0 __REG(IMX_SSI1_BASE + 0x00) ++#define SSI1_STX0_PHYS __PHYS_REG(IMX_SSI1_BASE + 0x00) ++#define SSI1_STX1 __REG(IMX_SSI1_BASE + 0x04) ++#define SSI1_STX1_PHYS __PHYS_REG(IMX_SSI1_BASE + 0x04) ++#define SSI1_SRX0 __REG(IMX_SSI1_BASE + 0x08) ++#define SSI1_SRX0_PHYS __PHYS_REG(IMX_SSI1_BASE + 0x08) ++#define SSI1_SRX1 __REG(IMX_SSI1_BASE + 0x0c) ++#define SSI1_SRX1_PHYS __PHYS_REG(IMX_SSI1_BASE + 0x0c) ++#define SSI1_SCR __REG(IMX_SSI1_BASE + 0x10) ++#define SSI1_SISR __REG(IMX_SSI1_BASE + 0x14) ++#define SSI1_SIER __REG(IMX_SSI1_BASE + 0x18) ++#define SSI1_STCR __REG(IMX_SSI1_BASE + 0x1c) ++#define SSI1_SRCR __REG(IMX_SSI1_BASE + 0x20) ++#define SSI1_STCCR __REG(IMX_SSI1_BASE + 0x24) ++#define SSI1_SRCCR __REG(IMX_SSI1_BASE + 0x28) ++#define SSI1_SFCSR __REG(IMX_SSI1_BASE + 0x2c) ++#define SSI1_STR __REG(IMX_SSI1_BASE + 0x30) ++#define SSI1_SOR __REG(IMX_SSI1_BASE + 0x34) ++#define SSI1_SACNT __REG(IMX_SSI1_BASE + 0x38) ++#define SSI1_SACADD __REG(IMX_SSI1_BASE + 0x3c) ++#define SSI1_SACDAT __REG(IMX_SSI1_BASE + 0x40) ++#define SSI1_SATAG __REG(IMX_SSI1_BASE + 0x44) ++#define SSI1_STMSK __REG(IMX_SSI1_BASE + 0x48) ++#define SSI1_SRMSK __REG(IMX_SSI1_BASE + 0x4c) ++ ++#define SSI2_STX0 __REG(IMX_SSI2_BASE + 0x00) ++#define SSI2_STX0_PHYS __PHYS_REG(IMX_SSI2_BASE + 0x00) ++#define SSI2_STX1 __REG(IMX_SSI2_BASE + 0x04) ++#define SSI2_STX1_PHYS __PHYS_REG(IMX_SSI2_BASE + 0x04) ++#define SSI2_SRX0 __REG(IMX_SSI2_BASE + 0x08) ++#define SSI2_SRX0_PHYS __PHYS_REG(IMX_SSI2_BASE + 0x08) ++#define SSI2_SRX1 __REG(IMX_SSI2_BASE + 0x0c) ++#define SSI2_SRX1_PHYS __PHYS_REG(IMX_SSI2_BASE + 0x0c) ++#define SSI2_SCR __REG(IMX_SSI2_BASE + 0x10) ++#define SSI2_SISR __REG(IMX_SSI2_BASE + 0x14) ++#define SSI2_SIER __REG(IMX_SSI2_BASE + 0x18) ++#define SSI2_STCR __REG(IMX_SSI2_BASE + 0x1c) ++#define SSI2_SRCR __REG(IMX_SSI2_BASE + 0x20) ++#define SSI2_STCCR __REG(IMX_SSI2_BASE + 0x24) ++#define SSI2_SRCCR __REG(IMX_SSI2_BASE + 0x28) ++#define SSI2_SFCSR __REG(IMX_SSI2_BASE + 0x2c) ++#define SSI2_STR __REG(IMX_SSI2_BASE + 0x30) ++#define SSI2_SOR __REG(IMX_SSI2_BASE + 0x34) ++#define SSI2_SACNT __REG(IMX_SSI2_BASE + 0x38) ++#define SSI2_SACADD __REG(IMX_SSI2_BASE + 0x3c) ++#define SSI2_SACDAT __REG(IMX_SSI2_BASE + 0x40) ++#define SSI2_SATAG __REG(IMX_SSI2_BASE + 0x44) ++#define SSI2_STMSK __REG(IMX_SSI2_BASE + 0x48) ++#define SSI2_SRMSK __REG(IMX_SSI2_BASE + 0x4c) ++ ++#define SSI_SCR_CLK_IST (1 << 9) ++#define SSI_SCR_TCH_EN (1 << 8) ++#define SSI_SCR_SYS_CLK_EN (1 << 7) ++#define SSI_SCR_I2S_MODE_NORM (0 << 5) ++#define SSI_SCR_I2S_MODE_MSTR (1 << 5) ++#define SSI_SCR_I2S_MODE_SLAVE (2 << 5) ++#define SSI_SCR_SYN (1 << 4) ++#define SSI_SCR_NET (1 << 3) ++#define SSI_SCR_RE (1 << 2) ++#define SSI_SCR_TE (1 << 1) ++#define SSI_SCR_SSIEN (1 << 0) ++ ++#define SSI_SISR_CMDAU (1 << 18) ++#define SSI_SISR_CMDDU (1 << 17) ++#define SSI_SISR_RXT (1 << 16) ++#define SSI_SISR_RDR1 (1 << 15) ++#define SSI_SISR_RDR0 (1 << 14) ++#define SSI_SISR_TDE1 (1 << 13) ++#define SSI_SISR_TDE0 (1 << 12) ++#define SSI_SISR_ROE1 (1 << 11) ++#define SSI_SISR_ROE0 (1 << 10) ++#define SSI_SISR_TUE1 (1 << 9) ++#define SSI_SISR_TUE0 (1 << 8) ++#define SSI_SISR_TFS (1 << 7) ++#define SSI_SISR_RFS (1 << 6) ++#define SSI_SISR_TLS (1 << 5) ++#define SSI_SISR_RLS (1 << 4) ++#define SSI_SISR_RFF1 (1 << 3) ++#define SSI_SISR_RFF0 (1 << 2) ++#define SSI_SISR_TFE1 (1 << 1) ++#define SSI_SISR_TFE0 (1 << 0) ++ ++#define SSI_SIER_RDMAE (1 << 22) ++#define SSI_SIER_RIE (1 << 21) ++#define SSI_SIER_TDMAE (1 << 20) ++#define SSI_SIER_TIE (1 << 19) ++#define SSI_SIER_CMDAU_EN (1 << 18) ++#define SSI_SIER_CMDDU_EN (1 << 17) ++#define SSI_SIER_RXT_EN (1 << 16) ++#define SSI_SIER_RDR1_EN (1 << 15) ++#define SSI_SIER_RDR0_EN (1 << 14) ++#define SSI_SIER_TDE1_EN (1 << 13) ++#define SSI_SIER_TDE0_EN (1 << 12) ++#define SSI_SIER_ROE1_EN (1 << 11) ++#define SSI_SIER_ROE0_EN (1 << 10) ++#define SSI_SIER_TUE1_EN (1 << 9) ++#define SSI_SIER_TUE0_EN (1 << 8) ++#define SSI_SIER_TFS_EN (1 << 7) ++#define SSI_SIER_RFS_EN (1 << 6) ++#define SSI_SIER_TLS_EN (1 << 5) ++#define SSI_SIER_RLS_EN (1 << 4) ++#define SSI_SIER_RFF1_EN (1 << 3) ++#define SSI_SIER_RFF0_EN (1 << 2) ++#define SSI_SIER_TFE1_EN (1 << 1) ++#define SSI_SIER_TFE0_EN (1 << 0) ++ ++#define SSI_STCR_TXBIT0 (1 << 9) ++#define SSI_STCR_TFEN1 (1 << 8) ++#define SSI_STCR_TFEN0 (1 << 7) ++#define SSI_STCR_TFDIR (1 << 6) ++#define SSI_STCR_TXDIR (1 << 5) ++#define SSI_STCR_TSHFD (1 << 4) ++#define SSI_STCR_TSCKP (1 << 3) ++#define SSI_STCR_TFSI (1 << 2) ++#define SSI_STCR_TFSL (1 << 1) ++#define SSI_STCR_TEFS (1 << 0) ++ ++#define SSI_SRCR_RXBIT0 (1 << 9) ++#define SSI_SRCR_RFEN1 (1 << 8) ++#define SSI_SRCR_RFEN0 (1 << 7) ++#define SSI_SRCR_RFDIR (1 << 6) ++#define SSI_SRCR_RXDIR (1 << 5) ++#define SSI_SRCR_RSHFD (1 << 4) ++#define SSI_SRCR_RSCKP (1 << 3) ++#define SSI_SRCR_RFSI (1 << 2) ++#define SSI_SRCR_RFSL (1 << 1) ++#define SSI_SRCR_REFS (1 << 0) ++ ++#define SSI_STCCR_DIV2 (1 << 18) ++#define SSI_STCCR_PSR (1 << 15) ++#define SSI_STCCR_WL(x) ((((x) - 2) >> 1) << 13) ++#define SSI_STCCR_DC(x) (((x) & 0x1f) << 8) ++#define SSI_STCCR_PM(x) (((x) & 0xff) << 0) ++ ++#define SSI_SRCCR_DIV2 (1 << 18) ++#define SSI_SRCCR_PSR (1 << 15) ++#define SSI_SRCCR_WL(x) ((((x) - 2) >> 1) << 13) ++#define SSI_SRCCR_DC(x) (((x) & 0x1f) << 8) ++#define SSI_SRCCR_PM(x) (((x) & 0xff) << 0) ++ ++ ++#define SSI_SFCSR_RFCNT1(x) (((x) & 0xf) << 28) ++#define SSI_SFCSR_TFCNT1(x) (((x) & 0xf) << 24) ++#define SSI_SFCSR_RFWM1(x) (((x) & 0xf) << 20) ++#define SSI_SFCSR_TFWM1(x) (((x) & 0xf) << 16) ++#define SSI_SFCSR_RFCNT0(x) (((x) & 0xf) << 12) ++#define SSI_SFCSR_TFCNT0(x) (((x) & 0xf) << 8) ++#define SSI_SFCSR_RFWM0(x) (((x) & 0xf) << 4) ++#define SSI_SFCSR_TFWM0(x) (((x) & 0xf) << 0) ++ ++#define SSI_STR_TEST (1 << 15) ++#define SSI_STR_RCK2TCK (1 << 14) ++#define SSI_STR_RFS2TFS (1 << 13) ++#define SSI_STR_RXSTATE(x) (((x) & 0xf) << 8) ++#define SSI_STR_TXD2RXD (1 << 7) ++#define SSI_STR_TCK2RCK (1 << 6) ++#define SSI_STR_TFS2RFS (1 << 5) ++#define SSI_STR_TXSTATE(x) (((x) & 0xf) << 0) ++ ++#define SSI_SOR_CLKOFF (1 << 6) ++#define SSI_SOR_RX_CLR (1 << 5) ++#define SSI_SOR_TX_CLR (1 << 4) ++#define SSI_SOR_INIT (1 << 3) ++#define SSI_SOR_WAIT(x) (((x) & 0x3) << 1) ++#define SSI_SOR_SYNRST (1 << 0) ++ ++#define SSI_SACNT_FRDIV(x) (((x) & 0x3f) << 5) ++#define SSI_SACNT_WR (x << 4) ++#define SSI_SACNT_RD (x << 3) ++#define SSI_SACNT_TIF (x << 2) ++#define SSI_SACNT_FV (x << 1) ++#define SSI_SACNT_A97EN (x << 0) ++ ++ ++/* AUDMUX registers */ ++#define AUDMUX_HPCR1 __REG(IMX_AUDMUX_BASE + 0x00) ++#define AUDMUX_HPCR2 __REG(IMX_AUDMUX_BASE + 0x04) ++#define AUDMUX_HPCR3 __REG(IMX_AUDMUX_BASE + 0x08) ++#define AUDMUX_PPCR1 __REG(IMX_AUDMUX_BASE + 0x10) ++#define AUDMUX_PPCR2 __REG(IMX_AUDMUX_BASE + 0x14) ++#define AUDMUX_PPCR3 __REG(IMX_AUDMUX_BASE + 0x18) ++ ++#define AUDMUX_HPCR_TFSDIR (1 << 31) ++#define AUDMUX_HPCR_TCLKDIR (1 << 30) ++#define AUDMUX_HPCR_TFCSEL_TX (0 << 26) ++#define AUDMUX_HPCR_TFCSEL_RX (8 << 26) ++#define AUDMUX_HPCR_TFCSEL(x) (((x) & 0x7) << 26) ++#define AUDMUX_HPCR_RFSDIR (1 << 25) ++#define AUDMUX_HPCR_RCLKDIR (1 << 24) ++#define AUDMUX_HPCR_RFCSEL_TX (0 << 20) ++#define AUDMUX_HPCR_RFCSEL_RX (8 << 20) ++#define AUDMUX_HPCR_RFCSEL(x) (((x) & 0x7) << 20) ++#define AUDMUX_HPCR_RXDSEL(x) (((x) & 0x7) << 13) ++#define AUDMUX_HPCR_SYN (1 << 12) ++#define AUDMUX_HPCR_TXRXEN (1 << 10) ++#define AUDMUX_HPCR_INMEN (1 << 8) ++#define AUDMUX_HPCR_INMMASK(x) (((x) & 0xff) << 0) ++ ++#define AUDMUX_PPCR_TFSDIR (1 << 31) ++#define AUDMUX_PPCR_TCLKDIR (1 << 30) ++#define AUDMUX_PPCR_TFCSEL_TX (0 << 26) ++#define AUDMUX_PPCR_TFCSEL_RX (8 << 26) ++#define AUDMUX_PPCR_TFCSEL(x) (((x) & 0x7) << 26) ++#define AUDMUX_PPCR_RFSDIR (1 << 25) ++#define AUDMUX_PPCR_RCLKDIR (1 << 24) ++#define AUDMUX_PPCR_RFCSEL_TX (0 << 20) ++#define AUDMUX_PPCR_RFCSEL_RX (8 << 20) ++#define AUDMUX_PPCR_RFCSEL(x) (((x) & 0x7) << 20) ++#define AUDMUX_PPCR_RXDSEL(x) (((x) & 0x7) << 13) ++#define AUDMUX_PPCR_SYN (1 << 12) ++#define AUDMUX_PPCR_TXRXEN (1 << 10) ++ ++ ++#endif +Index: linux-2.6-pxa-new/sound/soc/s3c24xx/s3c24xx-i2s.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/s3c24xx/s3c24xx-i2s.c +@@ -0,0 +1,271 @@ ++/* ++ * s3c24xx-i2s.c -- ALSA Soc Audio Layer ++ * ++ * Copyright 2005 Wolfson Microelectronics PLC. ++ * Author: Graeme Gregory ++ * graeme.gregory@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * 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. ++ * ++ * Revision history ++ * 10th Nov 2006 Initial version. ++ */ ++ ++#include <linux/init.h> ++#include <linux/module.h> ++#include <linux/device.h> ++#include <linux/delay.h> ++#include <linux/clk.h> ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/initval.h> ++#include <sound/soc.h> ++ ++#include <asm/hardware.h> ++#include <asm/io.h> ++#include <asm/arch/regs-iis.h> ++#include <asm/arch/regs-gpio.h> ++#include <asm/arch/regs-clock.h> ++#include <asm/arch/audio.h> ++#include <asm/dma.h> ++#include <asm/arch/dma.h> ++ ++#include "s3c24xx-pcm.h" ++ ++/* used to disable sysclk if external crystal is used */ ++static int extclk = 0; ++module_param(extclk, int, 0); ++MODULE_PARM_DESC(extclk, "set to 1 to disable s3c24XX i2s sysclk"); ++ ++#define S3C24XX_I2S_DAIFMT \ ++ (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_LEFT_J | SND_SOC_DAIFMT_NB_NF) ++ ++#define S3C24XX_I2S_DIR \ ++ (SND_SOC_DAIDIR_PLAYBACK | SND_SOC_DAIDIR_CAPTURE) ++ ++#define S3C24XX_I2S_RATES \ ++ (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \ ++ SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ ++ SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000) ++ ++/* priv is divider */ ++static struct snd_soc_dai_mode s3c24xx_i2s_modes[] = ++{ ++ /* s3c24xx I2S frame and clock master modes */ ++ { ++ .fmt = S3C24XX_I2S_DAIFMT | SND_SOC_DAIFMT_CBS_CFS, ++ .pcmfmt = SNDRV_PCM_FMTBIT_S16_LE, ++ .pcmrate = SNDRV_PCM_RATE_44100, ++ .pcmdir = S3C24XX_I2S_DIR, ++ .flags = SND_SOC_DAI_BFS_RATE, ++ .fs = 384, ++ .bfs = 32, ++ .priv = 0x00 ++ }, ++}; ++ ++static struct s3c2410_dma_client s3c24xx_dma_client_out = { ++ .name = "I2S PCM Stereo out" ++}; ++ ++static struct s3c2410_dma_client s3c24xx_dma_client_in = { ++ .name = "I2S PCM Stereo in" ++}; ++ ++static s3c24xx_pcm_dma_params_t s3c24xx_i2s_pcm_stereo_out = { ++ .client = &s3c24xx_dma_client_out, ++ .channel = DMACH_I2S_OUT, ++ .dma_addr = S3C2410_PA_IIS+S3C2410_IISFIFO ++}; ++ ++static s3c24xx_pcm_dma_params_t s3c24xx_i2s_pcm_stereo_in = { ++ .client = &s3c24xx_dma_client_in, ++ .channel = DMACH_I2S_IN, ++ .dma_addr = S3C2410_PA_IIS+S3C2410_IISFIFO ++}; ++ ++ ++struct s3c24xx_i2s_port { ++ int master; ++}; ++static struct s3c24xx_i2s_port s3c24xx_i2s; ++ ++/* Empty for the s3c24xx platforms */ ++static int s3c24xx_i2s_startup(struct snd_pcm_substream *substream) ++{ ++ return 0; ++} ++ ++static int s3c24xx_i2s_hw_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ unsigned long iiscon; ++ unsigned long iismod; ++ unsigned long iisfcon; ++ ++ s3c24xx_i2s.master = 0; ++ if(rtd->cpu_dai->dai_runtime.fmt & SND_SOC_DAIFMT_CBS_CFS) ++ s3c24xx_i2s.master = 1; ++ ++ /* Configure the I2S pins in correct mode */ ++ s3c2410_gpio_cfgpin(S3C2410_GPE0,S3C2410_GPE0_I2SLRCK); ++ if (s3c24xx_i2s.master && !extclk){ ++ printk("Setting Clock Output as we are Master\n"); ++ s3c2410_gpio_cfgpin(S3C2410_GPE1,S3C2410_GPE1_I2SSCLK); ++ } ++ s3c2410_gpio_cfgpin(S3C2410_GPE2,S3C2410_GPE2_CDCLK); ++ s3c2410_gpio_cfgpin(S3C2410_GPE3,S3C2410_GPE3_I2SSDI); ++ s3c2410_gpio_cfgpin(S3C2410_GPE4,S3C2410_GPE4_I2SSDO); ++ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ { ++ rtd->cpu_dai->dma_data = &s3c24xx_i2s_pcm_stereo_out; ++ } ++ else ++ { ++ rtd->cpu_dai->dma_data = &s3c24xx_i2s_pcm_stereo_in; ++ } ++ ++ /* Working copies of registers */ ++ iiscon=readl(S3C24XX_VA_IIS+S3C2410_IISCON); ++ iismod=readl(S3C24XX_VA_IIS+S3C2410_IISMOD); ++ iisfcon=readl(S3C24XX_VA_IIS+S3C2410_IISFCON); ++ /* is port used by another stream */ ++ if (!(iiscon & S3C2410_IISCON_IISEN)) { ++ ++ /* Clear the registers */ ++ ++ iismod |= S3C2410_IISMOD_32FS | S3C2410_IISMOD_384FS; ++ ++ if (!s3c24xx_i2s.master) ++ iismod |= S3C2410_IISMOD_SLAVE; ++ ++ if (rtd->cpu_dai->dai_runtime.fmt & SND_SOC_DAIFMT_LEFT_J) ++ iismod |= S3C2410_IISMOD_MSB; ++ } ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ { ++ iismod |= S3C2410_IISMOD_TXMODE; ++ iiscon |= S3C2410_IISCON_TXDMAEN; ++ iisfcon |= S3C2410_IISFCON_TXDMA | S3C2410_IISFCON_TXENABLE; ++ } ++ else ++ { ++ iismod |= S3C2410_IISMOD_RXMODE; ++ iiscon |= S3C2410_IISCON_RXDMAEN; ++ iisfcon |= S3C2410_IISFCON_RXDMA | S3C2410_IISFCON_RXENABLE; ++ } ++ ++ writel(iiscon, S3C24XX_VA_IIS+S3C2410_IISCON); ++ writel(iismod, S3C24XX_VA_IIS+S3C2410_IISMOD); ++ writel(iisfcon, S3C24XX_VA_IIS+S3C2410_IISFCON); ++ ++ printk("IISCON: %lx IISMOD: %lx", iiscon, iismod); ++ ++ return 0; ++} ++ ++static int s3c24xx_i2s_trigger(struct snd_pcm_substream *substream, int cmd) ++{ ++ int ret = 0; ++ unsigned long iiscon; ++ ++ switch (cmd) { ++ case SNDRV_PCM_TRIGGER_START: ++ /* Enable the IIS unit */ ++ iiscon = readl(S3C24XX_VA_IIS+S3C2410_IISCON); ++ iiscon |= S3C2410_IISCON_IISEN; ++ writel(iiscon, S3C24XX_VA_IIS+S3C2410_IISCON); ++ break; ++ case SNDRV_PCM_TRIGGER_RESUME: ++ case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: ++ case SNDRV_PCM_TRIGGER_STOP: ++ case SNDRV_PCM_TRIGGER_SUSPEND: ++ case SNDRV_PCM_TRIGGER_PAUSE_PUSH: ++ break; ++ default: ++ ret = -EINVAL; ++ } ++ ++ return ret; ++} ++ ++static void s3c24xx_i2s_shutdown(struct snd_pcm_substream *substream) ++{ ++ unsigned long iismod, iiscon; ++ ++ iismod=readl(S3C24XX_VA_IIS+S3C2410_IISMOD); ++ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { ++ iismod &= ~S3C2410_IISMOD_TXMODE; ++ } else { ++ iismod &= ~S3C2410_IISMOD_RXMODE; ++ } ++ ++ writel(iismod,S3C24XX_VA_IIS+S3C2410_IISMOD); ++ ++ iiscon=readl(S3C24XX_VA_IIS+S3C2410_IISCON); ++ ++ if (iismod & ( S3C2410_IISMOD_TXMODE | S3C2410_IISMOD_RXMODE )) { ++ iiscon &= ! S3C2410_IISCON_IISEN; ++ writel(iiscon,S3C24XX_VA_IIS+S3C2410_IISCON); ++ } ++} ++ ++#ifdef CONFIG_PM ++static int s3c24xx_i2s_suspend(struct platform_device *dev, ++ struct snd_soc_cpu_dai *dai) ++{ ++} ++ ++static int s3c24xx_i2s_resume(struct platform_device *pdev, ++ struct snd_soc_cpu_dai *dai) ++{ ++} ++ ++#else ++#define s3c24xx_i2s_suspend NULL ++#define s3c24xx_i2s_resume NULL ++#endif ++ ++/* s3c24xx I2S sysclock is always 384 FS */ ++static unsigned int s3c24xx_i2s_config_sysclk(struct snd_soc_cpu_dai *iface, ++ struct snd_soc_clock_info *info, unsigned int clk) ++{ ++ return info->rate * 384; ++} ++ ++struct snd_soc_cpu_dai s3c24xx_i2s_dai = { ++ .name = "s3c24xx-i2s", ++ .id = 0, ++ .type = SND_SOC_DAI_I2S, ++ .suspend = s3c24xx_i2s_suspend, ++ .resume = s3c24xx_i2s_resume, ++ .config_sysclk = s3c24xx_i2s_config_sysclk, ++ .playback = { ++ .channels_min = 2, ++ .channels_max = 2,}, ++ .capture = { ++ .channels_min = 2, ++ .channels_max = 2,}, ++ .ops = { ++ .startup = s3c24xx_i2s_startup, ++ .shutdown = s3c24xx_i2s_shutdown, ++ .trigger = s3c24xx_i2s_trigger, ++ .hw_params = s3c24xx_i2s_hw_params,}, ++ .caps = { ++ .num_modes = ARRAY_SIZE(s3c24xx_i2s_modes), ++ .mode = s3c24xx_i2s_modes,}, ++}; ++ ++EXPORT_SYMBOL_GPL(s3c24xx_i2s_dai); ++ ++/* Module information */ ++MODULE_AUTHOR("Graeme Gregory, graeme.gregory@wolfsonmicro.com, www.wolfsonmicro.com"); ++MODULE_DESCRIPTION("s3c24xx I2S SoC Interface"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/s3c24xx/s3c24xx-pcm.c +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/s3c24xx/s3c24xx-pcm.c +@@ -0,0 +1,362 @@ ++/* ++ * s3c24xx-pcm.c -- ALSA Soc Audio Layer ++ * ++ * Copyright 2005 Wolfson Microelectronics PLC. ++ * Author: Graeme Gregory ++ * graeme.gregory@wolfsonmicro.com or linux@wolfsonmicro.com ++ * ++ * 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. ++ * ++ * Revision history ++ * 10th Nov 2006 Initial version. ++ */ ++ ++#include <linux/module.h> ++#include <linux/init.h> ++#include <linux/platform_device.h> ++#include <linux/slab.h> ++#include <linux/dma-mapping.h> ++ ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/pcm.h> ++#include <sound/pcm_params.h> ++#include <sound/soc.h> ++ ++#include <asm/dma.h> ++#include <asm/io.h> ++#include <asm/hardware.h> ++#include <asm/arch/dma.h> ++#include <asm/arch/audio.h> ++ ++#include "s3c24xx-pcm.h" ++ ++static const struct snd_pcm_hardware s3c24xx_pcm_hardware = { ++ .info = SNDRV_PCM_INFO_MMAP | ++ SNDRV_PCM_INFO_MMAP_VALID | ++ SNDRV_PCM_INFO_INTERLEAVED | ++ SNDRV_PCM_INFO_PAUSE | ++ SNDRV_PCM_INFO_RESUME, ++ .formats = SNDRV_PCM_FMTBIT_S16_LE, ++ .period_bytes_min = 32, ++ .period_bytes_max = 8192, ++ .periods_min = 1, ++ .periods_max = 8192, ++ .buffer_bytes_max = 256 * 1024, ++ .fifo_size = 32, ++}; ++ ++struct s3c24xx_runtime_data { ++ dma_addr_t dma_buffer; ++ dma_addr_t dma_buffer_end; ++ size_t period_size; ++ dma_addr_t period_ptr; ++ s3c24xx_pcm_dma_params_t *params; ++}; ++ ++/* Move the pointer onto the next period, dealing with wrap around. ++ */ ++void static next_period(struct s3c24xx_runtime_data *prtd) ++{ ++ prtd->period_ptr+=prtd->period_size; ++ if(prtd->period_ptr>=prtd->dma_buffer_end) ++ { ++ prtd->period_ptr=prtd->dma_buffer; ++ } ++} ++ ++void s3c24xx_audio_buffdone(struct s3c2410_dma_chan *channel, ++ void *dev_id, int size, ++ enum s3c2410_dma_buffresult result) ++{ ++ struct snd_pcm_substream *substream = dev_id; ++ struct s3c24xx_runtime_data *prtd = substream->runtime->private_data; ++ ++ if(result==S3C2410_RES_OK) ++ { ++ next_period(prtd); ++ s3c2410_dma_enqueue(prtd->params->channel, substream, prtd->period_ptr, prtd->period_size); ++ } ++ snd_pcm_period_elapsed(substream); ++ ++} ++ ++static int s3c24xx_pcm_hw_params(struct snd_pcm_substream *substream, ++ struct snd_pcm_hw_params *params) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct s3c24xx_runtime_data *prtd = runtime->private_data; ++ struct snd_soc_pcm_runtime *rtd = substream->private_data; ++ s3c24xx_pcm_dma_params_t *dma = rtd->cpu_dai->dma_data; ++ int ret; ++ ++ printk("Entered s3c24xx hw_params\n"); ++ ++ snd_pcm_set_runtime_buffer(substream, &substream->dma_buffer); ++ runtime->dma_bytes = params_buffer_bytes(params); ++ ++ prtd->params=dma; ++ if(ret=s3c2410_dma_request(prtd->params->channel, ++ prtd->params->client,NULL)) ++ { ++ printk("Failed to get dma channel %d for %s\n",prtd->params->channel, ++ prtd->params->client->name); ++ return ret; ++ } ++ ++ //s3c2410_dma_setflags(prtd->params->channel,S3C2410_DMAF_AUTOSTART); ++ if(substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ { ++ s3c2410_dma_devconfig(prtd->params->channel, S3C2410_DMASRC_MEM, ++ S3C2410_DISRCC_INC | S3C2410_DISRCC_APB, ++ prtd->params->dma_addr); ++ } ++ else ++ { ++ s3c2410_dma_devconfig(prtd->params->channel, S3C2410_DMASRC_HW, ++ S3C2410_DISRCC_INC | S3C2410_DISRCC_APB, ++ prtd->params->dma_addr); ++ } ++ ++ s3c2410_dma_config(prtd->params->channel,2,S3C2410_DCON_HANDSHAKE); ++ ++ s3c2410_dma_set_buffdone_fn(prtd->params->channel, s3c24xx_audio_buffdone); ++ ++ prtd->dma_buffer = runtime->dma_addr; ++ prtd->dma_buffer_end = runtime->dma_addr + runtime->dma_bytes; ++ prtd->period_size = params_period_bytes(params); ++ ++ return 0; ++} ++ ++static int s3c24xx_pcm_hw_free(struct snd_pcm_substream *substream) ++{ ++ struct s3c24xx_runtime_data *prtd = substream->runtime->private_data; ++ ++ printk("Entered s3c24xx hw_free\n"); ++ ++ return 0; ++} ++ ++static int s3c24xx_pcm_prepare(struct snd_pcm_substream *substream) ++{ ++ struct s3c24xx_runtime_data *prtd = substream->runtime->private_data; ++ ++ printk("Entered s3c24xx prepare\n"); ++ ++ /* Set the period that is to be queued in DMA */ ++ prtd->period_ptr = prtd->dma_buffer; ++ ++ /* queue the first period */ ++ s3c2410_dma_enqueue(prtd->params->channel, substream, prtd->period_ptr, prtd->period_size); ++ ++ /* Move to next period to be queued */ ++ next_period(prtd); ++ ++ /* queue the second buffer */ ++ s3c2410_dma_enqueue(prtd->params->channel, substream, prtd->period_ptr, prtd->period_size); ++ ++ ++ return 0; ++} ++ ++static int s3c24xx_pcm_trigger(struct snd_pcm_substream *substream, int cmd) ++{ ++ struct s3c24xx_runtime_data *prtd = substream->runtime->private_data; ++ int ret = 0; ++ ++ printk("Entered s3c24xx trigger\n"); ++ switch (cmd) { ++ case SNDRV_PCM_TRIGGER_START: ++ case SNDRV_PCM_TRIGGER_RESUME: ++ case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: ++ s3c2410_dma_ctrl(prtd->params->channel, S3C2410_DMAOP_START); ++ break; ++ ++ case SNDRV_PCM_TRIGGER_STOP: ++ case SNDRV_PCM_TRIGGER_SUSPEND: ++ case SNDRV_PCM_TRIGGER_PAUSE_PUSH: ++ s3c2410_dma_ctrl(prtd->params->channel, S3C2410_DMAOP_STOP); ++ break; ++ ++ default: ++ ret = -EINVAL; ++ } ++ ++ return ret; ++} ++ ++static snd_pcm_uframes_t s3c24xx_pcm_pointer(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct s3c24xx_runtime_data *prtd = runtime->private_data; ++ dma_addr_t dst,src; ++ snd_pcm_uframes_t x; ++ ++ printk("Entered s3c24xx pointer\n"); ++ ++ s3c2410_dma_getposition(prtd->params->channel, &src, &dst); ++ ++ printk("DMA Position: %lx, %lx\n", src, dst); ++ ++ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ++ { ++ x = bytes_to_frames(runtime, src - prtd->dma_buffer); ++ } ++ else ++ { ++ x = bytes_to_frames(runtime, dst - prtd->dma_buffer); ++ } ++ ++ if (x == runtime->buffer_size) ++ x=0; ++ return x; ++ ++} ++ ++static int s3c24xx_pcm_open(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct s3c24xx_runtime_data *prtd; ++ int ret; ++ ++ printk("Entered s3c24xx open\n"); ++ ++ snd_soc_set_runtime_hwparams(substream, &s3c24xx_pcm_hardware); ++ ++ if((prtd = kzalloc(sizeof(struct s3c24xx_runtime_data), GFP_KERNEL)) == NULL) ++ { ++ ret = -ENOMEM; ++ goto out; ++ } ++ ++ runtime->private_data = prtd; ++ return 0; ++ ++out: ++ return ret; ++} ++ ++static int s3c24xx_pcm_close(struct snd_pcm_substream *substream) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ struct s3c24xx_runtime_data *prtd = runtime->private_data; ++ ++ printk("Entered s3c24xx close\n"); ++ ++ s3c2410_dma_free(prtd->params->channel, prtd->params->client); ++ ++ return 0; ++} ++ ++static int ++s3c24xx_pcm_mmap(struct snd_pcm_substream *substream, struct vm_area_struct *vma) ++{ ++ struct snd_pcm_runtime *runtime = substream->runtime; ++ ++ printk("Entered s3c24xx mmap\n"); ++ ++ return dma_mmap_writecombine(substream->pcm->card->dev, vma, ++ runtime->dma_area, ++ runtime->dma_addr, ++ runtime->dma_bytes); ++} ++ ++struct snd_pcm_ops s3c24xx_pcm_ops = { ++ .open = s3c24xx_pcm_open, ++ .close = s3c24xx_pcm_close, ++ .ioctl = snd_pcm_lib_ioctl, ++ .hw_params = s3c24xx_pcm_hw_params, ++ .hw_free = s3c24xx_pcm_hw_free, ++ .prepare = s3c24xx_pcm_prepare, ++ .trigger = s3c24xx_pcm_trigger, ++ .pointer = s3c24xx_pcm_pointer, ++ .mmap = s3c24xx_pcm_mmap, ++}; ++ ++static int s3c24xx_pcm_preallocate_dma_buffer(struct snd_pcm *pcm, int stream) ++{ ++ struct snd_pcm_substream *substream = pcm->streams[stream].substream; ++ struct snd_dma_buffer *buf = &substream->dma_buffer; ++ size_t size = s3c24xx_pcm_hardware.buffer_bytes_max; ++ ++ printk("Entered s3c24xx preaccolate_dma_buffer\n"); ++ ++ buf->dev.type = SNDRV_DMA_TYPE_DEV; ++ buf->dev.dev = pcm->card->dev; ++ buf->private_data = NULL; ++ buf->area = dma_alloc_writecombine(pcm->card->dev, size, ++ &buf->addr, GFP_KERNEL); ++ if (!buf->area) ++ return -ENOMEM; ++ buf->bytes = size; ++ return 0; ++} ++ ++static void s3c24xx_pcm_free_dma_buffers(struct snd_pcm *pcm) ++{ ++ struct snd_pcm_substream *substream; ++ struct snd_dma_buffer *buf; ++ int stream; ++ ++ printk("Entered s3c24xx free_dma_buffers\n"); ++ ++ for (stream = 0; stream < 2; stream++) { ++ substream = pcm->streams[stream].substream; ++ if (!substream) ++ continue; ++ ++ buf = &substream->dma_buffer; ++ if (!buf->area) ++ continue; ++ ++ dma_free_writecombine(pcm->card->dev, buf->bytes, ++ buf->area, buf->addr); ++ buf->area = NULL; ++ } ++} ++ ++static u64 s3c24xx_pcm_dmamask = 0xffffffff; ++ ++int s3c24xx_pcm_new(struct snd_card *card, struct snd_soc_codec_dai *dai, ++ struct snd_pcm *pcm) ++{ ++ int ret = 0; ++ ++ printk("Entered s3c24xx new\n"); ++ ++ if (!card->dev->dma_mask) ++ card->dev->dma_mask = &s3c24xx_pcm_dmamask; ++ if (!card->dev->coherent_dma_mask) ++ card->dev->coherent_dma_mask = 0xffffffff; ++ ++ if (dai->playback.channels_min) { ++ ret = s3c24xx_pcm_preallocate_dma_buffer(pcm, SNDRV_PCM_STREAM_PLAYBACK); ++ if (ret) ++ goto out; ++ } ++ ++ if (dai->capture.channels_min) { ++ ret = s3c24xx_pcm_preallocate_dma_buffer(pcm, SNDRV_PCM_STREAM_CAPTURE); ++ if (ret) ++ goto out; ++ } ++ out: ++ return ret; ++} ++ ++struct snd_soc_platform s3c24xx_soc_platform = { ++ .name = "s3c24xx-audio", ++ .pcm_ops = &s3c24xx_pcm_ops, ++ .pcm_new = s3c24xx_pcm_new, ++ .pcm_free = s3c24xx_pcm_free_dma_buffers, ++}; ++ ++EXPORT_SYMBOL_GPL(s3c24xx_soc_platform); ++ ++MODULE_AUTHOR("Graeme Gregory"); ++MODULE_DESCRIPTION("Samsung S3C24XX PCM DMA module"); ++MODULE_LICENSE("GPL"); +Index: linux-2.6-pxa-new/sound/soc/s3c24xx/Kconfig +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/s3c24xx/Kconfig +@@ -0,0 +1,26 @@ ++menu "SoC Audio for the Atmel AT91" ++ ++config SND_S3C24XX_SOC ++ tristate "SoC Audio for the Samsung S3C24xx System-on-Chip" ++ depends on ARCH_S3C2410 && SND ++ select SND_PCM ++ help ++ Say Y or M if you want to add support for codecs attached to ++ the Samsung S3C24xx. ++ ++config SND_S3C24XX_SOC_I2S ++ tristate ++ ++config SND_S3C24XX_SOC_AC97 ++ tristate ++ ++# graeme - add mach dep ++config SND_S3C24XX_SOC_SMDK2440 ++ tristate "SoC I2S Audio support for SMDK2440" ++ depends on SND_S3C24XX_SOC ++ select SND_S3C24XX_SOC_I2S ++ select SND_SOC_UDA1380 ++ help ++ Say Y if you want to add support for SoC audio on ++ ++endmenu +Index: linux-2.6-pxa-new/sound/soc/s3c24xx/Makefile +=================================================================== +--- /dev/null ++++ linux-2.6-pxa-new/sound/soc/s3c24xx/Makefile +@@ -0,0 +1,11 @@ ++# S3C24xx Platform Support ++snd-soc-s3c24xx-objs := s3c24xx-pcm.o ++snd-soc-at91-i2s-objs := s3c24xx-i2s.o ++ ++obj-$(CONFIG_SND_S3C24XX_SOC) += snd-soc-s3c24xx.o ++obj-$(CONFIG_SND_S3C24XX_SOC_I2S) += snd-soc-s3c24xx-i2s.o ++ ++# S3C24xx Machine Support ++snd-soc-smdk2440-uda1380-objs := smdk2440_uda1380.o ++ ++obj-$(CONFIG_SND_S3C24XX_SOC_SMDK2440) += snd-soc-smdk2440-uda1380.o diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/config-nr-tty-devices.patch b/packages/linux/logicpd-pxa270-2.6.19.2/config-nr-tty-devices.patch new file mode 100644 index 0000000000..870ba5f643 --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/config-nr-tty-devices.patch @@ -0,0 +1,1068 @@ +diff -urN oldtree/drivers/char/Kconfig newtree/drivers/char/Kconfig +--- oldtree/drivers/char/Kconfig 2006-03-27 13:28:15.000000000 -0500 ++++ newtree/drivers/char/Kconfig 2006-03-28 10:37:24.195120750 -0500 +@@ -57,6 +57,18 @@ + + If unsure, say Y. + ++config NR_TTY_DEVICES ++ int "Maximum tty device number" ++ depends on VT ++ default 63 ++ ---help--- ++ This is the highest numbered device created in /dev. You will actually have ++ NR_TTY_DEVICES+1 devices in /dev. The default is 63, which will result in ++ 64 /dev entries. The lowest number you can set is 11, anything below that, ++ and it will default to 11. 63 is also the upper limit so we don't overrun ++ the serial consoles. ++ ++ + config HW_CONSOLE + bool + depends on VT && !S390 && !UML +diff -urN oldtree/drivers/char/Kconfig.orig newtree/drivers/char/Kconfig.orig +--- oldtree/drivers/char/Kconfig.orig 1969-12-31 19:00:00.000000000 -0500 ++++ newtree/drivers/char/Kconfig.orig 2006-03-28 10:37:24.207121500 -0500 +@@ -0,0 +1,1017 @@ ++# ++# Character device configuration ++# ++ ++menu "Character devices" ++ ++config VT ++ bool "Virtual terminal" if EMBEDDED ++ select INPUT ++ default y if !VIOCONS ++ ---help--- ++ If you say Y here, you will get support for terminal devices with ++ display and keyboard devices. These are called "virtual" because you ++ can run several virtual terminals (also called virtual consoles) on ++ one physical terminal. This is rather useful, for example one ++ virtual terminal can collect system messages and warnings, another ++ one can be used for a text-mode user session, and a third could run ++ an X session, all in parallel. Switching between virtual terminals ++ is done with certain key combinations, usually Alt-<function key>. ++ ++ The setterm command ("man setterm") can be used to change the ++ properties (such as colors or beeping) of a virtual terminal. The ++ man page console_codes(4) ("man console_codes") contains the special ++ character sequences that can be used to change those properties ++ directly. The fonts used on virtual terminals can be changed with ++ the setfont ("man setfont") command and the key bindings are defined ++ with the loadkeys ("man loadkeys") command. ++ ++ You need at least one virtual terminal device in order to make use ++ of your keyboard and monitor. Therefore, only people configuring an ++ embedded system would want to say N here in order to save some ++ memory; the only way to log into such a system is then via a serial ++ or network connection. ++ ++ If unsure, say Y, or else you won't be able to do much with your new ++ shiny Linux system :-) ++ ++config VT_CONSOLE ++ bool "Support for console on virtual terminal" if EMBEDDED ++ depends on VT ++ default y ++ ---help--- ++ The system console is the device which receives all kernel messages ++ and warnings and which allows logins in single user mode. If you ++ answer Y here, a virtual terminal (the device used to interact with ++ a physical terminal) can be used as system console. This is the most ++ common mode of operations, so you should say Y here unless you want ++ the kernel messages be output only to a serial port (in which case ++ you should say Y to "Console on serial port", below). ++ ++ If you do say Y here, by default the currently visible virtual ++ terminal (/dev/tty0) will be used as system console. You can change ++ that with a kernel command line option such as "console=tty3" which ++ would use the third virtual terminal as system console. (Try "man ++ bootparam" or see the documentation of your boot loader (lilo or ++ loadlin) about how to pass options to the kernel at boot time.) ++ ++ If unsure, say Y. ++ ++config HW_CONSOLE ++ bool ++ depends on VT && !S390 && !UML ++ default y ++ ++config SERIAL_NONSTANDARD ++ bool "Non-standard serial port support" ++ ---help--- ++ Say Y here if you have any non-standard serial boards -- boards ++ which aren't supported using the standard "dumb" serial driver. ++ This includes intelligent serial boards such as Cyclades, ++ Digiboards, etc. These are usually used for systems that need many ++ serial ports because they serve many terminals or dial-in ++ connections. ++ ++ Note that the answer to this question won't directly affect the ++ kernel: saying N will just cause the configurator to skip all ++ the questions about non-standard serial boards. ++ ++ Most people can say N here. ++ ++config COMPUTONE ++ tristate "Computone IntelliPort Plus serial support" ++ depends on SERIAL_NONSTANDARD && BROKEN_ON_SMP ++ ---help--- ++ This driver supports the entire family of Intelliport II/Plus ++ controllers with the exception of the MicroChannel controllers and ++ products previous to the Intelliport II. These are multiport cards, ++ which give you many serial ports. You would need something like this ++ to connect more than two modems to your Linux box, for instance in ++ order to become a dial-in server. If you have a card like that, say ++ Y here and read <file:Documentation/computone.txt>. ++ ++ To compile this driver as modules, choose M here: the ++ modules will be called ip2 and ip2main. ++ ++config ROCKETPORT ++ tristate "Comtrol RocketPort support" ++ depends on SERIAL_NONSTANDARD ++ help ++ This driver supports Comtrol RocketPort and RocketModem PCI boards. ++ These boards provide 2, 4, 8, 16, or 32 high-speed serial ports or ++ modems. For information about the RocketPort/RocketModem boards ++ and this driver read <file:Documentation/rocket.txt>. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called rocket. ++ ++ If you want to compile this driver into the kernel, say Y here. If ++ you don't have a Comtrol RocketPort/RocketModem card installed, say N. ++ ++config CYCLADES ++ tristate "Cyclades async mux support" ++ depends on SERIAL_NONSTANDARD ++ ---help--- ++ This driver supports Cyclades Z and Y multiserial boards. ++ You would need something like this to connect more than two modems to ++ your Linux box, for instance in order to become a dial-in server. ++ ++ For information about the Cyclades-Z card, read ++ <file:drivers/char/README.cycladesZ>. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called cyclades. ++ ++ If you haven't heard about it, it's safe to say N. ++ ++config CYZ_INTR ++ bool "Cyclades-Z interrupt mode operation (EXPERIMENTAL)" ++ depends on EXPERIMENTAL && CYCLADES ++ help ++ The Cyclades-Z family of multiport cards allows 2 (two) driver op ++ modes: polling and interrupt. In polling mode, the driver will check ++ the status of the Cyclades-Z ports every certain amount of time ++ (which is called polling cycle and is configurable). In interrupt ++ mode, it will use an interrupt line (IRQ) in order to check the ++ status of the Cyclades-Z ports. The default op mode is polling. If ++ unsure, say N. ++ ++config DIGIEPCA ++ tristate "Digiboard Intelligent Async Support" ++ depends on SERIAL_NONSTANDARD ++ ---help--- ++ This is a driver for Digi International's Xx, Xeve, and Xem series ++ of cards which provide multiple serial ports. You would need ++ something like this to connect more than two modems to your Linux ++ box, for instance in order to become a dial-in server. This driver ++ supports the original PC (ISA) boards as well as PCI, and EISA. If ++ you have a card like this, say Y here and read the file ++ <file:Documentation/digiepca.txt>. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called epca. ++ ++config ESPSERIAL ++ tristate "Hayes ESP serial port support" ++ depends on SERIAL_NONSTANDARD && ISA && BROKEN_ON_SMP && ISA_DMA_API ++ help ++ This is a driver which supports Hayes ESP serial ports. Both single ++ port cards and multiport cards are supported. Make sure to read ++ <file:Documentation/hayes-esp.txt>. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called esp. ++ ++ If unsure, say N. ++ ++config MOXA_INTELLIO ++ tristate "Moxa Intellio support" ++ depends on SERIAL_NONSTANDARD && BROKEN_ON_SMP ++ help ++ Say Y here if you have a Moxa Intellio multiport serial card. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called moxa. ++ ++config MOXA_SMARTIO ++ tristate "Moxa SmartIO support" ++ depends on SERIAL_NONSTANDARD ++ help ++ Say Y here if you have a Moxa SmartIO multiport serial card. ++ ++ This driver can also be built as a module ( = code which can be ++ inserted in and removed from the running kernel whenever you want). ++ The module will be called mxser. If you want to do that, say M ++ here. ++ ++config ISI ++ tristate "Multi-Tech multiport card support (EXPERIMENTAL)" ++ depends on SERIAL_NONSTANDARD ++ help ++ This is a driver for the Multi-Tech cards which provide several ++ serial ports. The driver is experimental and can currently only be ++ built as a module. The module will be called isicom. ++ If you want to do that, choose M here. ++ ++config SYNCLINK ++ tristate "Microgate SyncLink card support" ++ depends on SERIAL_NONSTANDARD && PCI && ISA_DMA_API ++ help ++ Provides support for the SyncLink ISA and PCI multiprotocol serial ++ adapters. These adapters support asynchronous and HDLC bit ++ synchronous communication up to 10Mbps (PCI adapter). ++ ++ This driver can only be built as a module ( = code which can be ++ inserted in and removed from the running kernel whenever you want). ++ The module will be called synclink. If you want to do that, say M ++ here. ++ ++config SYNCLINKMP ++ tristate "SyncLink Multiport support" ++ depends on SERIAL_NONSTANDARD ++ help ++ Enable support for the SyncLink Multiport (2 or 4 ports) ++ serial adapter, running asynchronous and HDLC communications up ++ to 2.048Mbps. Each ports is independently selectable for ++ RS-232, V.35, RS-449, RS-530, and X.21 ++ ++ This driver may be built as a module ( = code which can be ++ inserted in and removed from the running kernel whenever you want). ++ The module will be called synclinkmp. If you want to do that, say M ++ here. ++ ++config N_HDLC ++ tristate "HDLC line discipline support" ++ depends on SERIAL_NONSTANDARD ++ help ++ Allows synchronous HDLC communications with tty device drivers that ++ support synchronous HDLC such as the Microgate SyncLink adapter. ++ ++ This driver can only be built as a module ( = code which can be ++ inserted in and removed from the running kernel whenever you want). ++ The module will be called n_hdlc. If you want to do that, say M ++ here. ++ ++config RISCOM8 ++ tristate "SDL RISCom/8 card support" ++ depends on SERIAL_NONSTANDARD && BROKEN_ON_SMP ++ help ++ This is a driver for the SDL Communications RISCom/8 multiport card, ++ which gives you many serial ports. You would need something like ++ this to connect more than two modems to your Linux box, for instance ++ in order to become a dial-in server. If you have a card like that, ++ say Y here and read the file <file:Documentation/riscom8.txt>. ++ ++ Also it's possible to say M here and compile this driver as kernel ++ loadable module; the module will be called riscom8. ++ ++config SPECIALIX ++ tristate "Specialix IO8+ card support" ++ depends on SERIAL_NONSTANDARD ++ help ++ This is a driver for the Specialix IO8+ multiport card (both the ++ ISA and the PCI version) which gives you many serial ports. You ++ would need something like this to connect more than two modems to ++ your Linux box, for instance in order to become a dial-in server. ++ ++ If you have a card like that, say Y here and read the file ++ <file:Documentation/specialix.txt>. Also it's possible to say M here ++ and compile this driver as kernel loadable module which will be ++ called specialix. ++ ++config SPECIALIX_RTSCTS ++ bool "Specialix DTR/RTS pin is RTS" ++ depends on SPECIALIX ++ help ++ The Specialix IO8+ card can only support either RTS or DTR. If you ++ say N here, the driver will use the pin as "DTR" when the tty is in ++ software handshake mode. If you say Y here or hardware handshake is ++ on, it will always be RTS. Read the file ++ <file:Documentation/specialix.txt> for more information. ++ ++config SX ++ tristate "Specialix SX (and SI) card support" ++ depends on SERIAL_NONSTANDARD ++ help ++ This is a driver for the SX and SI multiport serial cards. ++ Please read the file <file:Documentation/sx.txt> for details. ++ ++ This driver can only be built as a module ( = code which can be ++ inserted in and removed from the running kernel whenever you want). ++ The module will be called sx. If you want to do that, say M here. ++ ++config RIO ++ tristate "Specialix RIO system support" ++ depends on SERIAL_NONSTANDARD && BROKEN_ON_SMP ++ help ++ This is a driver for the Specialix RIO, a smart serial card which ++ drives an outboard box that can support up to 128 ports. Product ++ information is at <http://www.perle.com/support/documentation.html#multiport>. ++ There are both ISA and PCI versions. ++ ++config RIO_OLDPCI ++ bool "Support really old RIO/PCI cards" ++ depends on RIO ++ help ++ Older RIO PCI cards need some initialization-time configuration to ++ determine the IRQ and some control addresses. If you have a RIO and ++ this doesn't seem to work, try setting this to Y. ++ ++config STALDRV ++ bool "Stallion multiport serial support" ++ depends on SERIAL_NONSTANDARD ++ help ++ Stallion cards give you many serial ports. You would need something ++ like this to connect more than two modems to your Linux box, for ++ instance in order to become a dial-in server. If you say Y here, ++ you will be asked for your specific card model in the next ++ questions. Make sure to read <file:Documentation/stallion.txt> in ++ this case. If you have never heard about all this, it's safe to ++ say N. ++ ++config STALLION ++ tristate "Stallion EasyIO or EC8/32 support" ++ depends on STALDRV && BROKEN_ON_SMP ++ help ++ If you have an EasyIO or EasyConnection 8/32 multiport Stallion ++ card, then this is for you; say Y. Make sure to read ++ <file:Documentation/stallion.txt>. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called stallion. ++ ++config ISTALLION ++ tristate "Stallion EC8/64, ONboard, Brumby support" ++ depends on STALDRV && BROKEN_ON_SMP ++ help ++ If you have an EasyConnection 8/64, ONboard, Brumby or Stallion ++ serial multiport card, say Y here. Make sure to read ++ <file:Documentation/stallion.txt>. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called istallion. ++ ++config AU1000_UART ++ bool "Enable Au1000 UART Support" ++ depends on SERIAL_NONSTANDARD && MIPS ++ help ++ If you have an Alchemy AU1000 processor (MIPS based) and you want ++ to use serial ports, say Y. Otherwise, say N. ++ ++config AU1000_SERIAL_CONSOLE ++ bool "Enable Au1000 serial console" ++ depends on AU1000_UART ++ help ++ If you have an Alchemy AU1000 processor (MIPS based) and you want ++ to use a console on a serial port, say Y. Otherwise, say N. ++ ++config QTRONIX_KEYBOARD ++ bool "Enable Qtronix 990P Keyboard Support" ++ depends on IT8712 ++ help ++ Images of Qtronix keyboards are at ++ <http://www.qtronix.com/keyboard.html>. ++ ++config IT8172_CIR ++ bool ++ depends on QTRONIX_KEYBOARD ++ default y ++ ++config IT8172_SCR0 ++ bool "Enable Smart Card Reader 0 Support " ++ depends on IT8712 ++ help ++ Say Y here to support smart-card reader 0 (SCR0) on the Integrated ++ Technology Express, Inc. ITE8172 SBC. Vendor page at ++ <http://www.ite.com.tw/ia/brief_it8172bsp.htm>; picture of the ++ board at <http://www.mvista.com/partners/semiconductor/ite.html>. ++ ++config IT8172_SCR1 ++ bool "Enable Smart Card Reader 1 Support " ++ depends on IT8712 ++ help ++ Say Y here to support smart-card reader 1 (SCR1) on the Integrated ++ Technology Express, Inc. ITE8172 SBC. Vendor page at ++ <http://www.ite.com.tw/ia/brief_it8172bsp.htm>; picture of the ++ board at <http://www.mvista.com/partners/semiconductor/ite.html>. ++ ++config A2232 ++ tristate "Commodore A2232 serial support (EXPERIMENTAL)" ++ depends on EXPERIMENTAL && ZORRO && BROKEN_ON_SMP ++ ---help--- ++ This option supports the 2232 7-port serial card shipped with the ++ Amiga 2000 and other Zorro-bus machines, dating from 1989. At ++ a max of 19,200 bps, the ports are served by a 6551 ACIA UART chip ++ each, plus a 8520 CIA, and a master 6502 CPU and buffer as well. The ++ ports were connected with 8 pin DIN connectors on the card bracket, ++ for which 8 pin to DB25 adapters were supplied. The card also had ++ jumpers internally to toggle various pinning configurations. ++ ++ This driver can be built as a module; but then "generic_serial" ++ will also be built as a module. This has to be loaded before ++ "ser_a2232". If you want to do this, answer M here. ++ ++config SGI_SNSC ++ bool "SGI Altix system controller communication support" ++ depends on (IA64_SGI_SN2 || IA64_GENERIC) ++ help ++ If you have an SGI Altix and you want to enable system ++ controller communication from user space (you want this!), ++ say Y. Otherwise, say N. ++ ++config SGI_TIOCX ++ bool "SGI TIO CX driver support" ++ depends on (IA64_SGI_SN2 || IA64_GENERIC) ++ help ++ If you have an SGI Altix and you have fpga devices attached ++ to your TIO, say Y here, otherwise say N. ++ ++config SGI_MBCS ++ tristate "SGI FPGA Core Services driver support" ++ depends on SGI_TIOCX ++ help ++ If you have an SGI Altix with an attached SABrick ++ say Y or M here, otherwise say N. ++ ++source "drivers/serial/Kconfig" ++ ++config UNIX98_PTYS ++ bool "Unix98 PTY support" if EMBEDDED ++ default y ++ ---help--- ++ A pseudo terminal (PTY) is a software device consisting of two ++ halves: a master and a slave. The slave device behaves identical to ++ a physical terminal; the master device is used by a process to ++ read data from and write data to the slave, thereby emulating a ++ terminal. Typical programs for the master side are telnet servers ++ and xterms. ++ ++ Linux has traditionally used the BSD-like names /dev/ptyxx for ++ masters and /dev/ttyxx for slaves of pseudo terminals. This scheme ++ has a number of problems. The GNU C library glibc 2.1 and later, ++ however, supports the Unix98 naming standard: in order to acquire a ++ pseudo terminal, a process opens /dev/ptmx; the number of the pseudo ++ terminal is then made available to the process and the pseudo ++ terminal slave can be accessed as /dev/pts/<number>. What was ++ traditionally /dev/ttyp2 will then be /dev/pts/2, for example. ++ ++ All modern Linux systems use the Unix98 ptys. Say Y unless ++ you're on an embedded system and want to conserve memory. ++ ++config LEGACY_PTYS ++ bool "Legacy (BSD) PTY support" ++ default y ++ ---help--- ++ A pseudo terminal (PTY) is a software device consisting of two ++ halves: a master and a slave. The slave device behaves identical to ++ a physical terminal; the master device is used by a process to ++ read data from and write data to the slave, thereby emulating a ++ terminal. Typical programs for the master side are telnet servers ++ and xterms. ++ ++ Linux has traditionally used the BSD-like names /dev/ptyxx ++ for masters and /dev/ttyxx for slaves of pseudo ++ terminals. This scheme has a number of problems, including ++ security. This option enables these legacy devices; on most ++ systems, it is safe to say N. ++ ++ ++config LEGACY_PTY_COUNT ++ int "Maximum number of legacy PTY in use" ++ depends on LEGACY_PTYS ++ range 1 256 ++ default "256" ++ ---help--- ++ The maximum number of legacy PTYs that can be used at any one time. ++ The default is 256, and should be more than enough. Embedded ++ systems may want to reduce this to save memory. ++ ++ When not in use, each legacy PTY occupies 12 bytes on 32-bit ++ architectures and 24 bytes on 64-bit architectures. ++ ++config PRINTER ++ tristate "Parallel printer support" ++ depends on PARPORT ++ ---help--- ++ If you intend to attach a printer to the parallel port of your Linux ++ box (as opposed to using a serial printer; if the connector at the ++ printer has 9 or 25 holes ["female"], then it's serial), say Y. ++ Also read the Printing-HOWTO, available from ++ <http://www.tldp.org/docs.html#howto>. ++ ++ It is possible to share one parallel port among several devices ++ (e.g. printer and ZIP drive) and it is safe to compile the ++ corresponding drivers into the kernel. ++ ++ To compile this driver as a module, choose M here and read ++ <file:Documentation/parport.txt>. The module will be called lp. ++ ++ If you have several parallel ports, you can specify which ports to ++ use with the "lp" kernel command line option. (Try "man bootparam" ++ or see the documentation of your boot loader (lilo or loadlin) about ++ how to pass options to the kernel at boot time.) The syntax of the ++ "lp" command line option can be found in <file:drivers/char/lp.c>. ++ ++ If you have more than 8 printers, you need to increase the LP_NO ++ macro in lp.c and the PARPORT_MAX macro in parport.h. ++ ++config LP_CONSOLE ++ bool "Support for console on line printer" ++ depends on PRINTER ++ ---help--- ++ If you want kernel messages to be printed out as they occur, you ++ can have a console on the printer. This option adds support for ++ doing that; to actually get it to happen you need to pass the ++ option "console=lp0" to the kernel at boot time. ++ ++ If the printer is out of paper (or off, or unplugged, or too ++ busy..) the kernel will stall until the printer is ready again. ++ By defining CONSOLE_LP_STRICT to 0 (at your own risk) you ++ can make the kernel continue when this happens, ++ but it'll lose the kernel messages. ++ ++ If unsure, say N. ++ ++config PPDEV ++ tristate "Support for user-space parallel port device drivers" ++ depends on PARPORT ++ ---help--- ++ Saying Y to this adds support for /dev/parport device nodes. This ++ is needed for programs that want portable access to the parallel ++ port, for instance deviceid (which displays Plug-and-Play device ++ IDs). ++ ++ This is the parallel port equivalent of SCSI generic support (sg). ++ It is safe to say N to this -- it is not needed for normal printing ++ or parallel port CD-ROM/disk support. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called ppdev. ++ ++ If unsure, say N. ++ ++config TIPAR ++ tristate "Texas Instruments parallel link cable support" ++ depends on PARPORT ++ ---help--- ++ If you own a Texas Instruments graphing calculator and use a ++ parallel link cable, then you might be interested in this driver. ++ ++ If you enable this driver, you will be able to communicate with ++ your calculator through a set of device nodes under /dev. The ++ main advantage of this driver is that you don't have to be root ++ to use this precise link cable (depending on the permissions on ++ the device nodes, though). ++ ++ To compile this driver as a module, choose M here: the ++ module will be called tipar. ++ ++ If you don't know what a parallel link cable is or what a Texas ++ Instruments graphing calculator is, then you probably don't need this ++ driver. ++ ++ If unsure, say N. ++ ++config HVC_CONSOLE ++ bool "pSeries Hypervisor Virtual Console support" ++ depends on PPC_PSERIES ++ help ++ pSeries machines when partitioned support a hypervisor virtual ++ console. This driver allows each pSeries partition to have a console ++ which is accessed via the HMC. ++ ++config HVCS ++ tristate "IBM Hypervisor Virtual Console Server support" ++ depends on PPC_PSERIES ++ help ++ Partitionable IBM Power5 ppc64 machines allow hosting of ++ firmware virtual consoles from one Linux partition by ++ another Linux partition. This driver allows console data ++ from Linux partitions to be accessed through TTY device ++ interfaces in the device tree of a Linux partition running ++ this driver. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called hvcs.ko. Additionally, this module ++ will depend on arch specific APIs exported from hvcserver.ko ++ which will also be compiled when this driver is built as a ++ module. ++ ++source "drivers/char/ipmi/Kconfig" ++ ++source "drivers/char/watchdog/Kconfig" ++ ++config DS1620 ++ tristate "NetWinder thermometer support" ++ depends on ARCH_NETWINDER ++ help ++ Say Y here to include support for the thermal management hardware ++ found in the NetWinder. This driver allows the user to control the ++ temperature set points and to read the current temperature. ++ ++ It is also possible to say M here to build it as a module (ds1620) ++ It is recommended to be used on a NetWinder, but it is not a ++ necessity. ++ ++config NWBUTTON ++ tristate "NetWinder Button" ++ depends on ARCH_NETWINDER ++ ---help--- ++ If you say Y here and create a character device node /dev/nwbutton ++ with major and minor numbers 10 and 158 ("man mknod"), then every ++ time the orange button is pressed a number of times, the number of ++ times the button was pressed will be written to that device. ++ ++ This is most useful for applications, as yet unwritten, which ++ perform actions based on how many times the button is pressed in a ++ row. ++ ++ Do not hold the button down for too long, as the driver does not ++ alter the behaviour of the hardware reset circuitry attached to the ++ button; it will still execute a hard reset if the button is held ++ down for longer than approximately five seconds. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called nwbutton. ++ ++ Most people will answer Y to this question and "Reboot Using Button" ++ below to be able to initiate a system shutdown from the button. ++ ++config NWBUTTON_REBOOT ++ bool "Reboot Using Button" ++ depends on NWBUTTON ++ help ++ If you say Y here, then you will be able to initiate a system ++ shutdown and reboot by pressing the orange button a number of times. ++ The number of presses to initiate the shutdown is two by default, ++ but this can be altered by modifying the value of NUM_PRESSES_REBOOT ++ in nwbutton.h and recompiling the driver or, if you compile the ++ driver as a module, you can specify the number of presses at load ++ time with "insmod button reboot_count=<something>". ++ ++config NWFLASH ++ tristate "NetWinder flash support" ++ depends on ARCH_NETWINDER ++ ---help--- ++ If you say Y here and create a character device /dev/flash with ++ major 10 and minor 160 you can manipulate the flash ROM containing ++ the NetWinder firmware. Be careful as accidentally overwriting the ++ flash contents can render your computer unbootable. On no account ++ allow random users access to this device. :-) ++ ++ To compile this driver as a module, choose M here: the ++ module will be called nwflash. ++ ++ If you're not sure, say N. ++ ++config HW_RANDOM ++ tristate "Intel/AMD/VIA HW Random Number Generator support" ++ depends on (X86 || IA64) && PCI ++ ---help--- ++ This driver provides kernel-side support for the Random Number ++ Generator hardware found on Intel i8xx-based motherboards, ++ AMD 76x-based motherboards, and Via Nehemiah CPUs. ++ ++ Provides a character driver, used to read() entropy data. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called hw_random. ++ ++ If unsure, say N. ++ ++config NVRAM ++ tristate "/dev/nvram support" ++ depends on ATARI || X86 || ARM || GENERIC_NVRAM ++ ---help--- ++ If you say Y here and create a character special file /dev/nvram ++ with major number 10 and minor number 144 using mknod ("man mknod"), ++ you get read and write access to the extra bytes of non-volatile ++ memory in the real time clock (RTC), which is contained in every PC ++ and most Ataris. The actual number of bytes varies, depending on the ++ nvram in the system, but is usually 114 (128-14 for the RTC). ++ ++ This memory is conventionally called "CMOS RAM" on PCs and "NVRAM" ++ on Ataris. /dev/nvram may be used to view settings there, or to ++ change them (with some utility). It could also be used to frequently ++ save a few bits of very important data that may not be lost over ++ power-off and for which writing to disk is too insecure. Note ++ however that most NVRAM space in a PC belongs to the BIOS and you ++ should NEVER idly tamper with it. See Ralf Brown's interrupt list ++ for a guide to the use of CMOS bytes by your BIOS. ++ ++ On Atari machines, /dev/nvram is always configured and does not need ++ to be selected. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called nvram. ++ ++config RTC ++ tristate "Enhanced Real Time Clock Support" ++ depends on !PPC32 && !PARISC && !IA64 && !M68K && (!SPARC || PCI) ++ ---help--- ++ If you say Y here and create a character special file /dev/rtc with ++ major number 10 and minor number 135 using mknod ("man mknod"), you ++ will get access to the real time clock (or hardware clock) built ++ into your computer. ++ ++ Every PC has such a clock built in. It can be used to generate ++ signals from as low as 1Hz up to 8192Hz, and can also be used ++ as a 24 hour alarm. It reports status information via the file ++ /proc/driver/rtc and its behaviour is set by various ioctls on ++ /dev/rtc. ++ ++ If you run Linux on a multiprocessor machine and said Y to ++ "Symmetric Multi Processing" above, you should say Y here to read ++ and set the RTC in an SMP compatible fashion. ++ ++ If you think you have a use for such a device (such as periodic data ++ sampling), then say Y here, and read <file:Documentation/rtc.txt> ++ for details. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called rtc. ++ ++config SGI_DS1286 ++ tristate "SGI DS1286 RTC support" ++ depends on SGI_IP22 ++ help ++ If you say Y here and create a character special file /dev/rtc with ++ major number 10 and minor number 135 using mknod ("man mknod"), you ++ will get access to the real time clock built into your computer. ++ Every SGI has such a clock built in. It reports status information ++ via the file /proc/rtc and its behaviour is set by various ioctls on ++ /dev/rtc. ++ ++config SGI_IP27_RTC ++ bool "SGI M48T35 RTC support" ++ depends on SGI_IP27 ++ help ++ If you say Y here and create a character special file /dev/rtc with ++ major number 10 and minor number 135 using mknod ("man mknod"), you ++ will get access to the real time clock built into your computer. ++ Every SGI has such a clock built in. It reports status information ++ via the file /proc/rtc and its behaviour is set by various ioctls on ++ /dev/rtc. ++ ++config GEN_RTC ++ tristate "Generic /dev/rtc emulation" ++ depends on RTC!=y && !IA64 && !ARM && !M32R && !SPARC ++ ---help--- ++ If you say Y here and create a character special file /dev/rtc with ++ major number 10 and minor number 135 using mknod ("man mknod"), you ++ will get access to the real time clock (or hardware clock) built ++ into your computer. ++ ++ It reports status information via the file /proc/driver/rtc and its ++ behaviour is set by various ioctls on /dev/rtc. If you enable the ++ "extended RTC operation" below it will also provide an emulation ++ for RTC_UIE which is required by some programs and may improve ++ precision in some cases. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called genrtc. ++ ++config GEN_RTC_X ++ bool "Extended RTC operation" ++ depends on GEN_RTC ++ help ++ Provides an emulation for RTC_UIE which is required by some programs ++ and may improve precision of the generic RTC support in some cases. ++ ++config EFI_RTC ++ bool "EFI Real Time Clock Services" ++ depends on IA64 ++ ++config DS1302 ++ tristate "DS1302 RTC support" ++ depends on M32R && (PLAT_M32700UT || PLAT_OPSPUT) ++ help ++ If you say Y here and create a character special file /dev/rtc with ++ major number 121 and minor number 0 using mknod ("man mknod"), you ++ will get access to the real time clock (or hardware clock) built ++ into your computer. ++ ++config S3C2410_RTC ++ bool "S3C2410 RTC Driver" ++ depends on ARCH_S3C2410 ++ help ++ RTC (Realtime Clock) driver for the clock inbuilt into the ++ Samsung S3C2410. This can provide periodic interrupt rates ++ from 1Hz to 64Hz for user programs, and wakeup from Alarm. ++ ++config RTC_VR41XX ++ tristate "NEC VR4100 series Real Time Clock Support" ++ depends on CPU_VR41XX ++ ++config COBALT_LCD ++ bool "Support for Cobalt LCD" ++ depends on MIPS_COBALT ++ help ++ This option enables support for the LCD display and buttons found ++ on Cobalt systems through a misc device. ++ ++config DTLK ++ tristate "Double Talk PC internal speech card support" ++ help ++ This driver is for the DoubleTalk PC, a speech synthesizer ++ manufactured by RC Systems (<http://www.rcsys.com/>). It is also ++ called the `internal DoubleTalk'. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called dtlk. ++ ++config R3964 ++ tristate "Siemens R3964 line discipline" ++ ---help--- ++ This driver allows synchronous communication with devices using the ++ Siemens R3964 packet protocol. Unless you are dealing with special ++ hardware like PLCs, you are unlikely to need this. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called n_r3964. ++ ++ If unsure, say N. ++ ++config APPLICOM ++ tristate "Applicom intelligent fieldbus card support" ++ depends on PCI ++ ---help--- ++ This driver provides the kernel-side support for the intelligent ++ fieldbus cards made by Applicom International. More information ++ about these cards can be found on the WWW at the address ++ <http://www.applicom-int.com/>, or by email from David Woodhouse ++ <dwmw2@infradead.org>. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called applicom. ++ ++ If unsure, say N. ++ ++config SONYPI ++ tristate "Sony Vaio Programmable I/O Control Device support (EXPERIMENTAL)" ++ depends on EXPERIMENTAL && X86 && PCI && INPUT && !64BIT ++ ---help--- ++ This driver enables access to the Sony Programmable I/O Control ++ Device which can be found in many (all ?) Sony Vaio laptops. ++ ++ If you have one of those laptops, read ++ <file:Documentation/sonypi.txt>, and say Y or M here. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called sonypi. ++ ++config TANBAC_TB0219 ++ tristate "TANBAC TB0219 base board support" ++ depends TANBAC_TB022X ++ ++menu "Ftape, the floppy tape device driver" ++ ++config FTAPE ++ tristate "Ftape (QIC-80/Travan) support" ++ depends on BROKEN_ON_SMP && (ALPHA || X86) ++ ---help--- ++ If you have a tape drive that is connected to your floppy ++ controller, say Y here. ++ ++ Some tape drives (like the Seagate "Tape Store 3200" or the Iomega ++ "Ditto 3200" or the Exabyte "Eagle TR-3") come with a "high speed" ++ controller of their own. These drives (and their companion ++ controllers) are also supported if you say Y here. ++ ++ If you have a special controller (such as the CMS FC-10, FC-20, ++ Mountain Mach-II, or any controller that is based on the Intel 82078 ++ FDC like the high speed controllers by Seagate and Exabyte and ++ Iomega's "Ditto Dash") you must configure it by selecting the ++ appropriate entries from the "Floppy tape controllers" sub-menu ++ below and possibly modify the default values for the IRQ and DMA ++ channel and the IO base in ftape's configuration menu. ++ ++ If you want to use your floppy tape drive on a PCI-bus based system, ++ please read the file <file:drivers/char/ftape/README.PCI>. ++ ++ The ftape kernel driver is also available as a runtime loadable ++ module. To compile this driver as a module, choose M here: the ++ module will be called ftape. ++ ++ Note that the Ftape-HOWTO is out of date (sorry) and documents the ++ older version 2.08 of this software but still contains useful ++ information. There is a web page with more recent documentation at ++ <http://www.instmath.rwth-aachen.de/~heine/ftape/>. This page ++ always contains the latest release of the ftape driver and useful ++ information (backup software, ftape related patches and ++ documentation, FAQ). Note that the file system interface has ++ changed quite a bit compared to previous versions of ftape. Please ++ read <file:Documentation/ftape.txt>. ++ ++source "drivers/char/ftape/Kconfig" ++ ++endmenu ++ ++source "drivers/char/agp/Kconfig" ++ ++source "drivers/char/drm/Kconfig" ++ ++source "drivers/char/pcmcia/Kconfig" ++ ++config MWAVE ++ tristate "ACP Modem (Mwave) support" ++ depends on X86 ++ select SERIAL_8250 ++ ---help--- ++ The ACP modem (Mwave) for Linux is a WinModem. It is composed of a ++ kernel driver and a user level application. Together these components ++ support direct attachment to public switched telephone networks (PSTNs) ++ and support selected world wide countries. ++ ++ This version of the ACP Modem driver supports the IBM Thinkpad 600E, ++ 600, and 770 that include on board ACP modem hardware. ++ ++ The modem also supports the standard communications port interface ++ (ttySx) and is compatible with the Hayes AT Command Set. ++ ++ The user level application needed to use this driver can be found at ++ the IBM Linux Technology Center (LTC) web site: ++ <http://www.ibm.com/linux/ltc/>. ++ ++ If you own one of the above IBM Thinkpads which has the Mwave chipset ++ in it, say Y. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called mwave. ++ ++config SCx200_GPIO ++ tristate "NatSemi SCx200 GPIO Support" ++ depends on SCx200 ++ help ++ Give userspace access to the GPIO pins on the National ++ Semiconductor SCx200 processors. ++ ++ If compiled as a module, it will be called scx200_gpio. ++ ++config GPIO_VR41XX ++ tristate "NEC VR4100 series General-purpose I/O Unit support" ++ depends on CPU_VR41XX ++ ++config RAW_DRIVER ++ tristate "RAW driver (/dev/raw/rawN) (OBSOLETE)" ++ help ++ The raw driver permits block devices to be bound to /dev/raw/rawN. ++ Once bound, I/O against /dev/raw/rawN uses efficient zero-copy I/O. ++ See the raw(8) manpage for more details. ++ ++ The raw driver is deprecated and will be removed soon. ++ Applications should simply open the device (eg /dev/hda1) ++ with the O_DIRECT flag. ++ ++config MAX_RAW_DEVS ++ int "Maximum number of RAW devices to support (1-8192)" ++ depends on RAW_DRIVER ++ default "256" ++ help ++ The maximum number of RAW devices that are supported. ++ Default is 256. Increase this number in case you need lots of ++ raw devices. ++ ++config HPET ++ bool "HPET - High Precision Event Timer" if (X86 || IA64) ++ default n ++ depends on ACPI ++ help ++ If you say Y here, you will have a miscdevice named "/dev/hpet/". Each ++ open selects one of the timers supported by the HPET. The timers are ++ non-periodioc and/or periodic. ++ ++config HPET_RTC_IRQ ++ bool "HPET Control RTC IRQ" if !HPET_EMULATE_RTC ++ default n ++ depends on HPET ++ help ++ If you say Y here, you will disable RTC_IRQ in drivers/char/rtc.c. It ++ is assumed the platform called hpet_alloc with the RTC IRQ values for ++ the HPET timers. ++ ++config HPET_MMAP ++ bool "Allow mmap of HPET" ++ default y ++ depends on HPET ++ help ++ If you say Y here, user applications will be able to mmap ++ the HPET registers. ++ ++ In some hardware implementations, the page containing HPET ++ registers may also contain other things that shouldn't be ++ exposed to the user. If this applies to your hardware, ++ say N here. ++ ++config HANGCHECK_TIMER ++ tristate "Hangcheck timer" ++ depends on X86 || IA64 || PPC64 || ARCH_S390 ++ help ++ The hangcheck-timer module detects when the system has gone ++ out to lunch past a certain margin. It can reboot the system ++ or merely print a warning. ++ ++config MMTIMER ++ tristate "MMTIMER Memory mapped RTC for SGI Altix" ++ depends on IA64_GENERIC || IA64_SGI_SN2 ++ default y ++ help ++ The mmtimer device allows direct userspace access to the ++ Altix system timer. ++ ++source "drivers/char/tpm/Kconfig" ++ ++config TELCLOCK ++ tristate "Telecom clock driver for MPBL0010 ATCA SBC" ++ depends on EXPERIMENTAL ++ default n ++ help ++ The telecom clock device is specific to the MPBL0010 ATCA computer and ++ allows direct userspace access to the configuration of the telecom clock ++ configuration settings. This device is used for hardware synchronization ++ across the ATCA backplane fabric. Upon loading, the driver exports a ++ sysfs directory, /sys/devices/platform/telco_clock, with a number of ++ files for controlling the behavior of this hardware. ++ ++endmenu ++ +diff -urN oldtree/include/linux/vt.h newtree/include/linux/vt.h +--- oldtree/include/linux/vt.h 2006-03-27 13:28:15.000000000 -0500 ++++ newtree/include/linux/vt.h 2006-03-28 10:37:24.211121750 -0500 +@@ -10,8 +10,19 @@ + * resizing). + */ + #define MIN_NR_CONSOLES 1 /* must be at least 1 */ +-#define MAX_NR_CONSOLES 63 /* serial lines start at 64 */ +-#define MAX_NR_USER_CONSOLES 63 /* must be root to allocate above this */ ++#if (CONFIG_NR_TTY_DEVICES < 11) ++/* Lower Limit */ ++#define MAX_NR_CONSOLES 11 ++#define MAX_NR_USER_CONSOLES 11 ++#elif (CONFIG_NR_TTY_DEVICES > 63) ++/* Upper Limit */ ++#define MAX_NR_CONSOLES 63 ++#define MAX_NR_USER_CONSOLES 63 ++#else ++/* They chose a sensible number */ ++#define MAX_NR_CONSOLES CONFIG_NR_TTY_DEVICES ++#define MAX_NR_USER_CONSOLES CONFIG_NR_TTY_DEVICES ++#endif + /* Note: the ioctl VT_GETSTATE does not work for + consoles 16 and higher (since it returns a short) */ + diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/defconfig-logicpd-pxa270 b/packages/linux/logicpd-pxa270-2.6.19.2/defconfig-logicpd-pxa270 new file mode 100644 index 0000000000..20ca536446 --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/defconfig-logicpd-pxa270 @@ -0,0 +1,1444 @@ +# +# Automatically generated make config: don't edit +# Linux kernel version: 2.6.19.2 +# Tue Jan 30 21:59:05 2007 +# +CONFIG_ARM=y +# CONFIG_GENERIC_TIME is not set +CONFIG_MMU=y +CONFIG_GENERIC_HARDIRQS=y +CONFIG_TRACE_IRQFLAGS_SUPPORT=y +CONFIG_HARDIRQS_SW_RESEND=y +CONFIG_GENERIC_IRQ_PROBE=y +CONFIG_RWSEM_GENERIC_SPINLOCK=y +CONFIG_GENERIC_HWEIGHT=y +CONFIG_GENERIC_CALIBRATE_DELAY=y +CONFIG_ARCH_MTD_XIP=y +CONFIG_VECTORS_BASE=0xffff0000 +CONFIG_DEFCONFIG_LIST="/lib/modules/$UNAME_RELEASE/.config" + +# +# Code maturity level options +# +CONFIG_EXPERIMENTAL=y +CONFIG_BROKEN_ON_SMP=y +CONFIG_INIT_ENV_ARG_LIMIT=32 + +# +# General setup +# +CONFIG_LOCALVERSION="" +CONFIG_LOCALVERSION_AUTO=y +CONFIG_SWAP=y +CONFIG_SYSVIPC=y +# CONFIG_IPC_NS is not set +# CONFIG_POSIX_MQUEUE is not set +# CONFIG_BSD_PROCESS_ACCT is not set +# CONFIG_TASKSTATS is not set +# CONFIG_UTS_NS is not set +# CONFIG_AUDIT is not set +# CONFIG_IKCONFIG is not set +# CONFIG_RELAY is not set +CONFIG_INITRAMFS_SOURCE="" +CONFIG_CC_OPTIMIZE_FOR_SIZE=y +CONFIG_SYSCTL=y +CONFIG_EMBEDDED=y +CONFIG_UID16=y +CONFIG_SYSCTL_SYSCALL=y +CONFIG_KALLSYMS=y +# CONFIG_KALLSYMS_ALL is not set +# CONFIG_KALLSYMS_EXTRA_PASS is not set +CONFIG_HOTPLUG=y +CONFIG_PRINTK=y +CONFIG_BUG=y +CONFIG_ELF_CORE=y +CONFIG_BASE_FULL=y +CONFIG_FUTEX=y +CONFIG_EPOLL=y +CONFIG_SHMEM=y +CONFIG_SLAB=y +CONFIG_VM_EVENT_COUNTERS=y +CONFIG_RT_MUTEXES=y +# CONFIG_TINY_SHMEM is not set +CONFIG_BASE_SMALL=0 +# CONFIG_SLOB is not set + +# +# Loadable module support +# +CONFIG_MODULES=y +CONFIG_MODULE_UNLOAD=y +CONFIG_MODULE_FORCE_UNLOAD=y +# CONFIG_MODVERSIONS is not set +# CONFIG_MODULE_SRCVERSION_ALL is not set +# CONFIG_KMOD is not set + +# +# Block layer +# +CONFIG_BLOCK=y +# CONFIG_BLK_DEV_IO_TRACE is not set + +# +# IO Schedulers +# +CONFIG_IOSCHED_NOOP=y +CONFIG_IOSCHED_AS=y +CONFIG_IOSCHED_DEADLINE=y +CONFIG_IOSCHED_CFQ=y +CONFIG_DEFAULT_AS=y +# CONFIG_DEFAULT_DEADLINE is not set +# CONFIG_DEFAULT_CFQ is not set +# CONFIG_DEFAULT_NOOP is not set +CONFIG_DEFAULT_IOSCHED="anticipatory" + +# +# System Type +# +# CONFIG_ARCH_AAEC2000 is not set +# CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_REALVIEW is not set +# CONFIG_ARCH_VERSATILE is not set +# CONFIG_ARCH_AT91 is not set +# CONFIG_ARCH_CLPS7500 is not set +# CONFIG_ARCH_CLPS711X is not set +# CONFIG_ARCH_CO285 is not set +# CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_EP93XX is not set +# CONFIG_ARCH_FOOTBRIDGE is not set +# CONFIG_ARCH_NETX is not set +# CONFIG_ARCH_H720X is not set +# CONFIG_ARCH_IMX is not set +# CONFIG_ARCH_IOP32X is not set +# CONFIG_ARCH_IOP33X is not set +# CONFIG_ARCH_IXP4XX is not set +# CONFIG_ARCH_IXP2000 is not set +# CONFIG_ARCH_IXP23XX is not set +# CONFIG_ARCH_L7200 is not set +# CONFIG_ARCH_PNX4008 is not set +CONFIG_ARCH_PXA=y +# CONFIG_ARCH_RPC is not set +# CONFIG_ARCH_SA1100 is not set +# CONFIG_ARCH_S3C2410 is not set +# CONFIG_ARCH_SHARK is not set +# CONFIG_ARCH_LH7A40X is not set +# CONFIG_ARCH_OMAP is not set + +# +# Intel PXA2xx Implementations +# +# CONFIG_ARCH_LUBBOCK is not set +CONFIG_MACH_LOGICPD_PXA270=y +# CONFIG_MACH_MAINSTONE is not set +# CONFIG_ARCH_PXA_IDP is not set +# CONFIG_PXA_SHARPSL is not set +# CONFIG_MACH_TRIZEPS4 is not set +CONFIG_PXA27x=y +# CONFIG_PXA_KEYS is not set +CONFIG_IWMMXT=y + +# +# Processor Type +# +CONFIG_CPU_32=y +CONFIG_CPU_XSCALE=y +CONFIG_CPU_32v5=y +CONFIG_CPU_ABRT_EV5T=y +CONFIG_CPU_CACHE_VIVT=y +CONFIG_CPU_TLB_V4WBI=y +CONFIG_CPU_CP15=y +CONFIG_CPU_CP15_MMU=y + +# +# Processor Features +# +# CONFIG_ARM_THUMB is not set +# CONFIG_CPU_DCACHE_DISABLE is not set +CONFIG_XSCALE_PMU=y + +# +# Bus support +# + +# +# PCCARD (PCMCIA/CardBus) support +# +# CONFIG_PCCARD is not set + +# +# Kernel Features +# +# CONFIG_PREEMPT is not set +CONFIG_NO_IDLE_HZ=y +CONFIG_HZ=100 +CONFIG_AEABI=y +CONFIG_OABI_COMPAT=y +# CONFIG_ARCH_DISCONTIGMEM_ENABLE is not set +CONFIG_SELECT_MEMORY_MODEL=y +CONFIG_FLATMEM_MANUAL=y +# CONFIG_DISCONTIGMEM_MANUAL is not set +# CONFIG_SPARSEMEM_MANUAL is not set +CONFIG_FLATMEM=y +CONFIG_FLAT_NODE_MEM_MAP=y +# CONFIG_SPARSEMEM_STATIC is not set +CONFIG_SPLIT_PTLOCK_CPUS=4096 +# CONFIG_RESOURCES_64BIT is not set +CONFIG_ALIGNMENT_TRAP=y + +# +# Boot options +# +CONFIG_ZBOOT_ROM_TEXT=0x0 +CONFIG_ZBOOT_ROM_BSS=0x0 +CONFIG_CMDLINE="console=ttyS0,115200 root=/dev/mtdblock2 rootfstype=jffs2 mem=64M " +# CONFIG_XIP_KERNEL is not set + +# +# CPU Frequency scaling +# +# CONFIG_CPU_FREQ is not set +# CONFIG_KEXEC is not set + +# +# Floating point emulation +# + +# +# At least one emulation must be selected +# +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_NWFPE_XP is not set +CONFIG_FPE_FASTFPE=y + +# +# Userspace binary formats +# +CONFIG_BINFMT_ELF=y +# CONFIG_BINFMT_AOUT is not set +# CONFIG_BINFMT_MISC is not set + +# +# Power management options +# +CONFIG_PM=y +# CONFIG_PM_LEGACY is not set +# CONFIG_PM_DEBUG is not set +# CONFIG_PM_SYSFS_DEPRECATED is not set +CONFIG_APM=y + +# +# Networking +# +CONFIG_NET=y + +# +# Networking options +# +CONFIG_NETDEBUG=y +# CONFIG_PACKET is not set +CONFIG_UNIX=y +CONFIG_XFRM=y +# CONFIG_XFRM_USER is not set +# CONFIG_XFRM_SUB_POLICY is not set +# CONFIG_NET_KEY is not set +CONFIG_INET=y +# CONFIG_IP_MULTICAST is not set +CONFIG_IP_ADVANCED_ROUTER=y +CONFIG_ASK_IP_FIB_HASH=y +# CONFIG_IP_FIB_TRIE is not set +CONFIG_IP_FIB_HASH=y +# CONFIG_IP_MULTIPLE_TABLES is not set +# CONFIG_IP_ROUTE_MULTIPATH is not set +# CONFIG_IP_ROUTE_VERBOSE is not set +CONFIG_IP_PNP=y +CONFIG_IP_PNP_DHCP=y +CONFIG_IP_PNP_BOOTP=y +CONFIG_IP_PNP_RARP=y +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set +# CONFIG_SYN_COOKIES is not set +# CONFIG_INET_AH is not set +# CONFIG_INET_ESP is not set +# CONFIG_INET_IPCOMP is not set +# CONFIG_INET_XFRM_TUNNEL is not set +# CONFIG_INET_TUNNEL is not set +CONFIG_INET_XFRM_MODE_TRANSPORT=y +CONFIG_INET_XFRM_MODE_TUNNEL=y +CONFIG_INET_XFRM_MODE_BEET=y +CONFIG_INET_DIAG=y +CONFIG_INET_TCP_DIAG=y +# CONFIG_TCP_CONG_ADVANCED is not set +CONFIG_TCP_CONG_CUBIC=y +CONFIG_DEFAULT_TCP_CONG="cubic" + +# +# IP: Virtual Server Configuration +# +# CONFIG_IP_VS is not set +CONFIG_IPV6=y +CONFIG_IPV6_PRIVACY=y +CONFIG_IPV6_ROUTER_PREF=y +CONFIG_IPV6_ROUTE_INFO=y +CONFIG_INET6_AH=y +CONFIG_INET6_ESP=y +CONFIG_INET6_IPCOMP=y +# CONFIG_IPV6_MIP6 is not set +CONFIG_INET6_XFRM_TUNNEL=y +CONFIG_INET6_TUNNEL=y +CONFIG_INET6_XFRM_MODE_TRANSPORT=y +CONFIG_INET6_XFRM_MODE_TUNNEL=y +CONFIG_INET6_XFRM_MODE_BEET=y +# CONFIG_INET6_XFRM_MODE_ROUTEOPTIMIZATION is not set +CONFIG_IPV6_SIT=y +CONFIG_IPV6_TUNNEL=y +# CONFIG_IPV6_MULTIPLE_TABLES is not set +# CONFIG_NETWORK_SECMARK is not set +CONFIG_NETFILTER=y +# CONFIG_NETFILTER_DEBUG is not set +CONFIG_BRIDGE_NETFILTER=y + +# +# Core Netfilter Configuration +# +# CONFIG_NETFILTER_NETLINK is not set +# CONFIG_NF_CONNTRACK is not set +CONFIG_NETFILTER_XTABLES=m +# CONFIG_NETFILTER_XT_TARGET_CLASSIFY is not set +# CONFIG_NETFILTER_XT_TARGET_MARK is not set +# CONFIG_NETFILTER_XT_TARGET_NFQUEUE is not set +# CONFIG_NETFILTER_XT_MATCH_COMMENT is not set +# CONFIG_NETFILTER_XT_MATCH_DCCP is not set +# CONFIG_NETFILTER_XT_MATCH_DSCP is not set +# CONFIG_NETFILTER_XT_MATCH_ESP is not set +# CONFIG_NETFILTER_XT_MATCH_LENGTH is not set +# CONFIG_NETFILTER_XT_MATCH_LIMIT is not set +# CONFIG_NETFILTER_XT_MATCH_MAC is not set +# CONFIG_NETFILTER_XT_MATCH_MARK is not set +# CONFIG_NETFILTER_XT_MATCH_POLICY is not set +# CONFIG_NETFILTER_XT_MATCH_MULTIPORT is not set +# CONFIG_NETFILTER_XT_MATCH_PHYSDEV is not set +# CONFIG_NETFILTER_XT_MATCH_PKTTYPE is not set +# CONFIG_NETFILTER_XT_MATCH_QUOTA is not set +# CONFIG_NETFILTER_XT_MATCH_REALM is not set +# CONFIG_NETFILTER_XT_MATCH_SCTP is not set +# CONFIG_NETFILTER_XT_MATCH_STATISTIC is not set +# CONFIG_NETFILTER_XT_MATCH_STRING is not set +# CONFIG_NETFILTER_XT_MATCH_TCPMSS is not set + +# +# IP: Netfilter Configuration +# +# CONFIG_IP_NF_CONNTRACK is not set +# CONFIG_IP_NF_QUEUE is not set +# CONFIG_IP_NF_IPTABLES is not set +# CONFIG_IP_NF_ARPTABLES is not set + +# +# IPv6: Netfilter Configuration (EXPERIMENTAL) +# +# CONFIG_IP6_NF_QUEUE is not set +# CONFIG_IP6_NF_IPTABLES is not set + +# +# Bridge: Netfilter Configuration +# +# CONFIG_BRIDGE_NF_EBTABLES is not set + +# +# DCCP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_DCCP is not set + +# +# SCTP Configuration (EXPERIMENTAL) +# +# CONFIG_IP_SCTP is not set + +# +# TIPC Configuration (EXPERIMENTAL) +# +# CONFIG_TIPC is not set +# CONFIG_ATM is not set +CONFIG_BRIDGE=m +# CONFIG_VLAN_8021Q is not set +# CONFIG_DECNET is not set +CONFIG_LLC=m +# CONFIG_LLC2 is not set +# CONFIG_IPX is not set +# CONFIG_ATALK is not set +# CONFIG_X25 is not set +# CONFIG_LAPB is not set +# CONFIG_ECONET is not set +# CONFIG_WAN_ROUTER is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network testing +# +# CONFIG_NET_PKTGEN is not set +# CONFIG_HAMRADIO is not set +# CONFIG_IRDA is not set +CONFIG_BT=y +CONFIG_BT_L2CAP=y +CONFIG_BT_SCO=y +CONFIG_BT_RFCOMM=y +CONFIG_BT_RFCOMM_TTY=y +CONFIG_BT_BNEP=y +CONFIG_BT_BNEP_MC_FILTER=y +CONFIG_BT_BNEP_PROTO_FILTER=y +CONFIG_BT_HIDP=y + +# +# Bluetooth device drivers +# +CONFIG_BT_HCIUSB=y +# CONFIG_BT_HCIUSB_SCO is not set +# CONFIG_BT_HCIUART is not set +# CONFIG_BT_HCIBCM203X is not set +# CONFIG_BT_HCIBPA10X is not set +# CONFIG_BT_HCIBFUSB is not set +# CONFIG_BT_HCIVHCI is not set +CONFIG_IEEE80211=y +# CONFIG_IEEE80211_DEBUG is not set +CONFIG_IEEE80211_CRYPT_WEP=y +# CONFIG_IEEE80211_CRYPT_CCMP is not set +CONFIG_IEEE80211_SOFTMAC=y +# CONFIG_IEEE80211_SOFTMAC_DEBUG is not set +CONFIG_WIRELESS_EXT=y + +# +# Device Drivers +# + +# +# Generic Driver Options +# +CONFIG_STANDALONE=y +CONFIG_PREVENT_FIRMWARE_BUILD=y +# CONFIG_FW_LOADER is not set +# CONFIG_DEBUG_DRIVER is not set +# CONFIG_SYS_HYPERVISOR is not set + +# +# Connector - unified userspace <-> kernelspace linker +# +# CONFIG_CONNECTOR is not set + +# +# Memory Technology Devices (MTD) +# +CONFIG_MTD=y +CONFIG_MTD_DEBUG=y +CONFIG_MTD_DEBUG_VERBOSE=0 +# CONFIG_MTD_CONCAT is not set +CONFIG_MTD_PARTITIONS=y +CONFIG_MTD_REDBOOT_PARTS=y +CONFIG_MTD_REDBOOT_DIRECTORY_BLOCK=-1 +# CONFIG_MTD_REDBOOT_PARTS_UNALLOCATED is not set +# CONFIG_MTD_REDBOOT_PARTS_READONLY is not set +# CONFIG_MTD_CMDLINE_PARTS is not set +# CONFIG_MTD_AFS_PARTS is not set + +# +# User Modules And Translation Layers +# +CONFIG_MTD_CHAR=y +CONFIG_MTD_BLOCK=y +# CONFIG_FTL is not set +# CONFIG_NFTL is not set +# CONFIG_INFTL is not set +# CONFIG_RFD_FTL is not set +# CONFIG_SSFDC is not set + +# +# RAM/ROM/Flash chip drivers +# +CONFIG_MTD_CFI=y +# CONFIG_MTD_JEDECPROBE is not set +CONFIG_MTD_GEN_PROBE=y +CONFIG_MTD_CFI_ADV_OPTIONS=y +CONFIG_MTD_CFI_NOSWAP=y +# CONFIG_MTD_CFI_BE_BYTE_SWAP is not set +# CONFIG_MTD_CFI_LE_BYTE_SWAP is not set +CONFIG_MTD_CFI_GEOMETRY=y +CONFIG_MTD_MAP_BANK_WIDTH_1=y +CONFIG_MTD_MAP_BANK_WIDTH_2=y +CONFIG_MTD_MAP_BANK_WIDTH_4=y +# CONFIG_MTD_MAP_BANK_WIDTH_8 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_16 is not set +# CONFIG_MTD_MAP_BANK_WIDTH_32 is not set +# CONFIG_MTD_CFI_I1 is not set +CONFIG_MTD_CFI_I2=y +# CONFIG_MTD_CFI_I4 is not set +# CONFIG_MTD_CFI_I8 is not set +# CONFIG_MTD_OTP is not set +CONFIG_MTD_CFI_INTELEXT=y +# CONFIG_MTD_CFI_AMDSTD is not set +# CONFIG_MTD_CFI_STAA is not set +CONFIG_MTD_CFI_UTIL=y +# CONFIG_MTD_RAM is not set +# CONFIG_MTD_ROM is not set +# CONFIG_MTD_ABSENT is not set +# CONFIG_MTD_OBSOLETE_CHIPS is not set +# CONFIG_MTD_XIP is not set + +# +# Mapping drivers for chip access +# +# CONFIG_MTD_COMPLEX_MAPPINGS is not set +# CONFIG_MTD_PHYSMAP is not set +CONFIG_MTD_LOGICPD_PXA270=y +# CONFIG_MTD_ARM_INTEGRATOR is not set +# CONFIG_MTD_SHARP_SL is not set +# CONFIG_MTD_PLATRAM is not set + +# +# Self-contained MTD device drivers +# +# CONFIG_MTD_SLRAM is not set +# CONFIG_MTD_PHRAM is not set +# CONFIG_MTD_MTDRAM is not set +# CONFIG_MTD_BLOCK2MTD is not set + +# +# Disk-On-Chip Device Drivers +# +# CONFIG_MTD_DOC2000 is not set +# CONFIG_MTD_DOC2001 is not set +# CONFIG_MTD_DOC2001PLUS is not set + +# +# NAND Flash Device Drivers +# +# CONFIG_MTD_NAND is not set + +# +# OneNAND Flash Device Drivers +# +# CONFIG_MTD_ONENAND is not set + +# +# Parallel port support +# +# CONFIG_PARPORT is not set + +# +# Plug and Play support +# + +# +# Block devices +# +CONFIG_BLK_DEV_LOGICPD_CF=y +# CONFIG_BLK_DEV_COW_COMMON is not set +# CONFIG_BLK_DEV_LOOP is not set +# CONFIG_BLK_DEV_NBD is not set +# CONFIG_BLK_DEV_UB is not set +# CONFIG_BLK_DEV_RAM is not set +# CONFIG_BLK_DEV_INITRD is not set +# CONFIG_CDROM_PKTCDVD is not set +# CONFIG_ATA_OVER_ETH is not set + +# +# ATA/ATAPI/MFM/RLL support +# +CONFIG_IDE=y +CONFIG_IDE_MAX_HWIFS=4 +CONFIG_BLK_DEV_IDE=y + +# +# Please see Documentation/ide.txt for help/info on IDE drives +# +# CONFIG_BLK_DEV_IDE_SATA is not set +CONFIG_BLK_DEV_IDEDISK=y +# CONFIG_IDEDISK_MULTI_MODE is not set +# CONFIG_BLK_DEV_IDECD is not set +# CONFIG_BLK_DEV_IDETAPE is not set +# CONFIG_BLK_DEV_IDEFLOPPY is not set +# CONFIG_BLK_DEV_IDESCSI is not set +# CONFIG_IDE_TASK_IOCTL is not set + +# +# IDE chipset support/bugfixes +# +# CONFIG_IDE_GENERIC is not set +# CONFIG_IDE_ARM is not set +# CONFIG_BLK_DEV_IDEDMA is not set +# CONFIG_IDEDMA_AUTO is not set +# CONFIG_BLK_DEV_HD is not set + +# +# SCSI device support +# +# CONFIG_RAID_ATTRS is not set +CONFIG_SCSI=y +# CONFIG_SCSI_NETLINK is not set +CONFIG_SCSI_PROC_FS=y + +# +# SCSI support type (disk, tape, CD-ROM) +# +# CONFIG_BLK_DEV_SD is not set +# CONFIG_CHR_DEV_ST is not set +# CONFIG_CHR_DEV_OSST is not set +# CONFIG_BLK_DEV_SR is not set +# CONFIG_CHR_DEV_SG is not set +# CONFIG_CHR_DEV_SCH is not set + +# +# Some SCSI devices (e.g. CD jukebox) support multiple LUNs +# +# CONFIG_SCSI_MULTI_LUN is not set +# CONFIG_SCSI_CONSTANTS is not set +# CONFIG_SCSI_LOGGING is not set + +# +# SCSI Transports +# +# CONFIG_SCSI_SPI_ATTRS is not set +# CONFIG_SCSI_FC_ATTRS is not set +# CONFIG_SCSI_ISCSI_ATTRS is not set +# CONFIG_SCSI_SAS_ATTRS is not set +# CONFIG_SCSI_SAS_LIBSAS is not set + +# +# SCSI low-level drivers +# +# CONFIG_ISCSI_TCP is not set +# CONFIG_SCSI_DEBUG is not set + +# +# Serial ATA (prod) and Parallel ATA (experimental) drivers +# +# CONFIG_ATA is not set + +# +# Multi-device support (RAID and LVM) +# +# CONFIG_MD is not set + +# +# Fusion MPT device support +# +# CONFIG_FUSION is not set + +# +# IEEE 1394 (FireWire) support +# + +# +# I2O device support +# + +# +# Network device support +# +CONFIG_NETDEVICES=y +# CONFIG_DUMMY is not set +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +# CONFIG_TUN is not set + +# +# PHY device support +# +# CONFIG_PHYLIB is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +CONFIG_MII=y +CONFIG_SMC91X=y +# CONFIG_DM9000 is not set +# CONFIG_SMC911X is not set + +# +# Ethernet (1000 Mbit) +# + +# +# Ethernet (10000 Mbit) +# + +# +# Token Ring devices +# + +# +# Wireless LAN (non-hamradio) +# +# CONFIG_NET_RADIO is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set +# CONFIG_PPP is not set +# CONFIG_SLIP is not set +# CONFIG_SHAPER is not set +# CONFIG_NETCONSOLE is not set +# CONFIG_NETPOLL is not set +# CONFIG_NET_POLL_CONTROLLER is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Input device support +# +CONFIG_INPUT=y +# CONFIG_INPUT_FF_MEMLESS is not set + +# +# Userland interfaces +# +CONFIG_INPUT_MOUSEDEV=y +CONFIG_INPUT_MOUSEDEV_PSAUX=y +CONFIG_INPUT_MOUSEDEV_SCREEN_X=1024 +CONFIG_INPUT_MOUSEDEV_SCREEN_Y=768 +# CONFIG_INPUT_JOYDEV is not set +# CONFIG_INPUT_TSDEV is not set +CONFIG_INPUT_EVDEV=y +# CONFIG_INPUT_EVBUG is not set +# CONFIG_INPUT_POWER is not set + +# +# Input Device Drivers +# +CONFIG_INPUT_KEYBOARD=y +CONFIG_KEYBOARD_ATKBD=y +# CONFIG_KEYBOARD_SUNKBD is not set +# CONFIG_KEYBOARD_LKKBD is not set +# CONFIG_KEYBOARD_XTKBD is not set +# CONFIG_KEYBOARD_NEWTON is not set +# CONFIG_KEYBOARD_STOWAWAY is not set +# CONFIG_INPUT_MOUSE is not set +# CONFIG_INPUT_JOYSTICK is not set +CONFIG_INPUT_TOUCHSCREEN=y +CONFIG_UCB1400=y +CONFIG_TOUCHSCREEN_UCB1400=m +# CONFIG_TOUCHSCREEN_GUNZE is not set +# CONFIG_TOUCHSCREEN_ELO is not set +# CONFIG_TOUCHSCREEN_MTOUCH is not set +# CONFIG_TOUCHSCREEN_MK712 is not set +# CONFIG_TOUCHSCREEN_PENMOUNT is not set +# CONFIG_TOUCHSCREEN_TOUCHRIGHT is not set +# CONFIG_TOUCHSCREEN_TOUCHWIN is not set +CONFIG_INPUT_MISC=y +CONFIG_INPUT_UINPUT=y + +# +# Hardware I/O ports +# +CONFIG_SERIO=y +# CONFIG_SERIO_SERPORT is not set +CONFIG_SERIO_LIBPS2=y +CONFIG_SERIO_RAW=y +# CONFIG_GAMEPORT is not set + +# +# Character devices +# +CONFIG_VT=y +CONFIG_VT_CONSOLE=y +CONFIG_NR_TTY_DEVICES=11 +CONFIG_HW_CONSOLE=y +# CONFIG_VT_HW_CONSOLE_BINDING is not set +# CONFIG_SERIAL_NONSTANDARD is not set + +# +# Serial drivers +# +# CONFIG_SERIAL_8250 is not set + +# +# Non-8250 serial port support +# +CONFIG_SERIAL_PXA=y +CONFIG_SERIAL_PXA_CONSOLE=y +CONFIG_SERIAL_CORE=y +CONFIG_SERIAL_CORE_CONSOLE=y +CONFIG_UNIX98_PTYS=y +# CONFIG_LEGACY_PTYS is not set + +# +# IPMI +# +# CONFIG_IPMI_HANDLER is not set + +# +# Watchdog Cards +# +CONFIG_WATCHDOG=y +# CONFIG_WATCHDOG_NOWAYOUT is not set + +# +# Watchdog Device Drivers +# +# CONFIG_SOFT_WATCHDOG is not set +CONFIG_SA1100_WATCHDOG=y + +# +# USB-based Watchdog Cards +# +# CONFIG_USBPCWATCHDOG is not set +CONFIG_HW_RANDOM=y +# CONFIG_NVRAM is not set +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set + +# +# Ftape, the floppy tape device driver +# +# CONFIG_RAW_DRIVER is not set + +# +# TPM devices +# +# CONFIG_TCG_TPM is not set + +# +# I2C support +# +# CONFIG_I2C is not set + +# +# SPI support +# +# CONFIG_SPI is not set +# CONFIG_SPI_MASTER is not set + +# +# Dallas's 1-wire bus +# +# CONFIG_W1 is not set + +# +# Hardware Monitoring support +# +CONFIG_HWMON=y +# CONFIG_HWMON_VID is not set +# CONFIG_SENSORS_ABITUGURU is not set +# CONFIG_SENSORS_F71805F is not set +# CONFIG_SENSORS_VT1211 is not set +# CONFIG_HWMON_DEBUG_CHIP is not set + +# +# Misc devices +# +# CONFIG_TIFM_CORE is not set + +# +# LED devices +# +CONFIG_NEW_LEDS=y +CONFIG_LEDS_CLASS=y + +# +# LED drivers +# + +# +# LED Triggers +# +# CONFIG_LEDS_TRIGGERS is not set + +# +# Multimedia devices +# +# CONFIG_VIDEO_DEV is not set + +# +# Digital Video Broadcasting Devices +# +# CONFIG_DVB is not set +# CONFIG_USB_DABUSB is not set + +# +# Graphics support +# +CONFIG_FIRMWARE_EDID=y +CONFIG_FB=y +CONFIG_FB_CFB_FILLRECT=y +CONFIG_FB_CFB_COPYAREA=y +CONFIG_FB_CFB_IMAGEBLIT=y +# CONFIG_FB_MACMODES is not set +# CONFIG_FB_BACKLIGHT is not set +# CONFIG_FB_MODE_HELPERS is not set +# CONFIG_FB_TILEBLITTING is not set +# CONFIG_FB_S1D13XXX is not set +CONFIG_FB_PXA=y +# CONFIG_FB_PXA_LCD_QVGA is not set +# CONFIG_FB_PXA_LCD_VGA is not set +# CONFIG_FB_PXA_LPD_LQ64D343 is not set +# CONFIG_FB_PXA_LPD_LQ035Q7DB02 is not set +# CONFIG_FB_PXA_LPD_LQ057Q3DC02 is not set +# CONFIG_FB_PXA_LPD_LQ10D368 is not set +CONFIG_FB_PXA_LPD_OSD024TTEA2=y +# CONFIG_FB_PXA_OVERLAY is not set +CONFIG_FB_PXA_PARAMETERS=y +# CONFIG_FB_MBX is not set +# CONFIG_FB_VIRTUAL is not set + +# +# Console display driver support +# +# CONFIG_VGA_CONSOLE is not set +CONFIG_DUMMY_CONSOLE=y +# CONFIG_FRAMEBUFFER_CONSOLE is not set + +# +# Logo configuration +# +# CONFIG_LOGO is not set +# CONFIG_BACKLIGHT_LCD_SUPPORT is not set + +# +# Sound +# +CONFIG_SOUND=y + +# +# Advanced Linux Sound Architecture +# +CONFIG_SND=y +CONFIG_SND_TIMER=y +CONFIG_SND_PCM=y +CONFIG_SND_SEQUENCER=y +# CONFIG_SND_SEQ_DUMMY is not set +CONFIG_SND_OSSEMUL=y +# CONFIG_SND_MIXER_OSS is not set +CONFIG_SND_PCM_OSS=y +CONFIG_SND_PCM_OSS_PLUGINS=y +# CONFIG_SND_SEQUENCER_OSS is not set +# CONFIG_SND_DYNAMIC_MINORS is not set +CONFIG_SND_SUPPORT_OLD_API=y +CONFIG_SND_VERBOSE_PROCFS=y +# CONFIG_SND_VERBOSE_PRINTK is not set +# CONFIG_SND_DEBUG is not set + +# +# Generic devices +# +CONFIG_SND_AC97_CODEC=y +CONFIG_SND_AC97_BUS=y +# CONFIG_SND_DUMMY is not set +# CONFIG_SND_VIRMIDI is not set +# CONFIG_SND_MTPAV is not set +# CONFIG_SND_SERIAL_U16550 is not set +# CONFIG_SND_MPU401 is not set + +# +# ALSA ARM devices +# +CONFIG_SND_PXA2XX_PCM=y +CONFIG_SND_PXA2XX_AC97=y + +# +# USB devices +# +# CONFIG_SND_USB_AUDIO is not set + +# +# SoC audio support +# +# CONFIG_SND_SOC is not set + +# +# Open Sound System +# +# CONFIG_SOUND_PRIME is not set + +# +# USB support +# +CONFIG_USB_ARCH_HAS_HCD=y +CONFIG_USB_ARCH_HAS_OHCI=y +# CONFIG_USB_ARCH_HAS_EHCI is not set +CONFIG_USB=y +# CONFIG_USB_DEBUG is not set + +# +# Miscellaneous USB options +# +CONFIG_USB_DEVICEFS=y +# CONFIG_USB_BANDWIDTH is not set +# CONFIG_USB_DYNAMIC_MINORS is not set +# CONFIG_USB_SUSPEND is not set +# CONFIG_USB_OTG is not set + +# +# USB Host Controller Drivers +# +# CONFIG_USB_ISP116X_HCD is not set +CONFIG_USB_OHCI_HCD=y +# CONFIG_USB_OHCI_BIG_ENDIAN is not set +CONFIG_USB_OHCI_LITTLE_ENDIAN=y +# CONFIG_USB_SL811_HCD is not set + +# +# USB Device Class drivers +# +# CONFIG_USB_ACM is not set +# CONFIG_USB_PRINTER is not set + +# +# NOTE: USB_STORAGE enables SCSI, and 'SCSI disk support' +# + +# +# may also be needed; see USB_STORAGE Help for more information +# +CONFIG_USB_STORAGE=y +# CONFIG_USB_STORAGE_DEBUG is not set +# CONFIG_USB_STORAGE_DATAFAB is not set +# CONFIG_USB_STORAGE_FREECOM is not set +# CONFIG_USB_STORAGE_ISD200 is not set +# CONFIG_USB_STORAGE_DPCM is not set +# CONFIG_USB_STORAGE_USBAT is not set +# CONFIG_USB_STORAGE_SDDR09 is not set +# CONFIG_USB_STORAGE_SDDR55 is not set +# CONFIG_USB_STORAGE_JUMPSHOT is not set +# CONFIG_USB_STORAGE_ALAUDA is not set +# CONFIG_USB_STORAGE_KARMA is not set +CONFIG_USB_LIBUSUAL=y + +# +# USB Input Devices +# +CONFIG_USB_HID=y +CONFIG_USB_HIDINPUT=y +# CONFIG_USB_HIDINPUT_POWERBOOK is not set +# CONFIG_HID_FF is not set +# CONFIG_USB_HIDDEV is not set +# CONFIG_USB_AIPTEK is not set +# CONFIG_USB_WACOM is not set +# CONFIG_USB_ACECAD is not set +# CONFIG_USB_KBTAB is not set +# CONFIG_USB_POWERMATE is not set +# CONFIG_USB_TOUCHSCREEN is not set +# CONFIG_USB_YEALINK is not set +# CONFIG_USB_XPAD is not set +# CONFIG_USB_ATI_REMOTE is not set +# CONFIG_USB_ATI_REMOTE2 is not set +# CONFIG_USB_KEYSPAN_REMOTE is not set +# CONFIG_USB_APPLETOUCH is not set + +# +# USB Imaging devices +# +# CONFIG_USB_MDC800 is not set +# CONFIG_USB_MICROTEK is not set + +# +# USB Network Adapters +# +# CONFIG_USB_CATC is not set +# CONFIG_USB_KAWETH is not set +# CONFIG_USB_PEGASUS is not set +# CONFIG_USB_RTL8150 is not set +CONFIG_USB_USBNET_MII=m +CONFIG_USB_USBNET=m +CONFIG_USB_NET_AX8817X=m +CONFIG_USB_NET_CDCETHER=m +# CONFIG_USB_NET_GL620A is not set +CONFIG_USB_NET_NET1080=m +# CONFIG_USB_NET_PLUSB is not set +# CONFIG_USB_NET_MCS7830 is not set +CONFIG_USB_NET_RNDIS_HOST=m +# CONFIG_USB_NET_CDC_SUBSET is not set +CONFIG_USB_NET_ZAURUS=m +CONFIG_USB_MON=y + +# +# USB port drivers +# + +# +# USB Serial Converter support +# +CONFIG_USB_SERIAL=y +# CONFIG_USB_SERIAL_CONSOLE is not set +CONFIG_USB_SERIAL_GENERIC=y +# CONFIG_USB_SERIAL_AIRCABLE is not set +# CONFIG_USB_SERIAL_AIRPRIME is not set +# CONFIG_USB_SERIAL_ARK3116 is not set +# CONFIG_USB_SERIAL_BELKIN is not set +# CONFIG_USB_SERIAL_WHITEHEAT is not set +# CONFIG_USB_SERIAL_DIGI_ACCELEPORT is not set +# CONFIG_USB_SERIAL_CP2101 is not set +# CONFIG_USB_SERIAL_CYPRESS_M8 is not set +# CONFIG_USB_SERIAL_EMPEG is not set +# CONFIG_USB_SERIAL_FTDI_SIO is not set +# CONFIG_USB_SERIAL_FUNSOFT is not set +# CONFIG_USB_SERIAL_VISOR is not set +# CONFIG_USB_SERIAL_IPAQ is not set +# CONFIG_USB_SERIAL_IR is not set +# CONFIG_USB_SERIAL_EDGEPORT is not set +# CONFIG_USB_SERIAL_EDGEPORT_TI is not set +# CONFIG_USB_SERIAL_GARMIN is not set +# CONFIG_USB_SERIAL_IPW is not set +# CONFIG_USB_SERIAL_KEYSPAN_PDA is not set +# CONFIG_USB_SERIAL_KEYSPAN is not set +# CONFIG_USB_SERIAL_KLSI is not set +# CONFIG_USB_SERIAL_KOBIL_SCT is not set +# CONFIG_USB_SERIAL_MCT_U232 is not set +# CONFIG_USB_SERIAL_MOS7720 is not set +# CONFIG_USB_SERIAL_MOS7840 is not set +# CONFIG_USB_SERIAL_NAVMAN is not set +CONFIG_USB_SERIAL_PL2303=y +# CONFIG_USB_SERIAL_HP4X is not set +# CONFIG_USB_SERIAL_SAFE is not set +# CONFIG_USB_SERIAL_SIERRAWIRELESS is not set +# CONFIG_USB_SERIAL_TI is not set +# CONFIG_USB_SERIAL_CYBERJACK is not set +# CONFIG_USB_SERIAL_XIRCOM is not set +# CONFIG_USB_SERIAL_OPTION is not set +# CONFIG_USB_SERIAL_OMNINET is not set + +# +# USB Miscellaneous drivers +# +# CONFIG_USB_EMI62 is not set +# CONFIG_USB_EMI26 is not set +# CONFIG_USB_ADUTUX is not set +# CONFIG_USB_AUERSWALD is not set +# CONFIG_USB_RIO500 is not set +# CONFIG_USB_LEGOTOWER is not set +# CONFIG_USB_LCD is not set +# CONFIG_USB_LED is not set +# CONFIG_USB_CYPRESS_CY7C63 is not set +# CONFIG_USB_CYTHERM is not set +# CONFIG_USB_PHIDGET is not set +# CONFIG_USB_IDMOUSE is not set +# CONFIG_USB_FTDI_ELAN is not set +# CONFIG_USB_APPLEDISPLAY is not set +# CONFIG_USB_LD is not set +# CONFIG_USB_TRANCEVIBRATOR is not set +# CONFIG_USB_TEST is not set + +# +# USB DSL modem support +# + +# +# USB Gadget Support +# +CONFIG_USB_GADGET=m +# CONFIG_USB_GADGET_DEBUG_FILES is not set +CONFIG_USB_GADGET_SELECTED=y +# CONFIG_USB_GADGET_NET2280 is not set +# CONFIG_USB_GADGET_PXA2XX is not set +CONFIG_USB_GADGET_PXA27X=y +CONFIG_USB_PXA27X=m +# CONFIG_USB_GADGET_GOKU is not set +# CONFIG_USB_GADGET_LH7A40X is not set +# CONFIG_USB_GADGET_OMAP is not set +# CONFIG_USB_GADGET_AT91 is not set +# CONFIG_USB_GADGET_DUMMY_HCD is not set +# CONFIG_USB_GADGET_DUALSPEED is not set +# CONFIG_USB_ZERO is not set +CONFIG_USB_ETH=m +CONFIG_USB_ETH_RNDIS=y +CONFIG_USB_GADGETFS=m +CONFIG_USB_FILE_STORAGE=m +CONFIG_USB_FILE_STORAGE_TEST=y +# CONFIG_USB_G_SERIAL is not set +# CONFIG_USB_MIDI_GADGET is not set + +# +# MMC/SD Card support +# +# CONFIG_MMC is not set + +# +# Real Time Clock +# +CONFIG_RTC_LIB=y +CONFIG_RTC_CLASS=y +CONFIG_RTC_HCTOSYS=y +CONFIG_RTC_HCTOSYS_DEVICE="rtc0" +# CONFIG_RTC_DEBUG is not set + +# +# RTC interfaces +# +CONFIG_RTC_INTF_SYSFS=y +CONFIG_RTC_INTF_PROC=y +CONFIG_RTC_INTF_DEV=y +# CONFIG_RTC_INTF_DEV_UIE_EMUL is not set + +# +# RTC drivers +# +# CONFIG_RTC_DRV_DS1553 is not set +# CONFIG_RTC_DRV_DS1742 is not set +# CONFIG_RTC_DRV_M48T86 is not set +CONFIG_RTC_DRV_SA1100=y +# CONFIG_RTC_DRV_TEST is not set +# CONFIG_RTC_DRV_V3020 is not set + +# +# File systems +# +CONFIG_EXT2_FS=y +CONFIG_EXT2_FS_XATTR=y +CONFIG_EXT2_FS_POSIX_ACL=y +CONFIG_EXT2_FS_SECURITY=y +CONFIG_EXT2_FS_XIP=y +CONFIG_FS_XIP=y +CONFIG_EXT3_FS=y +CONFIG_EXT3_FS_XATTR=y +CONFIG_EXT3_FS_POSIX_ACL=y +CONFIG_EXT3_FS_SECURITY=y +# CONFIG_EXT4DEV_FS is not set +CONFIG_JBD=y +# CONFIG_JBD_DEBUG is not set +CONFIG_FS_MBCACHE=y +# CONFIG_REISERFS_FS is not set +# CONFIG_JFS_FS is not set +CONFIG_FS_POSIX_ACL=y +# CONFIG_XFS_FS is not set +# CONFIG_GFS2_FS is not set +# CONFIG_OCFS2_FS is not set +# CONFIG_MINIX_FS is not set +# CONFIG_ROMFS_FS is not set +CONFIG_INOTIFY=y +CONFIG_INOTIFY_USER=y +# CONFIG_QUOTA is not set +CONFIG_DNOTIFY=y +# CONFIG_AUTOFS_FS is not set +# CONFIG_AUTOFS4_FS is not set +# CONFIG_FUSE_FS is not set + +# +# CD-ROM/DVD Filesystems +# +# CONFIG_ISO9660_FS is not set +# CONFIG_UDF_FS is not set + +# +# DOS/FAT/NT Filesystems +# +CONFIG_FAT_FS=y +CONFIG_MSDOS_FS=y +CONFIG_VFAT_FS=y +CONFIG_FAT_DEFAULT_CODEPAGE=437 +CONFIG_FAT_DEFAULT_IOCHARSET="iso8859-1" +# CONFIG_NTFS_FS is not set + +# +# Pseudo filesystems +# +CONFIG_PROC_FS=y +CONFIG_PROC_SYSCTL=y +CONFIG_SYSFS=y +CONFIG_TMPFS=y +# CONFIG_TMPFS_POSIX_ACL is not set +# CONFIG_HUGETLB_PAGE is not set +CONFIG_RAMFS=y +CONFIG_CONFIGFS_FS=y + +# +# Miscellaneous filesystems +# +# CONFIG_ADFS_FS is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_HFSPLUS_FS is not set +# CONFIG_BEFS_FS is not set +# CONFIG_BFS_FS is not set +# CONFIG_EFS_FS is not set +# CONFIG_JFFS_FS is not set +CONFIG_JFFS2_FS=y +CONFIG_JFFS2_FS_DEBUG=0 +CONFIG_JFFS2_FS_WRITEBUFFER=y +# CONFIG_JFFS2_SUMMARY is not set +# CONFIG_JFFS2_FS_XATTR is not set +# CONFIG_JFFS2_COMPRESSION_OPTIONS is not set +CONFIG_JFFS2_ZLIB=y +CONFIG_JFFS2_RTIME=y +# CONFIG_JFFS2_RUBIN is not set +# CONFIG_CRAMFS is not set +# CONFIG_VXFS_FS is not set +# CONFIG_HPFS_FS is not set +# CONFIG_QNX4FS_FS is not set +# CONFIG_SYSV_FS is not set +# CONFIG_UFS_FS is not set + +# +# Network File Systems +# +CONFIG_NFS_FS=y +CONFIG_NFS_V3=y +# CONFIG_NFS_V3_ACL is not set +# CONFIG_NFS_V4 is not set +# CONFIG_NFS_DIRECTIO is not set +# CONFIG_NFSD is not set +CONFIG_ROOT_NFS=y +CONFIG_LOCKD=y +CONFIG_LOCKD_V4=y +CONFIG_NFS_COMMON=y +CONFIG_SUNRPC=y +# CONFIG_RPCSEC_GSS_KRB5 is not set +# CONFIG_RPCSEC_GSS_SPKM3 is not set +# CONFIG_SMB_FS is not set +# CONFIG_CIFS is not set +# CONFIG_NCP_FS is not set +# CONFIG_CODA_FS is not set +# CONFIG_AFS_FS is not set +# CONFIG_9P_FS is not set + +# +# Partition Types +# +# CONFIG_PARTITION_ADVANCED is not set +CONFIG_MSDOS_PARTITION=y + +# +# Native Language Support +# +CONFIG_NLS=y +CONFIG_NLS_DEFAULT="iso8859-1" +CONFIG_NLS_CODEPAGE_437=y +# CONFIG_NLS_CODEPAGE_737 is not set +# CONFIG_NLS_CODEPAGE_775 is not set +# CONFIG_NLS_CODEPAGE_850 is not set +# CONFIG_NLS_CODEPAGE_852 is not set +# CONFIG_NLS_CODEPAGE_855 is not set +# CONFIG_NLS_CODEPAGE_857 is not set +# CONFIG_NLS_CODEPAGE_860 is not set +# CONFIG_NLS_CODEPAGE_861 is not set +# CONFIG_NLS_CODEPAGE_862 is not set +# CONFIG_NLS_CODEPAGE_863 is not set +# CONFIG_NLS_CODEPAGE_864 is not set +# CONFIG_NLS_CODEPAGE_865 is not set +# CONFIG_NLS_CODEPAGE_866 is not set +# CONFIG_NLS_CODEPAGE_869 is not set +# CONFIG_NLS_CODEPAGE_936 is not set +# CONFIG_NLS_CODEPAGE_950 is not set +# CONFIG_NLS_CODEPAGE_932 is not set +# CONFIG_NLS_CODEPAGE_949 is not set +# CONFIG_NLS_CODEPAGE_874 is not set +# CONFIG_NLS_ISO8859_8 is not set +# CONFIG_NLS_CODEPAGE_1250 is not set +# CONFIG_NLS_CODEPAGE_1251 is not set +# CONFIG_NLS_ASCII is not set +CONFIG_NLS_ISO8859_1=y +# CONFIG_NLS_ISO8859_2 is not set +# CONFIG_NLS_ISO8859_3 is not set +# CONFIG_NLS_ISO8859_4 is not set +# CONFIG_NLS_ISO8859_5 is not set +# CONFIG_NLS_ISO8859_6 is not set +# CONFIG_NLS_ISO8859_7 is not set +# CONFIG_NLS_ISO8859_9 is not set +# CONFIG_NLS_ISO8859_13 is not set +# CONFIG_NLS_ISO8859_14 is not set +# CONFIG_NLS_ISO8859_15 is not set +# CONFIG_NLS_KOI8_R is not set +# CONFIG_NLS_KOI8_U is not set +# CONFIG_NLS_UTF8 is not set + +# +# Profiling support +# +# CONFIG_PROFILING is not set + +# +# Kernel hacking +# +# CONFIG_PRINTK_TIME is not set +CONFIG_ENABLE_MUST_CHECK=y +CONFIG_MAGIC_SYSRQ=y +# CONFIG_UNUSED_SYMBOLS is not set +CONFIG_DEBUG_KERNEL=y +CONFIG_LOG_BUF_SHIFT=14 +CONFIG_DETECT_SOFTLOCKUP=y +# CONFIG_SCHEDSTATS is not set +# CONFIG_DEBUG_SLAB is not set +# CONFIG_DEBUG_RT_MUTEXES is not set +# CONFIG_RT_MUTEX_TESTER is not set +# CONFIG_DEBUG_SPINLOCK is not set +# CONFIG_DEBUG_MUTEXES is not set +# CONFIG_DEBUG_RWSEMS is not set +# CONFIG_DEBUG_SPINLOCK_SLEEP is not set +# CONFIG_DEBUG_LOCKING_API_SELFTESTS is not set +# CONFIG_DEBUG_KOBJECT is not set +# CONFIG_DEBUG_BUGVERBOSE is not set +CONFIG_DEBUG_INFO=y +# CONFIG_DEBUG_FS is not set +# CONFIG_DEBUG_VM is not set +# CONFIG_DEBUG_LIST is not set +CONFIG_FRAME_POINTER=y +CONFIG_FORCED_INLINING=y +# CONFIG_HEADERS_CHECK is not set +# CONFIG_RCU_TORTURE_TEST is not set +CONFIG_DEBUG_USER=y +# CONFIG_DEBUG_WAITQ is not set +CONFIG_DEBUG_ERRORS=y +CONFIG_DEBUG_LL=y +# CONFIG_DEBUG_ICEDCC is not set + +# +# Security options +# +# CONFIG_KEYS is not set +# CONFIG_SECURITY is not set + +# +# Cryptographic options +# +CONFIG_CRYPTO=y +CONFIG_CRYPTO_ALGAPI=y +CONFIG_CRYPTO_BLKCIPHER=y +CONFIG_CRYPTO_HASH=y +CONFIG_CRYPTO_MANAGER=y +CONFIG_CRYPTO_HMAC=y +# CONFIG_CRYPTO_NULL is not set +# CONFIG_CRYPTO_MD4 is not set +CONFIG_CRYPTO_MD5=y +CONFIG_CRYPTO_SHA1=y +# CONFIG_CRYPTO_SHA256 is not set +# CONFIG_CRYPTO_SHA512 is not set +# CONFIG_CRYPTO_WP512 is not set +# CONFIG_CRYPTO_TGR192 is not set +CONFIG_CRYPTO_ECB=y +CONFIG_CRYPTO_CBC=y +CONFIG_CRYPTO_DES=y +# CONFIG_CRYPTO_BLOWFISH is not set +# CONFIG_CRYPTO_TWOFISH is not set +# CONFIG_CRYPTO_SERPENT is not set +# CONFIG_CRYPTO_AES is not set +# CONFIG_CRYPTO_CAST5 is not set +# CONFIG_CRYPTO_CAST6 is not set +# CONFIG_CRYPTO_TEA is not set +CONFIG_CRYPTO_ARC4=y +# CONFIG_CRYPTO_KHAZAD is not set +# CONFIG_CRYPTO_ANUBIS is not set +CONFIG_CRYPTO_DEFLATE=y +# CONFIG_CRYPTO_MICHAEL_MIC is not set +# CONFIG_CRYPTO_CRC32C is not set +# CONFIG_CRYPTO_TEST is not set + +# +# Hardware crypto devices +# + +# +# Library routines +# +# CONFIG_CRC_CCITT is not set +# CONFIG_CRC16 is not set +CONFIG_CRC32=y +# CONFIG_LIBCRC32C is not set +CONFIG_ZLIB_INFLATE=y +CONFIG_ZLIB_DEFLATE=y +CONFIG_PLIST=y diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/input_power-r6.patch b/packages/linux/logicpd-pxa270-2.6.19.2/input_power-r6.patch new file mode 100644 index 0000000000..c0dd9125fc --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/input_power-r6.patch @@ -0,0 +1,222 @@ + drivers/input/Kconfig | 15 ++++ + drivers/input/power.c | 152 ++++++++++++++++++++++++++++++++++---------------- + 2 files changed, 121 insertions(+), 46 deletions(-) + +Index: git/drivers/input/power.c +=================================================================== +--- git.orig/drivers/input/power.c 2006-10-31 16:31:03.000000000 +0000 ++++ git/drivers/input/power.c 2006-10-31 17:38:34.000000000 +0000 +@@ -2,6 +2,7 @@ + * $Id: power.c,v 1.10 2001/09/25 09:17:15 vojtech Exp $ + * + * Copyright (c) 2001 "Crazy" James Simmons ++ * Copyright (c) 2005 Dmitry Torokhov + * + * Input driver Power Management. + * +@@ -34,66 +35,125 @@ + #include <linux/tty.h> + #include <linux/delay.h> + #include <linux/pm.h> ++#ifdef CONFIG_ACPI ++#include <linux/acpi.h> ++#endif ++#ifdef CONFIG_APM ++#if defined(CONFIG_ARM) && !defined(CONFIG_ARM26) ++#include <asm/apm.h> ++#endif ++#endif ++ + + static struct input_handler power_handler; ++//static long suspend_time_pressed; ++//static DEFINE_SPINLOCK(suspend_time_lock); + +-/* +- * Power management can't be done in a interrupt context. So we have to +- * use keventd. +- */ +-static int suspend_button_pushed = 0; +-static void suspend_button_task_handler(void *data) ++#ifdef CONFIG_ACPI ++/* FIXME: This should be in ACPI */ ++static void acpi_queue_event(int suspend) + { +- udelay(200); /* debounce */ +- suspend_button_pushed = 0; +-} ++ struct acpi_device dev; + +-static DECLARE_WORK(suspend_button_task, suspend_button_task_handler, NULL); ++ sprintf(acpi_device_name(&dev), "%s", suspend ? "KEY_SUSPEND" : "KEY_POWER"); ++ sprintf(acpi_device_class(&dev), "button/%s", suspend ? "sleep" : "power"); ++ acpi_bus_generate_event(&dev, 1, 1); ++} ++#endif + +-static void power_event(struct input_handle *handle, unsigned int type, +- unsigned int code, int down) ++static void system_power_event(unsigned int keycode) + { +- struct input_dev *dev = handle->dev; ++// unsigned long flags; + +- printk("Entering power_event\n"); ++ switch (keycode) { ++ case KEY_SUSPEND: + +- if (type == EV_PWR) { +- switch (code) { +- case KEY_SUSPEND: +- printk("Powering down entire device\n"); ++ /* ignore jitter */ ++ // spin_lock_irqsave(&suspend_time_lock, flags); ++ // if (time_before(jiffies, suspend_time_pressed + msecs_to_jiffies(200))) { ++ // spin_unlock_irqrestore(&suspend_time_lock, flags); ++ // break; ++ // } ++ // suspend_time_pressed = jiffies; ++ // spin_unlock_irqrestore(&suspend_time_lock, flags); + +- if (!suspend_button_pushed) { +- suspend_button_pushed = 1; +- schedule_work(&suspend_button_task); +- } ++ // if (!PM_IS_ACTIVE()) { ++ // printk(KERN_INFO "power: PM is not active, ignoring suspend request\n"); ++ // break; ++ // } ++ ++ printk(KERN_INFO "power: requesting system suspend...\n"); ++#ifdef CONFIG_ACPI ++ if (!acpi_disabled) { ++ acpi_queue_event(1); + break; +- case KEY_POWER: +- /* Hum power down the machine. */ ++ } ++#endif ++ ++#ifdef CONFIG_APM ++#if defined(CONFIG_ARM) && !defined(CONFIG_ARM26) ++ /* only ARM has apm_queue_event */ ++ apm_queue_event(APM_USER_SUSPEND); ++#endif ++#endif ++ break; ++ ++ case KEY_POWER: ++ ++ // if (!PM_IS_ACTIVE()) { ++ // printk(KERN_INFO "power: PM is not active, ignoring shutdown request\n"); ++ // break; ++ // } ++#ifdef CONFIG_ACPI ++ if (!acpi_disabled) { ++ printk(KERN_INFO "power: requesting system shutdown...\n"); ++ acpi_queue_event(0); + break; +- default: +- return; +- } ++ } ++#endif ++ ++#ifdef CONFIG_APM ++ printk(KERN_INFO "power: shutdown via APM is not supported...\n"); ++#endif ++ break; ++ ++ default: ++ break; + } ++} + +- if (type == EV_KEY) { +- switch (code) { +- case KEY_SUSPEND: +- printk("Powering down input device\n"); +- /* This is risky. See pm.h for details. */ +- if (dev->state != PM_RESUME) +- dev->state = PM_RESUME; +- else +- dev->state = PM_SUSPEND; +- pm_send(dev->pm_dev, dev->state, dev); +- break; +- case KEY_POWER: +- /* Turn the input device off completely ? */ +- break; +- default: +- return; +- } ++static void device_power_event(struct input_device *dev, unsigned int keycode) ++{ ++ switch (keycode) { ++ case KEY_SUSPEND: ++ case KEY_POWER: ++ printk(KERN_DEBUG "power.c: device-level power management is not supported yet\n"); ++ break; ++ ++ default: ++ break; ++ } ++} ++ ++static void power_event(struct input_handle *handle, unsigned int type, ++ unsigned int code, int value) ++{ ++ /* only react on key down events */ ++ if (value != 1) ++ return; ++ ++ switch (type) { ++ case EV_PWR: ++ system_power_event(code); ++ break; ++ ++ case EV_KEY: ++ device_power_event(handle->dev, code); ++ break; ++ ++ default: ++ break; + } +- return; + } + + static struct input_handle *power_connect(struct input_handler *handler, +@@ -107,7 +167,7 @@ static struct input_handle *power_connec + + handle->dev = dev; + handle->handler = handler; +- ++ handle->name = "power"; + input_open_device(handle); + + printk(KERN_INFO "power.c: Adding power management to input layer\n"); +Index: git/drivers/input/Kconfig +=================================================================== +--- git.orig/drivers/input/Kconfig 2006-10-31 16:31:03.000000000 +0000 ++++ git/drivers/input/Kconfig 2006-10-31 16:31:07.000000000 +0000 +@@ -144,6 +144,21 @@ config INPUT_EVBUG + + To compile this driver as a module, choose M here: the + module will be called evbug. ++ ++config INPUT_POWER ++ tristate "Power management interface" ++ depends on INPUT ++ ---help--- ++ Say Y here if you have an input device (keyboard) that has ++ sleep/power keys and you would like use these keys to ++ initiate power management operations. Note that on many ++ machines power and sleep keys are not part of the input ++ system and only accessible when using ACPI button driver. ++ ++ If unsure, say N. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called power. + + comment "Input Device Drivers" + diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/kexec-arm-r3.patch b/packages/linux/logicpd-pxa270-2.6.19.2/kexec-arm-r3.patch new file mode 100644 index 0000000000..e107f6de4a --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/kexec-arm-r3.patch @@ -0,0 +1,275 @@ +Index: linux-2.6.17/arch/arm/Kconfig +=================================================================== +--- linux-2.6.17.orig/arch/arm/Kconfig 2006-06-20 13:22:58.163642800 +0200 ++++ linux-2.6.17/arch/arm/Kconfig 2006-06-20 13:22:59.596424984 +0200 +@@ -326,6 +326,23 @@ + depends on CPU_XSCALE && !XSCALE_PMU_TIMER + default y + ++config KEXEC ++ bool "Kexec system call (EXPERIMENTAL)" ++ depends on EXPERIMENTAL ++ help ++ kexec is a system call that implements the ability to shutdown your ++ current kernel, and to start another kernel. It is like a reboot ++ but it is indepedent of the system firmware. And like a reboot ++ you can start any kernel with it, not just Linux. ++ ++ The name comes from the similiarity to the exec system call. ++ ++ It is an ongoing process to be certain the hardware in a machine ++ is properly shutdown, so do not be surprised if this code does not ++ initially work for you. It may help to enable device hotplugging ++ support. As of this writing the exact hardware interface is ++ strongly in flux, so no good recommendation can be made. ++ + endmenu + + source "arch/arm/common/Kconfig" +Index: linux-2.6.17/arch/arm/kernel/calls.S +=================================================================== +--- linux-2.6.17.orig/arch/arm/kernel/calls.S 2006-06-20 13:22:50.813760152 +0200 ++++ linux-2.6.17/arch/arm/kernel/calls.S 2006-06-20 13:22:59.596424984 +0200 +@@ -198,7 +198,7 @@ + CALL(sys_sigaltstack_wrapper) + CALL(sys_sendfile) + CALL(sys_ni_syscall) +- CALL(sys_ni_syscall) ++ CALL(sys_kexec_load) + /* 190 */ CALL(sys_vfork_wrapper) + CALL(sys_getrlimit) + CALL(sys_mmap2) +Index: linux-2.6.17/arch/arm/kernel/machine_kexec.c +=================================================================== +--- /dev/null 1970-01-01 00:00:00.000000000 +0000 ++++ linux-2.6.17/arch/arm/kernel/machine_kexec.c 2006-06-23 11:21:58.195936184 +0200 +@@ -0,0 +1,78 @@ ++/* ++ * machine_kexec.c - handle transition of Linux booting another kernel ++ */ ++ ++#include <linux/mm.h> ++#include <linux/kexec.h> ++#include <linux/delay.h> ++#include <linux/reboot.h> ++#include <asm/pgtable.h> ++#include <asm/pgalloc.h> ++#include <asm/mmu_context.h> ++#include <asm/io.h> ++#include <asm/cacheflush.h> ++#include <asm/mach-types.h> ++ ++const extern unsigned char relocate_new_kernel[]; ++const extern unsigned int relocate_new_kernel_size; ++ ++extern void setup_mm_for_reboot(char mode); ++ ++extern unsigned long kexec_start_address; ++extern unsigned long kexec_indirection_page; ++extern unsigned long kexec_mach_type; ++ ++/* ++ * Provide a dummy crash_notes definition while crash dump arrives to arm. ++ * This prevents breakage of crash_notes attribute in kernel/ksysfs.c. ++ */ ++ ++int machine_kexec_prepare(struct kimage *image) ++{ ++ return 0; ++} ++ ++void machine_kexec_cleanup(struct kimage *image) ++{ ++} ++ ++void machine_shutdown(void) ++{ ++} ++ ++void machine_crash_shutdown(struct pt_regs *regs) ++{ ++} ++ ++void machine_kexec(struct kimage *image) ++{ ++ unsigned long page_list; ++ unsigned long reboot_code_buffer_phys; ++ void *reboot_code_buffer; ++ ++ ++ page_list = image->head & PAGE_MASK; ++ ++ /* we need both effective and real address here */ ++ reboot_code_buffer_phys = ++ page_to_pfn(image->control_code_page) << PAGE_SHIFT; ++ reboot_code_buffer = page_address(image->control_code_page); ++ ++ /* Prepare parameters for reboot_code_buffer*/ ++ kexec_start_address = image->start; ++ kexec_indirection_page = page_list; ++ kexec_mach_type = machine_arch_type; ++ ++ /* copy our kernel relocation code to the control code page */ ++ memcpy(reboot_code_buffer, ++ relocate_new_kernel, relocate_new_kernel_size); ++ ++ ++ flush_icache_range((unsigned long) reboot_code_buffer, ++ (unsigned long) reboot_code_buffer + KEXEC_CONTROL_CODE_SIZE); ++ printk(KERN_INFO "Bye!\n"); ++ ++ cpu_proc_fin(); ++ setup_mm_for_reboot(0); /* mode is not used, so just pass 0*/ ++ cpu_reset(reboot_code_buffer_phys); ++} +Index: linux-2.6.17/arch/arm/kernel/Makefile +=================================================================== +--- linux-2.6.17.orig/arch/arm/kernel/Makefile 2006-06-20 13:22:50.814760000 +0200 ++++ linux-2.6.17/arch/arm/kernel/Makefile 2006-06-20 13:22:59.597424832 +0200 +@@ -20,6 +20,7 @@ + obj-$(CONFIG_ISA_DMA) += dma-isa.o + obj-$(CONFIG_PCI) += bios32.o + obj-$(CONFIG_SMP) += smp.o ++obj-$(CONFIG_KEXEC) += machine_kexec.o relocate_kernel.o + obj-$(CONFIG_OABI_COMPAT) += sys_oabi-compat.o + + obj-$(CONFIG_IWMMXT) += iwmmxt.o +Index: linux-2.6.17/arch/arm/kernel/relocate_kernel.S +=================================================================== +--- /dev/null 1970-01-01 00:00:00.000000000 +0000 ++++ linux-2.6.17/arch/arm/kernel/relocate_kernel.S 2006-06-20 13:22:59.597424832 +0200 +@@ -0,0 +1,74 @@ ++/* ++ * relocate_kernel.S - put the kernel image in place to boot ++ */ ++ ++#include <asm/kexec.h> ++ ++ .globl relocate_new_kernel ++relocate_new_kernel: ++ ++ ldr r0,kexec_indirection_page ++ ldr r1,kexec_start_address ++ ++ ++0: /* top, read another word for the indirection page */ ++ ldr r3, [r0],#4 ++ ++ /* Is it a destination page. Put destination address to r4 */ ++ tst r3,#1,0 ++ beq 1f ++ bic r4,r3,#1 ++ b 0b ++1: ++ /* Is it an indirection page */ ++ tst r3,#2,0 ++ beq 1f ++ bic r0,r3,#2 ++ b 0b ++1: ++ ++ /* are we done ? */ ++ tst r3,#4,0 ++ beq 1f ++ b 2f ++ ++1: ++ /* is it source ? */ ++ tst r3,#8,0 ++ beq 0b ++ bic r3,r3,#8 ++ mov r6,#1024 ++9: ++ ldr r5,[r3],#4 ++ str r5,[r4],#4 ++ subs r6,r6,#1 ++ bne 9b ++ b 0b ++ ++2: ++ /* Jump to relocated kernel */ ++ mov lr,r1 ++ mov r0,#0 ++ ldr r1,kexec_mach_type ++ mov r2,#0 ++ mov pc,lr ++ ++ .globl kexec_start_address ++kexec_start_address: ++ .long 0x0 ++ ++ .globl kexec_indirection_page ++kexec_indirection_page: ++ .long 0x0 ++ ++ .globl kexec_mach_type ++kexec_mach_type: ++ .long 0x0 ++ ++relocate_new_kernel_end: ++ ++ .globl relocate_new_kernel_size ++relocate_new_kernel_size: ++ .long relocate_new_kernel_end - relocate_new_kernel ++ ++ +Index: linux-2.6.17/include/asm-arm/kexec.h +=================================================================== +--- /dev/null 1970-01-01 00:00:00.000000000 +0000 ++++ linux-2.6.17/include/asm-arm/kexec.h 2006-06-20 13:22:59.598424680 +0200 +@@ -0,0 +1,30 @@ ++#ifndef _ARM_KEXEC_H ++#define _ARM_KEXEC_H ++ ++#ifdef CONFIG_KEXEC ++ ++/* Maximum physical address we can use pages from */ ++#define KEXEC_SOURCE_MEMORY_LIMIT (-1UL) ++/* Maximum address we can reach in physical address mode */ ++#define KEXEC_DESTINATION_MEMORY_LIMIT (-1UL) ++/* Maximum address we can use for the control code buffer */ ++#define KEXEC_CONTROL_MEMORY_LIMIT TASK_SIZE ++ ++#define KEXEC_CONTROL_CODE_SIZE 4096 ++ ++#define KEXEC_ARCH KEXEC_ARCH_ARM ++ ++#ifndef __ASSEMBLY__ ++ ++#define MAX_NOTE_BYTES 1024 ++ ++struct kimage; ++/* Provide a dummy definition to avoid build failures. */ ++static inline void crash_setup_regs(struct pt_regs *newregs, ++ struct pt_regs *oldregs) { } ++ ++#endif /* __ASSEMBLY__ */ ++ ++#endif /* CONFIG_KEXEC */ ++ ++#endif /* _ARM_KEXEC_H */ +Index: linux-2.6.17/include/asm-arm/unistd.h +=================================================================== +--- linux-2.6.17.orig/include/asm-arm/unistd.h 2006-06-20 13:22:50.815759848 +0200 ++++ linux-2.6.17/include/asm-arm/unistd.h 2006-06-20 13:22:59.598424680 +0200 +@@ -216,7 +216,7 @@ + #define __NR_sigaltstack (__NR_SYSCALL_BASE+186) + #define __NR_sendfile (__NR_SYSCALL_BASE+187) + /* 188 reserved */ +- /* 189 reserved */ ++#define __NR_sys_kexec_load (__NR_SYSCALL_BASE+189) /* 189 was reserved, temporarily use it for sys_kexec_load */ + #define __NR_vfork (__NR_SYSCALL_BASE+190) + #define __NR_ugetrlimit (__NR_SYSCALL_BASE+191) /* SuS compliant getrlimit */ + #define __NR_mmap2 (__NR_SYSCALL_BASE+192) +Index: linux-2.6.17/include/linux/kexec.h +=================================================================== +--- linux-2.6.17.orig/include/linux/kexec.h 2006-06-20 13:22:50.815759848 +0200 ++++ linux-2.6.17/include/linux/kexec.h 2006-06-20 13:22:59.600424376 +0200 +@@ -119,6 +119,7 @@ + #define KEXEC_ARCH_PPC (20 << 16) + #define KEXEC_ARCH_PPC64 (21 << 16) + #define KEXEC_ARCH_IA_64 (50 << 16) ++#define KEXEC_ARCH_ARM (40 << 16) + #define KEXEC_ARCH_S390 (22 << 16) + #define KEXEC_ARCH_SH (42 << 16) + diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/logicpd-pxa270-cf-hack.patch b/packages/linux/logicpd-pxa270-2.6.19.2/logicpd-pxa270-cf-hack.patch new file mode 100644 index 0000000000..ec1256261b --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/logicpd-pxa270-cf-hack.patch @@ -0,0 +1,727 @@ +Index: drivers/block/Kconfig +=================================================================== +RCS file: /cvs/eps/dev_eng/sw/products/Linux/PXAEngine/pxa/linux-2.6.17-rc5/drivers/block/Kconfig,v +retrieving revision 1.1.1.1 +retrieving revision 1.2 +diff -c -3 -p -r1.1.1.1 -r1.2 +*** drivers/block/Kconfig 29 May 2006 00:55:20 -0000 1.1.1.1 +--- drivers/block/Kconfig 1 Jun 2006 17:05:41 -0000 1.2 +*************** +*** 4,9 **** +--- 4,16 ---- + + menu "Block devices" + ++ config BLK_DEV_LOGICPD_CF ++ bool "LogicPD memory-mapped CompactFlash card support" ++ depends on MACH_LOGICPD_PXA270 ++ ---help--- ++ If you want to use the memory-mapped comapct flash card on ++ the LogicPD SDK, say Y. ++ + config BLK_DEV_FD + tristate "Normal floppy disk support" + depends on ARCH_MAY_HAVE_PC_FDC +Index: drivers/block/Makefile +=================================================================== +RCS file: /cvs/eps/dev_eng/sw/products/Linux/PXAEngine/pxa/linux-2.6.17-rc5/drivers/block/Makefile,v +retrieving revision 1.1.1.1 +retrieving revision 1.2 +diff -c -3 -p -r1.1.1.1 -r1.2 +*** drivers/block/Makefile 29 May 2006 00:55:20 -0000 1.1.1.1 +--- drivers/block/Makefile 1 Jun 2006 17:05:59 -0000 1.2 +*************** +*** 5,10 **** +--- 5,12 ---- + # Rewritten to use lists instead of if-statements. + # + ++ obj-$(CONFIG_BLK_DEV_LOGICPD_CF)+= lpd270-cf.o ++ + obj-$(CONFIG_MAC_FLOPPY) += swim3.o + obj-$(CONFIG_BLK_DEV_FD) += floppy.o + obj-$(CONFIG_AMIGA_FLOPPY) += amiflop.o +Index: drivers/block/lpd270-cf.c +=================================================================== +RCS file: drivers/block/lpd270-cf.c +diff -N drivers/block/lpd270-cf.c +*** /dev/null 1 Jan 1970 00:00:00 -0000 +--- drivers/block/lpd270-cf.c 1 Jun 2006 16:23:35 -0000 1.1 +*************** +*** 0 **** +--- 1,675 ---- ++ /* ++ * Support for LogicPD SDK Memory-mapped CompactFlash interface ++ * ++ * Copyright 2006 Logic Product Development <peterb@logicpd.com> ++ * ++ * This file is subject to the terms and conditions of the GNU General Public ++ * License. See the file COPYING in the main directory of this archive ++ * for more details. ++ */ ++ ++ ++ /* Uncomment the following if you want verbose error reports. */ ++ /* #define VERBOSE_ERRORS */ ++ ++ #include <linux/blkdev.h> ++ #include <linux/errno.h> ++ #include <linux/signal.h> ++ #include <linux/interrupt.h> ++ #include <linux/timer.h> ++ #include <linux/fs.h> ++ #include <linux/kernel.h> ++ #include <linux/genhd.h> ++ #include <linux/slab.h> ++ #include <linux/string.h> ++ #include <linux/ioport.h> ++ #include <linux/mc146818rtc.h> /* CMOS defines */ ++ #include <linux/init.h> ++ #include <linux/blkpg.h> ++ #include <linux/hdreg.h> ++ ++ #define REALLY_SLOW_IO ++ #include <asm/system.h> ++ #include <asm/io.h> ++ #include <asm/uaccess.h> ++ #include <asm/delay.h> ++ ++ #ifdef __arm__ ++ #undef HD_IRQ ++ #endif ++ #include <asm/irq.h> ++ #ifdef __arm__ ++ #define HD_IRQ IRQ_HARDDISK ++ #endif ++ ++ #define DEBUG ++ ++ /* Hd controller regster ports */ ++ ++ #define HD_DATA 0x1f0 /* _CTL when writing */ ++ #define HD_ERROR 0x1f1 /* see err-bits */ ++ #define HD_NSECTOR 0x1f2 /* nr of sectors to read/write */ ++ #define HD_SECTOR 0x1f3 /* starting sector */ ++ #define HD_LCYL 0x1f4 /* starting cylinder */ ++ #define HD_HCYL 0x1f5 /* high byte of starting cyl */ ++ #define HD_CURENT 0x1f6 /* 101dhhhh , d=drive, hhhh=head */ ++ #define HD_STATUS 0x1f7 /* see status-bits */ ++ #define HD_FEATURE HD_ERROR /* same io address, read=error, write=feature */ ++ #define HD_PRECOMP HD_FEATURE /* obsolete use of this port - predates IDE */ ++ #define HD_COMMAND HD_STATUS /* same io address, read=status, write=cmd */ ++ ++ #define HD_CMD 0x3f6 /* used for resets */ ++ #define HD_ALTSTATUS 0x3f6 /* same as HD_STATUS but doesn't clear irq */ ++ ++ /* Bits of HD_STATUS */ ++ #define ERR_STAT 0x01 ++ #define INDEX_STAT 0x02 ++ #define ECC_STAT 0x04 /* Corrected error */ ++ #define DRQ_STAT 0x08 ++ #define SEEK_STAT 0x10 ++ #define SERVICE_STAT SEEK_STAT ++ #define WRERR_STAT 0x20 ++ #define READY_STAT 0x40 ++ #define BUSY_STAT 0x80 ++ ++ /* Bits for HD_ERROR */ ++ #define MARK_ERR 0x01 /* Bad address mark */ ++ #define TRK0_ERR 0x02 /* couldn't find track 0 */ ++ #define ABRT_ERR 0x04 /* Command aborted */ ++ #define MCR_ERR 0x08 /* media change request */ ++ #define ID_ERR 0x10 /* ID field not found */ ++ #define MC_ERR 0x20 /* media changed */ ++ #define ECC_ERR 0x40 /* Uncorrectable ECC error */ ++ #define BBD_ERR 0x80 /* pre-EIDE meaning: block marked bad */ ++ #define ICRC_ERR 0x80 /* new meaning: CRC error during transfer */ ++ ++ static DEFINE_SPINLOCK(hd_lock); ++ static struct request_queue *hd_queue; ++ ++ #define MAJOR_NR HD_MAJOR ++ #define QUEUE (hd_queue) ++ #define CURRENT elv_next_request(hd_queue) ++ ++ #define TIMEOUT_VALUE (6*HZ) ++ #define HD_DELAY 0 ++ ++ #define MAX_ERRORS 16 /* Max read/write errors/sector */ ++ #define RESET_FREQ 8 /* Reset controller every 8th retry */ ++ #define RECAL_FREQ 4 /* Recalibrate every 4th retry */ ++ #define MAX_HD 2 ++ ++ #define STAT_OK (READY_STAT|SEEK_STAT) ++ #define OK_STATUS(s) (((s)&(STAT_OK|(BUSY_STAT|WRERR_STAT|ERR_STAT)))==STAT_OK) ++ ++ static int driveno = 0; ++ static int debug = 0; ++ ++ // Start of CF registers ++ #define CPLD_ATA_REG_BASE 0x14001800 ++ static unsigned char *reg_base; ++ ++ static inline unsigned int read_reg(unsigned char *base, unsigned int reg) ++ { ++ volatile unsigned short val; ++ ++ if (reg & 1) ++ val = *((volatile unsigned short *)(base + reg - 1)) >> 8; ++ else ++ val = *((volatile unsigned short *)(base + reg)); ++ ++ if (debug) ++ printk("%s: %02x %04x \n", __FUNCTION__, reg, val); ++ ++ return val; ++ } ++ ++ static inline void write_reg(unsigned char *base, unsigned int reg, unsigned int val) ++ { ++ if (debug) ++ printk("%s: %02x %04x\n", __FUNCTION__, reg, val); ++ if (reg & 1) ++ *((volatile unsigned short *)(base + reg - 1)) = (val << 8); ++ else ++ *((volatile unsigned short *)(base + reg)) = val; ++ } ++ ++ #define CB_DATA 0x08 ++ #define CB_ERR 0x0d ++ #define CB_SC_SN 0x02 ++ #define CB_CYL 0x04 ++ #define CB_STAT 0x07 ++ #define CB_DH_CMD 0x06 ++ #define CB_ASTAT 0x0e ++ #define CB_DC 0x0e ++ #define CB_DA 0x0f ++ ++ #define CB_STAT_BSY 0x80 ++ #define CB_STAT_DRQ 0x08 ++ #define CB_STAT_SEEK 0x10 ++ #define CB_STAT_DF 0x20 ++ #define CB_STAT_READY 0x40 ++ #define CB_STAT_ERR 0x01 ++ #define CB_DC_HD15 0x08 ++ #define CB_DC_NIEN 0x02 ++ ++ #define CMD_IDENTIFY_DEVICE 0xec ++ #define CMD_READ_SECTORS 0x20 ++ #define CMD_WRITE_SECTORS 0x30 ++ ++ #define TIMEOUT 0x800000 ++ ++ int cfide_card_present(void) ++ { ++ unsigned char data, data1, data2; ++ ++ /* Flip Sector Count */ ++ data = read_reg(reg_base, CB_SC_SN); ++ data1 = (~data) & 0xff; ++ write_reg(reg_base, CB_SC_SN, data1); ++ ++ /* write to the data register to waggle the bus */ ++ write_reg(reg_base, CB_DATA, data); ++ ++ /* Read back the sector count and if it matches what we put there ++ then the CF is present */ ++ data2 = read_reg(reg_base, CB_SC_SN); ++ if (data2 == data1) ++ return 1; ++ else { ++ printk("data %02x data1 %02x data2 %02x\n", data, data1, data2); ++ return 0; ++ } ++ } ++ ++ /* ++ * This struct defines the HD's and their types. ++ */ ++ struct hd_i_struct { ++ unsigned int head,sect,cyl,wpcom,lzone,ctl; ++ int unit; ++ int recalibrate; ++ int special_op; ++ }; ++ ++ #ifdef HD_TYPE ++ static struct hd_i_struct hd_info[] = { HD_TYPE }; ++ static int NR_HD = ((sizeof (hd_info))/(sizeof (struct hd_i_struct))); ++ #else ++ static struct hd_i_struct hd_info[MAX_HD]; ++ static int NR_HD; ++ #endif ++ ++ static struct gendisk *hd_gendisk[MAX_HD]; ++ ++ ++ ++ #if (HD_DELAY > 0) ++ ++ #include <asm/i8253.h> ++ ++ unsigned long last_req; ++ ++ unsigned long read_timer(void) ++ { ++ unsigned long t, flags; ++ int i; ++ ++ spin_lock_irqsave(&i8253_lock, flags); ++ t = jiffies * 11932; ++ outb_p(0, 0x43); ++ i = inb_p(0x40); ++ i |= inb(0x40) << 8; ++ spin_unlock_irqrestore(&i8253_lock, flags); ++ return(t - i); ++ } ++ #endif ++ ++ static void __init hd_setup(char *str, int *ints) ++ { ++ int hdind = 0; ++ ++ if (ints[0] != 3) ++ return; ++ if (hd_info[0].head != 0) ++ hdind=1; ++ hd_info[hdind].head = ints[2]; ++ hd_info[hdind].sect = ints[3]; ++ hd_info[hdind].cyl = ints[1]; ++ hd_info[hdind].wpcom = 0; ++ hd_info[hdind].lzone = ints[1]; ++ hd_info[hdind].ctl = (ints[2] > 8 ? 8 : 0); ++ NR_HD = hdind+1; ++ } ++ ++ ++ ++ void cfide_wait_fin(void) ++ { ++ unsigned long timer; ++ ++ // printk("%s:%d\n", __FUNCTION__, __LINE__); ++ ++ udelay(500); // wait 500us ++ ++ for (timer = 0; ++ timer < TIMEOUT && (read_reg(reg_base, CB_STAT) & CB_STAT_BSY); ++timer) ++ yield(); ++ ++ if (timer == TIMEOUT) ++ printk("%s:%d\n", __FUNCTION__, __LINE__); ++ } ++ ++ void cfide_wait_drq(void) ++ { ++ unsigned long timer; ++ ++ for (timer = 0; ++ timer < TIMEOUT && !(read_reg(reg_base, CB_STAT) & CB_STAT_DRQ); ++timer) ++ yield(); ++ ++ if (timer == TIMEOUT) ++ printk("%s:%d\n", __FUNCTION__, __LINE__); ++ } ++ ++ static union { ++ struct hd_driveid id; ++ short sh[512/2]; ++ } info_buf; ++ ++ ++ /* Read cnt sectors from the flash, starting at lba, storing the data ++ at dest */ ++ static int cfide_read_sectors(uint8_t *dest, uint32_t lba, uint32_t cnt) ++ { ++ uint8_t sect, head, devHead, status, devCtrl; ++ uint16_t cyl; ++ uint32_t orig_lba = lba; ++ uint32_t i,j; ++ uint16_t data; ++ ++ // printk("%s: dest %p lba %u cnt %u\n", __FUNCTION__, dest, lba, cnt); ++ ++ if (lba + cnt > info_buf.id.lba_capacity) { ++ printk("%s: %u+%u is larger than %u\n", __FUNCTION__, lba, cnt, info_buf.id.lba_capacity); ++ return -EINVAL; ++ } ++ ++ if (cnt > 255) { ++ printk("%s: cnt %u is too large\n", __FUNCTION__, cnt); ++ return -EINVAL; ++ } ++ ++ ++ /* translate from LBA */ ++ sect = lba & 0xff; ++ lba >>= 8; ++ cyl = lba & 0xffff; ++ lba >>= 16; ++ head = (lba & 0x0f) | 0x40; ++ ++ devCtrl = CB_DC_HD15 | CB_DC_NIEN; ++ devHead = driveno | head; ++ ++ write_reg(reg_base, CB_DC, devCtrl); ++ write_reg(reg_base, CB_SC_SN, ((uint16_t)cnt & 0xff) | ((uint16_t)sect << 8)); ++ write_reg(reg_base, CB_CYL, cyl); ++ ++ write_reg(reg_base, CB_DH_CMD, devHead | (CMD_READ_SECTORS << 8)); ++ ++ for (j=0; j<cnt; ++j) { ++ udelay(1); // spin for a moment to let the controller raise BSY ++ ++ cfide_wait_fin(); ++ cfide_wait_drq(); ++ for (i=0; i<256; ++i) { ++ data = read_reg(reg_base, CB_DATA); ++ #if 0 ++ *dest++ = data>>8; ++ *dest++ = data; ++ #else ++ *dest++ = data; ++ *dest++ = data>>8; ++ #endif ++ } ++ ++ ++ cfide_wait_fin(); ++ ++ status = read_reg(reg_base, CB_STAT); ++ if (status & (CB_STAT_DF|CB_STAT_ERR)) { ++ printk("%s: error at block %d status %#x\n", __FUNCTION__, orig_lba+j, status); ++ break; ++ } ++ ++ } ++ ++ return 0; ++ } ++ ++ /* Write cnt sectors to the flash, starting at lba, reading the data ++ from src */ ++ static int cfide_write_sectors(uint8_t *src, uint32_t lba, uint32_t cnt) ++ { ++ uint8_t sect, head, devHead, status, devCtrl; ++ uint16_t cyl; ++ uint32_t orig_lba = lba; ++ uint32_t i,j; ++ uint16_t data; ++ ++ if (lba + cnt > info_buf.id.lba_capacity) { ++ printk("%s: %u+%u is larger than %u\n", __FUNCTION__, lba, cnt, info_buf.id.lba_capacity); ++ return -EINVAL; ++ } ++ ++ if (cnt > 255) { ++ printk("%s: cnt %u is too large\n", __FUNCTION__, cnt); ++ return -EINVAL; ++ } ++ ++ /* translate from LBA */ ++ sect = lba & 0xff; ++ lba >>= 8; ++ cyl = lba & 0xffff; ++ lba >>= 16; ++ head = (lba & 0x0f) | 0x40; ++ ++ devCtrl = CB_DC_HD15 | CB_DC_NIEN; ++ devHead = driveno | head; ++ ++ write_reg(reg_base, CB_DC, devCtrl); ++ write_reg(reg_base, CB_SC_SN, ((uint16_t)cnt & 0xff) | ((uint16_t)sect << 8)); ++ write_reg(reg_base, CB_CYL, cyl); ++ ++ write_reg(reg_base, CB_DH_CMD, devHead | (CMD_WRITE_SECTORS << 8)); ++ ++ for (j=0; j<cnt; ++j) { ++ udelay(1); // spin for a moment to let the controller raise BSY ++ ++ cfide_wait_fin(); ++ ++ cfide_wait_drq(); ++ ++ for (i=0; i<256; ++i) { ++ #if 0 ++ data = (*src++ << 8); ++ data |= *src++; ++ #else ++ data = *src++; ++ data |= (*src++ << 8); ++ #endif ++ write_reg(reg_base, CB_DATA, data); ++ } ++ ++ cfide_wait_fin(); ++ ++ status = read_reg(reg_base, CB_STAT); ++ if (status & (CB_STAT_DF|CB_STAT_ERR)) { ++ printk("%s: error at block %d status %#x\n", __FUNCTION__, orig_lba+j, status); ++ break; ++ } ++ ++ } ++ return 0; ++ } ++ ++ ++ static void cfide_transfer(unsigned long sector, ++ unsigned long nsect, char *buffer, int write) ++ { ++ int ret; ++ if (write) ++ ret = cfide_write_sectors(buffer, sector, nsect); ++ else ++ ret = cfide_read_sectors(buffer, sector, nsect); ++ if (ret) ++ printk("%s:%d\n", __FUNCTION__, __LINE__); ++ ++ } ++ ++ ++ static void do_hd_request (request_queue_t * q) ++ { ++ struct request *req; ++ ++ // printk("%s:%d q %p\n", __FUNCTION__, __LINE__, q); ++ ++ while ((req = elv_next_request(q)) != NULL) { ++ if (blk_fs_request(req)) { ++ cfide_transfer(req->sector, req->current_nr_sectors, ++ req->buffer, rq_data_dir(req)); ++ end_request(req, 1); ++ } else { ++ printk (KERN_NOTICE "Skip non-fs request\n"); ++ end_request(req, 0); ++ continue; ++ } ++ } ++ } ++ ++ static int hd_getgeo(struct block_device *bdev, struct hd_geometry *geo) ++ { ++ struct hd_i_struct *disk = bdev->bd_disk->private_data; ++ ++ geo->heads = disk->head; ++ geo->sectors = disk->sect; ++ geo->cylinders = disk->cyl; ++ return 0; ++ } ++ ++ ++ static struct block_device_operations hd_fops = { ++ .getgeo = hd_getgeo, ++ }; ++ ++ void cfide_fetch_info(void) ++ { ++ int i; ++ ++ // printk("%s:%d\n", __FUNCTION__, __LINE__); ++ ++ /* Select the drive and wait for it to finish */ ++ driveno &= 1; ++ write_reg(reg_base, CB_DH_CMD, driveno); ++ cfide_wait_fin(); ++ ++ // printk("%s:%d\n", __FUNCTION__, __LINE__); ++ ++ write_reg(reg_base, CB_DH_CMD, driveno | (CMD_IDENTIFY_DEVICE << 8)); ++ cfide_wait_fin(); ++ ++ // printk("%s:%d\n", __FUNCTION__, __LINE__); ++ ++ ++ cfide_wait_drq(); ++ ++ // printk("%s:%d\n", __FUNCTION__, __LINE__); ++ ++ for (i=0; i<512; i+=2) ++ info_buf.sh[i/2] = read_reg(reg_base, CB_DATA); ++ ++ /* Fix lba_capcity */ ++ info_buf.id.lba_capacity = (info_buf.id.lba_capacity>>16) | (info_buf.id.lba_capacity<<16); ++ // printk("%s:%d lba_capacity %#x\n", __FUNCTION__, __LINE__, info_buf.id.lba_capacity); ++ } ++ ++ /* ++ * This is the hard disk IRQ description. The SA_INTERRUPT in sa_flags ++ * means we run the IRQ-handler with interrupts disabled: this is bad for ++ * interrupt latency, but anything else has led to problems on some ++ * machines. ++ * ++ * We enable interrupts in some of the routines after making sure it's ++ * safe. ++ */ ++ ++ static int __init hd_init(void) ++ { ++ int drive; ++ ++ printk("%s:%d MAJOR_NR %d\n", __FUNCTION__, __LINE__, MAJOR_NR); ++ ++ if (register_blkdev(MAJOR_NR,"hd")) ++ return -1; ++ ++ reg_base = (unsigned char *) ioremap_nocache(CPLD_ATA_REG_BASE, 0x1000); ++ if (!reg_base) { ++ printk("%s:%d\n", __FUNCTION__, __LINE__); ++ return -ENOMEM; ++ } ++ ++ /* If no card present, return */ ++ if (!cfide_card_present()) { ++ printk("No CompactFlash card detected\n"); ++ iounmap(reg_base); ++ return 0; ++ } ++ ++ /* Fetchthe device info */ ++ cfide_fetch_info(); ++ ++ hd_queue = blk_init_queue(do_hd_request, &hd_lock); ++ printk("%s:%d hd_queue %p\n", __FUNCTION__, __LINE__, hd_queue); ++ if (!hd_queue) { ++ unregister_blkdev(MAJOR_NR,"hd"); ++ return -ENOMEM; ++ } ++ ++ ++ blk_queue_max_sectors(hd_queue, 255); ++ blk_queue_hardsect_size(hd_queue, 512); ++ ++ #if 1 ++ hd_info[0].cyl = info_buf.id.cyls; ++ hd_info[0].head = info_buf.id.heads; ++ hd_info[0].wpcom = 0; ++ hd_info[0].ctl = 0; ++ hd_info[0].lzone = 0; ++ hd_info[0].sect = info_buf.id.sectors; ++ NR_HD++; ++ ++ printk("%s:%d NR_HD %d\n", __FUNCTION__, __LINE__, NR_HD); ++ ++ #else ++ #ifdef __i386__ ++ if (!NR_HD) { ++ extern struct drive_info drive_info; ++ unsigned char *BIOS = (unsigned char *) &drive_info; ++ unsigned long flags; ++ int cmos_disks; ++ ++ for (drive=0 ; drive<2 ; drive++) { ++ hd_info[drive].cyl = *(unsigned short *) BIOS; ++ hd_info[drive].head = *(2+BIOS); ++ hd_info[drive].wpcom = *(unsigned short *) (5+BIOS); ++ hd_info[drive].ctl = *(8+BIOS); ++ hd_info[drive].lzone = *(unsigned short *) (12+BIOS); ++ hd_info[drive].sect = *(14+BIOS); ++ #ifdef does_not_work_for_everybody_with_scsi_but_helps_ibm_vp ++ if (hd_info[drive].cyl && NR_HD == drive) ++ NR_HD++; ++ #endif ++ BIOS += 16; ++ } ++ ++ /* ++ We query CMOS about hard disks : it could be that ++ we have a SCSI/ESDI/etc controller that is BIOS ++ compatible with ST-506, and thus showing up in our ++ BIOS table, but not register compatible, and therefore ++ not present in CMOS. ++ ++ Furthermore, we will assume that our ST-506 drives ++ <if any> are the primary drives in the system, and ++ the ones reflected as drive 1 or 2. ++ ++ The first drive is stored in the high nibble of CMOS ++ byte 0x12, the second in the low nibble. This will be ++ either a 4 bit drive type or 0xf indicating use byte 0x19 ++ for an 8 bit type, drive 1, 0x1a for drive 2 in CMOS. ++ ++ Needless to say, a non-zero value means we have ++ an AT controller hard disk for that drive. ++ ++ Currently the rtc_lock is a bit academic since this ++ driver is non-modular, but someday... ? Paul G. ++ */ ++ ++ spin_lock_irqsave(&rtc_lock, flags); ++ cmos_disks = CMOS_READ(0x12); ++ spin_unlock_irqrestore(&rtc_lock, flags); ++ ++ if (cmos_disks & 0xf0) { ++ if (cmos_disks & 0x0f) ++ NR_HD = 2; ++ else ++ NR_HD = 1; ++ } ++ } ++ #endif /* __i386__ */ ++ #ifdef __arm__ ++ if (!NR_HD) { ++ /* We don't know anything about the drive. This means ++ * that you *MUST* specify the drive parameters to the ++ * kernel yourself. ++ */ ++ printk("hd: no drives specified - use hd=cyl,head,sectors" ++ " on kernel command line\n"); ++ } ++ #endif ++ #endif ++ if (!NR_HD) ++ goto out; ++ ++ for (drive=0 ; drive < NR_HD ; drive++) { ++ struct gendisk *disk = alloc_disk(64); ++ struct hd_i_struct *p = &hd_info[drive]; ++ if (!disk) ++ goto Enomem; ++ disk->major = MAJOR_NR; ++ disk->first_minor = drive << 6; ++ disk->fops = &hd_fops; ++ sprintf(disk->disk_name, "hd%c", 'a'+drive); ++ disk->private_data = p; ++ set_capacity(disk, p->head * p->sect * p->cyl); ++ disk->queue = hd_queue; ++ p->unit = drive; ++ hd_gendisk[drive] = disk; ++ printk ("%s: %luMB, CHS=%d/%d/%d\n", ++ disk->disk_name, (unsigned long)get_capacity(disk)/2048, ++ p->cyl, p->head, p->sect); ++ } ++ ++ /* Let them fly */ ++ for(drive=0; drive < NR_HD; drive++) ++ add_disk(hd_gendisk[drive]); ++ ++ return 0; ++ ++ out: ++ unregister_blkdev(MAJOR_NR,"hd"); ++ blk_cleanup_queue(hd_queue); ++ return -1; ++ Enomem: ++ while (drive--) ++ put_disk(hd_gendisk[drive]); ++ goto out; ++ } ++ ++ static int __init parse_hd_setup (char *line) { ++ int ints[6]; ++ ++ (void) get_options(line, ARRAY_SIZE(ints), ints); ++ hd_setup(NULL, ints); ++ ++ return 1; ++ } ++ __setup("hd=", parse_hd_setup); ++ ++ module_init(hd_init); ++ ++ /* ++ * Local variables: ++ * c-indent-level: 4 ++ * tab-width: 4 ++ * End: ++ */ diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/logicpd-pxa270-flash.patch b/packages/linux/logicpd-pxa270-2.6.19.2/logicpd-pxa270-flash.patch new file mode 100644 index 0000000000..43990058c2 --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/logicpd-pxa270-flash.patch @@ -0,0 +1,244 @@ +Index: drivers/mtd/maps/Kconfig +=================================================================== +RCS file: /cvs/eps/dev_eng/sw/products/Linux/PXAEngine/pxa/linux-2.6.17-rc5/drivers/mtd/maps/Kconfig,v +retrieving revision 1.1.1.1 +retrieving revision 1.2 +diff -c -3 -p -r1.1.1.1 -r1.2 +*** drivers/mtd/maps/Kconfig 29 May 2006 00:58:44 -0000 1.1.1.1 +--- drivers/mtd/maps/Kconfig 30 May 2006 13:04:33 -0000 1.2 +*************** config MTD_MAINSTONE +*** 137,142 **** +--- 137,149 ---- + This provides a driver for the on-board flash of the Intel + 'Mainstone PXA27x evaluation board. + ++ config MTD_LOGICPD_PXA270 ++ tristate "CFI Flash device mapped on LogicPD PXA270 Card Engine" ++ depends on MACH_LOGICPD_PXA270 && MTD_CFI_INTELEXT && MTD_PARTITIONS ++ help ++ This provides a driver for the on-board flash of the LogicPD ++ PXA270 Card Engine. ++ + config MTD_OCTAGON + tristate "JEDEC Flash device mapped on Octagon 5066 SBC" + depends on X86 && MTD_JEDEC && MTD_COMPLEX_MAPPINGS +Index: drivers/mtd/maps/Makefile +=================================================================== +RCS file: /cvs/eps/dev_eng/sw/products/Linux/PXAEngine/pxa/linux-2.6.17-rc5/drivers/mtd/maps/Makefile,v +retrieving revision 1.1.1.1 +retrieving revision 1.2 +diff -c -3 -p -r1.1.1.1 -r1.2 +*** drivers/mtd/maps/Makefile 29 May 2006 00:58:44 -0000 1.1.1.1 +--- drivers/mtd/maps/Makefile 30 May 2006 13:04:33 -0000 1.2 +*************** obj-$(CONFIG_MTD_ICHXROM) += ichxrom.o +*** 22,27 **** +--- 22,28 ---- + obj-$(CONFIG_MTD_TSUNAMI) += tsunami_flash.o + obj-$(CONFIG_MTD_LUBBOCK) += lubbock-flash.o + obj-$(CONFIG_MTD_MAINSTONE) += mainstone-flash.o ++ obj-$(CONFIG_MTD_LOGICPD_PXA270)+= lpd270-flash.o + obj-$(CONFIG_MTD_MBX860) += mbx860.o + obj-$(CONFIG_MTD_CEIVA) += ceiva.o + obj-$(CONFIG_MTD_OCTAGON) += octagon-5066.o +Index: drivers/mtd/maps/lpd270-flash.c +=================================================================== +--- /dev/null 2006-05-22 10:25:23.000000000 -0400 ++++ drivers/mtd/maps/lpd270-flash.c 2006-12-29 13:19:54.000000000 -0500 +@@ -0,0 +1,197 @@ ++ /* ++ * $Id: lpd270-flash.c,v 1.1 2006/05/30 13:03:55 LOGIC+peterb Exp $ ++ * ++ * Map driver for the Lpd270 developer platform. ++ * ++ * Author: Nicolas Pitre ++ * Copyright: (C) 2001 MontaVista Software Inc. ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++ #include <linux/module.h> ++ #include <linux/types.h> ++ #include <linux/kernel.h> ++ #include <linux/init.h> ++ #include <linux/dma-mapping.h> ++ #include <linux/slab.h> ++ ++ #include <linux/mtd/mtd.h> ++ #include <linux/mtd/map.h> ++ #include <linux/mtd/partitions.h> ++ ++ #include <asm/io.h> ++ #include <asm/hardware.h> ++ #include <asm/arch/pxa-regs.h> ++ // #include <asm/arch/lpd270.h> ++ ++ ++ #define ROM_ADDR 0x00000000 ++ #define FLASH_ADDR 0x04000000 ++ ++ #define WINDOW_SIZE 0x04000000 ++ ++ static void lpd270_map_inval_cache(struct map_info *map, unsigned long from, ++ ssize_t len) ++ { ++ consistent_sync((char *)map->cached + from, len, DMA_FROM_DEVICE); ++ } ++ ++ static struct map_info lpd270_maps[2] = { { ++ .size = WINDOW_SIZE, ++ .phys = PXA_CS0_PHYS, ++ .inval_cache = lpd270_map_inval_cache, ++ }, { ++ .size = WINDOW_SIZE, ++ .phys = PXA_CS1_PHYS, ++ .inval_cache = lpd270_map_inval_cache, ++ } }; ++ ++ static struct mtd_partition lpd270_partitions[] = { ++ { ++ .name = "Bootloader", ++ .size = 0x000C0000, ++ .offset = 0, ++ .mask_flags = MTD_WRITEABLE /* force read-only */ ++ },{ ++ .name = "YAFFS", ++ .size = 0x00F40000, ++ .offset = 0x000C0000, ++ },{ ++ .name = "Filesystem", ++ .size = MTDPART_SIZ_FULL, ++ .offset = 0x01000000 ++ } ++ }; ++ ++ static struct mtd_info *mymtds[2]; ++ static struct mtd_partition *parsed_parts[2]; ++ static int nr_parsed_parts[2]; ++ ++ static const char *probes[] = { "RedBoot", "cmdlinepart", NULL }; ++ ++ static int __init init_lpd270(void) ++ { ++ int SW7 = 0; /* FIXME: get from SCR (Mst doc section 3.2.1.1) */ ++ int ret = 0, i; ++ ++ lpd270_maps[0].bankwidth = (BOOT_DEF & 1) ? 2 : 4; ++ lpd270_maps[1].bankwidth = 4; ++ ++ /* Compensate for SW7 which swaps the flash banks */ ++ lpd270_maps[SW7].name = "processor flash"; ++ lpd270_maps[SW7 ^ 1].name = "main board flash"; ++ ++ printk(KERN_CRIT "FLASH ENABLED!!!!!!!!!! Lpd270 configured to boot from %s\n", ++ lpd270_maps[0].name); ++ ++ // Set VPP in CPLD to allow writes to the flash ++ { ++ unsigned short *p; ++ ++ p = ioremap(0x08000000, PAGE_SIZE); ++ if (!p) { ++ printk("%s:%d ioremap failed\n", __FUNCTION__, __LINE__); ++ } else { ++ printk("%s:%d CTRL REG %04x\n", __FUNCTION__, __LINE__, *p); ++ *p |= (1<<3); // FL_VPEN ++ iounmap(p); ++ } ++ } ++ ++ for (i = 0; i < 2; i++) { ++ lpd270_maps[i].virt = ioremap_nocache(lpd270_maps[i].phys, ++ WINDOW_SIZE); ++ if (!lpd270_maps[i].virt) { ++ printk(KERN_WARNING "Failed to ioremap %s\n", ++ lpd270_maps[i].name); ++ if (!ret) ++ ret = -ENOMEM; ++ continue; ++ } ++ #if 0 ++ lpd270_maps[i].cached = ++ ioremap_cached(lpd270_maps[i].phys, WINDOW_SIZE); ++ if (!lpd270_maps[i].cached) ++ printk(KERN_WARNING "Failed to ioremap cached %s\n", ++ lpd270_maps[i].name); ++ #endif ++ simple_map_init(&lpd270_maps[i]); ++ ++ printk(KERN_NOTICE ++ "Probing %s at physical address 0x%08lx" ++ " (%d-bit bankwidth)\n", ++ lpd270_maps[i].name, lpd270_maps[i].phys, ++ lpd270_maps[i].bankwidth * 8); ++ ++ mymtds[i] = do_map_probe("cfi_probe", &lpd270_maps[i]); ++ ++ if (!mymtds[i]) { ++ iounmap((void *)lpd270_maps[i].virt); ++ if (lpd270_maps[i].cached) ++ iounmap(lpd270_maps[i].cached); ++ if (!ret) ++ ret = -EIO; ++ continue; ++ } ++ mymtds[i]->owner = THIS_MODULE; ++ ++ ret = parse_mtd_partitions(mymtds[i], probes, ++ &parsed_parts[i], 0); ++ ++ if (ret > 0) ++ nr_parsed_parts[i] = ret; ++ } ++ ++ if (!mymtds[0] && !mymtds[1]) ++ return ret; ++ ++ for (i = 0; i < 2; i++) { ++ if (!mymtds[i]) { ++ printk(KERN_WARNING "%s is absent. Skipping\n", ++ lpd270_maps[i].name); ++ } else if (nr_parsed_parts[i]) { ++ add_mtd_partitions(mymtds[i], parsed_parts[i], ++ nr_parsed_parts[i]); ++ } else if (!i) { ++ printk("Using static partitions on %s\n", ++ lpd270_maps[i].name); ++ add_mtd_partitions(mymtds[i], lpd270_partitions, ++ ARRAY_SIZE(lpd270_partitions)); ++ } else { ++ printk("Registering %s as whole device\n", ++ lpd270_maps[i].name); ++ add_mtd_device(mymtds[i]); ++ } ++ } ++ return 0; ++ } ++ ++ static void __exit cleanup_lpd270(void) ++ { ++ int i; ++ for (i = 0; i < 2; i++) { ++ if (!mymtds[i]) ++ continue; ++ ++ if (nr_parsed_parts[i] || !i) ++ del_mtd_partitions(mymtds[i]); ++ else ++ del_mtd_device(mymtds[i]); ++ ++ map_destroy(mymtds[i]); ++ iounmap((void *)lpd270_maps[i].virt); ++ if (lpd270_maps[i].cached) ++ iounmap(lpd270_maps[i].cached); ++ kfree(parsed_parts[i]); ++ } ++ } ++ ++ module_init(init_lpd270); ++ module_exit(cleanup_lpd270); ++ ++ MODULE_LICENSE("GPL"); ++ MODULE_AUTHOR("Nicolas Pitre <nico@cam.org>"); ++ MODULE_DESCRIPTION("MTD map driver for Logic PXA270"); diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/logicpd-pxa270-hardware-id-hack.patch b/packages/linux/logicpd-pxa270-2.6.19.2/logicpd-pxa270-hardware-id-hack.patch new file mode 100644 index 0000000000..975cf7a200 --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/logicpd-pxa270-hardware-id-hack.patch @@ -0,0 +1,31 @@ + +Index: arch/arm/kernel/head.S +=================================================================== +RCS file: /cvs/eps/dev_eng/sw/products/Linux/PXAEngine/pxa/linux-2.6.17-rc5/arch/arm/kernel/head.S,v +retrieving revision 1.1.1.1 +diff -c -3 -p -r1.1.1.1 head.S +*** arch/arm/kernel/head.S 29 May 2006 00:53:47 -0000 1.1.1.1 +--- arch/arm/kernel/head.S 1 Jun 2006 17:37:16 -0000 +*************** ENTRY(stext) +*** 74,79 **** +--- 74,93 ---- + msr cpsr_c, #PSR_F_BIT | PSR_I_BIT | MODE_SVC @ ensure svc mode + @ and irqs disabled + mrc p15, 0, r9, c0, c0 @ get processor id ++ ++ /* LPD-- ++ * This fakes out the Linux kernel into believing that it is ++ * running on a Mainstone hardware platform. The LogicLoader (LoLo) ++ * doesn't currently pass kernel parameters correctly for 2.6 ARM ++ * kernels. Therefore, we just hardcode it here. ++ * ++ * --LPD ++ */ ++ mov r0, #0 ++ mov r1, #0x300 ++ orr r1, r1, #0x0a0 ++ orr r1, r1, #0x002 ++ + bl __lookup_processor_type @ r5=procinfo r9=cpuid + movs r10, r5 @ invalid processor (r5=0)? + beq __error_p @ yes, error 'p' diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/logicpd-pxa270-lcd-osd024ttea2.patch b/packages/linux/logicpd-pxa270-2.6.19.2/logicpd-pxa270-lcd-osd024ttea2.patch new file mode 100644 index 0000000000..c70af61fc3 --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/logicpd-pxa270-lcd-osd024ttea2.patch @@ -0,0 +1,81 @@ +--- arch/arm/mach-pxa/lpd270.c~org 2006-11-29 16:57:37.000000000 -0500 ++++ arch/arm/mach-pxa/lpd270.c 2006-12-29 12:56:05.000000000 -0500 +@@ -384,6 +384,30 @@ + .pxafb_backlight_power = lpd270_backlight_power, + }; + ++/* 2.4" TFT QVGA */ ++static struct pxafb_mode_info osd_osd024ttea2_mode = { ++ .pixclock = 208000, /*280nS is the Min speed, 208000 = 230nS (scoped)*/ ++ .xres = 240, ++ .yres = 320, ++ .bpp = 16, ++ .hsync_len = 10, ++ .left_margin = 20, ++ .right_margin = 10, ++ .vsync_len = 1, ++ .upper_margin = 0x03, ++ .lower_margin = 0x06, ++ .sync = 0, ++}; ++ ++static struct pxafb_mach_info osd_osd024ttea2 = { ++ .modes = &osd_osd024ttea2_mode, ++ .num_modes = 1, ++ .lccr0 = 0x07800080, ++ .lccr3 = 0x04000001, ++ .pxafb_backlight_power = lpd270_backlight_power, ++}; ++ ++ + static struct pxafb_mach_info *lpd270_lcd_to_use; + + static int __init lpd270_set_lcd(char *str) +@@ -400,6 +424,8 @@ + lpd270_lcd_to_use = &sharp_lq10d368; + } else if (!strnicmp(str, "lq035q7db02-20", 14)) { + lpd270_lcd_to_use = &sharp_lq035q7db02_20; ++ } else if (!strnicmp(str, "osd024ttea2", 11)) { ++ lpd270_lcd_to_use = &osd_osd024ttea2; + } else { + printk(KERN_INFO "lpd270: unknown lcd panel [%s]\n", str); + } +--- drivers/video/Kconfig~org 2006-12-29 12:59:13.000000000 -0500 ++++ drivers/video/Kconfig 2006-12-29 13:02:17.000000000 -0500 +@@ -1530,6 +1530,35 @@ + config FB_PXA_LCD_VGA + bool "VGA (640x480)" + ++config FB_PXA_LPD_LQ64D343 ++ bool "6.4\" TFT VGA 640x480" ++ depends on FB_PXA ++ ---help--- ++ Sharp LQ64D343 LCD panel ++ ++config FB_PXA_LPD_LQ035Q7DB02 ++ bool "3.5\" TFT QVGA 240x320" ++ depends on FB_PXA ++ ---help--- ++ Sharp LQ035Q7DB02 LCD panel ++ ++config FB_PXA_LPD_LQ057Q3DC02 ++ bool "5.7\" TFT QVGA 240x320" ++ depends on FB_PXA ++ ---help--- ++ Sharp LQ057Q3DC02 LCD panel ++ ++config FB_PXA_LPD_LQ10D368 ++ bool "10.4\" TFT VGA 640x480" ++ depends on FB_PXA ++ ---help--- ++ Sharp LQ10D368 LCD panel ++ ++config FB_PXA_LPD_OSD024TTEA2 ++ bool "2.4\" TFT VGA 320x240" ++ help ++ OSD OSD024TTEA2 LCD panel. ++ + endchoice + + config FB_PXA_OVERLAY + diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/logicpd-pxa270-smc91x.patch b/packages/linux/logicpd-pxa270-2.6.19.2/logicpd-pxa270-smc91x.patch new file mode 100644 index 0000000000..b6fed75941 --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/logicpd-pxa270-smc91x.patch @@ -0,0 +1,35 @@ +Index: drivers/net/smc91x.h +=================================================================== +RCS file: /cvs/eps/dev_eng/sw/products/Linux/PXAEngine/pxa/linux-2.6.17-rc5/drivers/net/smc91x.h,v +retrieving revision 1.1.1.1 +retrieving revision 1.2 +diff -c -3 -p -r1.1.1.1 -r1.2 +*** drivers/net/smc91x.h 29 May 2006 00:59:41 -0000 1.1.1.1 +--- drivers/net/smc91x.h 29 May 2006 17:42:44 -0000 1.2 +*************** +*** 129,134 **** +--- 129,152 ---- + #define SMC_insb(a, r, p, l) readsb((a) + (r), p, (l)) + #define SMC_outsb(a, r, p, l) writesb((a) + (r), p, (l)) + ++ #elif defined(CONFIG_MACH_LOGICPD_PXA270) ++ ++ #define SMC_CAN_USE_8BIT 0 ++ #define SMC_CAN_USE_16BIT 1 ++ #define SMC_CAN_USE_32BIT 0 ++ #define SMC_IO_SHIFT 0 ++ #define SMC_NOWAIT 1 ++ #define SMC_USE_PXA_DMA 1 ++ ++ // #define SMC_inb(a, r) readb((a) + (r)) ++ #define SMC_inw(a, r) readw((a) + (r)) ++ // #define SMC_inl(a, r) readl((a) + (r)) ++ // #define SMC_outb(v, a, r) writeb(v, (a) + (r)) ++ #define SMC_outw(v, a, r) writew(v, (a) + (r)) ++ // #define SMC_outl(v, a, r) writel(v, (a) + (r)) ++ #define SMC_insw(a, r, p, l) readsw((a) + (r), p, l) ++ #define SMC_outsw(a, r, p, l) writesw((a) + (r), p, l) ++ + #elif defined(CONFIG_ARCH_INNOKOM) || \ + defined(CONFIG_MACH_MAINSTONE) || \ + defined(CONFIG_ARCH_PXA_IDP) || \ diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/pm_changes-r1.patch b/packages/linux/logicpd-pxa270-2.6.19.2/pm_changes-r1.patch new file mode 100644 index 0000000000..900bc046bf --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/pm_changes-r1.patch @@ -0,0 +1,44 @@ +Index: linux-2.6.15/arch/arm/mach-pxa/corgi.c +=================================================================== +--- linux-2.6.15.orig/arch/arm/mach-pxa/corgi.c 2006-03-04 11:27:13.000000000 +0000 ++++ linux-2.6.15/arch/arm/mach-pxa/corgi.c 2006-03-04 11:29:10.000000000 +0000 +@@ -60,6 +60,7 @@ + static struct scoop_config corgi_scoop_setup = { + .io_dir = CORGI_SCOOP_IO_DIR, + .io_out = CORGI_SCOOP_IO_OUT, ++ .suspend_clr = 0xffff, + }; + + struct platform_device corgiscoop_device = { +@@ -322,7 +323,7 @@ + PGSR1 = 0x00FF0080; + PGSR2 = 0x0001C004; + /* Stop 3.6MHz and drive HIGH to PCMCIA and CS */ +- PCFR |= PCFR_OPDE; ++ PCFR = PCFR_OPDE; + + corgi_ssp_set_machinfo(&corgi_ssp_machinfo); + +Index: linux-2.6.15/arch/arm/mach-pxa/corgi_pm.c +=================================================================== +--- linux-2.6.15.orig/arch/arm/mach-pxa/corgi_pm.c 2006-03-04 11:26:52.000000000 +0000 ++++ linux-2.6.15/arch/arm/mach-pxa/corgi_pm.c 2006-03-04 11:29:10.000000000 +0000 +@@ -97,6 +97,9 @@ + if (!machine_is_corgi()) + wakeup_mask |= GPIO_bit(CORGI_GPIO_MAIN_BAT_LOW); + ++ if (!machine_is_corgi()) ++ GPDR0 &= ~(GPIO_bit(15) | GPIO_bit(8)); /* Float n_CS1 */ ++ + PWER = wakeup_mask | PWER_RTC; + PRER = wakeup_mask; + PFER = wakeup_mask; +@@ -113,6 +116,8 @@ + + static void corgi_postsuspend(void) + { ++ if (!machine_is_corgi()) ++ GPDR0 |= GPIO_bit(15) | GPIO_bit(8); /* Un-Float n_CS1 */ + } + + /* diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/pxa25x_cpufreq-r1.patch b/packages/linux/logicpd-pxa270-2.6.19.2/pxa25x_cpufreq-r1.patch new file mode 100644 index 0000000000..dfbf7cada0 --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/pxa25x_cpufreq-r1.patch @@ -0,0 +1,398 @@ +Index: git/arch/arm/Kconfig +=================================================================== +--- git.orig/arch/arm/Kconfig 2006-07-04 21:48:58.000000000 +0100 ++++ git/arch/arm/Kconfig 2006-07-05 09:46:49.000000000 +0100 +@@ -690,7 +690,7 @@ + + endmenu + +-if (ARCH_SA1100 || ARCH_INTEGRATOR || ARCH_OMAP) ++if (ARCH_SA1100 || ARCH_INTEGRATOR || ARCH_OMAP || ARCH_PXA) + + menu "CPU Frequency scaling" + +@@ -717,6 +717,12 @@ + + If in doubt, say Y. + ++config CPU_FREQ_PXA25x ++ bool ++ select CPU_FREQ_TABLE ++ depends on CPU_FREQ && PXA25x ++ default y ++ + endmenu + + endif +Index: git/arch/arm/mach-pxa/Makefile +=================================================================== +--- git.orig/arch/arm/mach-pxa/Makefile 2006-07-05 09:44:45.000000000 +0100 ++++ git/arch/arm/mach-pxa/Makefile 2006-07-05 09:45:43.000000000 +0100 +@@ -29,6 +29,9 @@ + + obj-$(CONFIG_LEDS) += $(led-y) + ++# CPU freq support ++obj-$(CONFIG_CPU_FREQ_PXA25x) += cpu-pxa25x.o ++ + # Misc features + obj-$(CONFIG_PM) += pm.o sleep.o + obj-$(CONFIG_PXA_KEYS) += pxa_keys.o +Index: git/arch/arm/mach-pxa/cpu-pxa25x.c +=================================================================== +--- /dev/null 1970-01-01 00:00:00.000000000 +0000 ++++ git/arch/arm/mach-pxa/cpu-pxa25x.c 2006-07-05 09:45:43.000000000 +0100 +@@ -0,0 +1,353 @@ ++/* ++ * linux/arch/arm/mach-pxa/cpu-pxa.c ++ * ++ * Copyright (C) 2002,2003 Intrinsyc Software ++ * ++ * 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; if not, write to the Free Software ++ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA ++ * ++ * History: ++ * 31-Jul-2002 : Initial version [FB] ++ * 29-Jan-2003 : added PXA255 support [FB] ++ * 20-Apr-2003 : ported to v2.5 (Dustin McIntire, Sensoria Corp.) ++ * 18-Jul-2005 : updated for latest kernel (2.6.13-rc), cleanup for submission ++ * (Ian Campbell, Arcom Control Systems) ++ * ++ * Note: ++ * This driver may change the memory bus clock rate, but will not do any ++ * platform specific access timing changes... for example if you have flash ++ * memory connected to CS0, you will need to register a platform specific ++ * notifier which will adjust the memory access strobes to maintain a ++ * minimum strobe width. ++ * ++ */ ++ ++#include <linux/kernel.h> ++#include <linux/module.h> ++#include <linux/sched.h> ++#include <linux/init.h> ++#include <linux/cpufreq.h> ++#include <linux/moduleparam.h> ++ ++#include <asm/hardware.h> ++ ++#include <asm/arch/pxa-regs.h> ++ ++#define dprintk(msg...) cpufreq_debug_printk(CPUFREQ_DEBUG_DRIVER, "pxa25x", msg) ++ ++struct pxa_freqs { ++ unsigned int khz; ++ unsigned int membus; ++ unsigned int cccr; ++ unsigned int div2; ++}; ++ ++/* Define the refresh period in mSec for the SDRAM and the number of rows */ ++#define SDRAM_TREF 64 /* standard 64ms SDRAM */ ++#define SDRAM_ROWS 8192 /* 64MB=8192 32MB=4096 */ ++#define MDREFR_DRI(x) ((x*SDRAM_TREF)/(SDRAM_ROWS*32)) ++ ++#define CCLKCFG_TURBO 0x1 ++#define CCLKCFG_FCS 0x2 ++#define PXA25x_MIN_FREQ 99532 ++#define PXA25x_MAX_FREQ 398131 ++#define MDREFR_DB2_MASK (MDREFR_K2DB2 | MDREFR_K1DB2) ++#define MDREFR_DRI_MASK 0xFFF ++ ++/* Use the run mode frequencies for performance */ ++static struct pxa_freqs pxa25x_performance_freqs[] = { ++ /* CPU MEMBUS CCCR DIV2 */ ++ {99532, 99532, 0x121, 1}, /* run= 99, turbo= 99, PXbus=50, SDRAM=50 */ ++ {132710, 132710, 0x123, 1}, /* run=133, turbo=133, PXbus=66, SDRAM=66 */ ++ {199065, 99532, 0x141, 0}, /* run=199, turbo=199, PXbus=99, SDRAM=99 */ ++ {265421, 132710, 0x143, 1}, /* run=265, turbo=265, PXbus=133, SDRAM=66 */ ++ {331776, 165888, 0x145, 1}, /* run=331, turbo=331, PXbus=166, SDRAM=83 */ ++ {398131, 99532, 0x161, 0}, /* run=398, turbo=398, PXbus=196, SDRAM=99 */ ++}; ++ ++static struct cpufreq_frequency_table pxa25x_performance_freq_table[ARRAY_SIZE(pxa25x_performance_freqs)+1]; ++ ++/* Use the turbo mode frequencies for powersave */ ++static struct pxa_freqs pxa25x_powersave_freqs[] = { ++ /* CPU MEMBUS CCCR DIV2 */ ++ {99532, 99532, 0x121, 1}, /* run=99, turbo= 99, PXbus=50, SDRAM=50 */ ++ {199065, 99532, 0x221, 0}, /* run=99, turbo=199, PXbus=50, SDRAM=99 */ ++ {298598, 99532, 0x321, 0}, /* run=99, turbo=287, PXbus=50, SDRAM=99 */ ++ {398131, 99532, 0x241, 0}, /* run=199, turbo=398, PXbus=99, SDRAM=99 */ ++}; ++ ++static struct cpufreq_frequency_table pxa25x_powersave_freq_table[ARRAY_SIZE(pxa25x_powersave_freqs)+1]; ++ ++extern unsigned get_clk_frequency_khz(int info); ++ ++static unsigned int max_frequency = PXA25x_MAX_FREQ; ++static int performance = 0; ++ ++/* ++ * This option can be used if you have one of the 200MHz PXA25x parts by adding ++ * cpu_pxa25x.max_frequency=199065 to the kernel command line ++ */ ++module_param(max_frequency, int, 0); ++MODULE_PARM_DESC(max_frequency, "Set the maximum cpu frequency"); ++ ++module_param(performance, int, 0); ++MODULE_PARM_DESC(performance, "Use performance instead of powersave frequency tables"); ++ ++static void pxa_select_freq_table(struct cpufreq_policy *policy, ++ struct pxa_freqs ** settings, ++ struct cpufreq_frequency_table **table) ++{ ++ cpufreq_frequency_table_put_attr(policy->cpu); ++ ++ if (performance) { ++ dprintk("selecting performance tables\n"); ++ cpufreq_frequency_table_get_attr(pxa25x_performance_freq_table, policy->cpu); ++ if (settings) ++ *settings = pxa25x_performance_freqs; ++ if (table) ++ *table = pxa25x_performance_freq_table; ++ } else { ++ dprintk("selecting powersave tables\n"); ++ cpufreq_frequency_table_get_attr(pxa25x_powersave_freq_table, policy->cpu); ++ if (settings) ++ *settings = pxa25x_powersave_freqs; ++ if (table) ++ *table = pxa25x_powersave_freq_table; ++ } ++} ++ ++/* find a valid frequency point */ ++static int pxa_verify_policy(struct cpufreq_policy *policy) ++{ ++ int ret; ++ struct cpufreq_frequency_table *pxa_freqs_table; ++ ++ pxa_select_freq_table(policy, NULL, &pxa_freqs_table); ++ ++ ret = cpufreq_frequency_table_verify(policy, pxa_freqs_table); ++ ++ dprintk("verified CPU policy: %dKhz min to %dKhz max\n", ++ policy->min, policy->max); ++ ++ return ret; ++} ++ ++static int pxa_set_target(struct cpufreq_policy *policy, ++ unsigned int target_freq, unsigned int relation) ++{ ++ int idx; ++ struct cpufreq_freqs freqs; ++ struct pxa_freqs *pxa_freq_settings; ++ struct cpufreq_frequency_table *pxa_freqs_table; ++ unsigned long flags; ++ unsigned int unused; ++ unsigned int preset_mdrefr, postset_mdrefr; ++ void *ramstart; ++ ++ /* Get the current policy */ ++ pxa_select_freq_table(policy, &pxa_freq_settings, &pxa_freqs_table); ++ ++ /* Lookup the next frequency */ ++ if (cpufreq_frequency_table_target(policy, pxa_freqs_table, ++ target_freq, relation, &idx)) ++ return -EINVAL; ++ ++ freqs.old = get_clk_frequency_khz(0); ++ freqs.new = pxa_freq_settings[idx].khz; ++ freqs.cpu = policy->cpu; ++ ++ if (freqs.new == freqs.old && pxa_freq_settings[idx].cccr == CCCR) ++ return 0; ++ ++ dprintk("changing CPU frequency to %d.%03d Mhz (SDRAM %d Mhz, CCCR %#04x)\n", ++ freqs.new / 1000, freqs.new % 1000, ++ (pxa_freq_settings[idx].div2) ? ++ (pxa_freq_settings[idx].membus / 2000) : ++ (pxa_freq_settings[idx].membus / 1000), ++ pxa_freq_settings[idx].cccr); ++ ++ ramstart = phys_to_virt(0xa0000000); ++ ++ /* ++ * Tell everyone what we're about to do... ++ * you should add a notify client with any platform specific ++ * Vcc changing capability ++ */ ++ cpufreq_notify_transition(&freqs, CPUFREQ_PRECHANGE); ++ ++ /* Calculate the next MDREFR. If we're slowing down the SDRAM clock ++ * we need to preset the smaller DRI before the change. If we're speeding ++ * up we need to set the larger DRI value after the change. ++ */ ++ preset_mdrefr = postset_mdrefr = MDREFR; ++ if ((MDREFR & MDREFR_DRI_MASK) > MDREFR_DRI(pxa_freq_settings[idx].membus)) { ++ preset_mdrefr = (preset_mdrefr & ~MDREFR_DRI_MASK) | ++ MDREFR_DRI(pxa_freq_settings[idx].membus); ++ } ++ postset_mdrefr = (postset_mdrefr & ~MDREFR_DRI_MASK) | ++ MDREFR_DRI(pxa_freq_settings[idx].membus); ++ ++ /* If we're dividing the memory clock by two for the SDRAM clock, this ++ * must be set prior to the change. Clearing the divide must be done ++ * after the change. ++ */ ++ if (pxa_freq_settings[idx].div2) { ++ preset_mdrefr |= MDREFR_DB2_MASK; ++ postset_mdrefr |= MDREFR_DB2_MASK; ++ } else { ++ postset_mdrefr &= ~MDREFR_DB2_MASK; ++ } ++ ++ local_irq_save(flags); ++ ++ /* Set new the CCCR */ ++ CCCR = pxa_freq_settings[idx].cccr; ++ ++ __asm__ __volatile__(" \ ++ ldr r4, [%1] ; /* load MDREFR */ \ ++ b 2f ; \ ++ .align 5 ; \ ++1: \ ++ str %4, [%1] ; /* preset the MDREFR */ \ ++ mcr p14, 0, %2, c6, c0, 0 ; /* set CCLKCFG[FCS] */ \ ++ str %5, [%1] ; /* postset the MDREFR */ \ ++ \ ++ b 3f ; \ ++2: b 1b ; \ ++3: nop ; \ ++ " ++ : "=&r"(unused) ++ : "r"(&MDREFR), "r"(CCLKCFG_TURBO | CCLKCFG_FCS), ++ "r"(ramstart), "r"(preset_mdrefr), ++ "r"(postset_mdrefr) ++ : "r4", "r5"); ++ local_irq_restore(flags); ++ ++ /* ++ * Tell everyone what we've just done... ++ * you should add a notify client with any platform specific ++ * SDRAM refresh timer adjustments ++ */ ++ cpufreq_notify_transition(&freqs, CPUFREQ_POSTCHANGE); ++ ++ return 0; ++} ++ ++static unsigned int pxa_cpufreq_get(unsigned int cpu) ++{ ++ return get_clk_frequency_khz(0); ++} ++ ++static int pxa_cpufreq_init(struct cpufreq_policy *policy) ++{ ++ int i; ++ ++ if (policy->cpu != 0) ++ return -ENODEV; ++ ++ /* set default policy and cpuinfo */ ++ policy->governor = CPUFREQ_DEFAULT_GOVERNOR; ++ policy->cpuinfo.max_freq = max_frequency; ++ policy->cpuinfo.min_freq = PXA25x_MIN_FREQ; ++ policy->cpuinfo.transition_latency = 1000000; /* FIXME: 1 ms, assumed */ ++ policy->cur = get_clk_frequency_khz(0); /* current freq */ ++ policy->min = policy->max = policy->cur; ++ policy->min = pxa25x_powersave_freqs[0].khz; ++ policy->max = pxa25x_powersave_freqs[sizeof(pxa25x_powersave_freqs)/sizeof(*pxa25x_powersave_freqs)-1].khz; ++ ++ /* Generate the run cpufreq_frequency_table struct */ ++ for (i = 0; i < ARRAY_SIZE(pxa25x_performance_freqs); i++) { ++ pxa25x_performance_freq_table[i].frequency = pxa25x_performance_freqs[i].khz; ++ pxa25x_performance_freq_table[i].index = i; ++ } ++ pxa25x_performance_freq_table[i].frequency = CPUFREQ_TABLE_END; ++ ++ /* Generate the turbo cpufreq_frequency_table struct */ ++ for (i = 0; i < ARRAY_SIZE(pxa25x_powersave_freqs); i++) { ++ pxa25x_powersave_freq_table[i].frequency = pxa25x_powersave_freqs[i].khz; ++ pxa25x_powersave_freq_table[i].index = i; ++ } ++ pxa25x_powersave_freq_table[i].frequency = CPUFREQ_TABLE_END; ++ ++ /* calls cpufreq_frequency_table_get_attr */ ++ pxa_select_freq_table(policy, NULL, NULL); ++ ++ printk(KERN_INFO "pxa25x: CPU frequency change support initialized (%s tables)\n", ++ performance ? "performance" : "powersave"); ++ ++ return 0; ++} ++ ++static ssize_t show_pxa25x_freq_model_attr(struct cpufreq_policy * policy, char *buf) ++{ ++ return sprintf (buf, "%s\n", performance ? "performance" : "powersave"); ++} ++ ++static ssize_t store_pxa25x_freq_model_attr(struct cpufreq_policy * policy, const char *buf, size_t count) ++{ ++ unsigned int ret; ++ char str[16]; ++ ++ ret = sscanf(buf, "%15s", str); ++ if (ret != 1) ++ return -EINVAL; ++ ++ if (strnicmp(str,"performance",16)==0) ++ performance = 1; ++ else if (strnicmp(str,"powersave",16)==0) ++ performance = 0; ++ else ++ return -EINVAL; ++ ++ ret = cpufreq_update_policy(policy->cpu); ++ ++ return ret ? ret : count; ++} ++ ++struct freq_attr pxa25x_freq_model_attr = { ++ .attr = { .name = "pxa25x_freq_model", .mode = 0644, .owner=THIS_MODULE }, ++ .show = show_pxa25x_freq_model_attr, ++ .store = store_pxa25x_freq_model_attr, ++}; ++ ++static struct freq_attr* pxa_cpufreq_attr[] = { ++ &cpufreq_freq_attr_scaling_available_freqs, ++ &pxa25x_freq_model_attr, ++ NULL, ++}; ++ ++static struct cpufreq_driver pxa_cpufreq_driver = { ++ .verify = pxa_verify_policy, ++ .target = pxa_set_target, ++ .init = pxa_cpufreq_init, ++ .get = pxa_cpufreq_get, ++ .name = "pxa25x", ++ .attr = pxa_cpufreq_attr, ++}; ++ ++static int __init pxa_cpu_init(void) ++{ ++ return cpufreq_register_driver(&pxa_cpufreq_driver); ++} ++ ++static void __exit pxa_cpu_exit(void) ++{ ++ cpufreq_unregister_driver(&pxa_cpufreq_driver); ++} ++ ++MODULE_AUTHOR("Intrinsyc Software Inc."); ++MODULE_DESCRIPTION("CPU frequency changing driver for the PXA architecture"); ++MODULE_LICENSE("GPL"); ++module_init(pxa_cpu_init); ++module_exit(pxa_cpu_exit); diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/pxa27x_overlay-r4.patch b/packages/linux/logicpd-pxa270-2.6.19.2/pxa27x_overlay-r4.patch new file mode 100644 index 0000000000..8e319d5f3e --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/pxa27x_overlay-r4.patch @@ -0,0 +1,2469 @@ + drivers/video/Kconfig | 18 + drivers/video/Makefile | 1 + drivers/video/pxafb.c | 303 +++++-- + drivers/video/pxafb.h | 65 + + drivers/video/pxafb_overlay.c | 1550 ++++++++++++++++++++++++++++++++++++ + include/asm-arm/arch-pxa/pxa-regs.h | 112 ++ + 6 files changed, 1994 insertions(+), 55 deletions(-) + +Index: linux-2.6.19/drivers/video/Kconfig +=================================================================== +--- linux-2.6.19.orig/drivers/video/Kconfig 2006-12-16 18:35:40.000000000 +0000 ++++ linux-2.6.19/drivers/video/Kconfig 2006-12-16 18:36:17.000000000 +0000 +@@ -1520,6 +1520,24 @@ config FB_PXA + + If unsure, say N. + ++choice ++ prompt "PXA LCD type" ++ depends on FB_PXA ++ ++config FB_PXA_LCD_QVGA ++ bool "QVGA(320x240)" ++ ++config FB_PXA_LCD_VGA ++ bool "VGA (640x480)" ++ ++endchoice ++ ++config FB_PXA_OVERLAY ++ tristate "PXA LCD overlay support" ++ depends on FB_PXA ++ ---help--- ++ Frame buffer overlay driver for PXA27x ++ + config FB_PXA_PARAMETERS + bool "PXA LCD command line parameters" + default n +Index: linux-2.6.19/drivers/video/Makefile +=================================================================== +--- linux-2.6.19.orig/drivers/video/Makefile 2006-12-16 18:35:40.000000000 +0000 ++++ linux-2.6.19/drivers/video/Makefile 2006-12-16 18:36:17.000000000 +0000 +@@ -86,6 +86,7 @@ obj-$(CONFIG_FB_GBE) += gbe + obj-$(CONFIG_FB_CIRRUS) += cirrusfb.o + obj-$(CONFIG_FB_ASILIANT) += asiliantfb.o + obj-$(CONFIG_FB_PXA) += pxafb.o ++obj-$(CONFIG_FB_PXA_OVERLAY) += pxafb_overlay.o + obj-$(CONFIG_FB_W100) += w100fb.o + obj-$(CONFIG_FB_AU1100) += au1100fb.o + obj-$(CONFIG_FB_AU1200) += au1200fb.o +Index: linux-2.6.19/drivers/video/pxafb.c +=================================================================== +--- linux-2.6.19.orig/drivers/video/pxafb.c 2006-12-16 18:36:12.000000000 +0000 ++++ linux-2.6.19/drivers/video/pxafb.c 2006-12-16 18:38:29.000000000 +0000 +@@ -58,17 +58,49 @@ + #define LCCR0_INVALID_CONFIG_MASK (LCCR0_OUM|LCCR0_BM|LCCR0_QDM|LCCR0_DIS|LCCR0_EFM|LCCR0_IUM|LCCR0_SFM|LCCR0_LDM|LCCR0_ENB) + #define LCCR3_INVALID_CONFIG_MASK (LCCR3_HSP|LCCR3_VSP|LCCR3_PCD|LCCR3_BPP) + ++wait_queue_head_t fcs_wait_eof; ++int fcs_in_eof; ++static DECLARE_MUTEX(fcs_lcd_sem); ++ + static void (*pxafb_backlight_power)(int); + static void (*pxafb_lcd_power)(int, struct fb_var_screeninfo *); + + static int pxafb_activate_var(struct fb_var_screeninfo *var, struct pxafb_info *); +-static void set_ctrlr_state(struct pxafb_info *fbi, u_int state); ++void pxafb_set_ctrlr_state(struct pxafb_info *fbi, u_int state); + + #ifdef CONFIG_FB_PXA_PARAMETERS + #define PXAFB_OPTIONS_SIZE 256 + static char g_options[PXAFB_OPTIONS_SIZE] __initdata = ""; + #endif + ++static struct pxafb_rgb def_rgb_8 = { ++ red: { offset: 0, length: 8, }, ++ green: { offset: 0, length: 8, }, ++ blue: { offset: 0, length: 8, }, ++ transp: { offset: 0, length: 0, }, ++}; ++ ++static struct pxafb_rgb def_rgb_16 = { ++ red: { offset: 11, length: 5, }, ++ green: { offset: 5, length: 6, }, ++ blue: { offset: 0, length: 5, }, ++ transp: { offset: 0, length: 0, }, ++}; ++ ++static struct pxafb_rgb def_rgb_18 = { ++ red: { offset: 12, length: 6, }, ++ green: { offset: 6, length: 6, }, ++ blue: { offset: 0, length: 6, }, ++ transp: { offset: 0, length: 0, }, ++}; ++ ++static struct pxafb_rgb def_rgb_24 = { ++ red: { offset: 16, length: 8, }, ++ green: { offset: 8, length: 8, }, ++ blue: { offset: 0, length: 8, }, ++ transp: { offset: 0, length: 0, }, ++}; ++ + static inline void pxafb_schedule_work(struct pxafb_info *fbi, u_int state) + { + unsigned long flags; +@@ -190,6 +222,10 @@ static int pxafb_bpp_to_lccr3(struct fb_ + case 4: ret = LCCR3_4BPP; break; + case 8: ret = LCCR3_8BPP; break; + case 16: ret = LCCR3_16BPP; break; ++ case 18: ret = LCCR3_18BPP; break; ++ case 19: ret = LCCR3_19BPP; break; ++ case 24: ret = LCCR3_24BPP; break; ++ case 25: ret = LCCR3_25BPP; break; + } + return ret; + } +@@ -301,18 +337,34 @@ static int pxafb_check_var(struct fb_var + * The pixel packing format is described on page 7-11 of the + * PXA2XX Developer's Manual. + */ +- if (var->bits_per_pixel == 16) { +- var->red.offset = 11; var->red.length = 5; +- var->green.offset = 5; var->green.length = 6; +- var->blue.offset = 0; var->blue.length = 5; +- var->transp.offset = var->transp.length = 0; +- } else { +- var->red.offset = var->green.offset = var->blue.offset = var->transp.offset = 0; +- var->red.length = 8; +- var->green.length = 8; +- var->blue.length = 8; +- var->transp.length = 0; +- } ++ switch (var->bits_per_pixel) { ++ case 16: ++ /* 2 pixels per line */ ++ var->red = def_rgb_16.red; ++ var->green = def_rgb_16.green; ++ var->blue = def_rgb_16.blue; ++ var->transp = def_rgb_16.transp; ++ break; ++ case 18: ++ case 19: ++ var->red = def_rgb_18.red; ++ var->green = def_rgb_18.green; ++ var->blue = def_rgb_18.blue; ++ var->transp = def_rgb_18.transp; ++ break; ++ case 24: ++ case 25: ++ var->red = def_rgb_24.red; ++ var->green = def_rgb_24.green; ++ var->blue = def_rgb_24.blue; ++ var->transp = def_rgb_24.transp; ++ break; ++ default: ++ var->red = def_rgb_8.red; ++ var->green = def_rgb_8.green; ++ var->blue = def_rgb_8.blue; ++ var->transp = def_rgb_8.transp; ++ } + + #ifdef CONFIG_CPU_FREQ + pr_debug("pxafb: dma period = %d ps, clock = %d kHz\n", +@@ -326,7 +378,7 @@ static int pxafb_check_var(struct fb_var + static inline void pxafb_set_truecolor(u_int is_true_color) + { + pr_debug("pxafb: true_color = %d\n", is_true_color); +- // do your machine-specific setup if needed ++ /* do your machine-specific setup if needed */ + } + + /* +@@ -341,7 +393,8 @@ static int pxafb_set_par(struct fb_info + + pr_debug("pxafb: set_par\n"); + +- if (var->bits_per_pixel == 16) ++ if (var->bits_per_pixel == 16 || var->bits_per_pixel == 18 ||var->bits_per_pixel == 19 ++ || var->bits_per_pixel == 24 || var->bits_per_pixel == 25) + fbi->fb.fix.visual = FB_VISUAL_TRUECOLOR; + else if (!fbi->cmap_static) + fbi->fb.fix.visual = FB_VISUAL_PSEUDOCOLOR; +@@ -354,12 +407,25 @@ static int pxafb_set_par(struct fb_info + fbi->fb.fix.visual = FB_VISUAL_STATIC_PSEUDOCOLOR; + } + +- fbi->fb.fix.line_length = var->xres_virtual * +- var->bits_per_pixel / 8; +- if (var->bits_per_pixel == 16) +- fbi->palette_size = 0; +- else +- fbi->palette_size = var->bits_per_pixel == 1 ? 4 : 1 << var->bits_per_pixel; ++ switch (var->bits_per_pixel) { ++ case 16: ++ fbi->fb.fix.line_length = var->xres_virtual * 2; ++ fbi->palette_size = 0; ++ break; ++ case 18: ++ case 19: ++ fbi->fb.fix.line_length = var->xres_virtual * 3; ++ fbi->palette_size = 0; ++ break; ++ case 24: ++ case 25: ++ fbi->fb.fix.line_length = var->xres_virtual * 4; ++ fbi->palette_size = 0; ++ break; ++ default: ++ fbi->fb.fix.line_length = var->xres_virtual * var->bits_per_pixel / 8; ++ fbi->palette_size = var->bits_per_pixel == 1 ? 4 : 1 << var->bits_per_pixel; ++ } + + palette_mem_size = fbi->palette_size * sizeof(u16); + +@@ -373,7 +439,8 @@ static int pxafb_set_par(struct fb_info + */ + pxafb_set_truecolor(fbi->fb.fix.visual == FB_VISUAL_TRUECOLOR); + +- if (fbi->fb.var.bits_per_pixel == 16) ++ if (fbi->fb.var.bits_per_pixel == 16 || fbi->fb.var.bits_per_pixel == 18 ||fbi->fb.var.bits_per_pixel == 19 ++ || fbi->fb.var.bits_per_pixel == 24 || fbi->fb.var.bits_per_pixel == 25) + fb_dealloc_cmap(&fbi->fb.cmap); + else + fb_alloc_cmap(&fbi->fb.cmap, 1<<fbi->fb.var.bits_per_pixel, 0); +@@ -419,7 +486,7 @@ static int pxafb_set_par(struct fb_info + * 16 bpp mode does not really use the palette, so this will not + * blank the display in all modes. + */ +-static int pxafb_blank(int blank, struct fb_info *info) ++int pxafb_blank(int blank, struct fb_info *info) + { + struct pxafb_info *fbi = (struct pxafb_info *)info; + int i; +@@ -436,19 +503,20 @@ static int pxafb_blank(int blank, struct + for (i = 0; i < fbi->palette_size; i++) + pxafb_setpalettereg(i, 0, 0, 0, 0, info); + +- pxafb_schedule_work(fbi, C_DISABLE); +- //TODO if (pxafb_blank_helper) pxafb_blank_helper(blank); ++ pxafb_schedule_work(fbi, C_BLANK); ++ /* TODO if (pxafb_blank_helper) pxafb_blank_helper(blank); */ + break; + + case FB_BLANK_UNBLANK: +- //TODO if (pxafb_blank_helper) pxafb_blank_helper(blank); ++ /* TODO if (pxafb_blank_helper) pxafb_blank_helper(blank); */ + if (fbi->fb.fix.visual == FB_VISUAL_PSEUDOCOLOR || + fbi->fb.fix.visual == FB_VISUAL_STATIC_PSEUDOCOLOR) + fb_set_cmap(&fbi->fb.cmap, info); +- pxafb_schedule_work(fbi, C_ENABLE); ++ pxafb_schedule_work(fbi, C_UNBLANK); + } + return 0; + } ++EXPORT_SYMBOL(pxafb_blank); + + static int pxafb_mmap(struct fb_info *info, + struct vm_area_struct *vma) +@@ -582,6 +650,10 @@ static int pxafb_activate_var(struct fb_ + case 4: + case 8: + case 16: ++ case 18: ++ case 19: ++ case 24: ++ case 25: + break; + default: + printk(KERN_ERR "%s: invalid bit depth %d\n", +@@ -613,7 +685,10 @@ static int pxafb_activate_var(struct fb_ + + new_regs.lccr0 = fbi->lccr0 | + (LCCR0_LDM | LCCR0_SFM | LCCR0_IUM | LCCR0_EFM | +- LCCR0_QDM | LCCR0_BM | LCCR0_OUM); ++#ifdef CONFIG_PXA27x /* Enable overlay for PXA27x */ ++ LCCR0_OUC | LCCR0_CMDIM | LCCR0_RDSTM | ++#endif ++ LCCR0_QDM | LCCR0_BM | LCCR0_OUM); + + new_regs.lccr1 = + LCCR1_DisWdth(var->xres) + +@@ -672,13 +747,14 @@ static int pxafb_activate_var(struct fb_ + + fbi->dmadesc_fbhigh_cpu->fsadr = fbi->screen_dma; + fbi->dmadesc_fbhigh_cpu->fidr = 0; +- fbi->dmadesc_fbhigh_cpu->ldcmd = BYTES_PER_PANEL; ++ fbi->dmadesc_fbhigh_cpu->ldcmd = BYTES_PER_PANEL | LDCMD_EOFINT; + + fbi->dmadesc_palette_cpu->fsadr = fbi->palette_dma; + fbi->dmadesc_palette_cpu->fidr = 0; + fbi->dmadesc_palette_cpu->ldcmd = (fbi->palette_size * 2) | LDCMD_PAL; + +- if (var->bits_per_pixel == 16) { ++ if (var->bits_per_pixel == 16 || var->bits_per_pixel == 18 ||var->bits_per_pixel == 19 ++ || var->bits_per_pixel == 24 || var->bits_per_pixel == 25) { + /* palette shouldn't be loaded in true-color mode */ + fbi->dmadesc_fbhigh_cpu->fdadr = fbi->dmadesc_fbhigh_dma; + fbi->fdadr0 = fbi->dmadesc_fbhigh_dma; /* no pal just fbhigh */ +@@ -731,8 +807,8 @@ static int pxafb_activate_var(struct fb_ + } + + /* +- * NOTE! The following functions are purely helpers for set_ctrlr_state. +- * Do not call them directly; set_ctrlr_state does the correct serialisation ++ * NOTE! The following functions are purely helpers for pxafb_set_ctrlr_state. ++ * Do not call them directly; pxafb_set_ctrlr_state does the correct serialisation + * to ensure that things happen in the right way 100% of time time. + * -- rmk + */ +@@ -754,7 +830,8 @@ static inline void __pxafb_lcd_power(str + + static void pxafb_setup_gpio(struct pxafb_info *fbi) + { +- int gpio, ldd_bits; ++ int gpio; ++ int ldd_bits = 0; + unsigned int lccr0 = fbi->lccr0; + + /* +@@ -764,28 +841,56 @@ static void pxafb_setup_gpio(struct pxaf + /* 4 bit interface */ + if ((lccr0 & LCCR0_CMS) == LCCR0_Mono && + (lccr0 & LCCR0_SDS) == LCCR0_Sngl && +- (lccr0 & LCCR0_DPD) == LCCR0_4PixMono) ++ (lccr0 & LCCR0_DPD) == LCCR0_4PixMono) { + ldd_bits = 4; +- ++ } + /* 8 bit interface */ + else if (((lccr0 & LCCR0_CMS) == LCCR0_Mono && + ((lccr0 & LCCR0_SDS) == LCCR0_Dual || (lccr0 & LCCR0_DPD) == LCCR0_8PixMono)) || + ((lccr0 & LCCR0_CMS) == LCCR0_Color && +- (lccr0 & LCCR0_PAS) == LCCR0_Pas && (lccr0 & LCCR0_SDS) == LCCR0_Sngl)) ++ (lccr0 & LCCR0_PAS) == LCCR0_Pas && (lccr0 & LCCR0_SDS) == LCCR0_Sngl)) { + ldd_bits = 8; +- ++ } + /* 16 bit interface */ +- else if ((lccr0 & LCCR0_CMS) == LCCR0_Color && +- ((lccr0 & LCCR0_SDS) == LCCR0_Dual || (lccr0 & LCCR0_PAS) == LCCR0_Act)) +- ldd_bits = 16; ++ else if ((lccr0 & LCCR0_CMS) == LCCR0_Color && ++ ((lccr0 & LCCR0_SDS) == LCCR0_Dual || (lccr0 & LCCR0_PAS) == LCCR0_Act)) { ++ switch (fbi->fb.var.bits_per_pixel) { ++ case 16: ++#ifdef CONFIG_PXA27x ++ /* bits 58-77 */ ++ GPDR1 |= (0x3f << 26); ++ GPDR2 |= 0x00003fff; ++ ++ GAFR1_U = (GAFR1_U & ~(0xfff << 20)) | (0xaaa << 20); ++ GAFR2_L = (GAFR2_L & 0xf0000000) | 0x0aaaaaaa; ++#endif ++ ldd_bits = 16; ++ break; ++ case 18: ++ case 19: ++ case 24: ++ case 25: ++#ifdef CONFIG_PXA27x ++ /* bits 58-77 and 86, 87 */ ++ GPDR1 |= (0x3f << 26); ++ GPDR2 |= 0x00c03fff; + ++ GAFR1_U = (GAFR1_U & ~(0xfff << 20)) | (0xaaa << 20); ++ GAFR2_L = (GAFR2_L & 0xf0000000) | 0x0aaaaaaa; ++ GAFR2_U = (GAFR2_U & 0xffff0fff) | 0xa000; ++#endif ++ ldd_bits = 25; ++ break; ++ } ++ } + else { + printk(KERN_ERR "pxafb_setup_gpio: unable to determine bits per pixel\n"); + return; + } + +- for (gpio = 58; ldd_bits; gpio++, ldd_bits--) ++ for (gpio = 58; ldd_bits > 0; gpio++, ldd_bits--) { + pxa_gpio_mode(gpio | GPIO_ALT_FN_2_OUT); ++ } + pxa_gpio_mode(GPIO74_LCD_FCLK_MD); + pxa_gpio_mode(GPIO75_LCD_LCLK_MD); + pxa_gpio_mode(GPIO76_LCD_PCLK_MD); +@@ -805,6 +910,7 @@ static void pxafb_enable_controller(stru + /* enable LCD controller clock */ + pxa_set_cken(CKEN16_LCD, 1); + ++ down(&fcs_lcd_sem); + /* Sequence from 11.7.10 */ + LCCR3 = fbi->reg_lccr3; + LCCR2 = fbi->reg_lccr2; +@@ -815,6 +921,8 @@ static void pxafb_enable_controller(stru + FDADR1 = fbi->fdadr1; + LCCR0 |= LCCR0_ENB; + ++ up(&fcs_lcd_sem); ++ + pr_debug("FDADR0 0x%08x\n", (unsigned int) FDADR0); + pr_debug("FDADR1 0x%08x\n", (unsigned int) FDADR1); + pr_debug("LCCR0 0x%08x\n", (unsigned int) LCCR0); +@@ -829,6 +937,7 @@ static void pxafb_disable_controller(str + + pr_debug("pxafb: disabling LCD controller\n"); + ++ down(&fcs_lcd_sem); + set_current_state(TASK_UNINTERRUPTIBLE); + add_wait_queue(&fbi->ctrlr_wait, &wait); + +@@ -838,6 +947,7 @@ static void pxafb_disable_controller(str + + schedule_timeout(200 * HZ / 1000); + remove_wait_queue(&fbi->ctrlr_wait, &wait); ++ up(&fcs_lcd_sem); + + /* disable LCD controller clock */ + pxa_set_cken(CKEN16_LCD, 0); +@@ -855,6 +965,11 @@ static irqreturn_t pxafb_handle_irq(int + LCCR0 |= LCCR0_LDM; + wake_up(&fbi->ctrlr_wait); + } ++ if (lcsr & LCSR_EOF && fcs_in_eof) { ++ LCCR0 |= LCCR0_EFM; ++ fcs_in_eof = 0; ++ wake_up(&fcs_wait_eof); ++ } + + LCSR = lcsr; + return IRQ_HANDLED; +@@ -865,7 +980,7 @@ static irqreturn_t pxafb_handle_irq(int + * sleep when disabling the LCD controller, or if we get two contending + * processes trying to alter state. + */ +-static void set_ctrlr_state(struct pxafb_info *fbi, u_int state) ++void pxafb_set_ctrlr_state(struct pxafb_info *fbi, u_int state) + { + u_int old_state; + +@@ -887,7 +1002,9 @@ static void set_ctrlr_state(struct pxafb + */ + if (old_state != C_DISABLE && old_state != C_DISABLE_PM) { + fbi->state = state; +- //TODO __pxafb_lcd_power(fbi, 0); ++ /* TODO __pxafb_lcd_power(fbi, 0); */ ++ if(fbi->set_overlay_ctrlr_state) ++ fbi->set_overlay_ctrlr_state(fbi, C_DISABLE); + pxafb_disable_controller(fbi); + } + break; +@@ -901,6 +1018,8 @@ static void set_ctrlr_state(struct pxafb + fbi->state = state; + __pxafb_backlight_power(fbi, 0); + __pxafb_lcd_power(fbi, 0); ++ if(fbi->set_overlay_ctrlr_state) ++ fbi->set_overlay_ctrlr_state(fbi, C_DISABLE); + if (old_state != C_DISABLE_CLKCHANGE) + pxafb_disable_controller(fbi); + } +@@ -914,7 +1033,9 @@ static void set_ctrlr_state(struct pxafb + if (old_state == C_DISABLE_CLKCHANGE) { + fbi->state = C_ENABLE; + pxafb_enable_controller(fbi); +- //TODO __pxafb_lcd_power(fbi, 1); ++ /* TODO __pxafb_lcd_power(fbi, 1); */ ++ if(fbi->set_overlay_ctrlr_state) ++ fbi->set_overlay_ctrlr_state(fbi, C_ENABLE); + } + break; + +@@ -926,9 +1047,13 @@ static void set_ctrlr_state(struct pxafb + */ + if (old_state == C_ENABLE) { + __pxafb_lcd_power(fbi, 0); ++ if(fbi->set_overlay_ctrlr_state) ++ fbi->set_overlay_ctrlr_state(fbi, C_DISABLE); + pxafb_disable_controller(fbi); + pxafb_setup_gpio(fbi); + pxafb_enable_controller(fbi); ++ if(fbi->set_overlay_ctrlr_state) ++ fbi->set_overlay_ctrlr_state(fbi, C_ENABLE); + __pxafb_lcd_power(fbi, 1); + } + break; +@@ -954,11 +1079,46 @@ static void set_ctrlr_state(struct pxafb + pxafb_enable_controller(fbi); + __pxafb_lcd_power(fbi, 1); + __pxafb_backlight_power(fbi, 1); ++ if(fbi->set_overlay_ctrlr_state) ++ fbi->set_overlay_ctrlr_state(fbi, C_ENABLE); + } + break; ++ ++ case C_BLANK: ++ /* ++ * Disable controller, blank overlays if exist. ++ */ ++ if ((old_state != C_DISABLE) && (old_state != C_BLANK)) { ++ fbi->state = state; ++ __pxafb_backlight_power(fbi, 0); ++ __pxafb_lcd_power(fbi, 0); ++ if(fbi->set_overlay_ctrlr_state) ++ fbi->set_overlay_ctrlr_state(fbi, C_BLANK); ++ if (old_state != C_DISABLE_CLKCHANGE) ++ pxafb_disable_controller(fbi); ++ } ++ break; ++ ++ case C_UNBLANK: ++ /* ++ * Power up the LCD screen, enable controller, and ++ * turn on the backlight, unblank overlays if exist. ++ */ ++ if ((old_state != C_ENABLE) && (old_state != C_UNBLANK)) { ++ fbi->state = C_UNBLANK; ++ pxafb_setup_gpio(fbi); ++ pxafb_enable_controller(fbi); ++ __pxafb_lcd_power(fbi, 1); ++ __pxafb_backlight_power(fbi, 1); ++ if(fbi->set_overlay_ctrlr_state) ++ fbi->set_overlay_ctrlr_state(fbi, C_UNBLANK); ++ } ++ break; ++ + } + up(&fbi->ctrlr_sem); + } ++EXPORT_SYMBOL(pxafb_set_ctrlr_state); + + /* + * Our LCD controller task (which is called when we blank or unblank) +@@ -969,7 +1129,7 @@ static void pxafb_task(void *dummy) + struct pxafb_info *fbi = dummy; + u_int state = xchg(&fbi->task_state, -1); + +- set_ctrlr_state(fbi, state); ++ pxafb_set_ctrlr_state(fbi, state); + } + + #ifdef CONFIG_CPU_FREQ +@@ -984,19 +1144,29 @@ static int + pxafb_freq_transition(struct notifier_block *nb, unsigned long val, void *data) + { + struct pxafb_info *fbi = TO_INF(nb, freq_transition); +- //TODO struct cpufreq_freqs *f = data; ++ /* TODO struct cpufreq_freqs *f = data; */ ++ struct cpufreq_freqs *clkinfo; + u_int pcd; ++ u_int lccr3; + + switch (val) { + case CPUFREQ_PRECHANGE: +- set_ctrlr_state(fbi, C_DISABLE_CLKCHANGE); ++ pxafb_set_ctrlr_state(fbi, C_DISABLE_CLKCHANGE); + break; + + case CPUFREQ_POSTCHANGE: ++ clkinfo = (struct cpufreq_freqs *)data; ++ /* If leaving a 13kHz state with the LCD sustained */ ++ if ((clkinfo->old == 13000)) ++ break; ++ + pcd = get_pcd(fbi->fb.var.pixclock); ++ lccr3 = fbi->reg_lccr3; + set_hsync_time(fbi, pcd); + fbi->reg_lccr3 = (fbi->reg_lccr3 & ~0xff) | LCCR3_PixClkDiv(pcd); +- set_ctrlr_state(fbi, C_ENABLE_CLKCHANGE); ++ pxafb_set_ctrlr_state(fbi, C_ENABLE_CLKCHANGE); ++ if (lccr3 != fbi->reg_lccr3 && !((LCCR0 & LCCR0_DIS) || !(LCCR0 & LCCR0_ENB))) ++ LCCR3 = fbi->reg_lccr3; + break; + } + return 0; +@@ -1015,7 +1185,7 @@ pxafb_freq_policy(struct notifier_block + printk(KERN_DEBUG "min dma period: %d ps, " + "new clock %d kHz\n", pxafb_display_dma_period(var), + policy->max); +- // TODO: fill in min/max values ++ /* TODO: fill in min/max values */ + break; + #if 0 + case CPUFREQ_NOTIFY: +@@ -1041,7 +1211,7 @@ static int pxafb_suspend(struct platform + { + struct pxafb_info *fbi = platform_get_drvdata(dev); + +- set_ctrlr_state(fbi, C_DISABLE_PM); ++ pxafb_set_ctrlr_state(fbi, C_DISABLE_PM); + return 0; + } + +@@ -1049,7 +1219,11 @@ static int pxafb_resume(struct platform_ + { + struct pxafb_info *fbi = platform_get_drvdata(dev); + +- set_ctrlr_state(fbi, C_ENABLE_PM); ++ pxafb_set_ctrlr_state(fbi, C_ENABLE_PM); ++//RP#ifdef CONFIG_PXA27x ++//RP LCCR4 |= (1 << 31); /* Disable the PCD Divisor, PCDDIV */ ++//RP LCCR4 |= (5 << 17); /* Undocumented feature */ ++//RP#endif + return 0; + } + #else +@@ -1153,11 +1327,21 @@ static struct pxafb_info * __init pxafb_ + fbi->task_state = (u_char)-1; + + for (i = 0; i < inf->num_modes; i++) { +- smemlen = mode[i].xres * mode[i].yres * mode[i].bpp / 8; ++ if (mode[i].bpp <= 16) { /* 8, 16 bpp */ ++ smemlen = mode[i].xres * mode[i].yres * mode[i].bpp / 8; ++ } else if ( mode[i].bpp > 19 ) { /* 24, 25 bpp */ ++ smemlen = mode[i].xres * mode[i].yres * 4; ++ } else { /* 18, 19 bpp */ ++ /* packed format */ ++ smemlen = mode[i].xres * mode[i].yres * 3; ++ } ++ + if (smemlen > fbi->fb.fix.smem_len) + fbi->fb.fix.smem_len = smemlen; + } + ++ fbi->set_overlay_ctrlr_state = NULL; ++ + init_waitqueue_head(&fbi->ctrlr_wait); + INIT_WORK(&fbi->task, pxafb_task, fbi); + init_MUTEX(&fbi->ctrlr_sem); +@@ -1224,6 +1408,10 @@ static int __init pxafb_parse_options(st + case 4: + case 8: + case 16: ++ case 18: ++ case 19: ++ case 24: ++ case 25: + inf->modes[0].bpp = bpp; + dev_info(dev, "overriding bit depth: %d\n", bpp); + break; +@@ -1372,7 +1560,7 @@ int __init pxafb_probe(struct platform_d + fbi = pxafb_init_fbinfo(&dev->dev); + if (!fbi) { + dev_err(&dev->dev, "Failed to initialize framebuffer device\n"); +- ret = -ENOMEM; // only reason for pxafb_init_fbinfo to fail is kmalloc ++ ret = -ENOMEM; /* only reason for pxafb_init_fbinfo to fail is kmalloc */ + goto failed; + } + +@@ -1407,7 +1595,7 @@ int __init pxafb_probe(struct platform_d + } + + #ifdef CONFIG_PM +- // TODO ++ /* TODO */ + #endif + + #ifdef CONFIG_CPU_FREQ +@@ -1420,7 +1608,12 @@ int __init pxafb_probe(struct platform_d + /* + * Ok, now enable the LCD controller + */ +- set_ctrlr_state(fbi, C_ENABLE); ++ pxafb_set_ctrlr_state(fbi, C_ENABLE); ++//#ifdef CONFIG_PXA27x ++// LCCR4 |= (1 << 31); /* Disabel the PCD Divisor, PCDDIV */ ++// LCCR4 |= (5 << 17); /* Undocumented feature */ ++//#endif ++ init_waitqueue_head(&fcs_wait_eof); + + return 0; + +Index: linux-2.6.19/drivers/video/pxafb.h +=================================================================== +--- linux-2.6.19.orig/drivers/video/pxafb.h 2006-12-16 18:35:40.000000000 +0000 ++++ linux-2.6.19/drivers/video/pxafb.h 2006-12-16 18:36:17.000000000 +0000 +@@ -29,6 +29,60 @@ struct pxafb_lcd_reg { + unsigned int lccr3; + }; + ++struct pxafb_rgb { ++ struct fb_bitfield red; ++ struct fb_bitfield green; ++ struct fb_bitfield blue; ++ struct fb_bitfield transp; ++}; ++ ++#ifdef CONFIG_PXA27x ++/* PXA Overlay Framebuffer Support */ ++struct overlayfb_info ++{ ++ struct fb_info fb; ++ ++ struct fb_var_screeninfo old_var; ++ ++ struct semaphore mutex; ++ unsigned long refcount; ++ ++ struct pxafb_info *basefb; ++ ++ unsigned long map_cpu; ++ unsigned long screen_cpu; ++ unsigned long palette_cpu; ++ unsigned long map_size; ++ unsigned long palette_size; ++ ++ dma_addr_t screen_dma; ++ dma_addr_t map_dma; ++ dma_addr_t palette_dma; ++ ++ volatile u_char state; ++ ++ /* overlay specific info */ ++ unsigned long xpos; /* screen position (x, y)*/ ++ unsigned long ypos; ++ unsigned long format; ++ ++ /* additional */ ++ union { ++ struct pxafb_dma_descriptor *dma0; ++ struct pxafb_dma_descriptor *dma1; ++ struct { ++ struct pxafb_dma_descriptor *dma2; ++ struct pxafb_dma_descriptor *dma3; ++ struct pxafb_dma_descriptor *dma4; ++ }; ++ struct { ++ struct pxafb_dma_descriptor *dma5_pal; ++ struct pxafb_dma_descriptor *dma5_frame; ++ }; ++ }; ++}; ++#endif ++ + /* PXA LCD DMA descriptor */ + struct pxafb_dma_descriptor { + unsigned int fdadr; +@@ -87,6 +141,14 @@ struct pxafb_info { + wait_queue_head_t ctrlr_wait; + struct work_struct task; + ++#ifdef CONFIG_PXA27x ++ /* PXA Overlay Framebuffer Support */ ++ struct overlayfb_info *overlay1fb; ++ struct overlayfb_info *overlay2fb; ++ struct overlayfb_info *cursorfb; ++#endif ++ void (*set_overlay_ctrlr_state)(struct pxafb_info *, u_int); ++ + #ifdef CONFIG_CPU_FREQ + struct notifier_block freq_transition; + struct notifier_block freq_policy; +@@ -106,6 +168,9 @@ struct pxafb_info { + #define C_DISABLE_PM (5) + #define C_ENABLE_PM (6) + #define C_STARTUP (7) ++#define C_BLANK (8) ++#define C_UNBLANK (9) ++ + + #define PXA_NAME "PXA" + +Index: linux-2.6.19/drivers/video/pxafb_overlay.c +=================================================================== +--- /dev/null 1970-01-01 00:00:00.000000000 +0000 ++++ linux-2.6.19/drivers/video/pxafb_overlay.c 2006-12-16 18:36:17.000000000 +0000 +@@ -0,0 +1,1550 @@ ++/* ++ * linux/drivers/video/pxafb_overlay.c ++ * ++ * Copyright (c) 2004, Intel Corporation ++ * ++ * Code Status: ++ * 2004/10/28: <yan.yin@intel.com> ++ * - Ported to 2.6 kernel ++ * - Made overlay driver a loadable module ++ * - Merged overlay optimized patch ++ * 2004/03/10: <stanley.cai@intel.com> ++ * - Fixed Bugs ++ * - Added workaround for overlay1&2 ++ * 2003/08/27: <yu.tang@intel.com> ++ * - Added Overlay 1 & Overlay2 & Hardware Cursor support ++ * ++ * ++ * This software program is licensed subject to the GNU Lesser General ++ * Public License (LGPL). Version 2.1, February 1999, available at ++ * http://www.gnu.org/copyleft/lesser.html ++ * ++ * Intel PXA27x LCD Controller Frame Buffer Overlay Driver ++ * ++ * ++ */ ++ ++#include <linux/module.h> ++#include <linux/moduleparam.h> ++#include <linux/kernel.h> ++#include <linux/sched.h> ++#include <linux/errno.h> ++#include <linux/string.h> ++#include <linux/interrupt.h> ++#include <linux/slab.h> ++#include <linux/fb.h> ++#include <linux/delay.h> ++#include <linux/init.h> ++#include <linux/ioport.h> ++#include <linux/cpufreq.h> ++#include <linux/device.h> ++#include <linux/platform_device.h> ++#include <linux/dma-mapping.h> ++ ++#include <asm/hardware.h> ++#include <asm/io.h> ++#include <asm/irq.h> ++#include <asm/uaccess.h> ++#include <asm/arch/bitfield.h> ++#include <asm/arch/pxafb.h> ++#include <asm/arch/pxa-regs.h> ++ ++#include "pxafb.h" ++ ++/* LCD enhancement : Overlay 1 & 2 & Hardware Cursor */ ++ ++/* ++ * LCD enhancement : Overlay 1 ++ * ++ * Features: ++ * - support 16bpp (No palette) ++ */ ++/* ++ * debugging? ++ */ ++#define DEBUG 0 ++ ++#ifdef DEBUG ++#define dbg(fmt,arg...) printk(KERN_ALERT "%s(): " fmt "\n", __FUNCTION__, ##arg) ++#else ++#define dbg(fmt,arg...) ++#endif ++ ++static int overlay1fb_enable(struct fb_info *info); ++static int overlay2fb_enable(struct fb_info *info); ++static int cursorfb_enable(struct fb_info *info); ++ ++static int overlay1fb_disable(struct fb_info *info); ++static int overlay2fb_disable(struct fb_info *info); ++static int cursorfb_disable(struct fb_info *info); ++ ++static int overlay1fb_blank(int blank, struct fb_info *info); ++static int overlay2fb_blank(int blank, struct fb_info *info); ++static int cursorfb_blank(int blank, struct fb_info *info); ++ ++extern void pxafb_set_ctrlr_state(struct pxafb_info *fbi, u_int state); ++extern int pxafb_blank(int blank, struct fb_info *info); ++ ++static struct pxafb_rgb def_rgb_18 = { ++ red: { offset: 12, length: 6, }, ++ green: { offset: 6, length: 6, }, ++ blue: { offset: 0, length: 6, }, ++ transp: { offset: 0, length: 0, }, ++}; ++ ++static struct pxafb_rgb def_rgbt_16 = { ++ red: { offset: 10, length: 5, }, ++ green: { offset: 5, length: 5, }, ++ blue: { offset: 0, length: 5, }, ++ transp: { offset: 15, length: 1, }, ++}; ++ ++static struct pxafb_rgb def_rgbt_19 = { ++ red: { offset: 12, length: 6, }, ++ green: { offset: 6, length: 6, }, ++ blue: { offset: 0, length: 6, }, ++ transp: { offset: 18, length: 1, }, ++}; ++ ++static struct pxafb_rgb def_rgbt_24 = { ++ red: { offset: 16, length: 7, }, ++ green: { offset: 8, length: 8, }, ++ blue: { offset: 0, length: 8, }, ++ transp: { offset: 0, length: 0, }, ++}; ++ ++static struct pxafb_rgb def_rgbt_25 = { ++ red: { offset: 16, length: 8, }, ++ green: { offset: 8, length: 8, }, ++ blue: { offset: 0, length: 8, }, ++ transp: { offset: 24, length: 1, }, ++}; ++ ++#define CLEAR_LCD_INTR(reg, intr) do { \ ++ reg = (intr); \ ++}while(0) ++ ++#define WAIT_FOR_LCD_INTR(reg,intr,timeout) ({ \ ++ int __done =0; \ ++ int __t = timeout; \ ++ while (__t) { \ ++ __done = (reg) & (intr); \ ++ if (__done) break; \ ++ mdelay(10); \ ++ __t--; \ ++ } \ ++ if (!__t) dbg("wait " #intr " timeount");\ ++ __done; \ ++}) ++ ++#define DISABLE_OVERLAYS(fbi) do { \ ++ if (fbi->overlay1fb && (fbi->overlay1fb->state == C_ENABLE)) { \ ++ overlay1fb_disable((struct fb_info*)fbi->overlay1fb); \ ++ } \ ++ if (fbi->overlay2fb && (fbi->overlay2fb->state == C_ENABLE)) { \ ++ overlay2fb_disable((struct fb_info*)fbi->overlay2fb); \ ++ } \ ++ if (fbi->cursorfb && (fbi->cursorfb->state == C_ENABLE)) { \ ++ cursorfb_disable((struct fb_info*)fbi->cursorfb); \ ++ } \ ++}while(0) ++ ++#define ENABLE_OVERLAYS(fbi) do { \ ++ if (fbi->overlay1fb && (fbi->overlay1fb->state == C_DISABLE)){ \ ++ overlay1fb_enable((struct fb_info*)fbi->overlay1fb); \ ++ } \ ++ if (fbi->overlay2fb && (fbi->overlay2fb->state == C_DISABLE)){ \ ++ overlay2fb_enable((struct fb_info*)fbi->overlay2fb); \ ++ } \ ++ if (fbi->cursorfb && (fbi->cursorfb->state == C_DISABLE)){ \ ++ cursorfb_enable((struct fb_info*)fbi->cursorfb); \ ++ } \ ++}while(0) ++ ++#define BLANK_OVERLAYS(fbi) do { \ ++ if (fbi->overlay1fb && (fbi->overlay1fb->state == C_ENABLE)) { \ ++ overlay1fb_disable((struct fb_info*)fbi->overlay1fb); \ ++ fbi->overlay1fb->state = C_BLANK; \ ++ } \ ++ if (fbi->overlay2fb && (fbi->overlay2fb->state == C_ENABLE)) { \ ++ overlay2fb_disable((struct fb_info*)fbi->overlay2fb); \ ++ fbi->overlay2fb->state = C_BLANK; \ ++ } \ ++ if (fbi->cursorfb && (fbi->cursorfb->state == C_ENABLE)) { \ ++ cursorfb_disable((struct fb_info*)fbi->cursorfb); \ ++ fbi->cursorfb->state = C_BLANK; \ ++ } \ ++}while(0) ++ ++#define UNBLANK_OVERLAYS(fbi) do { \ ++ if (fbi->overlay1fb && (fbi->overlay1fb->state == C_BLANK)){ \ ++ overlay1fb_enable((struct fb_info*)fbi->overlay1fb); \ ++ fbi->overlay1fb->state = C_ENABLE; \ ++ } \ ++ if (fbi->overlay2fb && (fbi->overlay2fb->state == C_BLANK)){ \ ++ overlay2fb_enable((struct fb_info*)fbi->overlay2fb); \ ++ fbi->overlay2fb->state = C_ENABLE; \ ++ } \ ++ if (fbi->cursorfb && (fbi->cursorfb->state == C_BLANK)){ \ ++ cursorfb_enable((struct fb_info*)fbi->cursorfb); \ ++ fbi->cursorfb->state = C_ENABLE; \ ++ } \ ++}while(0) ++ ++static int overlay1fb_open(struct fb_info *info, int user) ++{ ++ struct overlayfb_info *fbi = (struct overlayfb_info*) info; ++ int ret = 0; ++ ++/* If basefb is disable, enable fb. */ ++ if (fbi->basefb && fbi->basefb->state != C_ENABLE) ++ pxafb_blank(VESA_NO_BLANKING, (struct fb_info *)(fbi->basefb)); ++ ++ down(&fbi->mutex); ++ ++ if (fbi->refcount) ++ ret = -EACCES; ++ else ++ fbi->refcount ++; ++ ++ up(&fbi->mutex); ++ ++ /* Initialize the variables in overlay1 framebuffer. */ ++ fbi->fb.var.xres = fbi->fb.var.yres = 0; ++ fbi->fb.var.bits_per_pixel = 0; ++ ++ return ret; ++} ++ ++static int overlay1fb_release(struct fb_info *info, int user) ++{ ++ struct overlayfb_info *fbi = (struct overlayfb_info*) info; ++ down(&fbi->mutex); ++ ++ if (fbi->refcount) ++ fbi->refcount --; ++ ++ up(&fbi->mutex); ++ /* disable overlay when released */ ++ overlay1fb_blank(1, info); ++ ++ return 0; ++} ++ ++static int overlay1fb_map_video_memory(struct fb_info *info) ++{ ++ struct overlayfb_info *fbi = (struct overlayfb_info*) info; ++ ++ if (fbi->map_cpu) ++ dma_free_writecombine(NULL, fbi->map_size, (void*)fbi->map_cpu, fbi->map_dma); ++ fbi->map_size = PAGE_ALIGN(fbi->fb.fix.smem_len + PAGE_SIZE); ++ ++ fbi->map_cpu = (unsigned long)dma_alloc_writecombine(NULL, fbi->map_size, ++ &fbi->map_dma, GFP_KERNEL ); ++ ++ if (!fbi->map_cpu) return -ENOMEM; ++ ++ fbi->screen_cpu = fbi->map_cpu + PAGE_SIZE; ++ fbi->screen_dma = fbi->map_dma + PAGE_SIZE; ++ ++ fbi->fb.fix.smem_start = fbi->screen_dma; ++ ++ /* setup dma descriptor */ ++ fbi->dma1 = (struct pxafb_dma_descriptor*) ++ (fbi->screen_cpu - sizeof(struct pxafb_dma_descriptor)); ++ ++ fbi->dma1->fdadr = (fbi->screen_dma - sizeof(struct pxafb_dma_descriptor)); ++ fbi->dma1->fsadr = fbi->screen_dma; ++ fbi->dma1->fidr = 0; ++ fbi->dma1->ldcmd = fbi->fb.fix.smem_len; ++ ++ return 0; ++} ++ ++static int overlay1fb_enable(struct fb_info *info) ++{ ++ struct overlayfb_info *fbi = (struct overlayfb_info*) info; ++ unsigned long bpp1; ++ ++ if (!fbi->map_cpu) return -EINVAL; ++ ++ switch(fbi->fb.var.bits_per_pixel){ ++ case 16: ++ bpp1 = 0x4; ++ break; ++ case 18: ++ bpp1 = 0x6; ++ break; ++ case 19: ++ bpp1 = 0x8; ++ break; ++ case 24: ++ bpp1 = 0x9; ++ break; ++ case 25: ++ bpp1 = 0xa; ++ break; ++ default: ++ return -EINVAL; ++ } ++ ++ /* disable branch/start/end of frame interrupt */ ++ LCCR5 |= (LCCR5_IUM1 | LCCR5_BSM1 | LCCR5_EOFM1 | LCCR5_SOFM1); ++ ++ if (fbi->state == C_DISABLE || fbi->state == C_BLANK) ++ FDADR1 = (fbi->dma1->fdadr); ++ else ++ FBR1 = fbi->dma1->fdadr | 0x1; ++ ++ /* enable overlay 1 window */ ++ OVL1C2 = (fbi->ypos << 10) | fbi->xpos; ++ OVL1C1 = OVL1C1_O1EN | (bpp1 << 20) | ((fbi->fb.var.yres-1)<<10) | (fbi->fb.var.xres-1); ++ ++ fbi->state = C_ENABLE; ++ ++ return 0; ++} ++ ++static int overlay1fb_disable(struct fb_info *info) ++{ ++ struct overlayfb_info *fbi = (struct overlayfb_info*)info; ++ int done; ++ ++ if ((fbi->state == C_DISABLE) || (fbi->state == C_BLANK)) ++ return 0; ++ ++ fbi->state = C_DISABLE; ++ ++ /* clear O1EN */ ++ OVL1C1 &= ~OVL1C1_O1EN; ++ ++ CLEAR_LCD_INTR(LCSR1, LCSR1_BS1); ++ FBR1 = 0x3; ++ done = WAIT_FOR_LCD_INTR(LCSR1, LCSR1_BS1, 100); ++ ++ if (!done) { ++ pr_debug(KERN_INFO "%s: timeout\n", __FUNCTION__); ++ return -1; ++ } ++ return 0; ++} ++ ++static int overlay1fb_blank(int blank, struct fb_info *info) ++{ ++ struct overlayfb_info *fbi = (struct overlayfb_info*) info; ++ int err=0; ++ ++ switch(blank) ++ { ++ case 0: ++ err = overlay1fb_enable(info); ++ if (err) { ++ fbi->state = C_DISABLE; ++ pxafb_set_ctrlr_state(fbi->basefb, C_REENABLE); ++ } ++ break; ++ case 1: ++ err = overlay1fb_disable(info); ++ if (err) { ++ fbi->state = C_DISABLE; ++ pxafb_set_ctrlr_state(fbi->basefb, C_REENABLE); ++ } ++ break; ++ default: ++ break; ++ } ++ ++ return err; ++} ++ ++static int overlay1fb_check_var( struct fb_var_screeninfo *var, struct fb_info *info) ++{ ++ int xpos, ypos; ++ struct overlayfb_info *fbi=(struct overlayfb_info*)info; ++ ++ /* must in base frame */ ++ xpos = (var->nonstd & 0x3ff); ++ ypos = ((var->nonstd>>10) & 0x3ff); ++ ++ if ( (xpos + var->xres) > fbi->basefb->fb.var.xres ) ++ return -EINVAL; ++ ++ if ( (ypos + var->yres) > fbi->basefb->fb.var.yres ) ++ return -EINVAL; ++ ++ switch (var->bits_per_pixel) { ++ case 16: ++ if ( var->xres & 0x1 ) { ++ printk("xres should be a multiple of 2 pixels!\n"); ++ return -EINVAL; ++ } ++ break; ++ case 18: ++ case 19: ++ if ( var->xres & 0x7 ) { ++ printk("xres should be a multiple of 8 pixels!\n"); ++ return -EINVAL; ++ } ++ break; ++ default: ++ break; ++ } ++ ++ fbi->old_var=*var; ++ ++ var->activate=FB_ACTIVATE_NOW; ++ ++ return 0; ++} ++ ++ ++static int overlay1fb_set_par(struct fb_info *info) ++{ ++ int nbytes=0, err=0, pixels_per_line=0; ++ ++ struct overlayfb_info *fbi=(struct overlayfb_info*)info; ++ struct fb_var_screeninfo *var = &fbi->fb.var; ++ ++ info->flags &= ~FBINFO_MISC_USEREVENT; ++ ++ if (fbi->state == C_BLANK) ++ return 0; ++ ++ if (fbi->state == C_DISABLE) ++ goto out1; ++ ++ /* only xpos & ypos change */ ++ if ( (var->xres == fbi->old_var.xres) && ++ (var->yres == fbi->old_var.yres) && ++ (var->bits_per_pixel == fbi->old_var.bits_per_pixel) ) ++ goto out2; ++ ++ out1: ++ switch(var->bits_per_pixel) { ++ case 16: ++ /* 2 pixels per line */ ++ pixels_per_line = (fbi->fb.var.xres + 0x1) & (~0x1); ++ nbytes = 2; ++ ++ var->red = def_rgbt_16.red; ++ var->green = def_rgbt_16.green; ++ var->blue = def_rgbt_16.blue; ++ var->transp = def_rgbt_16.transp; ++ ++ break; ++ case 18: ++ /* 8 pixels per line */ ++ pixels_per_line = (fbi->fb.var.xres + 0x7 ) & (~0x7); ++ nbytes = 3; ++ ++ var->red = def_rgb_18.red; ++ var->green = def_rgb_18.green; ++ var->blue = def_rgb_18.blue; ++ var->transp = def_rgb_18.transp; ++ ++ break; ++ case 19: ++ /* 8 pixels per line */ ++ pixels_per_line = (fbi->fb.var.xres + 0x7 ) & (~0x7); ++ nbytes = 3; ++ ++ var->red = def_rgbt_19.red; ++ var->green = def_rgbt_19.green; ++ var->blue = def_rgbt_19.blue; ++ var->transp = def_rgbt_19.transp; ++ ++ break; ++ case 24: ++ pixels_per_line = fbi->fb.var.xres; ++ nbytes = 4; ++ ++ var->red = def_rgbt_24.red; ++ var->green = def_rgbt_24.green; ++ var->blue = def_rgbt_24.blue; ++ var->transp = def_rgbt_24.transp; ++ ++ break; ++ case 25: ++ pixels_per_line = fbi->fb.var.xres; ++ nbytes = 4; ++ ++ var->red = def_rgbt_25.red; ++ var->green = def_rgbt_25.green; ++ var->blue = def_rgbt_25.blue; ++ var->transp = def_rgbt_25.transp; ++ ++ break; ++ } ++ ++ fbi->fb.fix.line_length = nbytes * pixels_per_line; ++ fbi->fb.fix.smem_len = fbi->fb.fix.line_length * fbi->fb.var.yres; ++ ++ err= overlay1fb_map_video_memory((struct fb_info*)fbi); ++ ++ if (err) return err; ++ ++out2: ++ fbi->xpos = var->nonstd & 0x3ff; ++ fbi->ypos = (var->nonstd>>10) & 0x3ff; ++ ++ overlay1fb_enable(info); ++ ++ return 0; ++ ++} ++ ++static struct fb_ops overlay1fb_ops = { ++ .owner = THIS_MODULE, ++ .fb_open = overlay1fb_open, ++ .fb_release = overlay1fb_release, ++ .fb_check_var = overlay1fb_check_var, ++ .fb_set_par = overlay1fb_set_par, ++ .fb_blank = overlay1fb_blank, ++ .fb_fillrect = cfb_fillrect, ++ .fb_copyarea = cfb_copyarea, ++ .fb_imageblit = cfb_imageblit, ++}; ++ ++ /* ++ * LCD enhancement : Overlay 2 ++ * ++ * Features: ++ * - support planar YCbCr420/YCbCr422/YCbCr444; ++ */ ++static int overlay2fb_open(struct fb_info *info, int user) ++{ ++ struct overlayfb_info *fbi = (struct overlayfb_info*) info; ++ int ret = 0; ++ ++ /* if basefb is disable, enable fb. */ ++ if (fbi->basefb && fbi->basefb->state != C_ENABLE) ++ pxafb_blank(VESA_NO_BLANKING, (struct fb_info *)(fbi->basefb)); ++ ++ down(&fbi->mutex); ++ ++ if (fbi->refcount) ++ ret = -EACCES; ++ else ++ fbi->refcount ++; ++ ++ up(&fbi->mutex); ++ fbi->fb.var.xres = fbi->fb.var.yres = 0; ++ ++ return ret; ++} ++ ++static int overlay2fb_release(struct fb_info *info, int user) ++{ ++ struct overlayfb_info *fbi = (struct overlayfb_info*) info; ++ ++ down(&fbi->mutex); ++ ++ if (fbi->refcount) ++ fbi->refcount --; ++ ++ up(&fbi->mutex); ++ ++ /* disable overlay when released */ ++ overlay2fb_blank(1, info); ++ ++ return 0; ++} ++ ++static int overlay2fb_map_YUV_memory( struct fb_info *info) ++{ ++ struct overlayfb_info *fbi = (struct overlayfb_info*) info; ++ unsigned int ylen, cblen, crlen, aylen, acblen, acrlen; ++ unsigned int yoff, cboff, croff; ++ unsigned int xres,yres; ++ unsigned int nbytes; ++ ++ ylen = cblen = crlen = aylen = acblen = acrlen = 0; ++ yoff = cboff = croff = 0; ++ ++ if (fbi->map_cpu) ++ dma_free_writecombine(NULL, fbi->map_size, (void*)fbi->map_cpu, fbi->map_dma); ++ ++ yres = fbi->fb.var.yres; ++ ++ switch(fbi->format) { ++ case 0x4: /* YCbCr 4:2:0 planar */ ++ pr_debug("420 planar\n"); ++ /* 16 pixels per line */ ++ xres = (fbi->fb.var.xres + 0xf) & (~0xf); ++ fbi->fb.fix.line_length = xres; ++ ++ nbytes = xres * yres; ++ ylen = nbytes; ++ cblen = crlen = (nbytes/4); ++ ++ break; ++ case 0x3: /* YCbCr 4:2:2 planar */ ++ /* 8 pixles per line */ ++ pr_debug("422 planar\n"); ++ xres = (fbi->fb.var.xres + 0x7) & (~0x7); ++ fbi->fb.fix.line_length = xres; ++ ++ nbytes = xres * yres; ++ ylen = nbytes; ++ cblen = crlen = (nbytes/2); ++ ++ break; ++ case 0x2: /* YCbCr 4:4:4 planar */ ++ /* 4 pixels per line */ ++ pr_debug("444 planar\n"); ++ xres = (fbi->fb.var.xres + 0x3) & (~0x3); ++ fbi->fb.fix.line_length = xres; ++ ++ nbytes = xres * yres; ++ ylen = cblen = crlen = nbytes; ++ break; ++ } ++ ++ /* 16-bytes alignment for DMA */ ++ aylen = (ylen + 0xf) & (~0xf); ++ acblen = (cblen + 0xf) & (~0xf); ++ acrlen = (crlen + 0xf) & (~0xf); ++ ++ fbi->fb.fix.smem_len = aylen + acblen + acrlen; ++ ++ /* alloc memory */ ++ ++ fbi->map_size = PAGE_ALIGN(fbi->fb.fix.smem_len + PAGE_SIZE); ++ fbi->map_cpu = (unsigned long)dma_alloc_writecombine(NULL, fbi->map_size, ++ &fbi->map_dma, GFP_KERNEL ); ++ ++ if (!fbi->map_cpu) return -ENOMEM; ++ ++ fbi->screen_cpu = fbi->map_cpu + PAGE_SIZE; ++ fbi->screen_dma = fbi->map_dma + PAGE_SIZE; ++ ++ fbi->fb.fix.smem_start = fbi->screen_dma; ++ ++ /* setup dma for Planar format */ ++ fbi->dma2 = (struct pxafb_dma_descriptor*) ++ (fbi->screen_cpu - sizeof(struct pxafb_dma_descriptor)); ++ fbi->dma3 = fbi->dma2 - 1; ++ fbi->dma4 = fbi->dma3 - 1; ++ ++ /* offset */ ++ yoff = 0; ++ cboff = aylen; ++ croff = cboff + acblen; ++ ++ /* Y vector */ ++ fbi->dma2->fdadr = (fbi->screen_dma - sizeof(struct pxafb_dma_descriptor)); ++ fbi->dma2->fsadr = fbi->screen_dma + yoff; ++ fbi->dma2->fidr = 0; ++ fbi->dma2->ldcmd = ylen; ++ ++ /* Cb vector */ ++ fbi->dma3->fdadr = (fbi->dma2->fdadr - sizeof(struct pxafb_dma_descriptor)); ++ fbi->dma3->fsadr = (fbi->screen_dma + cboff); ++ fbi->dma3->fidr = 0; ++ fbi->dma3->ldcmd = cblen; ++ ++ /* Cr vector */ ++ ++ fbi->dma4->fdadr = (fbi->dma3->fdadr - sizeof(struct pxafb_dma_descriptor)); ++ fbi->dma4->fsadr = (fbi->screen_dma + croff); ++ fbi->dma4->fidr = 0; ++ fbi->dma4->ldcmd = crlen; ++ ++ /* adjust for user */ ++ fbi->fb.var.red.length = ylen; ++ fbi->fb.var.red.offset = yoff; ++ fbi->fb.var.green.length = cblen; ++ fbi->fb.var.green.offset = cboff; ++ fbi->fb.var.blue.length = crlen; ++ fbi->fb.var.blue.offset = croff; ++ ++ return 0; ++}; ++ ++static int overlay2fb_map_RGB_memory( struct fb_info *info) ++{ ++ struct overlayfb_info *fbi = (struct overlayfb_info*) info; ++ struct fb_var_screeninfo *var = &fbi->fb.var; ++ int pixels_per_line=0 , nbytes=0; ++ ++ if (fbi->map_cpu) ++ dma_free_writecombine(NULL, fbi->map_size, (void*)fbi->map_cpu, fbi->map_dma); ++ ++ switch(var->bits_per_pixel) { ++ case 16: ++ /* 2 pixels per line */ ++ pixels_per_line = (fbi->fb.var.xres + 0x1) & (~0x1); ++ nbytes = 2; ++ ++ var->red = def_rgbt_16.red; ++ var->green = def_rgbt_16.green; ++ var->blue = def_rgbt_16.blue; ++ var->transp = def_rgbt_16.transp; ++ break; ++ ++ case 18: ++ /* 8 pixels per line */ ++ pixels_per_line = (fbi->fb.var.xres + 0x7 ) & (~0x7); ++ nbytes = 3; ++ ++ var->red = def_rgb_18.red; ++ var->green = def_rgb_18.green; ++ var->blue = def_rgb_18.blue; ++ var->transp = def_rgb_18.transp; ++ ++ break; ++ case 19: ++ /* 8 pixels per line */ ++ pixels_per_line = (fbi->fb.var.xres + 0x7 ) & (~0x7); ++ nbytes = 3; ++ ++ var->red = def_rgbt_19.red; ++ var->green = def_rgbt_19.green; ++ var->blue = def_rgbt_19.blue; ++ var->transp = def_rgbt_19.transp; ++ ++ break; ++ case 24: ++ pixels_per_line = fbi->fb.var.xres; ++ nbytes = 4; ++ ++ var->red = def_rgbt_24.red; ++ var->green = def_rgbt_24.green; ++ var->blue = def_rgbt_24.blue; ++ var->transp = def_rgbt_24.transp; ++ ++ break; ++ ++ case 25: ++ pixels_per_line = fbi->fb.var.xres; ++ nbytes = 4; ++ ++ var->red = def_rgbt_25.red; ++ var->green = def_rgbt_25.green; ++ var->blue = def_rgbt_25.blue; ++ var->transp = def_rgbt_25.transp; ++ ++ break; ++ } ++ ++ fbi->fb.fix.line_length = nbytes * pixels_per_line ; ++ fbi->fb.fix.smem_len = fbi->fb.fix.line_length * fbi->fb.var.yres ; ++ ++ fbi->map_size = PAGE_ALIGN(fbi->fb.fix.smem_len + PAGE_SIZE); ++ fbi->map_cpu = (unsigned long)dma_alloc_writecombine(NULL, fbi->map_size, ++ &fbi->map_dma, GFP_KERNEL ); ++ ++ if (!fbi->map_cpu) return -ENOMEM; ++ ++ fbi->screen_cpu = fbi->map_cpu + PAGE_SIZE; ++ fbi->screen_dma = fbi->map_dma + PAGE_SIZE; ++ ++ fbi->fb.fix.smem_start = fbi->screen_dma; ++ ++ /* setup dma descriptor */ ++ fbi->dma2 = (struct pxafb_dma_descriptor*) ++ (fbi->screen_cpu - sizeof(struct pxafb_dma_descriptor)); ++ ++ fbi->dma2->fdadr = (fbi->screen_dma - sizeof(struct pxafb_dma_descriptor)); ++ fbi->dma2->fsadr = fbi->screen_dma; ++ fbi->dma2->fidr = 0; ++ fbi->dma2->ldcmd = fbi->fb.fix.smem_len; ++ ++ return 0; ++} ++ ++static int overlay2fb_enable(struct fb_info *info) ++{ ++ struct overlayfb_info *fbi = (struct overlayfb_info*) info; ++ unsigned long bpp2; ++ unsigned int xres, yres; ++ ++ if (!fbi->map_cpu) return -EINVAL; ++ ++ switch(fbi->fb.var.bits_per_pixel) { ++ case 16: ++ bpp2 = 0x4; ++ break; ++ case 18: ++ bpp2 = 0x6; ++ break; ++ case 19: ++ bpp2 = 0x8; ++ break; ++ case 24: ++ bpp2 = 0x9; ++ break; ++ case 25: ++ bpp2 = 0xa; ++ break; ++ default: ++ return -EINVAL; ++ } ++ ++ /* disable branch/start/end of frame interrupt */ ++ LCCR5 |= (LCCR5_IUM4 | LCCR5_IUM3 | LCCR5_IUM2 | ++ LCCR5_BSM4 | LCCR5_BSM3 | LCCR5_BSM2 | ++ LCCR5_EOFM4 | LCCR5_EOFM3 | LCCR5_EOFM2 | ++ LCCR5_SOFM4 | LCCR5_SOFM3 | LCCR5_SOFM2); ++ ++ if (fbi->format == 0) { ++ /* overlay2 RGB resolution, RGB and YUV have different xres value*/ ++ xres = fbi->fb.var.xres; ++ yres = fbi->fb.var.yres; ++ ++ OVL2C2 = (fbi->format << 20) | (fbi->ypos << 10) | fbi->xpos; ++ OVL2C1 = OVL2C1_O2EN | (bpp2 << 20) | ((yres-1)<<10) | (xres-1); ++ /* setup RGB DMA */ ++ if (fbi->state == C_DISABLE || fbi->state == C_BLANK) ++ FDADR2 = fbi->dma2->fdadr; ++ else ++ FBR2 = fbi->dma2->fdadr | 0x1; ++ } else { ++ /* overlay2 YUV resolution */ ++ xres = fbi->fb.fix.line_length; ++ yres = fbi->fb.var.yres; ++ ++ OVL2C2 = (fbi->format << 20) | (fbi->ypos << 10) | fbi->xpos; ++ OVL2C1 = OVL2C1_O2EN | (bpp2 << 20) | ((yres-1)<<10) | (xres-1); ++ ++ if (fbi->state == C_DISABLE || fbi->state == C_BLANK) { ++ FDADR2 = fbi->dma2->fdadr; ++ FDADR3 = fbi->dma3->fdadr; ++ FDADR4 = fbi->dma4->fdadr; ++ } else { ++ FBR2 = fbi->dma2->fdadr | 0x01; ++ FBR3 = fbi->dma3->fdadr | 0x01; ++ FBR4 = fbi->dma4->fdadr | 0x01; ++ } ++ } ++ ++ fbi->state = C_ENABLE; ++ return 0; ++} ++ ++static int overlay2fb_disable(struct fb_info *info) ++{ ++ struct overlayfb_info *fbi = (struct overlayfb_info*)info; ++ int done; ++ ++ if (fbi->state == C_DISABLE) ++ return 0; ++ if (fbi->state == C_BLANK) { ++ fbi->state = C_DISABLE; ++ return 0; ++ } ++ ++ fbi->state = C_DISABLE; ++ ++ /* clear O2EN */ ++ OVL2C1 &= ~OVL2C1_O2EN; ++ ++ /* Make overlay2 can't disable/enable ++ * correctly sometimes. ++ */ ++ CLEAR_LCD_INTR(LCSR1, LCSR1_BS2); ++ ++ if (fbi->format == 0) ++ FBR2 = 0x3; ++ else { ++ FBR2 = 0x3; ++ FBR3 = 0x3; ++ FBR4 = 0x3; ++ } ++ ++ done = WAIT_FOR_LCD_INTR(LCSR1, LCSR1_BS2, 100); ++ ++ if (!done) { ++ pr_debug(KERN_INFO "%s: timeout\n", __FUNCTION__); ++ return -1; ++ } ++ return 0; ++} ++ ++static int overlay2fb_blank(int blank, struct fb_info *info) ++{ ++ struct overlayfb_info *fbi = (struct overlayfb_info*) info; ++ int err=0; ++ ++ switch(blank) ++ { ++ case 0: ++ err = overlay2fb_enable(info); ++ if (err) { ++ fbi->state = C_DISABLE; ++ pxafb_set_ctrlr_state(fbi->basefb, C_REENABLE); ++ } ++ break; ++ case 1: ++ err = overlay2fb_disable(info); ++ if (err) { ++ fbi->state = C_DISABLE; ++ pxafb_set_ctrlr_state(fbi->basefb, C_REENABLE); ++ } ++ break; ++ default: ++ /* reserved */ ++ break; ++ } ++ ++ return err; ++} ++ ++ ++static int overlay2fb_check_var( struct fb_var_screeninfo *var, struct fb_info *info) ++{ ++ int xpos, ypos, xres, yres; ++ int format; ++ struct overlayfb_info *fbi=(struct overlayfb_info*)info; ++ ++ xres=yres=0; ++ ++ xpos = (var->nonstd & 0x3ff); ++ ypos = (var->nonstd >> 10) & 0x3ff; ++ format = (var->nonstd >>20) & 0x7; ++ ++ ++ /* Palnar YCbCr444, YCbCr422, YCbCr420 */ ++ if ( (format != 0x4) && (format != 0x3) && (format != 0x2) && (format !=0x0)) ++ return -EINVAL; ++ ++ /* dummy pixels */ ++ switch(format) { ++ case 0x0: /* RGB */ ++ xres = var->xres; ++ break; ++ case 0x2: /* 444 */ ++ xres = (var->xres + 0x3) & ~(0x3); ++ break; ++ case 0x3: /* 422 */ ++ xres = (var->xres + 0x7) & ~(0x7); ++ break; ++ case 0x4: /* 420 */ ++ xres = (var->xres + 0xf) & ~(0xf); ++ break; ++ } ++ yres = var->yres; ++ ++ if ( (xpos + xres) > fbi->basefb->fb.var.xres ) ++ return -EINVAL; ++ ++ if ( (ypos + yres) > fbi->basefb->fb.var.yres ) ++ return -EINVAL; ++ ++ fbi->old_var=*var; ++ ++ var->activate=FB_ACTIVATE_NOW; ++ ++ return 0; ++ ++} ++ ++ ++/* ++ * overlay2fb_set_var() ++ * ++ * var.nonstd is used as YCbCr format. ++ * var.red/green/blue is used as (Y/Cb/Cr) vector ++ */ ++ ++static int overlay2fb_set_par(struct fb_info *info) ++{ ++ unsigned int xpos, ypos; ++ int format, err; ++ ++ struct overlayfb_info *fbi=(struct overlayfb_info*)info; ++ struct fb_var_screeninfo *var = &fbi->fb.var; ++ ++ info->flags &= ~FBINFO_MISC_USEREVENT; ++ ++ if (fbi->state == C_BLANK) ++ return 0; ++ ++ if (fbi->state == C_DISABLE) ++ goto out1; ++ ++ if ( (var->xres == fbi->old_var.xres) && ++ (var->yres == fbi->old_var.yres) && ++ (var->bits_per_pixel == fbi->old_var.bits_per_pixel) && ++ (((var->nonstd>>20) & 0x7) == fbi->format) ) ++ goto out2; ++ ++out1: ++ xpos = var->nonstd & 0x3ff; ++ ypos = (var->nonstd>>10) & 0x3ff; ++ format = (var->nonstd>>20) & 0x7; ++ ++ ++ fbi->format = format; ++ if ( fbi->format==0 ) ++ err = overlay2fb_map_RGB_memory(info); ++ else ++ err = overlay2fb_map_YUV_memory(info); ++ ++ if (err) return err; ++ ++out2: ++ /* position */ ++ fbi->xpos = var->nonstd & 0x3ff; ++ fbi->ypos = (var->nonstd>>10) & 0x3ff; ++ ++ overlay2fb_enable(info); ++ ++ return 0; ++} ++ ++static struct fb_ops overlay2fb_ops = { ++ .owner = THIS_MODULE, ++ .fb_open = overlay2fb_open, ++ .fb_release = overlay2fb_release, ++ .fb_check_var = overlay2fb_check_var, ++ .fb_set_par = overlay2fb_set_par, ++ .fb_blank = overlay2fb_blank, ++ .fb_fillrect = cfb_fillrect, ++ .fb_copyarea = cfb_copyarea, ++ .fb_imageblit = cfb_imageblit, ++}; ++ ++/* Hardware cursor */ ++ ++/* Bulverde Cursor Modes */ ++struct cursorfb_mode{ ++ int xres; ++ int yres; ++ int bpp; ++}; ++ ++static struct cursorfb_mode cursorfb_modes[]={ ++ { 32, 32, 2}, ++ { 32, 32, 2}, ++ { 32, 32, 2}, ++ { 64, 64, 2}, ++ { 64, 64, 2}, ++ { 64, 64, 2}, ++ {128, 128, 1}, ++ {128, 128, 1} ++}; ++ ++static int cursorfb_enable(struct fb_info *info) ++{ ++ struct overlayfb_info *fbi = (struct overlayfb_info*) info; ++ ++ if (!fbi->map_cpu) return -EINVAL; ++ ++ CCR &= ~CCR_CEN; ++ ++ /* set palette format ++ * ++ * FIXME: if only cursor uses palette ++ */ ++ LCCR4 = (LCCR4 & (~(0x3<<15))) | (0x1<<15); ++ ++ /* disable branch/start/end of frame interrupt */ ++ LCCR5 |= (LCCR5_IUM5 | LCCR5_BSM5 | LCCR5_EOFM5 | LCCR5_SOFM5); ++ ++ /* load palette and frame data */ ++ if (fbi->state == C_DISABLE) { ++ FDADR5 = fbi->dma5_pal->fdadr; ++ udelay(1); ++ FDADR5 = fbi->dma5_frame->fdadr; ++ udelay(1); ++ ++ } ++ else { ++ FBR5 = fbi->dma5_pal->fdadr | 0x1; ++ udelay(1); ++ FBR5 = fbi->dma5_frame->fdadr | 0x1; ++ udelay(1); ++ } ++ ++ CCR = CCR_CEN | (fbi->ypos << 15) | (fbi->xpos << 5) | (fbi->format); ++ ++ fbi->state = C_ENABLE; ++ ++ return 0; ++} ++ ++static int cursorfb_disable(struct fb_info *info) ++{ ++ struct overlayfb_info *fbi = (struct overlayfb_info*)info; ++ int done, ret = 0; ++ ++ fbi->state = C_DISABLE; ++ ++ done = WAIT_FOR_LCD_INTR(LCSR1, LCSR1_BS5, 100); ++ if (!done) ret = -1; ++ ++ CCR &= ~CCR_CEN; ++ ++ return ret; ++} ++ ++static int cursorfb_setcolreg(u_int regno, u_int red, u_int green, u_int blue, ++ u_int trans, struct fb_info *info) ++{ ++ struct overlayfb_info *fbi = (struct overlayfb_info *)info; ++ u_int val, ret = 1; ++ u_int *pal=(u_int*) fbi->palette_cpu; ++ ++ /* 25bit with Transparcy for 16bpp format */ ++ if (regno < fbi->palette_size) { ++ val = ((trans << 24) & 0x1000000); ++ val |= ((red << 16) & 0x0ff0000); ++ val |= ((green << 8 ) & 0x000ff00); ++ val |= ((blue << 0) & 0x00000ff); ++ ++ pal[regno] = val; ++ ret = 0; ++ } ++ return ret; ++} ++ ++int cursorfb_blank(int blank, struct fb_info *info) ++{ ++ switch(blank) ++ { ++ case 0: ++ cursorfb_enable(info); ++ break; ++ case 1: ++ cursorfb_disable(info); ++ break; ++ default: ++ /* reserved */ ++ break; ++ } ++ return 0; ++} ++ ++static int cursorfb_check_var( struct fb_var_screeninfo *var, struct fb_info *info) ++{ ++ int xpos, ypos, xres, yres; ++ int mode; ++ struct cursorfb_mode *cursor; ++ struct overlayfb_info *fbi=(struct overlayfb_info*)info; ++ ++ mode = var->nonstd & 0x7; ++ xpos = (var->nonstd>>5) & 0x3ff; ++ ypos = (var->nonstd>>15) & 0x3ff; ++ ++ if (mode>7 || mode <0 ) ++ return -EINVAL; ++ ++ cursor = cursorfb_modes + mode; ++ ++ xres = cursor->xres; ++ yres = cursor->yres; ++ ++ if ( (xpos + xres) > fbi->basefb->fb.var.xres ) ++ return -EINVAL; ++ ++ if ( (ypos + yres) > fbi->basefb->fb.var.yres ) ++ return -EINVAL; ++ ++ return 0; ++ ++} ++ ++static int cursorfb_set_par(struct fb_info *info) ++{ ++ struct overlayfb_info *fbi = (struct overlayfb_info*) info; ++ struct fb_var_screeninfo *var = &fbi->fb.var; ++ struct cursorfb_mode *cursor; ++ int mode, xpos, ypos; ++ int err; ++ ++ info->flags &= ~FBINFO_MISC_USEREVENT; ++ ++ mode = var->nonstd & 0x7; ++ xpos = (var->nonstd>>5) & 0x3ff; ++ ypos = (var->nonstd>>15) & 0x3ff; ++ ++ if (mode != fbi->format) { ++ cursor = cursorfb_modes + mode; ++ ++ /* update "var" info */ ++ fbi->fb.var.xres = cursor->xres; ++ fbi->fb.var.yres = cursor->yres; ++ fbi->fb.var.bits_per_pixel = cursor->bpp; ++ ++ /* alloc video memory ++ * ++ * 4k is engouh for 128x128x1 cursor, ++ * - 2k for cursor pixels, ++ * - 2k for palette data, plus 2 dma descriptor ++ */ ++ if (!fbi->map_cpu) { ++ fbi->map_size = PAGE_SIZE; ++ fbi->map_cpu = (unsigned long)dma_alloc_writecombine(NULL, fbi->map_size, ++ &fbi->map_dma, GFP_KERNEL ); ++ if (!fbi->map_cpu) return -ENOMEM; ++ } ++ ++ cursor = cursorfb_modes + mode; ++ ++ /* update overlay & fix "info" */ ++ fbi->screen_cpu = fbi->map_cpu; ++ fbi->palette_cpu = fbi->map_cpu + (PAGE_SIZE/2); ++ fbi->screen_dma = fbi->map_dma; ++ fbi->palette_dma = fbi->map_dma + (PAGE_SIZE/2); ++ ++ fbi->format = mode; ++ fbi->palette_size = (1<<cursor->bpp) ; ++ fbi->fb.fix.smem_start = fbi->screen_dma; ++ fbi->fb.fix.smem_len = cursor->xres * cursor->yres * cursor->bpp / 8; ++ fbi->fb.fix.line_length = cursor->xres * cursor->bpp / 8 ; ++ ++ fbi->dma5_pal = (struct pxafb_dma_descriptor*)(fbi->map_cpu + PAGE_SIZE - 16 ); ++ fbi->dma5_pal->fdadr = (fbi->map_dma + PAGE_SIZE - 16); ++ fbi->dma5_pal->fsadr = fbi->palette_dma; ++ fbi->dma5_pal->fidr = 0; ++ fbi->dma5_pal->ldcmd = (fbi->palette_size<<2) | LDCMD_PAL; ++ ++ fbi->dma5_frame = (struct pxafb_dma_descriptor*)(fbi->map_cpu + PAGE_SIZE - 32 ); ++ fbi->dma5_frame->fdadr = (fbi->map_dma + PAGE_SIZE - 32); ++ fbi->dma5_frame->fsadr = fbi->screen_dma; ++ fbi->dma5_frame->fidr = 0; ++ fbi->dma5_frame->ldcmd = fbi->fb.fix.smem_len; ++ ++ /* alloc & set default cmap */ ++ err = fb_alloc_cmap(&fbi->fb.cmap, fbi->palette_size, 0); ++ if (err) return err; ++ err = fb_set_cmap(&fbi->fb.cmap, info); ++ if (err) return err; ++ } ++ ++ /* update overlay info */ ++ if( (xpos != fbi->xpos) || (ypos != fbi->ypos) ) { ++ fbi->xpos = xpos; ++ fbi->ypos = ypos; ++ } ++ ++ cursorfb_enable(info); ++ pxafb_set_ctrlr_state(fbi->basefb, C_REENABLE); ++ ++ return 0; ++} ++ ++static struct fb_ops cursorfb_ops = { ++ .owner = THIS_MODULE, ++ .fb_check_var = cursorfb_check_var, ++ .fb_set_par = cursorfb_set_par, ++ .fb_blank = cursorfb_blank, ++ .fb_fillrect = cfb_fillrect, ++ .fb_copyarea = cfb_copyarea, ++ .fb_imageblit = cfb_imageblit, ++ .fb_setcolreg = cursorfb_setcolreg, ++}; ++ ++static struct overlayfb_info * __init overlay1fb_init_fbinfo(void) ++{ ++ struct overlayfb_info *fbi; ++ ++ fbi = kmalloc(sizeof(struct overlayfb_info) + sizeof(u16) * 16, GFP_KERNEL); ++ if (!fbi) ++ return NULL; ++ ++ memset(fbi, 0, sizeof(struct overlayfb_info) ); ++ ++ fbi->refcount = 0; ++ init_MUTEX(&fbi->mutex); ++ ++ strcpy(fbi->fb.fix.id, "overlay1"); ++ ++ fbi->fb.fix.type = FB_TYPE_PACKED_PIXELS; ++ fbi->fb.fix.type_aux = 0; ++ fbi->fb.fix.xpanstep = 0; ++ fbi->fb.fix.ypanstep = 0; ++ fbi->fb.fix.ywrapstep = 0; ++ fbi->fb.fix.accel = FB_ACCEL_NONE; ++ ++ fbi->fb.var.nonstd = 0; ++ fbi->fb.var.activate = FB_ACTIVATE_NOW; ++ fbi->fb.var.height = -1; ++ fbi->fb.var.width = -1; ++ fbi->fb.var.accel_flags = 0; ++ fbi->fb.var.vmode = FB_VMODE_NONINTERLACED; ++ ++ ++ fbi->fb.fbops = &overlay1fb_ops; ++ fbi->fb.flags = FBINFO_FLAG_DEFAULT; ++ fbi->fb.node = -1; ++ fbi->fb.pseudo_palette = NULL; ++ ++ fbi->xpos = 0; ++ fbi->ypos = 0; ++ fbi->format = -1; ++ fbi->state = C_DISABLE; ++ ++ return fbi; ++} ++ ++static struct overlayfb_info * __init overlay2fb_init_fbinfo(void) ++{ ++ struct overlayfb_info *fbi; ++ ++ fbi = kmalloc(sizeof(struct overlayfb_info) + sizeof(u16) * 16, GFP_KERNEL); ++ if (!fbi) ++ return NULL; ++ ++ memset(fbi, 0, sizeof(struct overlayfb_info) ); ++ ++ fbi->refcount = 0; ++ init_MUTEX(&fbi->mutex); ++ ++ strcpy(fbi->fb.fix.id, "overlay2"); ++ ++ fbi->fb.fix.type = FB_TYPE_PACKED_PIXELS; ++ fbi->fb.fix.type_aux = 0; ++ fbi->fb.fix.xpanstep = 0; ++ fbi->fb.fix.ypanstep = 0; ++ fbi->fb.fix.ywrapstep = 0; ++ fbi->fb.fix.accel = FB_ACCEL_NONE; ++ ++ fbi->fb.var.nonstd = 0; ++ fbi->fb.var.activate = FB_ACTIVATE_NOW; ++ fbi->fb.var.height = -1; ++ fbi->fb.var.width = -1; ++ fbi->fb.var.accel_flags = 0; ++ fbi->fb.var.vmode = FB_VMODE_NONINTERLACED; ++ ++ fbi->fb.fbops = &overlay2fb_ops; ++ fbi->fb.flags = FBINFO_FLAG_DEFAULT; ++ fbi->fb.node = -1; ++ fbi->fb.pseudo_palette = NULL; ++ ++ fbi->xpos = 0; ++ fbi->ypos = 0; ++ fbi->format = -1; ++ fbi->state = C_DISABLE; ++ ++ return fbi; ++} ++ ++static struct overlayfb_info * __init cursorfb_init_fbinfo(void) ++{ ++ struct overlayfb_info *fbi; ++ ++ fbi = kmalloc(sizeof(struct overlayfb_info) + sizeof(u16) * 16, GFP_KERNEL); ++ if (!fbi) ++ return NULL; ++ ++ memset(fbi, 0, sizeof(struct overlayfb_info) ); ++ ++ fbi->refcount = 0; ++ init_MUTEX(&fbi->mutex); ++ ++ strcpy(fbi->fb.fix.id, "cursor"); ++ ++ fbi->fb.fix.type = FB_TYPE_PACKED_PIXELS; ++ fbi->fb.fix.type_aux = 0; ++ fbi->fb.fix.xpanstep = 0; ++ fbi->fb.fix.ypanstep = 0; ++ fbi->fb.fix.ywrapstep = 0; ++ fbi->fb.fix.accel = FB_ACCEL_NONE; ++ ++ fbi->fb.var.nonstd = 0; ++ fbi->fb.var.activate = FB_ACTIVATE_NOW; ++ fbi->fb.var.height = -1; ++ fbi->fb.var.width = -1; ++ fbi->fb.var.accel_flags = 0; ++ fbi->fb.var.vmode = FB_VMODE_NONINTERLACED; ++ ++ fbi->fb.fbops = &cursorfb_ops; ++ fbi->fb.flags = FBINFO_FLAG_DEFAULT; ++ fbi->fb.node = -1; ++ fbi->fb.pseudo_palette = NULL; ++ ++ ++ fbi->xpos = 0; ++ fbi->ypos = 0; ++ fbi->format = -1; ++ fbi->state = C_DISABLE; ++ ++ return fbi; ++} ++ ++ ++void pxa_set_overlay_ctrlr_state(struct pxafb_info *fbi, u_int state) ++{ ++ switch (state) { ++ case C_DISABLE: ++ DISABLE_OVERLAYS(fbi); ++ break; ++ case C_ENABLE: ++ ENABLE_OVERLAYS(fbi); ++ break; ++ case C_BLANK: ++ BLANK_OVERLAYS(fbi); ++ break; ++ case C_UNBLANK: ++ UNBLANK_OVERLAYS(fbi); ++ break; ++ default: ++ break; ++ } ++} ++ ++struct callback_data_t{ ++ char *name; ++ struct device *dev; ++}; ++ ++ ++static int find_dev(struct device *dev, void *callback_data) ++{ ++ int found=0; ++ struct callback_data_t * data=(struct callback_data_t *) callback_data; ++ ++ found = (strncmp(dev->kobj.name, data->name, KOBJ_NAME_LEN) == 0); ++ if(found == 1){ ++ data->dev = dev; ++ } ++ ++ return found; ++} ++struct device* find_bus_device(struct bus_type *bus, char *name) ++{ ++ struct callback_data_t callback_data; ++ ++ callback_data.name = name; ++ callback_data.dev = NULL; ++ bus_for_each_dev(bus, NULL, &callback_data, find_dev); ++ ++ return callback_data.dev; ++} ++ ++static int __devinit pxafb_overlay_init(void) ++{ ++ int ret; ++ struct overlayfb_info *overlay1fb, *overlay2fb, *cursorfb; ++ struct pxafb_info *fbi; ++ struct device *dev; ++ ++ ret = -1; ++ overlay1fb = overlay2fb = cursorfb = NULL; ++ fbi=NULL; ++ ++ dev=find_bus_device(&platform_bus_type, "pxa2xx-fb"); ++ if(dev ==NULL){ ++ printk(KERN_INFO "Base framebuffer not exists, failed to load overlay driver!\n"); ++ return ret; ++ } ++ ++ fbi = dev_get_drvdata(dev); ++ if(fbi ==NULL ){ ++ printk(KERN_INFO "Base framebuffer not initialized, failed to load overlay driver!\n"); ++ return ret; ++ } ++ ++ ++ /* Overlay 1 windows */ ++ overlay1fb = overlay1fb_init_fbinfo(); ++ ++ if(!overlay1fb) { ++ ret = -ENOMEM; ++ printk("overlay1fb_init_fbinfo failed\n"); ++ goto failed; ++ } ++ ++ ++ ret = register_framebuffer(&overlay1fb->fb); ++ if (ret<0) goto failed; ++ ++ /* Overlay 2 window */ ++ overlay2fb = overlay2fb_init_fbinfo(); ++ ++ if(!overlay2fb) { ++ ret = -ENOMEM; ++ printk("overlay2fb_init_fbinfo failed\n"); ++ goto failed; ++ } ++ ++ ret = register_framebuffer(&overlay2fb->fb); ++ if (ret<0) goto failed; ++ ++ /* Hardware cursor window */ ++ cursorfb = cursorfb_init_fbinfo(); ++ ++ if(!cursorfb) { ++ ret = -ENOMEM; ++ printk("cursorfb_init_fbinfo failed\n"); ++ goto failed; ++ } ++ ++ ret = register_framebuffer(&cursorfb->fb); ++ if (ret<0) goto failed; ++ ++ ++ /* set refernce to Overlays */ ++ fbi->overlay1fb = overlay1fb; ++ fbi->overlay2fb = overlay2fb; ++ fbi->cursorfb = cursorfb; ++ fbi->set_overlay_ctrlr_state=pxa_set_overlay_ctrlr_state; ++ ++ /* set refernce to BaseFrame */ ++ overlay1fb->basefb = fbi; ++ overlay2fb->basefb = fbi; ++ cursorfb->basefb = fbi; ++ ++ printk(KERN_INFO "Load PXA Overlay driver successfully!\n"); ++ ++ return 0; ++ ++failed: ++ if (overlay1fb) ++ kfree(overlay1fb); ++ if (overlay2fb) ++ kfree(overlay2fb); ++ if (cursorfb) ++ kfree(cursorfb); ++ printk(KERN_INFO "Load PXA Overlay driver failed!\n"); ++ return ret; ++} ++ ++static void __exit pxafb_overlay_exit(void) ++{ ++ struct pxafb_info *fbi; ++ struct device *dev; ++ ++ dev=find_bus_device(&platform_bus_type, "pxa2xx-fb"); ++ if(dev ==NULL){ ++ return ; ++ } ++ ++ fbi = dev_get_drvdata(dev); ++ if(fbi ==NULL ){ ++ return ; ++ } ++ ++ if (fbi->overlay1fb) { ++ unregister_framebuffer(&(fbi->overlay1fb->fb)); ++ kfree(fbi->overlay1fb); ++ fbi->overlay1fb = NULL; ++ } ++ ++ if (fbi->overlay2fb) { ++ unregister_framebuffer(&(fbi->overlay2fb->fb)); ++ kfree(fbi->overlay2fb); ++ fbi->overlay2fb = NULL; ++ } ++ ++ if (fbi->cursorfb) { ++ unregister_framebuffer(&(fbi->cursorfb->fb)); ++ kfree(fbi->cursorfb); ++ fbi->cursorfb = NULL; ++ } ++ ++ fbi->set_overlay_ctrlr_state = NULL; ++ ++ printk(KERN_INFO "Unload PXA Overlay driver successfully!\n"); ++ return ; ++} ++ ++ ++module_init(pxafb_overlay_init); ++module_exit(pxafb_overlay_exit); ++ ++MODULE_DESCRIPTION("Loadable framebuffer overlay driver for PXA"); ++MODULE_LICENSE("GPL"); ++ +Index: linux-2.6.19/include/asm-arm/arch-pxa/pxa-regs.h +=================================================================== +--- linux-2.6.19.orig/include/asm-arm/arch-pxa/pxa-regs.h 2006-12-16 18:35:40.000000000 +0000 ++++ linux-2.6.19/include/asm-arm/arch-pxa/pxa-regs.h 2006-12-16 18:36:17.000000000 +0000 +@@ -100,6 +100,7 @@ + #define DCSR_CLRCMPST (1 << 24) /* Clear Descriptor Compare Status */ + #define DCSR_CMPST (1 << 10) /* The Descriptor Compare Status */ + #define DCSR_ENRINTR (1 << 9) /* The end of Receive */ ++#define DCSR_EORINTR (1 << 9) /* The end of Receive */ + #endif + #define DCSR_REQPEND (1 << 8) /* Request Pending (read-only) */ + #define DCSR_STOPSTATE (1 << 3) /* Stop State (read-only) */ +@@ -794,11 +795,18 @@ + #define UDC_INT_PACKETCMP (0x1) + + #define UDCICR_INT(n,intr) (((intr) & 0x03) << (((n) & 0x0F) * 2)) ++/* Older defines, do not use. */ + #define UDCICR1_IECC (1 << 31) /* IntEn - Configuration Change */ + #define UDCICR1_IESOF (1 << 30) /* IntEn - Start of Frame */ + #define UDCICR1_IERU (1 << 29) /* IntEn - Resume */ + #define UDCICR1_IESU (1 << 28) /* IntEn - Suspend */ + #define UDCICR1_IERS (1 << 27) /* IntEn - Reset */ ++/* New defines. */ ++#define UDCISR1_IRCC (1 << 31) /* IntEn - Configuration Change */ ++#define UDCISR1_IRSOF (1 << 30) /* IntEn - Start of Frame */ ++#define UDCISR1_IRRU (1 << 29) /* IntEn - Resume */ ++#define UDCISR1_IRSU (1 << 28) /* IntEn - Suspend */ ++#define UDCISR1_IRRS (1 << 27) /* IntEn - Reset */ + + #define UDCISR0 __REG(0x4060000C) /* UDC Interrupt Status Register 0 */ + #define UDCISR1 __REG(0x40600010) /* UDC Interrupt Status Register 1 */ +@@ -1848,6 +1856,8 @@ + #define DFBR0 __REG(0x44000020) /* DMA Channel 0 Frame Branch Register */ + #define DFBR1 __REG(0x44000024) /* DMA Channel 1 Frame Branch Register */ + #define LCSR __REG(0x44000038) /* LCD Controller Status Register */ ++#define LCSR0 __REG(0x44000038) /* LCD Controller Status Register */ ++#define LCSR1 __REG(0x44000034) /* LCD Controller Status Register */ + #define LIIDR __REG(0x4400003C) /* LCD Controller Interrupt ID Register */ + #define TMEDRGBR __REG(0x44000040) /* TMED RGB Seed Register */ + #define TMEDCR __REG(0x44000044) /* TMED Control Register */ +@@ -1857,6 +1867,10 @@ + #define LCCR3_4BPP (2 << 24) + #define LCCR3_8BPP (3 << 24) + #define LCCR3_16BPP (4 << 24) ++#define LCCR3_18BPP (6 << 24) ++#define LCCR3_19BPP (8 << 24) ++#define LCCR3_24BPP (9 << 24) ++#define LCCR3_25BPP (10<< 24) + + #define FDADR0 __REG(0x44000200) /* DMA Channel 0 Frame Descriptor Address Register */ + #define FSADR0 __REG(0x44000204) /* DMA Channel 0 Frame Source Address Register */ +@@ -2021,6 +2035,104 @@ + + #define LDCMD_PAL (1 << 26) /* instructs DMA to load palette buffer */ + ++/* Overlay1 & Overlay2 & Hardware Cursor */ ++#define LCSR1_SOF1 (1 << 0) ++#define LCSR1_SOF2 (1 << 1) ++#define LCSR1_SOF3 (1 << 2) ++#define LCSR1_SOF4 (1 << 3) ++#define LCSR1_SOF5 (1 << 4) ++#define LCSR1_SOF6 (1 << 5) ++ ++#define LCSR1_EOF1 (1 << 8) ++#define LCSR1_EOF2 (1 << 9) ++#define LCSR1_EOF3 (1 << 10) ++#define LCSR1_EOF4 (1 << 11) ++#define LCSR1_EOF5 (1 << 12) ++#define LCSR1_EOF6 (1 << 13) ++ ++#define LCSR1_BS1 (1 << 16) ++#define LCSR1_BS2 (1 << 17) ++#define LCSR1_BS3 (1 << 18) ++#define LCSR1_BS4 (1 << 19) ++#define LCSR1_BS5 (1 << 20) ++#define LCSR1_BS6 (1 << 21) ++ ++#define LCSR1_IU2 (1 << 25) ++#define LCSR1_IU3 (1 << 26) ++#define LCSR1_IU4 (1 << 27) ++#define LCSR1_IU5 (1 << 28) ++#define LCSR1_IU6 (1 << 29) ++ ++#define LDCMD_SOFINT (1 << 22) ++#define LDCMD_EOFINT (1 << 21) ++ ++ ++#define LCCR5_SOFM1 (1<<0) /* Start Of Frame Mask for Overlay 1 (channel 1) */ ++#define LCCR5_SOFM2 (1<<1) /* Start Of Frame Mask for Overlay 2 (channel 2) */ ++#define LCCR5_SOFM3 (1<<2) /* Start Of Frame Mask for Overlay 2 (channel 3) */ ++#define LCCR5_SOFM4 (1<<3) /* Start Of Frame Mask for Overlay 2 (channel 4) */ ++#define LCCR5_SOFM5 (1<<4) /* Start Of Frame Mask for cursor (channel 5) */ ++#define LCCR5_SOFM6 (1<<5) /* Start Of Frame Mask for command data (channel 6) */ ++ ++#define LCCR5_EOFM1 (1<<8) /* End Of Frame Mask for Overlay 1 (channel 1) */ ++#define LCCR5_EOFM2 (1<<9) /* End Of Frame Mask for Overlay 2 (channel 2) */ ++#define LCCR5_EOFM3 (1<<10) /* End Of Frame Mask for Overlay 2 (channel 3) */ ++#define LCCR5_EOFM4 (1<<11) /* End Of Frame Mask for Overlay 2 (channel 4) */ ++#define LCCR5_EOFM5 (1<<12) /* End Of Frame Mask for cursor (channel 5) */ ++#define LCCR5_EOFM6 (1<<13) /* End Of Frame Mask for command data (channel 6) */ ++ ++#define LCCR5_BSM1 (1<<16) /* Branch mask for Overlay 1 (channel 1) */ ++#define LCCR5_BSM2 (1<<17) /* Branch mask for Overlay 2 (channel 2) */ ++#define LCCR5_BSM3 (1<<18) /* Branch mask for Overlay 2 (channel 3) */ ++#define LCCR5_BSM4 (1<<19) /* Branch mask for Overlay 2 (channel 4) */ ++#define LCCR5_BSM5 (1<<20) /* Branch mask for cursor (channel 5) */ ++#define LCCR5_BSM6 (1<<21) /* Branch mask for data command (channel 6) */ ++ ++#define LCCR5_IUM1 (1<<24) /* Input FIFO Underrun Mask for Overlay 1 */ ++#define LCCR5_IUM2 (1<<25) /* Input FIFO Underrun Mask for Overlay 2 */ ++#define LCCR5_IUM3 (1<<26) /* Input FIFO Underrun Mask for Overlay 2 */ ++#define LCCR5_IUM4 (1<<27) /* Input FIFO Underrun Mask for Overlay 2 */ ++#define LCCR5_IUM5 (1<<28) /* Input FIFO Underrun Mask for cursor */ ++#define LCCR5_IUM6 (1<<29) /* Input FIFO Underrun Mask for data command */ ++ ++#define OVL1C1_O1EN (1<<31) /* Enable bit for Overlay 1 */ ++#define OVL2C1_O2EN (1<<31) /* Enable bit for Overlay 2 */ ++#define CCR_CEN (1<<31) /* Enable bit for Cursor */ ++ ++/* LCD registers */ ++#define LCCR4 __REG(0x44000010) /* LCD Controller Control Register 4 */ ++#define LCCR5 __REG(0x44000014) /* LCD Controller Control Register 5 */ ++#define FBR0 __REG(0x44000020) /* DMA Channel 0 Frame Branch Register */ ++#define FBR1 __REG(0x44000024) /* DMA Channel 1 Frame Branch Register */ ++#define FBR2 __REG(0x44000028) /* DMA Channel 2 Frame Branch Register */ ++#define FBR3 __REG(0x4400002C) /* DMA Channel 3 Frame Branch Register */ ++#define FBR4 __REG(0x44000030) /* DMA Channel 4 Frame Branch Register */ ++#define FDADR2 __REG(0x44000220) /* DMA Channel 2 Frame Descriptor Address Register */ ++#define FSADR2 __REG(0x44000224) /* DMA Channel 2 Frame Source Address Register */ ++#define FIDR2 __REG(0x44000228) /* DMA Channel 2 Frame ID Register */ ++#define LDCMD2 __REG(0x4400022C) /* DMA Channel 2 Command Register */ ++#define FDADR3 __REG(0x44000230) /* DMA Channel 3 Frame Descriptor Address Register */ ++#define FSADR3 __REG(0x44000234) /* DMA Channel 3 Frame Source Address Register */ ++#define FIDR3 __REG(0x44000238) /* DMA Channel 3 Frame ID Register */ ++#define LDCMD3 __REG(0x4400023C) /* DMA Channel 3 Command Register */ ++#define FDADR4 __REG(0x44000240) /* DMA Channel 4 Frame Descriptor Address Register */ ++#define FSADR4 __REG(0x44000244) /* DMA Channel 4 Frame Source Address Register */ ++#define FIDR4 __REG(0x44000248) /* DMA Channel 4 Frame ID Register */ ++#define LDCMD4 __REG(0x4400024C) /* DMA Channel 4 Command Register */ ++#define FDADR5 __REG(0x44000250) /* DMA Channel 5 Frame Descriptor Address Register */ ++#define FSADR5 __REG(0x44000254) /* DMA Channel 5 Frame Source Address Register */ ++#define FIDR5 __REG(0x44000258) /* DMA Channel 5 Frame ID Register */ ++#define LDCMD5 __REG(0x4400025C) /* DMA Channel 5 Command Register */ ++ ++#define OVL1C1 __REG(0x44000050) /* Overlay 1 Control Register 1 */ ++#define OVL1C2 __REG(0x44000060) /* Overlay 1 Control Register 2 */ ++#define OVL2C1 __REG(0x44000070) /* Overlay 2 Control Register 1 */ ++#define OVL2C2 __REG(0x44000080) /* Overlay 2 Control Register 2 */ ++#define CCR __REG(0x44000090) /* Cursor Control Register */ ++ ++#define FBR5 __REG(0x44000110) /* DMA Channel 5 Frame Branch Register */ ++#define FBR6 __REG(0x44000114) /* DMA Channel 6 Frame Branch Register */ ++ + /* + * Memory controller + */ diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/pxa_irda_susres_fix-r0.patch b/packages/linux/logicpd-pxa270-2.6.19.2/pxa_irda_susres_fix-r0.patch new file mode 100644 index 0000000000..9f4885a85c --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/pxa_irda_susres_fix-r0.patch @@ -0,0 +1,93 @@ +--- + drivers/net/irda/pxaficp_ir.c | 28 ++++++++++++++-------------- + 1 file changed, 14 insertions(+), 14 deletions(-) + +Index: linux-2.6.19/drivers/net/irda/pxaficp_ir.c +=================================================================== +--- linux-2.6.19.orig/drivers/net/irda/pxaficp_ir.c 2006-11-29 21:57:37.000000000 +0000 ++++ linux-2.6.19/drivers/net/irda/pxaficp_ir.c 2006-12-16 22:07:27.000000000 +0000 +@@ -704,9 +704,9 @@ static int pxa_irda_stop(struct net_devi + return 0; + } + +-static int pxa_irda_suspend(struct device *_dev, pm_message_t state) ++static int pxa_irda_suspend(struct platform_device *_dev, pm_message_t state) + { +- struct net_device *dev = dev_get_drvdata(_dev); ++ struct net_device *dev = platform_get_drvdata(_dev); + struct pxa_irda *si; + + if (dev && netif_running(dev)) { +@@ -718,9 +718,9 @@ static int pxa_irda_suspend(struct devic + return 0; + } + +-static int pxa_irda_resume(struct device *_dev) ++static int pxa_irda_resume(struct platform_device *_dev) + { +- struct net_device *dev = dev_get_drvdata(_dev); ++ struct net_device *dev = platform_get_drvdata(_dev); + struct pxa_irda *si; + + if (dev && netif_running(dev)) { +@@ -746,9 +746,8 @@ static int pxa_irda_init_iobuf(iobuff_t + return io->head ? 0 : -ENOMEM; + } + +-static int pxa_irda_probe(struct device *_dev) ++static int pxa_irda_probe(struct platform_device *pdev) + { +- struct platform_device *pdev = to_platform_device(_dev); + struct net_device *dev; + struct pxa_irda *si; + unsigned int baudrate_mask; +@@ -805,7 +804,7 @@ static int pxa_irda_probe(struct device + err = register_netdev(dev); + + if (err == 0) +- dev_set_drvdata(&pdev->dev, dev); ++ platform_set_drvdata(pdev, dev); + + if (err) { + kfree(si->tx_buff.head); +@@ -822,9 +821,9 @@ err_mem_1: + return err; + } + +-static int pxa_irda_remove(struct device *_dev) ++static int pxa_irda_remove(struct platform_device *_dev) + { +- struct net_device *dev = dev_get_drvdata(_dev); ++ struct net_device *dev = platform_get_drvdata(_dev); + + if (dev) { + struct pxa_irda *si = netdev_priv(dev); +@@ -840,9 +839,10 @@ static int pxa_irda_remove(struct device + return 0; + } + +-static struct device_driver pxa_ir_driver = { +- .name = "pxa2xx-ir", +- .bus = &platform_bus_type, ++static struct platform_driver pxa_ir_driver = { ++ .driver = { ++ .name = "pxa2xx-ir", ++ }, + .probe = pxa_irda_probe, + .remove = pxa_irda_remove, + .suspend = pxa_irda_suspend, +@@ -851,12 +851,12 @@ static struct device_driver pxa_ir_drive + + static int __init pxa_irda_init(void) + { +- return driver_register(&pxa_ir_driver); ++ return platform_driver_register(&pxa_ir_driver); + } + + static void __exit pxa_irda_exit(void) + { +- driver_unregister(&pxa_ir_driver); ++ platform_driver_unregister(&pxa_ir_driver); + } + + module_init(pxa_irda_init); diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/pxa_keys-r5.patch b/packages/linux/logicpd-pxa270-2.6.19.2/pxa_keys-r5.patch new file mode 100644 index 0000000000..d263786f26 --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/pxa_keys-r5.patch @@ -0,0 +1,211 @@ +Index: linux-2.6.15-rc1/arch/arm/mach-pxa/Makefile +=================================================================== +--- linux-2.6.15-rc1.orig/arch/arm/mach-pxa/Makefile 2005-11-19 23:13:40.000000000 +0000 ++++ linux-2.6.15-rc1/arch/arm/mach-pxa/Makefile 2005-11-19 23:13:41.000000000 +0000 +@@ -31,6 +31,7 @@ + + # Misc features + obj-$(CONFIG_PM) += pm.o sleep.o ++obj-$(CONFIG_PXA_KEYS) += pxa_keys.o + obj-$(CONFIG_PXA_SSP) += ssp.o + + ifeq ($(CONFIG_PXA27x),y) +Index: linux-2.6.15-rc1/arch/arm/mach-pxa/Kconfig +=================================================================== +--- linux-2.6.15-rc1.orig/arch/arm/mach-pxa/Kconfig 2005-11-19 23:13:40.000000000 +0000 ++++ linux-2.6.15-rc1/arch/arm/mach-pxa/Kconfig 2005-11-19 23:13:41.000000000 +0000 +@@ -112,6 +112,10 @@ + help + Select code specific to PXA27x variants + ++config PXA_KEYS ++ tristate "PXA25x/27x simple keyboard driver" ++ depends on (PXA25x || PXA27x) && INPUT ++ + config IWMMXT + bool + help +Index: linux-2.6.15-rc1/arch/arm/mach-pxa/pxa_keys.c +=================================================================== +--- /dev/null 1970-01-01 00:00:00.000000000 +0000 ++++ linux-2.6.15-rc1/arch/arm/mach-pxa/pxa_keys.c 2005-11-19 23:32:30.000000000 +0000 +@@ -0,0 +1,157 @@ ++/* ++ * Driver interface for keys on PXA25x GPIO lines ++ * ++ * Copyright 2005 Phil Blundell ++ * ++ * This program is free software; you can redistribute it and/or modify ++ * it under the terms of the GNU General Public License version 2 as ++ * published by the Free Software Foundation. ++ */ ++ ++#include <linux/module.h> ++#include <linux/version.h> ++ ++#include <linux/init.h> ++#include <linux/fs.h> ++#include <linux/interrupt.h> ++#include <linux/sched.h> ++#include <linux/pm.h> ++#include <linux/sysctl.h> ++#include <linux/proc_fs.h> ++#include <linux/delay.h> ++#include <linux/platform_device.h> ++#include <linux/input.h> ++ ++#include <asm/irq.h> ++#include <asm/arch/hardware.h> ++#include <asm/arch/pxa-regs.h> ++#include <asm/arch/pxa_keys.h> ++ ++static irqreturn_t ++pxa_keys_isr (int irq, void *dev_id, struct pt_regs *regs) ++{ ++ int i; ++ struct pxa_keys_platform_data *k = dev_id; ++ ++ if (k->suspended) ++ return IRQ_HANDLED; ++ ++ for (i = 0; i < k->nbuttons; i++) { ++ int gpio = k->buttons[i].gpio; ++ if (irq == IRQ_GPIO(gpio)) { ++ int state = ((GPLR(gpio) & GPIO_bit(gpio)) ? 1 : 0) ++ ^ ((k->buttons[i].flags & PXAKEY_ACTIVE_LOW) != 0); ++ input_report_key (&k->input, k->buttons[i].keycode, state); ++ if ((k->buttons[i].flags & PXAKEY_PWR_KEY) ++ && time_after(jiffies, k->suspend_jiffies + HZ)) { ++ input_event(&k->input, EV_PWR, k->buttons[i].keycode, state); ++ k->suspend_jiffies=jiffies; ++ } ++ ++ } ++ } ++ ++ return IRQ_HANDLED; ++} ++ ++#ifdef CONFIG_PM ++static int pxa_keys_suspend(struct platform_device *pdev, pm_message_t state) ++{ ++ struct pxa_keys_platform_data *k = platform_get_drvdata(pdev); ++ k->suspended = 1; ++ return 0; ++} ++ ++static int pxa_keys_resume(struct platform_device *pdev) ++{ ++ struct pxa_keys_platform_data *k = platform_get_drvdata(pdev); ++ ++ /* Upon resume, ignore the suspend key for a short while */ ++ k->suspend_jiffies=jiffies; ++ k->suspended = 0; ++ return 0; ++} ++#else ++#define pxa_keys_suspend NULL ++#define pxa_keys_resume NULL ++#endif ++ ++static int ++pxa_keys_probe (struct platform_device *pdev) ++{ ++ struct pxa_keys_platform_data *k; ++ int i; ++ ++ k = pdev->dev.platform_data; ++ platform_set_drvdata(pdev, k); ++ ++ init_input_dev (&k->input); ++ ++ k->input.evbit[0] = BIT(EV_KEY) | BIT(EV_PWR) | BIT(EV_REP); ++ for (i = 0; i < k->nbuttons; i++) { ++ int code = k->buttons[i].keycode; ++ int irq = IRQ_GPIO (k->buttons[i].gpio); ++ int result; ++ ++ set_irq_type (irq, IRQT_BOTHEDGE); ++ result = request_irq (irq, pxa_keys_isr, SA_SAMPLE_RANDOM, ++ "pxa_keys", k); ++ if (result == 0) ++ set_bit (code, k->input.keybit); ++ else ++ printk("pxa_keys: unable to claim irq %d; error %d\n", irq, result); ++ } ++ ++ k->input.name = pdev->name; ++ k->input.private = k; ++ ++ k->suspend_jiffies=jiffies; ++ k->suspended=0; ++ ++ input_register_device (&k->input); ++ ++ return 0; ++} ++ ++static int ++pxa_keys_remove (struct platform_device *pdev) ++{ ++ struct pxa_keys_platform_data *k = pdev->dev.platform_data; ++ int i; ++ ++ for (i = 0; i < k->nbuttons; i++) { ++ int irq = IRQ_GPIO (k->buttons[i].gpio); ++ free_irq (irq, k); ++ } ++ ++ input_unregister_device (&k->input); ++ ++ return 0; ++} ++ ++struct platform_driver pxa_keys_device_driver = { ++ .probe = pxa_keys_probe, ++ .remove = pxa_keys_remove, ++ .suspend = pxa_keys_suspend, ++ .resume = pxa_keys_resume, ++ .driver = { ++ .name = "pxa2xx-keys", ++ }, ++}; ++ ++static int pxa_keys_init (void) ++{ ++ return platform_driver_register(&pxa_keys_device_driver); ++} ++ ++static void pxa_keys_cleanup (void) ++{ ++ platform_driver_unregister(&pxa_keys_device_driver); ++} ++ ++module_init (pxa_keys_init); ++module_exit (pxa_keys_cleanup); ++ ++MODULE_LICENSE("GPL"); ++MODULE_AUTHOR("Phil Blundell <pb@handhelds.org>"); ++MODULE_DESCRIPTION("Keyboard driver for PXA25x/PXA27x GPIOs"); +Index: linux-2.6.15-rc1/include/asm-arm/arch-pxa/pxa_keys.h +=================================================================== +--- /dev/null 1970-01-01 00:00:00.000000000 +0000 ++++ linux-2.6.15-rc1/include/asm-arm/arch-pxa/pxa_keys.h 2005-11-19 23:13:41.000000000 +0000 +@@ -0,0 +1,17 @@ ++#include <linux/input.h> ++ ++struct pxa_keys_button { ++ int keycode; ++ int gpio; ++ unsigned long flags; ++}; ++#define PXAKEY_ACTIVE_LOW (1 << 0) /* Key is Active Low */ ++#define PXAKEY_PWR_KEY (1 << 1) /* Key is a Power Key */ ++ ++struct pxa_keys_platform_data { ++ struct pxa_keys_button *buttons; ++ int nbuttons; ++ struct input_dev input; ++ unsigned int suspended; ++ unsigned long suspend_jiffies; ++}; diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/pxa_timerfix-r0.patch b/packages/linux/logicpd-pxa270-2.6.19.2/pxa_timerfix-r0.patch new file mode 100644 index 0000000000..959de60276 --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/pxa_timerfix-r0.patch @@ -0,0 +1,77 @@ +PXA Timers: Stop interrupts becomming enabled when they shouldn't be + +Calling save_time_delta() from within pxa_pm_enter() isn't allowed as it +will reenable interrupts in a function where they should be disabled +throughout. These calls can be made safely from the time.c suspend/resume +functions instead. + +Signed-Off-By: Richard Purdie <rpurdie@rpsys.net> + + +Index: linux-2.6.12/arch/arm/mach-pxa/pm.c +=================================================================== +--- linux-2.6.12.orig/arch/arm/mach-pxa/pm.c 2005-08-21 20:05:16.000000000 +0100 ++++ linux-2.6.12/arch/arm/mach-pxa/pm.c 2005-08-21 20:06:02.000000000 +0100 +@@ -78,7 +78,6 @@ + { + unsigned long sleep_save[SLEEP_SAVE_SIZE]; + unsigned long checksum = 0; +- struct timespec delta, rtc; + int i; + extern void pxa_cpu_pm_enter(suspend_state_t state); + +@@ -87,11 +86,6 @@ + iwmmxt_task_disable(NULL); + #endif + +- /* preserve current time */ +- rtc.tv_sec = RCNR; +- rtc.tv_nsec = 0; +- save_time_delta(&delta, &rtc); +- + SAVE(GPLR0); SAVE(GPLR1); SAVE(GPLR2); + SAVE(GPDR0); SAVE(GPDR1); SAVE(GPDR2); + SAVE(GRER0); SAVE(GRER1); SAVE(GRER2); +@@ -182,10 +176,6 @@ + + RESTORE(PSTR); + +- /* restore current time */ +- rtc.tv_sec = RCNR; +- restore_time_delta(&delta, &rtc); +- + #ifdef DEBUG + printk(KERN_DEBUG "*** made it back from resume\n"); + #endif +Index: linux-2.6.12/arch/arm/mach-pxa/time.c +=================================================================== +--- linux-2.6.12.orig/arch/arm/mach-pxa/time.c 2005-08-21 20:04:39.000000000 +0100 ++++ linux-2.6.12/arch/arm/mach-pxa/time.c 2005-08-21 20:05:24.000000000 +0100 +@@ -128,6 +128,7 @@ + + #ifdef CONFIG_PM + static unsigned long osmr[4], oier; ++static struct timespec delta, rtc; + + static void pxa_timer_suspend(void) + { +@@ -136,10 +137,19 @@ + osmr[2] = OSMR2; + osmr[3] = OSMR3; + oier = OIER; ++ ++ /* preserve current time */ ++ rtc.tv_sec = RCNR; ++ rtc.tv_nsec = 0; ++ save_time_delta(&delta, &rtc); + } + + static void pxa_timer_resume(void) + { ++ /* restore current time */ ++ rtc.tv_sec = RCNR; ++ restore_time_delta(&delta, &rtc); ++ + OSMR0 = osmr[0]; + OSMR1 = osmr[1]; + OSMR2 = osmr[2]; diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/pxafb_fix_params-r2.patch b/packages/linux/logicpd-pxa270-2.6.19.2/pxafb_fix_params-r2.patch new file mode 100644 index 0000000000..6ddb951df0 --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/pxafb_fix_params-r2.patch @@ -0,0 +1,90 @@ +Fix pxafb compile failures when CONFIG_FB_PXA_PARAMETERS is enabled after +recent structure changes. + +Signed-off-by: Richard Purdie <rpurdie@rpsys.net> + +--- + drivers/video/pxafb.c | 40 ++++++++++++++++++++-------------------- + 1 file changed, 20 insertions(+), 20 deletions(-) + +Index: git/drivers/video/pxafb.c +=================================================================== +--- git.orig/drivers/video/pxafb.c 2006-12-16 23:53:00.000000000 +0000 ++++ git/drivers/video/pxafb.c 2006-12-16 23:53:18.000000000 +0000 +@@ -1216,7 +1216,7 @@ static int __init pxafb_parse_options(st + done: + if (res_specified) { + dev_info(dev, "overriding resolution: %dx%d\n", xres, yres); +- inf->xres = xres; inf->yres = yres; ++ inf->modes[0].xres = xres; inf->modes[0].yres = yres; + } + if (bpp_specified) + switch (bpp) { +@@ -1225,48 +1225,48 @@ static int __init pxafb_parse_options(st + case 4: + case 8: + case 16: +- inf->bpp = bpp; ++ inf->modes[0].bpp = bpp; + dev_info(dev, "overriding bit depth: %d\n", bpp); + break; + default: + dev_err(dev, "Depth %d is not valid\n", bpp); + } + } else if (!strncmp(this_opt, "pixclock:", 9)) { +- inf->pixclock = simple_strtoul(this_opt+9, NULL, 0); +- dev_info(dev, "override pixclock: %ld\n", inf->pixclock); ++ inf->modes[0].pixclock = simple_strtoul(this_opt+9, NULL, 0); ++ dev_info(dev, "override pixclock: %ld\n", inf->modes[0].pixclock); + } else if (!strncmp(this_opt, "left:", 5)) { +- inf->left_margin = simple_strtoul(this_opt+5, NULL, 0); +- dev_info(dev, "override left: %u\n", inf->left_margin); ++ inf->modes[0].left_margin = simple_strtoul(this_opt+5, NULL, 0); ++ dev_info(dev, "override left: %u\n", inf->modes[0].left_margin); + } else if (!strncmp(this_opt, "right:", 6)) { +- inf->right_margin = simple_strtoul(this_opt+6, NULL, 0); +- dev_info(dev, "override right: %u\n", inf->right_margin); ++ inf->modes[0].right_margin = simple_strtoul(this_opt+6, NULL, 0); ++ dev_info(dev, "override right: %u\n", inf->modes[0].right_margin); + } else if (!strncmp(this_opt, "upper:", 6)) { +- inf->upper_margin = simple_strtoul(this_opt+6, NULL, 0); +- dev_info(dev, "override upper: %u\n", inf->upper_margin); ++ inf->modes[0].upper_margin = simple_strtoul(this_opt+6, NULL, 0); ++ dev_info(dev, "override upper: %u\n", inf->modes[0].upper_margin); + } else if (!strncmp(this_opt, "lower:", 6)) { +- inf->lower_margin = simple_strtoul(this_opt+6, NULL, 0); +- dev_info(dev, "override lower: %u\n", inf->lower_margin); ++ inf->modes[0].lower_margin = simple_strtoul(this_opt+6, NULL, 0); ++ dev_info(dev, "override lower: %u\n", inf->modes[0].lower_margin); + } else if (!strncmp(this_opt, "hsynclen:", 9)) { +- inf->hsync_len = simple_strtoul(this_opt+9, NULL, 0); +- dev_info(dev, "override hsynclen: %u\n", inf->hsync_len); ++ inf->modes[0].hsync_len = simple_strtoul(this_opt+9, NULL, 0); ++ dev_info(dev, "override hsynclen: %u\n", inf->modes[0].hsync_len); + } else if (!strncmp(this_opt, "vsynclen:", 9)) { +- inf->vsync_len = simple_strtoul(this_opt+9, NULL, 0); +- dev_info(dev, "override vsynclen: %u\n", inf->vsync_len); ++ inf->modes[0].vsync_len = simple_strtoul(this_opt+9, NULL, 0); ++ dev_info(dev, "override vsynclen: %u\n", inf->modes[0].vsync_len); + } else if (!strncmp(this_opt, "hsync:", 6)) { + if (simple_strtoul(this_opt+6, NULL, 0) == 0) { + dev_info(dev, "override hsync: Active Low\n"); +- inf->sync &= ~FB_SYNC_HOR_HIGH_ACT; ++ inf->modes[0].sync &= ~FB_SYNC_HOR_HIGH_ACT; + } else { + dev_info(dev, "override hsync: Active High\n"); +- inf->sync |= FB_SYNC_HOR_HIGH_ACT; ++ inf->modes[0].sync |= FB_SYNC_HOR_HIGH_ACT; + } + } else if (!strncmp(this_opt, "vsync:", 6)) { + if (simple_strtoul(this_opt+6, NULL, 0) == 0) { + dev_info(dev, "override vsync: Active Low\n"); +- inf->sync &= ~FB_SYNC_VERT_HIGH_ACT; ++ inf->modes[0].sync &= ~FB_SYNC_VERT_HIGH_ACT; + } else { + dev_info(dev, "override vsync: Active High\n"); +- inf->sync |= FB_SYNC_VERT_HIGH_ACT; ++ inf->modes[0].sync |= FB_SYNC_VERT_HIGH_ACT; + } + } else if (!strncmp(this_opt, "dpc:", 4)) { + if (simple_strtoul(this_opt+4, NULL, 0) == 0) { diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/ucb1400-touchscreen.patch b/packages/linux/logicpd-pxa270-2.6.19.2/ucb1400-touchscreen.patch new file mode 100644 index 0000000000..3986a2c10a --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/ucb1400-touchscreen.patch @@ -0,0 +1,747 @@ +This patch is slightly adjusted from + +http://www.arm.linux.org.uk/developer/patches/viewpatch.php?id=3073/1 +http://www.arm.linux.org.uk/developer/patches/viewpatch.php?id=3074/2 +http://www.arm.linux.org.uk/developer/patches/viewpatch.php?id=3075/2 + +in order to get it to apply cleanly to the released 2.6.15 codebase +and to put the Kconfig stuff in a more reasonable place in the tree. +Actually, I think Kconfig should probably separate the notion of the +touchscreen driver and the AC97-MCP layer thing; but that problem is +basically in the underlying mcp-based ucb1x00 driver layout in the +first place. +Index: linux-2.6.15gum/drivers/mfd/Makefile +=================================================================== +--- linux-2.6.15gum.orig/drivers/mfd/Makefile ++++ linux-2.6.15gum/drivers/mfd/Makefile +@@ -10,3 +10,6 @@ + ifeq ($(CONFIG_SA1100_ASSABET),y) + obj-$(CONFIG_MCP_UCB1200) += ucb1x00-assabet.o + endif ++ ++obj-$(CONFIG_TOUCHSCREEN_UCB1400) += mcp-ac97.o ucb1x00-core.o ucb1x00-ts.o ++ +Index: linux-2.6.15gum/drivers/mfd/mcp-core.c +=================================================================== +--- linux-2.6.15gum.orig/drivers/mfd/mcp-core.c ++++ linux-2.6.15gum/drivers/mfd/mcp-core.c +@@ -18,7 +18,6 @@ + #include <linux/slab.h> + #include <linux/string.h> + +-#include <asm/dma.h> + #include <asm/system.h> + + #include "mcp.h" +@@ -206,6 +205,7 @@ struct mcp *mcp_host_alloc(struct device + mcp->attached_device.bus = &mcp_bus_type; + mcp->attached_device.dma_mask = parent->dma_mask; + mcp->attached_device.release = mcp_release; ++ mcp->dev = &mcp->attached_device; + } + return mcp; + } +Index: linux-2.6.15gum/drivers/mfd/mcp-sa11x0.c +=================================================================== +--- linux-2.6.15gum.orig/drivers/mfd/mcp-sa11x0.c ++++ linux-2.6.15gum/drivers/mfd/mcp-sa11x0.c +@@ -31,8 +31,12 @@ + #include "mcp.h" + + struct mcp_sa11x0 { +- u32 mccr0; +- u32 mccr1; ++ u32 mccr0; ++ u32 mccr1; ++ dma_device_t dma_audio_rd; ++ dma_device_t dma_audio_wr; ++ dma_device_t dma_telco_rd; ++ dma_device_t dma_telco_wr; + }; + + #define priv(mcp) ((struct mcp_sa11x0 *)mcp_priv(mcp)) +@@ -159,10 +163,10 @@ static int mcp_sa11x0_probe(struct platf + mcp->owner = THIS_MODULE; + mcp->ops = &mcp_sa11x0; + mcp->sclk_rate = data->sclk_rate; +- mcp->dma_audio_rd = DMA_Ser4MCP0Rd; +- mcp->dma_audio_wr = DMA_Ser4MCP0Wr; +- mcp->dma_telco_rd = DMA_Ser4MCP1Rd; +- mcp->dma_telco_wr = DMA_Ser4MCP1Wr; ++ priv(mcp)->dma_audio_rd = DMA_Ser4MCP0Rd; ++ priv(mcp)->dma_audio_wr = DMA_Ser4MCP0Wr; ++ priv(mcp)->dma_telco_rd = DMA_Ser4MCP1Rd; ++ priv(mcp)->dma_telco_wr = DMA_Ser4MCP1Wr; + + platform_set_drvdata(pdev, mcp); + +Index: linux-2.6.15gum/drivers/mfd/mcp.h +=================================================================== +--- linux-2.6.15gum.orig/drivers/mfd/mcp.h ++++ linux-2.6.15gum/drivers/mfd/mcp.h +@@ -19,11 +19,8 @@ struct mcp { + int use_count; + unsigned int sclk_rate; + unsigned int rw_timeout; +- dma_device_t dma_audio_rd; +- dma_device_t dma_audio_wr; +- dma_device_t dma_telco_rd; +- dma_device_t dma_telco_wr; + struct device attached_device; ++ struct device *dev; + }; + + struct mcp_ops { +Index: linux-2.6.15gum/drivers/mfd/ucb1x00-assabet.c +=================================================================== +--- linux-2.6.15gum.orig/drivers/mfd/ucb1x00-assabet.c ++++ linux-2.6.15gum/drivers/mfd/ucb1x00-assabet.c +@@ -15,8 +15,6 @@ + #include <linux/proc_fs.h> + #include <linux/device.h> + +-#include <asm/dma.h> +- + #include "ucb1x00.h" + + #define UCB1X00_ATTR(name,input)\ +Index: linux-2.6.15gum/drivers/mfd/ucb1x00-core.c +=================================================================== +--- a/drivers/mfd/ucb1x00-core~org.c 2006-11-29 16:57:37.000000000 -0500 ++++ a/drivers/mfd/ucb1x00-core.c 2006-12-29 13:42:58.000000000 -0500 +@@ -22,6 +22,7 @@ + #include <linux/init.h> + #include <linux/errno.h> + #include <linux/interrupt.h> ++#include <linux/kthread.h> + #include <linux/device.h> + #include <linux/mutex.h> + +@@ -30,6 +31,14 @@ + + #include "ucb1x00.h" + ++#if (defined CONFIG_UCB1400) || (defined CONFIG_UCB1400_MODULE) ++#define UCB_IS_1400(id) ((id) == UCB_ID_1400) ++#define UCB_X_CSR1 0xe /* this fake entry will be translated by mcp */ ++#define UCB_X_CSR2 0xf /* this fake entry will be translated by mcp */ ++#else ++#define UCB_IS_1400(id) (0) ++#endif ++ + static DEFINE_MUTEX(ucb1x00_mutex); + static LIST_HEAD(ucb1x00_drivers); + static LIST_HEAD(ucb1x00_devices); +@@ -57,9 +66,9 @@ + spin_lock_irqsave(&ucb->io_lock, flags); + ucb->io_dir |= out; + ucb->io_dir &= ~in; ++ spin_unlock_irqrestore(&ucb->io_lock, flags); + + ucb1x00_reg_write(ucb, UCB_IO_DIR, ucb->io_dir); +- spin_unlock_irqrestore(&ucb->io_lock, flags); + } + + /** +@@ -85,9 +94,9 @@ + spin_lock_irqsave(&ucb->io_lock, flags); + ucb->io_out |= set; + ucb->io_out &= ~clear; ++ spin_unlock_irqrestore(&ucb->io_lock, flags); + + ucb1x00_reg_write(ucb, UCB_IO_DATA, ucb->io_out); +- spin_unlock_irqrestore(&ucb->io_lock, flags); + } + + /** +@@ -177,7 +186,7 @@ + schedule_timeout(1); + } + +- return UCB_ADC_DAT(val); ++ return UCB_IS_1400(ucb->id) ? (val & 0x3ff) : ((val & 0x7fe0) >> 5); + } + + /** +@@ -222,6 +231,47 @@ + return IRQ_HANDLED; + } + ++/* ++ * A restriction with interrupts exists when using the ucb1400, as ++ * the codec read/write routines may sleep while waiting for codec ++ * access completion and uses semaphores for access control to the ++ * AC97 bus. A complete codec read cycle could take anywhere from ++ * 60 to 100uSec so we *definitely* don't want to spin inside the ++ * interrupt handler waiting for codec access. So, we handle the ++ * interrupt by scheduling a RT kernel thread to run in process ++ * context instead of interrupt context. ++ */ ++static int ucb1x00_thread(void *_ucb) ++{ ++ struct task_struct *tsk = current; ++ struct ucb1x00 *ucb = _ucb; ++ ++ tsk->policy = SCHED_FIFO; ++ tsk->rt_priority = 1; ++ ++ while (!kthread_should_stop()) { ++ wait_for_completion_interruptible(&ucb->irq_wait); ++ if (try_to_freeze()) ++ continue; ++ ucb1x00_irq(ucb->irq, ucb); ++ enable_irq(ucb->irq); ++ } ++ ++ ucb->irq_task = NULL; ++ return 0; ++} ++ ++static irqreturn_t ucb1x00_threaded_irq(int irqnr, void *devid, struct pt_regs *regs) ++{ ++ struct ucb1x00 *ucb = devid; ++ if (irqnr == ucb->irq) { ++ disable_irq(ucb->irq); ++ complete(&ucb->irq_wait); ++ return IRQ_HANDLED; ++ } ++ return IRQ_NONE; ++} ++ + /** + * ucb1x00_hook_irq - hook a UCB1x00 interrupt + * @ucb: UCB1x00 structure describing chip +@@ -276,17 +326,27 @@ + if (idx < 16) { + spin_lock_irqsave(&ucb->lock, flags); + +- ucb1x00_enable(ucb); +- if (edges & UCB_RISING) { ++ if (edges & UCB_RISING) + ucb->irq_ris_enbl |= 1 << idx; +- ucb1x00_reg_write(ucb, UCB_IE_RIS, ucb->irq_ris_enbl); +- } +- if (edges & UCB_FALLING) { ++ if (edges & UCB_FALLING) + ucb->irq_fal_enbl |= 1 << idx; +- ucb1x00_reg_write(ucb, UCB_IE_FAL, ucb->irq_fal_enbl); +- } +- ucb1x00_disable(ucb); ++ + spin_unlock_irqrestore(&ucb->lock, flags); ++ ++ ucb1x00_enable(ucb); ++ ucb1x00_reg_write(ucb, UCB_IE_RIS, ucb->irq_ris_enbl); ++ ucb1x00_reg_write(ucb, UCB_IE_FAL, ucb->irq_fal_enbl); ++ ucb1x00_disable(ucb); ++ ++ /* This prevents spurious interrupts on the UCB1400 */ ++ ucb1x00_reg_write(ucb, UCB_IE_CLEAR, 1 << idx); ++ ucb1x00_reg_write(ucb, UCB_IE_CLEAR, 0); ++ ++ ucb1x00_reg_write(ucb, UCB_IE_RIS, ucb->irq_ris_enbl); ++ ucb1x00_reg_write(ucb, UCB_IE_FAL, ucb->irq_fal_enbl); ++ ++ ucb1x00_disable(ucb); ++ + } + } + +@@ -306,16 +366,20 @@ + spin_lock_irqsave(&ucb->lock, flags); + + ucb1x00_enable(ucb); +- if (edges & UCB_RISING) { ++ if (edges & UCB_RISING) + ucb->irq_ris_enbl &= ~(1 << idx); +- ucb1x00_reg_write(ucb, UCB_IE_RIS, ucb->irq_ris_enbl); +- } +- if (edges & UCB_FALLING) { ++ ++ if (edges & UCB_FALLING) + ucb->irq_fal_enbl &= ~(1 << idx); +- ucb1x00_reg_write(ucb, UCB_IE_FAL, ucb->irq_fal_enbl); +- } +- ucb1x00_disable(ucb); ++ + spin_unlock_irqrestore(&ucb->lock, flags); ++ ++ ucb1x00_enable(ucb); ++ ucb1x00_reg_write(ucb, UCB_IE_RIS, ucb->irq_ris_enbl); ++ ucb1x00_reg_write(ucb, UCB_IE_FAL, ucb->irq_fal_enbl); ++ ucb1x00_disable(ucb); ++ ++ + } + } + +@@ -348,16 +412,17 @@ + ucb->irq_ris_enbl &= ~(1 << idx); + ucb->irq_fal_enbl &= ~(1 << idx); + +- ucb1x00_enable(ucb); +- ucb1x00_reg_write(ucb, UCB_IE_RIS, ucb->irq_ris_enbl); +- ucb1x00_reg_write(ucb, UCB_IE_FAL, ucb->irq_fal_enbl); +- ucb1x00_disable(ucb); +- + irq->fn = NULL; + irq->devid = NULL; + ret = 0; + } + spin_unlock_irq(&ucb->lock); ++ ++ ucb1x00_enable(ucb); ++ ucb1x00_reg_write(ucb, UCB_IE_RIS, ucb->irq_ris_enbl); ++ ucb1x00_reg_write(ucb, UCB_IE_FAL, ucb->irq_fal_enbl); ++ ucb1x00_disable(ucb); ++ + return ret; + + bad: +@@ -479,7 +544,7 @@ + mcp_enable(mcp); + id = mcp_reg_read(mcp, UCB_ID); + +- if (id != UCB_ID_1200 && id != UCB_ID_1300 && id != UCB_ID_TC35143) { ++ if (id != UCB_ID_1200 && id != UCB_ID_1300 && id != UCB_ID_TC35143 && !UCB_IS_1400(id)) { + printk(KERN_WARNING "UCB1x00 ID not found: %04x\n", id); + goto err_disable; + } +@@ -492,29 +557,41 @@ + memset(ucb, 0, sizeof(struct ucb1x00)); + + ucb->cdev.class = &ucb1x00_class; +- ucb->cdev.dev = &mcp->attached_device; ++ ucb->cdev.dev = mcp->dev; + strlcpy(ucb->cdev.class_id, "ucb1x00", sizeof(ucb->cdev.class_id)); + + spin_lock_init(&ucb->lock); + spin_lock_init(&ucb->io_lock); + sema_init(&ucb->adc_sem, 1); ++ init_completion(&ucb->irq_wait); + + ucb->id = id; + ucb->mcp = mcp; + ucb->irq = ucb1x00_detect_irq(ucb); + if (ucb->irq == NO_IRQ) { +- printk(KERN_ERR "UCB1x00: IRQ probe failed\n"); +- ret = -ENODEV; +- goto err_free; ++ printk(KERN_ERR "UCB1x00: IRQ probe auto-detect failed.. hardcoding IRQ..\n"); ++ ucb->irq = 164; ++ //ret = -ENODEV; ++ //goto err_free; + } + +- ret = request_irq(ucb->irq, ucb1x00_irq, IRQF_TRIGGER_RISING, +- "UCB1x00", ucb); ++ ret = request_irq(ucb->irq, ++ UCB_IS_1400(id) ? ucb1x00_threaded_irq : ucb1x00_irq, ++ 0, "UCB1x00", ucb); ++ + if (ret) { + printk(KERN_ERR "ucb1x00: unable to grab irq%d: %d\n", + ucb->irq, ret); + goto err_free; + } ++ if (UCB_IS_1400(id)) { ++ ucb->irq_task = kthread_run(ucb1x00_thread, ucb, "kUCB1x00d"); ++ if (IS_ERR(ucb->irq_task)) { ++ ret = PTR_ERR(ucb->irq_task); ++ ucb->irq_task = NULL; ++ goto err_irq; ++ } ++ } + + mcp_set_drvdata(mcp, ucb); + +@@ -532,6 +609,8 @@ + goto out; + + err_irq: ++ if (UCB_IS_1400(id) && ucb->irq_task) ++ kthread_stop(ucb->irq_task); + free_irq(ucb->irq, ucb); + err_free: + kfree(ucb); +@@ -554,6 +633,8 @@ + } + mutex_unlock(&ucb1x00_mutex); + ++ if (UCB_IS_1400(ucb->id) && ucb->irq_task) ++ kthread_stop(ucb->irq_task); + free_irq(ucb->irq, ucb); + class_device_unregister(&ucb->cdev); + } + +Index: linux-2.6.15gum/drivers/mfd/ucb1x00-ts.c +=================================================================== +--- linux-2.6.15gum.orig/drivers/mfd/ucb1x00-ts.c ++++ linux-2.6.15gum/drivers/mfd/ucb1x00-ts.c +@@ -33,10 +33,8 @@ + #include <linux/slab.h> + #include <linux/kthread.h> + +-#include <asm/dma.h> +-#include <asm/semaphore.h> +-#include <asm/arch/collie.h> + #include <asm/mach-types.h> ++#include <asm/arch-sa1100/collie.h> + + #include "ucb1x00.h" + +@@ -45,13 +43,14 @@ + struct input_dev *idev; + struct ucb1x00 *ucb; + +- wait_queue_head_t irq_wait; ++ struct completion irq_wait; + struct task_struct *rtask; + u16 x_res; + u16 y_res; + + unsigned int restart:1; + unsigned int adcsync:1; ++ unsigned int go_thread; + }; + + static int adcsync; +@@ -205,7 +204,6 @@ + { + struct ucb1x00_ts *ts = _ts; + struct task_struct *tsk = current; +- DECLARE_WAITQUEUE(wait, tsk); + int valid; + + /* +@@ -217,10 +215,8 @@ + + valid = 0; + +- add_wait_queue(&ts->irq_wait, &wait); +- while (!kthread_should_stop()) { ++ while (!kthread_should_stop() && ts->go_thread) { + unsigned int x, y, p; +- signed long timeout; + + ts->restart = 0; + +@@ -242,8 +238,6 @@ + + + if (ucb1x00_ts_pen_down(ts)) { +- set_task_state(tsk, TASK_INTERRUPTIBLE); +- + ucb1x00_enable_irq(ts->ucb, UCB_IRQ_TSPX, machine_is_collie() ? UCB_RISING : UCB_FALLING); + ucb1x00_disable(ts->ucb); + +@@ -256,7 +250,16 @@ + valid = 0; + } + +- timeout = MAX_SCHEDULE_TIMEOUT; ++ /* ++ * Since ucb1x00_enable_irq() might sleep due ++ * to the way the UCB1400 regs are accessed, we ++ * can't use set_task_state() before that call, ++ * and not changing state before enabling the ++ * interrupt is racy. A completion handler avoids ++ * the issue. ++ */ ++ wait_for_completion_interruptible(&ts->irq_wait); ++ + } else { + ucb1x00_disable(ts->ucb); + +@@ -271,16 +274,12 @@ + } + + set_task_state(tsk, TASK_INTERRUPTIBLE); +- timeout = HZ / 100; ++ schedule_timeout(HZ/100); + } + + try_to_freeze(); +- +- schedule_timeout(timeout); + } + +- remove_wait_queue(&ts->irq_wait, &wait); +- + ts->rtask = NULL; + return 0; + } +@@ -293,7 +292,7 @@ + { + struct ucb1x00_ts *ts = id; + ucb1x00_disable_irq(ts->ucb, UCB_IRQ_TSPX, UCB_FALLING); +- wake_up(&ts->irq_wait); ++ complete(&ts->irq_wait); + } + + static int ucb1x00_ts_open(struct input_dev *idev) +@@ -303,7 +302,7 @@ + + BUG_ON(ts->rtask); + +- init_waitqueue_head(&ts->irq_wait); ++ init_completion(&ts->irq_wait); + ret = ucb1x00_hook_irq(ts->ucb, UCB_IRQ_TSPX, ucb1x00_ts_irq, ts); + if (ret < 0) + goto out; +@@ -318,6 +317,8 @@ + ucb1x00_adc_disable(ts->ucb); + + ts->rtask = kthread_run(ucb1x00_thread, ts, "ktsd"); ++ ts->go_thread = 1; //start thread! ++ + if (!IS_ERR(ts->rtask)) { + ret = 0; + } else { +@@ -337,9 +338,14 @@ + { + struct ucb1x00_ts *ts = idev->private; + ++ ++ ts->go_thread = 0; //Stop thread! ++ complete(&ts->irq_wait); ++ + if (ts->rtask) + kthread_stop(ts->rtask); + ++ + ucb1x00_enable(ts->ucb); + ucb1x00_free_irq(ts->ucb, UCB_IRQ_TSPX, ts); + ucb1x00_reg_write(ts->ucb, UCB_TS_CR, 0); +@@ -358,7 +364,7 @@ + * after sleep. + */ + ts->restart = 1; +- wake_up(&ts->irq_wait); ++ complete(&ts->irq_wait); + } + return 0; + } + +Index: linux-2.6.15gum/drivers/mfd/ucb1x00.h +=================================================================== +--- a/drivers/mfd/ucb1x00.h~org 2006-12-29 13:43:38.000000000 -0500 ++++ a/drivers/mfd/ucb1x00.h 2006-12-29 13:44:48.000000000 -0500 +@@ -94,6 +94,7 @@ + #define UCB_ID 0x0c + #define UCB_ID_1200 0x1004 + #define UCB_ID_1300 0x1005 ++#define UCB_ID_1400 0x4304 + #define UCB_ID_TC35143 0x9712 + + #define UCB_MODE 0x0d +@@ -111,6 +112,8 @@ + spinlock_t lock; + struct mcp *mcp; + unsigned int irq; ++ struct task_struct *irq_task; ++ struct completion irq_wait; + struct semaphore adc_sem; + spinlock_t io_lock; + u16 id; +Index: linux-2.6.15gum/drivers/mfd/mcp-ac97.c +=================================================================== +--- /dev/null ++++ linux-2.6.15gum/drivers/mfd/mcp-ac97.c +@@ -0,0 +1,153 @@ ++/* ++ * linux/drivers/misc/mcp-ac97.c ++ * ++ * Author: Nicolas Pitre ++ * Created: Jan 14, 2005 ++ * Copyright: (C) MontaVista Software Inc. ++ * ++ * This program is free software; you can redistribute it and/or ++ * modify it under the terms of the GNU General Public License ++ * version 2 as published by the Free Software Foundation. ++ * ++ * This module provides the minimum replacement for mcp-core.c allowing for ++ * the UCB1400 chip to be driven by the ucb1x00 driver over an AC97 link. ++ */ ++ ++#include <linux/module.h> ++#include <linux/init.h> ++#include <linux/errno.h> ++#include <linux/device.h> ++ ++#include <sound/driver.h> ++#include <sound/core.h> ++#include <sound/ac97_codec.h> ++ ++#include "mcp.h" ++ ++/* ucb1x00 SIB register to ucb1400 AC-link register mapping */ ++ ++static const unsigned char regmap[] = { ++ 0x5a, /* UCB_IO_DATA */ ++ 0X5C, /* UCB_IO_DIR */ ++ 0X5E, /* UCB_IE_RIS */ ++ 0x60, /* UCB_IE_FAL */ ++ 0x62, /* UCB_IE_STATUS */ ++ 0, /* UCB_TC_A */ ++ 0, /* UCB_TC_B */ ++ 0, /* UCB_AC_A */ ++ 0, /* UCB_AC_B */ ++ 0x64, /* UCB_TS_CR */ ++ 0x66, /* UCB_ADC_CR */ ++ 0x68, /* UCB_ADC_DATA */ ++ 0x7e, /* UCB_ID */ ++ 0, /* UCB_MODE */ ++}; ++ ++unsigned int mcp_reg_read(struct mcp *mcp, unsigned int reg) ++{ ++ ac97_t *ac97 = to_ac97_t(mcp->dev); ++ if (reg < ARRAY_SIZE(regmap)) { ++ reg = regmap[reg]; ++ if (reg) ++ return ac97->bus->ops->read(ac97, reg); ++ } ++ return -1; ++} ++EXPORT_SYMBOL(mcp_reg_read); ++ ++void mcp_reg_write(struct mcp *mcp, unsigned int reg, unsigned int val) ++{ ++ ac97_t *ac97 = to_ac97_t(mcp->dev); ++ if (reg < ARRAY_SIZE(regmap)) { ++ reg = regmap[reg]; ++ if (reg) ++ ac97->bus->ops->write(ac97, reg, val); ++ } ++} ++EXPORT_SYMBOL(mcp_reg_write); ++ ++void mcp_enable(struct mcp *mcp) ++{ ++} ++EXPORT_SYMBOL(mcp_enable); ++ ++void mcp_disable(struct mcp *mcp) ++{ ++} ++EXPORT_SYMBOL(mcp_disable); ++ ++#define to_mcp_driver(d) container_of(d, struct mcp_driver, drv) ++ ++static int mcp_probe(struct device *dev) ++{ ++ struct mcp_driver *drv = to_mcp_driver(dev->driver); ++ struct mcp *mcp; ++ int ret; ++ ++ ret = -ENOMEM; ++ mcp = kmalloc(sizeof(*mcp), GFP_KERNEL); ++ if (mcp) { ++ memset(mcp, 0, sizeof(*mcp)); ++ mcp->owner = THIS_MODULE; ++ mcp->dev = dev; ++ ret = drv->probe(mcp); ++ if (ret) ++ kfree(mcp); ++ } ++ if (!ret) ++ dev_set_drvdata(dev, mcp); ++ return ret; ++} ++ ++static int mcp_remove(struct device *dev) ++{ ++ struct mcp_driver *drv = to_mcp_driver(dev->driver); ++ struct mcp *mcp = dev_get_drvdata(dev); ++ ++ drv->remove(mcp); ++ dev_set_drvdata(dev, NULL); ++ kfree(mcp); ++ return 0; ++} ++ ++static int mcp_suspend(struct device *dev, pm_message_t state) ++{ ++ struct mcp_driver *drv = to_mcp_driver(dev->driver); ++ struct mcp *mcp = dev_get_drvdata(dev); ++ int ret = 0; ++ ++ if (drv->suspend) ++ ret = drv->suspend(mcp, state); ++ return ret; ++} ++ ++static int mcp_resume(struct device *dev) ++{ ++ struct mcp_driver *drv = to_mcp_driver(dev->driver); ++ struct mcp *mcp = dev_get_drvdata(dev); ++ int ret = 0; ++ ++ if (drv->resume) ++ ret = drv->resume(mcp); ++ return ret; ++} ++ ++int mcp_driver_register(struct mcp_driver *mcpdrv) ++{ ++ mcpdrv->drv.owner = THIS_MODULE; ++ mcpdrv->drv.bus = &ac97_bus_type; ++ mcpdrv->drv.probe = mcp_probe; ++ mcpdrv->drv.remove = mcp_remove; ++ mcpdrv->drv.suspend = mcp_suspend; ++ mcpdrv->drv.resume = mcp_resume; ++ return driver_register(&mcpdrv->drv); ++} ++EXPORT_SYMBOL(mcp_driver_register); ++ ++void mcp_driver_unregister(struct mcp_driver *mcpdrv) ++{ ++ driver_unregister(&mcpdrv->drv); ++} ++EXPORT_SYMBOL(mcp_driver_unregister); ++ ++MODULE_LICENSE("GPL"); +Index: linux-2.6.15gum/drivers/input/touchscreen/Kconfig +=================================================================== +--- linux-2.6.15gum.orig/drivers/input/touchscreen/Kconfig ++++ linux-2.6.15gum/drivers/input/touchscreen/Kconfig +@@ -11,6 +11,25 @@ menuconfig INPUT_TOUCHSCREEN + + if INPUT_TOUCHSCREEN + ++config UCB1400 ++ bool ++ ++config TOUCHSCREEN_UCB1400 ++ tristate "UCB1400 Touchscreen support" ++ depends on ARCH_LUBBOCK || MACH_MAINSTONE || ARCH_PXA ++ select SND_AC97_BUS ++ select UCB1400 ++ help ++ Say Y here if you have a touchscreen connected to a UCB1400 ADC chip ++ on the AC97 bus of a PXA255/PXA270 host. ++ ++ If unsure, say N. ++ ++ To compile this driver as a module, choose M here: the ++ module will be called ucb1x00-ts. It will also build the modules ++ ucb1x00-core and mcp-ac97 which provide the compatibility layers ++ down to the AC97 bus. ++ + config TOUCHSCREEN_BITSY + tristate "Compaq iPAQ H3600 (Bitsy) touchscreen" + depends on SA1100_BITSY +Index: linux-2.6.15gum/drivers/input/Kconfig +=================================================================== +--- linux-2.6.15gum.orig/drivers/input/Kconfig ++++ linux-2.6.15gum/drivers/input/Kconfig +@@ -87,7 +87,7 @@ config INPUT_JOYDEV + module will be called joydev. + + config INPUT_TSDEV +- tristate "Touchscreen interface" ++ tristate "Compaq touchscreen interface" + ---help--- + Say Y here if you have an application that only can understand the + Compaq touchscreen protocol for absolute pointer data. This is diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/usb_add_epalloc-r3.patch b/packages/linux/logicpd-pxa270-2.6.19.2/usb_add_epalloc-r3.patch new file mode 100644 index 0000000000..d9d576934a --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/usb_add_epalloc-r3.patch @@ -0,0 +1,282 @@ +Index: git/drivers/usb/gadget/epautoconf.c +=================================================================== +--- git.orig/drivers/usb/gadget/epautoconf.c 2006-07-15 09:10:59.000000000 +0100 ++++ git/drivers/usb/gadget/epautoconf.c 2006-07-15 10:27:00.000000000 +0100 +@@ -228,14 +228,19 @@ + * + * On failure, this returns a null endpoint descriptor. + */ +-struct usb_ep * __devinit usb_ep_autoconfig ( ++struct usb_ep * usb_ep_autoconfig ( + struct usb_gadget *gadget, +- struct usb_endpoint_descriptor *desc ++ struct usb_endpoint_descriptor *desc, ++ struct usb_endpoint_config *epconfig, int numconfigs + ) + { + struct usb_ep *ep; + u8 type; + ++ /* Use device specific ep allocation code if provided */ ++ if (gadget->ops->ep_alloc) ++ return gadget->ops->ep_alloc(gadget, desc, epconfig, numconfigs); ++ + type = desc->bmAttributes & USB_ENDPOINT_XFERTYPE_MASK; + + /* First, apply chip-specific "best usage" knowledge. +Index: git/drivers/usb/gadget/ether.c +=================================================================== +--- git.orig/drivers/usb/gadget/ether.c 2006-07-15 09:10:59.000000000 +0100 ++++ git/drivers/usb/gadget/ether.c 2006-07-15 10:25:23.000000000 +0100 +@@ -2195,7 +2195,8 @@ + struct eth_dev *dev; + struct net_device *net; + u8 cdc = 1, zlp = 1, rndis = 1; +- struct usb_ep *in_ep, *out_ep, *status_ep = NULL; ++ struct usb_ep *in_ep = NULL , *out_ep = NULL, *status_ep = NULL; ++ struct usb_endpoint_config ep_config[2]; + int status = -ENOMEM; + int gcnum; + +@@ -2291,7 +2292,26 @@ + + /* all we really need is bulk IN/OUT */ + usb_ep_autoconfig_reset (gadget); +- in_ep = usb_ep_autoconfig (gadget, &fs_source_desc); ++ ++ ep_config[0].config = DEV_CONFIG_VALUE; ++#if defined(DEV_CONFIG_CDC) ++ ep_config[0].interface = data_intf.bInterfaceNumber; ++ ep_config[0].altinterface = data_intf.bAlternateSetting; ++#else /* DEV_CONFIG_SUBSET */ ++ ep_config[0].interface = subset_data_intf.bInterfaceNumber; ++ ep_config[0].altinterface = subset_data_intf.bAlternateSetting; ++#endif ++ ++#ifdef CONFIG_USB_ETH_RNDIS ++ ep_config[1].config = DEV_RNDIS_CONFIG_VALUE; ++ ep_config[1].interface = rndis_data_intf.bInterfaceNumber; ++ ep_config[1].altinterface = rndis_data_intf.bAlternateSetting; ++ ++ in_ep = usb_ep_autoconfig(gadget, &fs_source_desc, &ep_config[0], 2); ++#else ++ in_ep = usb_ep_autoconfig(gadget, &fs_source_desc, &ep_config[0], 1); ++#endif ++ + if (!in_ep) { + autoconf_fail: + dev_err (&gadget->dev, +@@ -2301,7 +2321,12 @@ + } + in_ep->driver_data = in_ep; /* claim */ + +- out_ep = usb_ep_autoconfig (gadget, &fs_sink_desc); ++#ifdef CONFIG_USB_ETH_RNDIS ++ out_ep = usb_ep_autoconfig(gadget, &fs_sink_desc, &ep_config[0], 2); ++#else ++ out_ep = usb_ep_autoconfig(gadget, &fs_sink_desc, &ep_config[0], 1); ++#endif ++ + if (!out_ep) + goto autoconf_fail; + out_ep->driver_data = out_ep; /* claim */ +@@ -2311,7 +2336,25 @@ + * Since some hosts expect one, try to allocate one anyway. + */ + if (cdc || rndis) { +- status_ep = usb_ep_autoconfig (gadget, &fs_status_desc); ++#ifdef DEV_CONFIG_CDC ++ ep_config[0].config = DEV_CONFIG_VALUE; ++ ep_config[0].interface = control_intf.bInterfaceNumber; ++ ep_config[0].altinterface = control_intf.bAlternateSetting; ++#endif ++#ifdef CONFIG_USB_ETH_RNDIS ++ ep_config[1].config = DEV_RNDIS_CONFIG_VALUE; ++ ep_config[1].interface = rndis_control_intf.bInterfaceNumber; ++ ep_config[1].altinterface = rndis_control_intf.bAlternateSetting; ++#endif ++ ++#if defined(DEV_CONFIG_CDC) && defined(CONFIG_USB_ETH_RNDIS) ++ status_ep = usb_ep_autoconfig(gadget, &fs_status_desc, &ep_config[0], 2); ++#elif defined(CONFIG_USB_ETH_RNDIS) ++ status_ep = usb_ep_autoconfig(gadget, &fs_status_desc, &ep_config[1], 1); ++#else ++ status_ep = usb_ep_autoconfig(gadget, &fs_status_desc, &ep_config[0], 1); ++#endif ++ + if (status_ep) { + status_ep->driver_data = status_ep; /* claim */ + } else if (rndis) { +Index: git/drivers/usb/gadget/file_storage.c +=================================================================== +--- git.orig/drivers/usb/gadget/file_storage.c 2006-07-15 09:10:59.000000000 +0100 ++++ git/drivers/usb/gadget/file_storage.c 2006-07-15 10:25:23.000000000 +0100 +@@ -3836,6 +3836,7 @@ + struct usb_ep *ep; + struct usb_request *req; + char *pathbuf, *p; ++ struct usb_endpoint_config ep_config; + + fsg->gadget = gadget; + set_gadget_data(gadget, fsg); +@@ -3901,21 +3902,25 @@ + } + + /* Find all the endpoints we will use */ ++ ep_config.config = CONFIG_VALUE; ++ ep_config.interface = intf_desc.bInterfaceNumber; ++ ep_config.altinterface = intf_desc.bAlternateSetting; ++ + usb_ep_autoconfig_reset(gadget); +- ep = usb_ep_autoconfig(gadget, &fs_bulk_in_desc); ++ ep = usb_ep_autoconfig(gadget, &fs_bulk_in_desc, &ep_config, 1); + if (!ep) + goto autoconf_fail; + ep->driver_data = fsg; // claim the endpoint + fsg->bulk_in = ep; + +- ep = usb_ep_autoconfig(gadget, &fs_bulk_out_desc); ++ ep = usb_ep_autoconfig(gadget, &fs_bulk_out_desc, &ep_config, 1); + if (!ep) + goto autoconf_fail; + ep->driver_data = fsg; // claim the endpoint + fsg->bulk_out = ep; + + if (transport_is_cbi()) { +- ep = usb_ep_autoconfig(gadget, &fs_intr_in_desc); ++ ep = usb_ep_autoconfig(gadget, &fs_intr_in_desc, &ep_config, 1); + if (!ep) + goto autoconf_fail; + ep->driver_data = fsg; // claim the endpoint +Index: git/drivers/usb/gadget/serial.c +=================================================================== +--- git.orig/drivers/usb/gadget/serial.c 2006-07-15 09:10:59.000000000 +0100 ++++ git/drivers/usb/gadget/serial.c 2006-07-15 10:25:23.000000000 +0100 +@@ -1354,6 +1354,7 @@ + struct usb_ep *ep; + struct gs_dev *dev; + int gcnum; ++ struct usb_endpoint_config ep_config[2]; + + /* Some controllers can't support CDC ACM: + * - sh doesn't support multiple interfaces or configs; +@@ -1374,22 +1375,33 @@ + __constant_cpu_to_le16(GS_VERSION_NUM|0x0099); + } + ++ ep_config[0].config = GS_BULK_CONFIG_ID; ++ ep_config[0].interface = gs_bulk_interface_desc.bInterfaceNumber; ++ ep_config[0].altinterface = gs_bulk_interface_desc.bAlternateSetting; ++ ep_config[1].config = GS_ACM_CONFIG_ID; ++ ep_config[1].interface = gs_data_interface_desc.bInterfaceNumber; ++ ep_config[1].altinterface = gs_data_interface_desc.bAlternateSetting; ++ + usb_ep_autoconfig_reset(gadget); + +- ep = usb_ep_autoconfig(gadget, &gs_fullspeed_in_desc); ++ ep = usb_ep_autoconfig(gadget, &gs_fullspeed_in_desc, &ep_config[0], 2); + if (!ep) + goto autoconf_fail; + EP_IN_NAME = ep->name; + ep->driver_data = ep; /* claim the endpoint */ + +- ep = usb_ep_autoconfig(gadget, &gs_fullspeed_out_desc); ++ ep = usb_ep_autoconfig(gadget, &gs_fullspeed_out_desc, &ep_config[0], 2); + if (!ep) + goto autoconf_fail; + EP_OUT_NAME = ep->name; + ep->driver_data = ep; /* claim the endpoint */ + + if (use_acm) { +- ep = usb_ep_autoconfig(gadget, &gs_fullspeed_notify_desc); ++ ep_config[0].config = GS_ACM_CONFIG_ID; ++ ep_config[0].interface = gs_control_interface_desc.bInterfaceNumber; ++ ep_config[0].altinterface = gs_control_interface_desc.bAlternateSetting; ++ ++ ep = usb_ep_autoconfig(gadget, &gs_fullspeed_notify_desc, &ep_config[0], 1); + if (!ep) { + printk(KERN_ERR "gs_bind: cannot run ACM on %s\n", gadget->name); + goto autoconf_fail; +Index: git/drivers/usb/gadget/zero.c +=================================================================== +--- git.orig/drivers/usb/gadget/zero.c 2006-07-15 09:10:59.000000000 +0100 ++++ git/drivers/usb/gadget/zero.c 2006-07-15 10:25:23.000000000 +0100 +@@ -1144,6 +1144,7 @@ + struct zero_dev *dev; + struct usb_ep *ep; + int gcnum; ++ struct usb_endpoint_config ep_config[2]; + + /* FIXME this can't yet work right with SH ... it has only + * one configuration, numbered one. +@@ -1156,7 +1157,15 @@ + * but there may also be important quirks to address. + */ + usb_ep_autoconfig_reset (gadget); +- ep = usb_ep_autoconfig (gadget, &fs_source_desc); ++ ++ ep_config[0].config = CONFIG_SOURCE_SINK; ++ ep_config[0].interface = source_sink_intf.bInterfaceNumber; ++ ep_config[0].altinterface = source_sink_intf.bAlternateSetting; ++ ep_config[1].config = CONFIG_LOOPBACK; ++ ep_config[1].interface = loopback_intf.bInterfaceNumber; ++ ep_config[1].altinterface = loopback_intf.bAlternateSetting; ++ ++ ep = usb_ep_autoconfig(gadget, &fs_source_desc, &ep_config[0], 2); + if (!ep) { + autoconf_fail: + printk (KERN_ERR "%s: can't autoconfigure on %s\n", +@@ -1166,7 +1175,7 @@ + EP_IN_NAME = ep->name; + ep->driver_data = ep; /* claim */ + +- ep = usb_ep_autoconfig (gadget, &fs_sink_desc); ++ ep = usb_ep_autoconfig(gadget, &fs_sink_desc, &ep_config[0], 2); + if (!ep) + goto autoconf_fail; + EP_OUT_NAME = ep->name; +Index: git/include/linux/usb_gadget.h +=================================================================== +--- git.orig/include/linux/usb_gadget.h 2006-07-15 09:11:05.000000000 +0100 ++++ git/include/linux/usb_gadget.h 2006-07-15 10:29:40.000000000 +0100 +@@ -445,10 +445,28 @@ + + struct usb_gadget; + ++/** ++ * struct usb_endpoint_config - possible configurations of a given endpoint ++ * @config: the configuration number ++ * @interface: the interface number ++ * @altinterface: the altinterface number ++ * ++ * Used as an array to pass information about the possible configurations ++ * of a given endpoint to the bus controller. ++ */ ++struct usb_endpoint_config { ++ u8 config; ++ u8 interface; ++ u8 altinterface; ++}; ++ + /* the rest of the api to the controller hardware: device operations, + * which don't involve endpoints (or i/o). + */ + struct usb_gadget_ops { ++ struct usb_ep* (*ep_alloc)(struct usb_gadget *, ++ struct usb_endpoint_descriptor *, ++ struct usb_endpoint_config *, int); + int (*get_frame)(struct usb_gadget *); + int (*wakeup)(struct usb_gadget *); + int (*set_selfpowered) (struct usb_gadget *, int is_selfpowered); +@@ -872,7 +890,10 @@ + /* utility wrapping a simple endpoint selection policy */ + + extern struct usb_ep *usb_ep_autoconfig (struct usb_gadget *, +- struct usb_endpoint_descriptor *) __devinit; ++ struct usb_endpoint_descriptor *, ++ struct usb_endpoint_config *, ++ int numconfigs ++); + + extern void usb_ep_autoconfig_reset (struct usb_gadget *) __devinit; + diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/usb_pxa27x_udc-r3.patch b/packages/linux/logicpd-pxa270-2.6.19.2/usb_pxa27x_udc-r3.patch new file mode 100644 index 0000000000..556193d1ed --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/usb_pxa27x_udc-r3.patch @@ -0,0 +1,2878 @@ + arch/arm/mach-pxa/spitz.c | 27 + drivers/usb/gadget/Kconfig | 20 + drivers/usb/gadget/Makefile | 1 + drivers/usb/gadget/ether.c | 12 + drivers/usb/gadget/pxa27x_udc.c | 2412 ++++++++++++++++++++++++++++++++++++++++ + drivers/usb/gadget/pxa27x_udc.h | 298 ++++ + drivers/usb/gadget/pxa2xx_udc.h | 7 + 7 files changed, 2774 insertions(+), 3 deletions(-) + +Index: git/arch/arm/mach-pxa/spitz.c +=================================================================== +--- git.orig/arch/arm/mach-pxa/spitz.c 2006-12-16 22:57:10.000000000 +0000 ++++ git/arch/arm/mach-pxa/spitz.c 2006-12-16 22:57:15.000000000 +0000 +@@ -348,6 +348,32 @@ static struct pxamci_platform_data spitz + + + /* ++ * USB Client (Gadget/UDC) ++ */ ++static void spitz_udc_command(int cmd) ++{ ++ switch(cmd) { ++ case PXA2XX_UDC_CMD_CONNECT: ++ UP2OCR = UP2OCR_HXOE | UP2OCR_DMPUE | UP2OCR_DMPUBE; ++ break; ++ case PXA2XX_UDC_CMD_DISCONNECT: ++ //UP2OCR = UP2OCR_HXOE | UP2OCR_DMPUE | UP2OCR_DMPUBE; ++ break; ++ } ++} ++ ++static int spitz_udc_detect(void) ++{ ++ return 1; ++} ++ ++static struct pxa2xx_udc_mach_info spitz_udc_info __initdata = { ++ .udc_is_connected = spitz_udc_detect, ++ .udc_command = spitz_udc_command, ++}; ++ ++ ++/* + * USB Host (OHCI) + */ + static int spitz_ohci_init(struct device *dev) +@@ -498,6 +524,7 @@ static void __init common_init(void) + pxa_gpio_mode(SPITZ_GPIO_HSYNC | GPIO_IN); + + platform_add_devices(devices, ARRAY_SIZE(devices)); ++ pxa_set_udc_info(&spitz_udc_info); + pxa_set_mci_info(&spitz_mci_platform_data); + pxa_set_ohci_info(&spitz_ohci_platform_data); + pxa_set_ficp_info(&spitz_ficp_platform_data); +Index: git/drivers/usb/gadget/Kconfig +=================================================================== +--- git.orig/drivers/usb/gadget/Kconfig 2006-12-16 22:57:10.000000000 +0000 ++++ git/drivers/usb/gadget/Kconfig 2006-12-16 22:57:15.000000000 +0000 +@@ -121,6 +121,26 @@ config USB_PXA2XX_SMALL + default y if USB_ETH + default y if USB_G_SERIAL + ++config USB_GADGET_PXA27X ++ boolean "PXA 27x" ++ depends on ARCH_PXA && PXA27x ++ help ++ Intel's PXA 27x series XScale ARM-5TE processors include ++ an integrated full speed USB 1.1 device controller. ++ ++ It has 23 endpoints, as well as endpoint zero (for control ++ transfers). ++ ++ Say "y" to link the driver statically, or "m" to build a ++ dynamically linked module called "pxa27x_udc" and force all ++ gadget drivers to also be dynamically linked. ++ ++config USB_PXA27X ++ tristate ++ depends on USB_GADGET_PXA27X ++ default USB_GADGET ++ select USB_GADGET_SELECTED ++ + config USB_GADGET_GOKU + boolean "Toshiba TC86C001 'Goku-S'" + depends on PCI +Index: git/drivers/usb/gadget/Makefile +=================================================================== +--- git.orig/drivers/usb/gadget/Makefile 2006-12-16 22:57:10.000000000 +0000 ++++ git/drivers/usb/gadget/Makefile 2006-12-16 22:57:15.000000000 +0000 +@@ -4,6 +4,7 @@ + obj-$(CONFIG_USB_DUMMY_HCD) += dummy_hcd.o + obj-$(CONFIG_USB_NET2280) += net2280.o + obj-$(CONFIG_USB_PXA2XX) += pxa2xx_udc.o ++obj-$(CONFIG_USB_PXA27X) += pxa27x_udc.o + obj-$(CONFIG_USB_GOKU) += goku_udc.o + obj-$(CONFIG_USB_OMAP) += omap_udc.o + obj-$(CONFIG_USB_LH7A40X) += lh7a40x_udc.o +Index: git/drivers/usb/gadget/pxa27x_udc.c +=================================================================== +--- /dev/null 1970-01-01 00:00:00.000000000 +0000 ++++ git/drivers/usb/gadget/pxa27x_udc.c 2006-12-16 23:19:49.000000000 +0000 +@@ -0,0 +1,2412 @@ ++/* ++ * Handles the Intel 27x USB Device Controller (UDC) ++ * ++ * Copyright (C) 2002 Intrinsyc, Inc. (Frank Becker) ++ * Copyright (C) 2003 Robert Schwebel, Pengutronix ++ * Copyright (C) 2003 Benedikt Spranger, Pengutronix ++ * Copyright (C) 2003 David Brownell ++ * Copyright (C) 2003 Joshua Wise ++ * Copyright (C) 2004 Intel Corporation ++ * Copyright (C) 2005 SDG Systems, LLC (Aric Blumer) ++ * Copyright (C) 2005-2006 Openedhand Ltd. (Richard Purdie) ++ * ++ * 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; if not, write to the Free Software ++ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA ++ * ++ */ ++ ++#undef DEBUG ++//#define DEBUG 1 ++//#define VERBOSE DBG_VERBOSE ++ ++#include <linux/module.h> ++#include <linux/kernel.h> ++#include <linux/ioport.h> ++#include <linux/types.h> ++#include <linux/version.h> ++#include <linux/errno.h> ++#include <linux/platform_device.h> ++#include <linux/delay.h> ++#include <linux/sched.h> ++#include <linux/slab.h> ++#include <linux/init.h> ++#include <linux/timer.h> ++#include <linux/list.h> ++#include <linux/interrupt.h> ++#include <linux/proc_fs.h> ++#include <linux/mm.h> ++#include <linux/device.h> ++#include <linux/dma-mapping.h> ++ ++#include <asm/byteorder.h> ++#include <asm/dma.h> ++#include <asm/io.h> ++#include <asm/irq.h> ++#include <asm/system.h> ++#include <asm/mach-types.h> ++#include <asm/unaligned.h> ++#include <asm/hardware.h> ++#include <asm/arch/pxa-regs.h> ++ ++#include <linux/usb_ch9.h> ++#include <linux/usb_gadget.h> ++ ++#include <asm/arch/udc.h> ++ ++/* ++ * This driver handles the USB Device Controller (UDC) in Intel's PXA 27x ++ * series processors. ++ * ++ * Such controller drivers work with a gadget driver. The gadget driver ++ * returns descriptors, implements configuration and data protocols used ++ * by the host to interact with this device, and allocates endpoints to ++ * the different protocol interfaces. The controller driver virtualizes ++ * usb hardware so that the gadget drivers will be more portable. ++ * ++ * This UDC hardware wants to implement a bit too much USB protocol. The ++ * biggest issue is that the endpoints have to be setup before the controller ++ * can be enabled and each endpoint can only have one configuration, interface ++ * and alternative interface number. Once enabled, these cannot be changed ++ * without a controller reset. ++ * ++ * Intel Errata #22 mentions issues when changing alternate interface. ++ * The exact meaning of this remains uncertain as gadget drivers using alternate ++ * interfaces such as CDC-Ethernet appear to work... ++ */ ++ ++#define DRIVER_VERSION "01-01-2006" ++#define DRIVER_DESC "PXA 27x USB Device Controller driver" ++ ++static const char driver_name [] = "pxa27x_udc"; ++ ++static const char ep0name [] = "ep0"; ++ ++ ++#define USE_DMA ++//#undef USE_DMA ++ ++#ifdef CONFIG_PROC_FS ++#define UDC_PROC_FILE ++#endif ++ ++#include "pxa27x_udc.h" ++ ++#ifdef USE_DMA ++static int use_dma = 1; ++module_param(use_dma, bool, 0); ++MODULE_PARM_DESC(use_dma, "true to use dma"); ++ ++static void dma_nodesc_handler(int dmach, void *_ep); ++static void kick_dma(struct pxa27x_ep *ep, struct pxa27x_request *req); ++ ++#define DMASTR " (dma support)" ++ ++#else /* !USE_DMA */ ++#define DMASTR " (pio only)" ++#endif ++ ++#define UDCISR0_IR0 0x3 ++#define UDCISR_INT_MASK (UDC_INT_FIFOERROR | UDC_INT_PACKETCMP) ++#define UDCICR_INT_MASK UDCISR_INT_MASK ++ ++#define UDCCSR_MASK (UDCCSR_FST | UDCCSR_DME) ++ ++static void pxa27x_ep_fifo_flush(struct usb_ep *ep); ++static void nuke(struct pxa27x_ep *, int status); ++static void udc_init_ep(struct pxa27x_udc *dev); ++ ++ ++/* ++ * Endpoint Functions ++ */ ++static void pio_irq_enable(int ep_num) ++{ ++ if (ep_num < 16) ++ UDCICR0 |= 3 << (ep_num * 2); ++ else { ++ ep_num -= 16; ++ UDCICR1 |= 3 << (ep_num * 2); ++ } ++} ++ ++static void pio_irq_disable(int ep_num) ++{ ++ ep_num &= 0xf; ++ if (ep_num < 16) ++ UDCICR0 &= ~(3 << (ep_num * 2)); ++ else { ++ ep_num -= 16; ++ UDCICR1 &= ~(3 << (ep_num * 2)); ++ } ++} ++ ++/* The UDCCR reg contains mask and interrupt status bits, ++ * so using '|=' isn't safe as it may ack an interrupt. ++ */ ++#define UDCCR_MASK_BITS (UDCCR_OEN | UDCCR_UDE) ++ ++static inline void udc_set_mask_UDCCR(int mask) ++{ ++ UDCCR = (UDCCR & UDCCR_MASK_BITS) | (mask & UDCCR_MASK_BITS); ++} ++ ++static inline void udc_clear_mask_UDCCR(int mask) ++{ ++ UDCCR = (UDCCR & UDCCR_MASK_BITS) & ~(mask & UDCCR_MASK_BITS); ++} ++ ++static inline void udc_ack_int_UDCCR(int mask) ++{ ++ /* udccr contains the bits we dont want to change */ ++ __u32 udccr = UDCCR & UDCCR_MASK_BITS; ++ ++ UDCCR = udccr | (mask & ~UDCCR_MASK_BITS); ++} ++ ++/* ++ * Endpoint enable/disable ++ * ++ * Not much to do here as the ep_alloc function sets up most things. Once ++ * enabled, not much of the pxa27x configuration can be changed. ++ * ++ */ ++static int pxa27x_ep_enable(struct usb_ep *_ep, const struct usb_endpoint_descriptor *desc) ++{ ++ struct pxa27x_virt_ep *virt_ep = container_of(_ep, struct pxa27x_virt_ep, usb_ep); ++ struct pxa27x_ep *ep = virt_ep->pxa_ep; ++ struct pxa27x_udc *dev; ++ ++ if (!_ep || !desc || _ep->name == ep0name ++ || desc->bDescriptorType != USB_DT_ENDPOINT ++ || ep->fifo_size < le16_to_cpu(desc->wMaxPacketSize)) { ++ dev_err(ep->dev->dev, "%s, bad ep or descriptor\n", __FUNCTION__); ++ return -EINVAL; ++ } ++ ++ /* xfer types must match, except that interrupt ~= bulk */ ++ if( ep->ep_type != USB_ENDPOINT_XFER_BULK ++ && desc->bmAttributes != USB_ENDPOINT_XFER_INT) { ++ dev_err(ep->dev->dev, "%s, %s type mismatch\n", __FUNCTION__, _ep->name); ++ return -EINVAL; ++ } ++ ++ /* hardware _could_ do smaller, but driver doesn't */ ++ if ((desc->bmAttributes == USB_ENDPOINT_XFER_BULK ++ && le16_to_cpu (desc->wMaxPacketSize) ++ != BULK_FIFO_SIZE) ++ || !desc->wMaxPacketSize) { ++ dev_err(ep->dev->dev, "%s, bad %s maxpacket\n", __FUNCTION__, _ep->name); ++ return -ERANGE; ++ } ++ ++ dev = ep->dev; ++ if (!dev->driver || dev->gadget.speed == USB_SPEED_UNKNOWN) { ++ dev_err(ep->dev->dev, "%s, bogus device state\n", __FUNCTION__); ++ return -ESHUTDOWN; ++ } ++ ++ ep->desc = desc; ++ ep->dma = -1; ++ ep->stopped = 0; ++ ep->pio_irqs = ep->dma_irqs = 0; ++ ep->usb_ep->maxpacket = le16_to_cpu(desc->wMaxPacketSize); ++ ++ /* flush fifo (mostly for OUT buffers) */ ++ pxa27x_ep_fifo_flush(_ep); ++ ++ /* ... reset halt state too, if we could ... */ ++ ++#ifdef USE_DMA ++ /* for (some) bulk and ISO endpoints, try to get a DMA channel and ++ * bind it to the endpoint. otherwise use PIO. ++ */ ++ dev_dbg(ep->dev->dev, "%s: called attributes=%d\n", __FUNCTION__, ep->ep_type); ++ switch (ep->ep_type) { ++ case USB_ENDPOINT_XFER_ISOC: ++ if (le16_to_cpu(desc->wMaxPacketSize) % 32) ++ break; ++ // fall through ++ case USB_ENDPOINT_XFER_BULK: ++ if (!use_dma || !ep->reg_drcmr) ++ break; ++ ep->dma = pxa_request_dma((char *)_ep->name, (le16_to_cpu(desc->wMaxPacketSize) > 64) ++ ? DMA_PRIO_MEDIUM : DMA_PRIO_LOW, dma_nodesc_handler, ep); ++ if (ep->dma >= 0) { ++ *ep->reg_drcmr = DRCMR_MAPVLD | ep->dma; ++ dev_dbg(ep->dev->dev, "%s using dma%d\n", _ep->name, ep->dma); ++ } ++ default: ++ break; ++ } ++#endif ++ DBG(DBG_VERBOSE, "enabled %s\n", _ep->name); ++ return 0; ++} ++ ++static int pxa27x_ep_disable(struct usb_ep *_ep) ++{ ++ struct pxa27x_virt_ep *virt_ep = container_of(_ep, struct pxa27x_virt_ep, usb_ep); ++ struct pxa27x_ep *ep = virt_ep->pxa_ep; ++ unsigned long flags; ++ ++ if (!_ep || !ep->desc) { ++ dev_err(ep->dev->dev, "%s, %s not enabled\n", __FUNCTION__, ++ _ep ? _ep->name : NULL); ++ return -EINVAL; ++ } ++ local_irq_save(flags); ++ nuke(ep, -ESHUTDOWN); ++ ++#ifdef USE_DMA ++ if (ep->dma >= 0) { ++ *ep->reg_drcmr = 0; ++ pxa_free_dma(ep->dma); ++ ep->dma = -1; ++ } ++#endif ++ ++ /* flush fifo (mostly for IN buffers) */ ++ pxa27x_ep_fifo_flush(_ep); ++ ++ ep->desc = 0; ++ ep->stopped = 1; ++ ++ local_irq_restore(flags); ++ DBG(DBG_VERBOSE, "%s disabled\n", _ep->name); ++ return 0; ++} ++ ++ ++ ++/* for the pxa27x, these can just wrap kmalloc/kfree. gadget drivers ++ * must still pass correctly initialized endpoints, since other controller ++ * drivers may care about how it's currently set up (dma issues etc). ++ */ ++ ++/* ++ * pxa27x_ep_alloc_request - allocate a request data structure ++ */ ++static struct usb_request * ++pxa27x_ep_alloc_request(struct usb_ep *_ep, unsigned gfp_flags) ++{ ++ struct pxa27x_request *req; ++ ++ req = kzalloc(sizeof *req, gfp_flags); ++ if (!req) ++ return 0; ++ ++ INIT_LIST_HEAD(&req->queue); ++ return &req->req; ++} ++ ++ ++/* ++ * pxa27x_ep_free_request - deallocate a request data structure ++ */ ++static void ++pxa27x_ep_free_request(struct usb_ep *_ep, struct usb_request *_req) ++{ ++ struct pxa27x_request *req; ++ ++ req = container_of(_req, struct pxa27x_request, req); ++ WARN_ON(!list_empty(&req->queue)); ++ kfree(req); ++} ++ ++ ++/* PXA cache needs flushing with DMA I/O (it's dma-incoherent), but there's ++ * no device-affinity and the heap works perfectly well for i/o buffers. ++ * It wastes much less memory than dma_alloc_coherent() would, and even ++ * prevents cacheline (32 bytes wide) sharing problems. ++ */ ++static void * ++pxa27x_ep_alloc_buffer(struct usb_ep *_ep, unsigned bytes, dma_addr_t *dma, unsigned gfp_flags) ++{ ++ char *retval; ++ ++ retval = kmalloc(bytes, gfp_flags & ~(__GFP_DMA|__GFP_HIGHMEM)); ++ if (retval) ++ *dma = virt_to_bus(retval); ++ return retval; ++} ++ ++static void ++pxa27x_ep_free_buffer(struct usb_ep *_ep, void *buf, dma_addr_t dma, unsigned bytes) ++{ ++ kfree(buf); ++} ++ ++/*-------------------------------------------------------------------------*/ ++ ++/* ++ * done - retire a request; caller blocked irqs ++ */ ++static void done(struct pxa27x_ep *ep, struct pxa27x_request *req, int status) ++{ ++ list_del_init(&req->queue); ++ if (likely (req->req.status == -EINPROGRESS)) ++ req->req.status = status; ++ else ++ status = req->req.status; ++ ++ if (status && status != -ESHUTDOWN) ++ DBG(DBG_VERBOSE, "complete %s req %p stat %d len %u/%u\n", ++ ep->usb_ep->name, &req->req, status, ++ req->req.actual, req->req.length); ++ ++ /* don't modify queue heads during completion callback */ ++ req->req.complete(ep->usb_ep, &req->req); ++} ++ ++ ++static inline void ep0_idle(struct pxa27x_udc *dev) ++{ ++ dev->ep0state = EP0_IDLE; ++} ++ ++static int write_packet(volatile u32 *uddr, struct pxa27x_request *req, unsigned max) ++{ ++ u32 *buf; ++ int length, count, remain; ++ ++ buf = (u32*)(req->req.buf + req->req.actual); ++ prefetch(buf); ++ ++ /* how big will this packet be? */ ++ length = min(req->req.length - req->req.actual, max); ++ req->req.actual += length; ++ ++ remain = length & 0x3; ++ count = length & ~(0x3); ++ ++ //dev_dbg(ep->dev->dev, "Length %d, Remain %d, Count %d\n",length, remain, count); ++ ++ while (likely(count)) { ++ //dev_dbg(ep->dev->dev, "Sending:0x%x\n", *buf); ++ *uddr = *buf++; ++ count -= 4; ++ } ++ ++ if (remain) { ++ volatile u8* reg=(u8*)uddr; ++ char *rd =(u8*)buf; ++ ++ while (remain--) { ++ *reg=*rd++; ++ } ++ } ++ ++ return length; ++} ++ ++/* ++ * write to an IN endpoint fifo, as many packets as possible. ++ * irqs will use this to write the rest later. ++ * caller guarantees at least one packet buffer is ready (or a zlp). ++ */ ++static int ++write_fifo(struct pxa27x_ep *ep, struct pxa27x_request *req) ++{ ++ unsigned max; ++ ++ max = le16_to_cpu(ep->desc->wMaxPacketSize); ++ do { ++ int count, is_last, is_short; ++ ++ //dev_dbg(ep->dev->dev, "write_fifo7 %x\n", *ep->reg_udccsr); ++ ++ if (*ep->reg_udccsr & UDCCSR_PC) { ++ //dev_dbg(ep->dev->dev, "Transmit Complete\n"); ++ *ep->reg_udccsr = UDCCSR_PC | (*ep->reg_udccsr & UDCCSR_MASK); ++ } ++ ++ if (*ep->reg_udccsr & UDCCSR_TRN) { ++ //dev_dbg(ep->dev->dev, "Clearing Underrun\n"); ++ *ep->reg_udccsr = UDCCSR_TRN | (*ep->reg_udccsr & UDCCSR_MASK); ++ } ++ //dev_dbg(ep->dev->dev, "write_fifo8 %x\n", *ep->reg_udccsr); ++ ++ count = write_packet(ep->reg_udcdr, req, max); ++ ++ /* last packet is usually short (or a zlp) */ ++ if (unlikely (count != max)) ++ is_last = is_short = 1; ++ else { ++ if (likely(req->req.length != req->req.actual) ++ || req->req.zero) ++ is_last = 0; ++ else ++ is_last = 1; ++ /* interrupt/iso maxpacket may not fill the fifo */ ++ is_short = unlikely (max < ep->fifo_size); ++ } ++ ++ //dev_dbg(ep->dev->dev, "write_fifo0 %x\n", *ep->reg_udccsr); ++ ++ dev_dbg(ep->dev->dev, "wrote %s count:%d bytes%s%s %d left %p\n", ++ ep->usb_ep->name, count, ++ is_last ? "/L" : "", is_short ? "/S" : "", ++ req->req.length - req->req.actual, &req->req); ++ ++ /* let loose that packet. maybe try writing another one, ++ * double buffering might work. ++ */ ++ ++ if (is_short) ++ *ep->reg_udccsr = UDCCSR_SP | (*ep->reg_udccsr & UDCCSR_MASK); ++ ++ dev_dbg(ep->dev->dev, "write_fifo0.5 %x\n", *ep->reg_udccsr); ++ ++ /* requests complete when all IN data is in the FIFO */ ++ if (is_last) { ++ done(ep, req, 0); ++ if (list_empty(&ep->queue) || unlikely(ep->dma >= 0)) { ++ pio_irq_disable(ep->pxa_ep_num); ++ //dev_dbg(ep->dev->dev, "write_fifo1 %x\n", *ep->reg_udccsr); ++#ifdef USE_DMA ++ /* unaligned data and zlps couldn't use dma */ ++ if (unlikely(!list_empty(&ep->queue))) { ++ req = list_entry(ep->queue.next, ++ struct pxa27x_request, queue); ++ kick_dma(ep,req); ++ return 0; ++ } ++#endif ++ } ++ //dev_dbg(ep->dev->dev, "write_fifo2 %x\n", *ep->reg_udccsr); ++ return 1; ++ } ++ ++ // TODO experiment: how robust can fifo mode tweaking be? ++ // double buffering is off in the default fifo mode, which ++ // prevents TFS from being set here. ++ ++ } while (*ep->reg_udccsr & UDCCSR_FS); ++ //dev_dbg(ep->dev->dev, "write_fifo2 %x\n", *ep->reg_udccsr); ++ return 0; ++} ++ ++/* caller asserts req->pending (ep0 irq status nyet cleared); starts ++ * ep0 data stage. these chips want very simple state transitions. ++ */ ++static inline ++void ep0start(struct pxa27x_udc *dev, u32 flags, const char *tag) ++{ ++ UDCCSR0 = flags|UDCCSR0_SA|UDCCSR0_OPC; ++ UDCISR0 = UDCICR_INT(0, UDC_INT_FIFOERROR | UDC_INT_PACKETCMP); ++ dev->req_pending = 0; ++ DBG(DBG_VERY_NOISY, "%s %s, %02x/%02x\n", ++ __FUNCTION__, tag, UDCCSR0, flags); ++} ++ ++static int ++write_ep0_fifo(struct pxa27x_ep *ep, struct pxa27x_request *req) ++{ ++ unsigned count; ++ int is_short; ++ ++ count = write_packet(&UDCDR0, req, EP0_FIFO_SIZE); ++ ep->dev->stats.write.bytes += count; ++ ++ /* last packet "must be" short (or a zlp) */ ++ is_short = (count != EP0_FIFO_SIZE); ++ ++ DBG(DBG_VERY_NOISY, "ep0in %d bytes %d left %p\n", count, ++ req->req.length - req->req.actual, &req->req); ++ ++ if (unlikely (is_short)) { ++ if (ep->dev->req_pending) ++ ep0start(ep->dev, UDCCSR0_IPR, "short IN"); ++ else ++ UDCCSR0 = UDCCSR0_IPR; ++ ++ count = req->req.length; ++ done(ep, req, 0); ++ ep0_idle(ep->dev); ++#if 0 ++ /* This seems to get rid of lost status irqs in some cases: ++ * host responds quickly, or next request involves config ++ * change automagic, or should have been hidden, or ... ++ * ++ * FIXME get rid of all udelays possible... ++ */ ++ if (count >= EP0_FIFO_SIZE) { ++ count = 100; ++ do { ++ if ((UDCCSR0 & UDCCSR0_OPC) != 0) { ++ /* clear OPC, generate ack */ ++ UDCCSR0 = UDCCSR0_OPC; ++ break; ++ } ++ count--; ++ udelay(1); ++ } while (count); ++ } ++#endif ++ } else if (ep->dev->req_pending) ++ ep0start(ep->dev, 0, "IN"); ++ return is_short; ++} ++ ++ ++/* ++ * read_fifo - unload packet(s) from the fifo we use for usb OUT ++ * transfers and put them into the request. caller should have made ++ * sure there's at least one packet ready. ++ * ++ * returns true if the request completed because of short packet or the ++ * request buffer having filled (and maybe overran till end-of-packet). ++ */ ++static int read_fifo(struct pxa27x_ep *ep, struct pxa27x_request *req) ++{ ++ for (;;) { ++ u32 *buf; ++ int bufferspace, count, is_short; ++ ++ /* make sure there's a packet in the FIFO.*/ ++ if (unlikely ((*ep->reg_udccsr & UDCCSR_PC) == 0)) ++ break; ++ buf =(u32*) (req->req.buf + req->req.actual); ++ prefetchw(buf); ++ bufferspace = req->req.length - req->req.actual; ++ ++ /* read all bytes from this packet */ ++ if (likely (*ep->reg_udccsr & UDCCSR_BNE)) { ++ count = 0x3ff & *ep->reg_udcbcr; ++ req->req.actual += min(count, bufferspace); ++ } else /* zlp */ ++ count = 0; ++ ++ is_short = (count < ep->usb_ep->maxpacket); ++ dev_dbg(ep->dev->dev, "read %s udccsr:%02x, count:%d bytes%s req %p %d/%d\n", ++ ep->usb_ep->name, *ep->reg_udccsr, count, ++ is_short ? "/S" : "", ++ &req->req, req->req.actual, req->req.length); ++ ++ count = min(count, bufferspace); ++ while (likely (count > 0)) { ++ *buf++ = *ep->reg_udcdr; ++ count -= 4; ++ } ++ dev_dbg(ep->dev->dev, "Buf:0x%p\n", req->req.buf); ++ ++ *ep->reg_udccsr = UDCCSR_PC; ++ /* RPC/RSP/RNE could now reflect the other packet buffer */ ++ ++ /* completion */ ++ if (is_short || req->req.actual == req->req.length) { ++ done(ep, req, 0); ++ if (list_empty(&ep->queue)) ++ pio_irq_disable(ep->pxa_ep_num); ++ return 1; ++ } ++ ++ /* finished that packet. the next one may be waiting... */ ++ } ++ return 0; ++} ++ ++/* ++ * special ep0 version of the above. no UBCR0 or double buffering; status ++ * handshaking is magic. most device protocols don't need control-OUT. ++ * CDC vendor commands (and RNDIS), mass storage CB/CBI, and some other ++ * protocols do use them. ++ */ ++static int read_ep0_fifo(struct pxa27x_ep *ep, struct pxa27x_request *req) ++{ ++ u32 *buf, word; ++ unsigned bufferspace; ++ ++ buf = (u32*) (req->req.buf + req->req.actual); ++ bufferspace = req->req.length - req->req.actual; ++ ++ while (UDCCSR0 & UDCCSR0_RNE) { ++ word = UDCDR0; ++ ++ if (unlikely (bufferspace == 0)) { ++ /* this happens when the driver's buffer ++ * is smaller than what the host sent. ++ * discard the extra data. ++ */ ++ if (req->req.status != -EOVERFLOW) ++ dev_info(ep->dev->dev, "%s overflow\n", ep->usb_ep->name); ++ req->req.status = -EOVERFLOW; ++ } else { ++ *buf++ = word; ++ req->req.actual += 4; ++ bufferspace -= 4; ++ } ++ } ++ ++ UDCCSR0 = UDCCSR0_OPC ; ++ ++ /* completion */ ++ if (req->req.actual >= req->req.length) ++ return 1; ++ ++ /* finished that packet. the next one may be waiting... */ ++ return 0; ++} ++ ++#ifdef USE_DMA ++ ++#define MAX_IN_DMA ((DCMD_LENGTH + 1) - BULK_FIFO_SIZE) ++static void kick_dma(struct pxa27x_ep *ep, struct pxa27x_request *req) ++{ ++ u32 dcmd = 0; ++ u32 len = req->req.length; ++ u32 buf = req->req.dma; ++ u32 fifo = io_v2p((u32)ep->reg_udcdr); ++ ++ buf += req->req.actual; ++ len -= req->req.actual; ++ ep->dma_con = 0; ++ ++ DMSG("%s: req:0x%p length:%d, actual:%d dma:%d\n", ++ __FUNCTION__, &req->req, req->req.length, ++ req->req.actual,ep->dma); ++ ++ /* no-descriptor mode can be simple for bulk-in, iso-in, iso-out */ ++ DCSR(ep->dma) = DCSR_NODESC; ++ if (buf & 0x3) ++ DALGN |= 1 << ep->dma; ++ else ++ DALGN &= ~(1 << ep->dma); ++ ++ if (ep->dir_in) { ++ DSADR(ep->dma) = buf; ++ DTADR(ep->dma) = fifo; ++ if (len > MAX_IN_DMA) { ++ len= MAX_IN_DMA; ++ ep->dma_con =1 ; ++ } else if (len >= ep->usb_ep->maxpacket) { ++ if ((ep->dma_con = (len % ep->usb_ep->maxpacket) != 0)) ++ len = ep->usb_ep->maxpacket; ++ } ++ dcmd = len | DCMD_BURST32 | DCMD_WIDTH4 | DCMD_ENDIRQEN ++ | DCMD_FLOWTRG | DCMD_INCSRCADDR; ++ } else { ++ DSADR(ep->dma) = fifo; ++ DTADR(ep->dma) = buf; ++ dcmd = len | DCMD_BURST32 | DCMD_WIDTH4 | DCMD_ENDIRQEN ++ | DCMD_FLOWSRC | DCMD_INCTRGADDR; ++ } ++ *ep->reg_udccsr = UDCCSR_DME; ++ DCMD(ep->dma) = dcmd; ++ DCSR(ep->dma) = DCSR_NODESC | DCSR_EORIRQEN \ ++ | ((ep->dir_in) ? DCSR_STOPIRQEN : 0); ++ *ep->reg_drcmr = ep->dma | DRCMR_MAPVLD; ++ DCSR(ep->dma) |= DCSR_RUN; ++} ++ ++static void cancel_dma(struct pxa27x_ep *ep) ++{ ++ struct pxa27x_request *req; ++ u32 tmp; ++ ++ if (DCSR(ep->dma) == 0 || list_empty(&ep->queue)) ++ return; ++ ++ DMSG("hehe dma:%d,dcsr:0x%x\n", ep->dma, DCSR(ep->dma)); ++ DCSR(ep->dma) = 0; ++ while ((DCSR(ep->dma) & DCSR_STOPSTATE) == 0) ++ cpu_relax(); ++ ++ req = list_entry(ep->queue.next, struct pxa27x_request, queue); ++ tmp = DCMD(ep->dma) & DCMD_LENGTH; ++ req->req.actual = req->req.length - tmp; ++ ++ /* the last tx packet may be incomplete, so flush the fifo. ++ * FIXME correct req.actual if we can ++ */ ++ *ep->reg_udccsr = UDCCSR_FEF; ++} ++ ++static void dma_nodesc_handler(int dmach, void *_ep) ++{ ++ struct pxa27x_ep *ep = _ep; ++ struct pxa27x_request *req, *req_next; ++ u32 dcsr, tmp, completed; ++ ++ local_irq_disable(); ++ ++ req = list_entry(ep->queue.next, struct pxa27x_request, queue); ++ ++ DMSG("%s, buf:0x%p\n",__FUNCTION__, req->req.buf); ++ ++ ep->dma_irqs++; ++ ep->dev->stats.irqs++; ++ ++ completed = 0; ++ ++ dcsr = DCSR(dmach); ++ DCSR(ep->dma) &= ~DCSR_RUN; ++ ++ if (dcsr & DCSR_BUSERR) { ++ DCSR(dmach) = DCSR_BUSERR; ++ dev_err(ep->dev->dev, "DMA Bus Error\n"); ++ req->req.status = -EIO; ++ completed = 1; ++ } else if (dcsr & DCSR_ENDINTR) { ++ DCSR(dmach) = DCSR_ENDINTR; ++ if (ep->dir_in) { ++ tmp = req->req.length - req->req.actual; ++ /* Last packet is a short one*/ ++ if (tmp < ep->usb_ep->maxpacket) { ++ int count = 0; ++ ++ *ep->reg_udccsr = UDCCSR_SP | \ ++ (*ep->reg_udccsr & UDCCSR_MASK); ++ /*Wait for packet out */ ++ while( (count++ < 10000) && \ ++ !(*ep->reg_udccsr & UDCCSR_FS)); ++ if (count >= 10000) ++ DMSG("Failed to send packet\n"); ++ else ++ DMSG("%s: short packet sent len:%d," ++ "length:%d,actual:%d\n", __FUNCTION__, ++ tmp, req->req.length, req->req.actual); ++ req->req.actual = req->req.length; ++ completed = 1; ++ /* There are still packets to transfer */ ++ } else if ( ep->dma_con) { ++ DMSG("%s: more packets,length:%d,actual:%d\n", ++ __FUNCTION__,req->req.length, ++ req->req.actual); ++ req->req.actual += ep->usb_ep->maxpacket; ++ completed = 0; ++ } else { ++ DMSG("%s: no more packets,length:%d," ++ "actual:%d\n", __FUNCTION__, ++ req->req.length, req->req.actual); ++ req->req.actual = req->req.length; ++ completed = 1; ++ } ++ } else { ++ req->req.actual = req->req.length; ++ completed = 1; ++ } ++ } else if (dcsr & DCSR_EORINTR) { //Only happened in OUT DMA ++ int remain,udccsr ; ++ ++ DCSR(dmach) = DCSR_EORINTR; ++ remain = DCMD(dmach) & DCMD_LENGTH; ++ req->req.actual = req->req.length - remain; ++ ++ udccsr = *ep->reg_udccsr; ++ if (udccsr & UDCCSR_SP) { ++ *ep->reg_udccsr = UDCCSR_PC | (udccsr & UDCCSR_MASK); ++ completed = 1; ++ } ++ DMSG("%s: length:%d actual:%d\n", ++ __FUNCTION__, req->req.length, req->req.actual); ++ } else ++ DMSG("%s: Others dma:%d DCSR:0x%x DCMD:0x%x\n", ++ __FUNCTION__, dmach, DCSR(dmach), DCMD(dmach)); ++ ++ if (likely(completed)) { ++ if (req->queue.next != &ep->queue) { ++ req_next = list_entry(req->queue.next, ++ struct pxa27x_request, queue); ++ kick_dma(ep, req_next); ++ } ++ done(ep, req, 0); ++ } else { ++ kick_dma(ep, req); ++ } ++ ++ local_irq_enable(); ++} ++ ++#endif ++/*-------------------------------------------------------------------------*/ ++ ++static int ++pxa27x_ep_queue(struct usb_ep *_ep, struct usb_request *_req, unsigned gfp_flags) ++{ ++ struct pxa27x_virt_ep *virt_ep; ++ struct pxa27x_ep *ep; ++ struct pxa27x_request *req; ++ struct pxa27x_udc *dev; ++ unsigned long flags; ++ ++ virt_ep = container_of(_ep, struct pxa27x_virt_ep, usb_ep); ++ ep = virt_ep->pxa_ep; ++ ++ req = container_of(_req, struct pxa27x_request, req); ++ if (unlikely (!_req || !_req->complete || !_req->buf|| ++ !list_empty(&req->queue))) { ++ DMSG("%s, bad params\n", __FUNCTION__); ++ return -EINVAL; ++ } ++ ++ if (unlikely (!_ep || (!ep->desc && _ep->name != ep0name))) { ++ DMSG("%s, bad ep\n", __FUNCTION__); ++ return -EINVAL; ++ } ++ ++ DMSG("%s, ep point %d is queue\n", __FUNCTION__, ep->ep_num); ++ ++ dev = ep->dev; ++ if (unlikely (!dev->driver ++ || dev->gadget.speed == USB_SPEED_UNKNOWN)) { ++ DMSG("%s, bogus device state\n", __FUNCTION__); ++ return -ESHUTDOWN; ++ } ++ ++ /* iso is always one packet per request, that's the only way ++ * we can report per-packet status. that also helps with dma. ++ */ ++ if (unlikely (ep->ep_type == USB_ENDPOINT_XFER_ISOC ++ && req->req.length > le16_to_cpu ++ (ep->desc->wMaxPacketSize))) ++ return -EMSGSIZE; ++ ++#ifdef USE_DMA ++ // FIXME caller may already have done the dma mapping ++ if (ep->dma >= 0) { ++ _req->dma = dma_map_single(dev->dev, _req->buf, _req->length, ++ (ep->dir_in) ? DMA_TO_DEVICE : DMA_FROM_DEVICE); ++ } ++#endif ++ ++ DBG(DBG_NOISY, "%s queue req %p, len %d buf %p\n", ++ _ep->name, _req, _req->length, _req->buf); ++ ++ local_irq_save(flags); ++ ++ _req->status = -EINPROGRESS; ++ _req->actual = 0; ++ ++ /* kickstart this i/o queue? */ ++ if (list_empty(&ep->queue) && !ep->stopped) { ++ if (ep->desc == 0 /* ep0 */) { ++ unsigned length = _req->length; ++ ++ switch (dev->ep0state) { ++ case EP0_IN_DATA_PHASE: ++ dev->stats.write.ops++; ++ if (write_ep0_fifo(ep, req)) ++ req = 0; ++ break; ++ ++ case EP0_OUT_DATA_PHASE: ++ dev->stats.read.ops++; ++ if (dev->req_pending) ++ ep0start(dev, UDCCSR0_IPR, "OUT"); ++ if (length == 0 || ((UDCCSR0 & UDCCSR0_RNE) != 0 ++ && read_ep0_fifo(ep, req))) { ++ ep0_idle(dev); ++ done(ep, req, 0); ++ req = 0; ++ } ++ break; ++ case EP0_NO_ACTION: ++ ep0_idle(dev); ++ req=0; ++ break; ++ default: ++ DMSG("ep0 i/o, odd state %d\n", dev->ep0state); ++ local_irq_restore (flags); ++ return -EL2HLT; ++ } ++#ifdef USE_DMA ++ /* either start dma or prime pio pump */ ++ } else if (ep->dma >= 0) { ++ kick_dma(ep, req); ++#endif ++ /* can the FIFO can satisfy the request immediately? */ ++ } else if (ep->dir_in && (*ep->reg_udccsr & UDCCSR_FS) != 0 ++ && write_fifo(ep, req)) { ++ req = 0; ++ } else if ((*ep->reg_udccsr & UDCCSR_FS) != 0 ++ && read_fifo(ep, req)) { ++ req = 0; ++ } ++ DMSG("req:%p,ep->desc:%p,ep->dma:%d\n", req, ep->desc, ep->dma); ++ if (likely (req && ep->desc) && ep->dma < 0) ++ pio_irq_enable(ep->pxa_ep_num); ++ } ++ ++ /* pio or dma irq handler advances the queue. */ ++ if (likely (req != 0)) ++ list_add_tail(&req->queue, &ep->queue); ++ local_irq_restore(flags); ++ ++ return 0; ++} ++ ++ ++/* ++ * nuke - dequeue ALL requests ++ */ ++static void nuke(struct pxa27x_ep *ep, int status) ++{ ++ struct pxa27x_request *req; ++ ++ /* called with irqs blocked */ ++#ifdef USE_DMA ++ if (ep->dma >= 0 && !ep->stopped) ++ cancel_dma(ep); ++#endif ++ while (!list_empty(&ep->queue)) { ++ req = list_entry(ep->queue.next, struct pxa27x_request, queue); ++ done(ep, req, status); ++ } ++ if (ep->desc) ++ pio_irq_disable(ep->pxa_ep_num); ++} ++ ++ ++/* dequeue JUST ONE request */ ++static int pxa27x_ep_dequeue(struct usb_ep *_ep, struct usb_request *_req) ++{ ++ struct pxa27x_virt_ep *virt_ep = container_of(_ep, struct pxa27x_virt_ep, usb_ep); ++ struct pxa27x_ep *ep = virt_ep->pxa_ep; ++ struct pxa27x_request *req; ++ unsigned long flags; ++ ++ if (!_ep || _ep->name == ep0name) ++ return -EINVAL; ++ ++ local_irq_save(flags); ++ ++ /* make sure it's actually queued on this endpoint */ ++ list_for_each_entry(req, &ep->queue, queue) { ++ if (&req->req == _req) ++ break; ++ } ++ if (&req->req != _req) { ++ local_irq_restore(flags); ++ return -EINVAL; ++ } ++ ++#ifdef USE_DMA ++ if (ep->dma >= 0 && ep->queue.next == &req->queue && !ep->stopped) { ++ cancel_dma(ep); ++ done(ep, req, -ECONNRESET); ++ /* restart i/o */ ++ if (!list_empty(&ep->queue)) { ++ req = list_entry(ep->queue.next, ++ struct pxa27x_request, queue); ++ kick_dma(ep, req); ++ } ++ } else ++#endif ++ done(ep, req, -ECONNRESET); ++ ++ local_irq_restore(flags); ++ return 0; ++} ++ ++/*-------------------------------------------------------------------------*/ ++ ++static int pxa27x_ep_set_halt(struct usb_ep *_ep, int value) ++{ ++ struct pxa27x_virt_ep *virt_ep = container_of(_ep, struct pxa27x_virt_ep, usb_ep); ++ struct pxa27x_ep *ep = virt_ep->pxa_ep; ++ unsigned long flags; ++ ++ DMSG("%s is called\n", __FUNCTION__); ++ if (unlikely (!_ep || (!ep->desc && _ep->name != ep0name)) ++ || ep->ep_type == USB_ENDPOINT_XFER_ISOC) { ++ DMSG("%s, bad ep\n", __FUNCTION__); ++ return -EINVAL; ++ } ++ if (value == 0) { ++ /* this path (reset toggle+halt) is needed to implement ++ * SET_INTERFACE on normal hardware. but it can't be ++ * done from software on the PXA UDC, and the hardware ++ * forgets to do it as part of SET_INTERFACE automagic. ++ */ ++ DMSG("only host can clear %s halt\n", _ep->name); ++ return -EROFS; ++ } ++ ++ local_irq_save(flags); ++ ++ if (ep->dir_in && ((*ep->reg_udccsr & UDCCSR_FS) == 0 ++ || !list_empty(&ep->queue))) { ++ local_irq_restore(flags); ++ return -EAGAIN; ++ } ++ ++ /* FST bit is the same for control, bulk in, bulk out, interrupt in */ ++ *ep->reg_udccsr = UDCCSR_FST|UDCCSR_FEF; ++ ++ /* ep0 needs special care */ ++ if (!ep->desc) { ++ start_watchdog(ep->dev); ++ ep->dev->req_pending = 0; ++ ep->dev->ep0state = EP0_STALL; ++ ++ /* and bulk/intr endpoints like dropping stalls too */ ++ } else { ++ unsigned i; ++ for (i = 0; i < 1000; i += 20) { ++ if (*ep->reg_udccsr & UDCCSR_SST) ++ break; ++ udelay(20); ++ } ++ } ++ local_irq_restore(flags); ++ ++ DBG(DBG_VERBOSE, "%s halt\n", _ep->name); ++ return 0; ++} ++ ++static int pxa27x_ep_fifo_status(struct usb_ep *_ep) ++{ ++ struct pxa27x_virt_ep *virt_ep = container_of(_ep, struct pxa27x_virt_ep, usb_ep); ++ struct pxa27x_ep *ep = virt_ep->pxa_ep; ++ ++ if (!_ep) { ++ DMSG("%s, bad ep\n", __FUNCTION__); ++ return -ENODEV; ++ } ++ /* pxa can't report unclaimed bytes from IN fifos */ ++ if (ep->dir_in) ++ return -EOPNOTSUPP; ++ if (ep->dev->gadget.speed == USB_SPEED_UNKNOWN ++ || (*ep->reg_udccsr & UDCCSR_FS) == 0) ++ return 0; ++ else ++ return (*ep->reg_udcbcr & 0xfff) + 1; ++} ++ ++static void pxa27x_ep_fifo_flush(struct usb_ep *_ep) ++{ ++ struct pxa27x_virt_ep *virt_ep = container_of(_ep, struct pxa27x_virt_ep, usb_ep); ++ struct pxa27x_ep *ep = virt_ep->pxa_ep; ++ ++ DMSG("pxa27x_ep_fifo_flush\n"); ++ ++ if (!_ep || _ep->name == ep0name || !list_empty(&ep->queue)) { ++ DMSG("%s, bad ep\n", __FUNCTION__); ++ return; ++ } ++ ++ /* toggle and halt bits stay unchanged */ ++ ++ /* for OUT, just read and discard the FIFO contents. */ ++ if (!ep->dir_in) { ++ while (((*ep->reg_udccsr) & UDCCSR_BNE) != 0) ++ (void) *ep->reg_udcdr; ++ return; ++ } ++ ++ /* most IN status is the same, but ISO can't stall */ ++ *ep->reg_udccsr = UDCCSR_PC|UDCCSR_FST|UDCCSR_TRN ++ | (ep->ep_type == USB_ENDPOINT_XFER_ISOC) ++ ? 0 : UDCCSR_SST; ++} ++ ++ ++static struct usb_ep_ops pxa27x_ep_ops = { ++ .enable = pxa27x_ep_enable, ++ .disable = pxa27x_ep_disable, ++ ++ .alloc_request = pxa27x_ep_alloc_request, ++ .free_request = pxa27x_ep_free_request, ++ ++ .alloc_buffer = pxa27x_ep_alloc_buffer, ++ .free_buffer = pxa27x_ep_free_buffer, ++ ++ .queue = pxa27x_ep_queue, ++ .dequeue = pxa27x_ep_dequeue, ++ ++ .set_halt = pxa27x_ep_set_halt, ++ .fifo_status = pxa27x_ep_fifo_status, ++ .fifo_flush = pxa27x_ep_fifo_flush, ++}; ++ ++ ++/* --------------------------------------------------------------------------- ++ * device-scoped parts of the api to the usb controller hardware ++ * --------------------------------------------------------------------------- ++ */ ++ ++static inline unsigned int validate_fifo_size(u8 bmAttributes) ++{ ++ switch (bmAttributes & USB_ENDPOINT_XFERTYPE_MASK) { ++ case USB_ENDPOINT_XFER_CONTROL: ++ return EP0_FIFO_SIZE; ++ break; ++ case USB_ENDPOINT_XFER_ISOC: ++ return ISO_FIFO_SIZE; ++ break; ++ case USB_ENDPOINT_XFER_BULK: ++ return BULK_FIFO_SIZE; ++ break; ++ case USB_ENDPOINT_XFER_INT: ++ return INT_FIFO_SIZE; ++ break; ++ default: ++ break; ++ } ++} ++ ++static void pxa27x_ep_free(struct usb_gadget *gadget, struct usb_ep *_ep) ++{ ++ struct pxa27x_udc *dev = the_controller; ++ struct pxa27x_virt_ep *virt_ep; ++ int i; ++ ++ virt_ep = container_of(_ep, struct pxa27x_virt_ep, usb_ep); ++ ++ for (i = 1; i < UDC_EP_NUM; i++) { ++ if (dev->ep[i].usb_ep == &virt_ep->usb_ep) { ++ if (dev->ep[i].desc) { ++ virt_ep->pxa_ep = &dev->ep[i]; ++ pxa27x_ep_disable(&virt_ep->usb_ep); ++ } ++ dev->ep[i].usb_ep = NULL; ++ } ++ } ++ ++ if (!list_empty(&virt_ep->usb_ep.ep_list)) ++ list_del_init(&virt_ep->usb_ep.ep_list); ++ ++ kfree(virt_ep->usb_ep.name); ++ kfree(virt_ep); ++} ++ ++static void pxa27x_ep_freeall(struct usb_gadget *gadget) ++{ ++ struct pxa27x_udc *dev = the_controller; ++ int i; ++ ++ for (i = 1; i < UDC_EP_NUM; i++) { ++ if(dev->ep[i].usb_ep) ++ pxa27x_ep_free(gadget, dev->ep[i].usb_ep); ++ } ++} ++ ++#define NAME_SIZE 18 ++ ++static int pxa27x_find_free_ep(struct pxa27x_udc *dev) ++{ ++ int i; ++ for (i = 1; i < UDC_EP_NUM; i++) { ++ if(!dev->ep[i].assigned) ++ return i; ++ } ++ return -1; ++} ++ ++/* ++ * Endpoint Allocation/Configuration ++ * ++ * pxa27x endpoint configuration is fixed when the device is enabled. Any pxa ++ * endpoint is only active in one configuration, interface and alternate ++ * interface combination so to support gadget drivers, we map one usb_ep to ++ * one of several pxa ep's. One pxa endpoint is assigned per configuration ++ * combination. ++ */ ++static struct usb_ep* pxa27x_ep_alloc(struct usb_gadget *gadget, struct usb_endpoint_descriptor *desc, ++ struct usb_endpoint_config *epconfig, int configs) ++{ ++ struct pxa27x_udc *dev = the_controller; ++ struct pxa27x_virt_ep *virt_ep; ++ unsigned int i, fifo_size; ++ char *name; ++ ++ if (unlikely(configs < 1)) { ++ dev_err(dev->dev, "%s: Error in config data\n", __FUNCTION__); ++ return NULL; ++ } ++ ++ virt_ep = kmalloc(sizeof(struct pxa27x_virt_ep), GFP_KERNEL); ++ name = kmalloc(NAME_SIZE, GFP_KERNEL); ++ if (!virt_ep || !name) { ++ dev_err(dev->dev, "%s: -ENOMEM\n", __FUNCTION__); ++ kfree(name); ++ kfree(virt_ep); ++ return NULL; ++ } ++ ++ if (!(desc->wMaxPacketSize)) { ++ fifo_size = validate_fifo_size(desc->bmAttributes); ++ desc->wMaxPacketSize = fifo_size; ++ } else { ++ fifo_size = desc->wMaxPacketSize; ++ } ++ ++ DMSG("pxa27x_ep_alloc: bLength: %d, bDescriptorType: %x, bEndpointAddress: %x,\n" ++ " bmAttributes: %x, wMaxPacketSize: %d\n", desc->bLength, ++ desc->bDescriptorType, desc->bEndpointAddress, desc->bmAttributes, ++ desc->wMaxPacketSize); ++ ++ if (!(desc->bEndpointAddress & 0xF)) ++ desc->bEndpointAddress |= dev->ep_num; ++ ++ for (i = 0; i < configs; i++) ++ { ++ struct pxa27x_ep *pxa_ep; ++ int j; ++ ++ DMSG("pxa27x_ep_alloc: config: %d, interface: %d, altinterface: %x,\n", ++ epconfig->config, epconfig->interface, epconfig->altinterface); ++ ++ j = pxa27x_find_free_ep(dev); ++ ++ if (unlikely(j < 0)) { ++ dev_err(dev->dev, "pxa27x_ep_alloc: Failed to find a spare endpoint\n"); ++ pxa27x_ep_free(gadget, &virt_ep->usb_ep); ++ return NULL; ++ } ++ ++ pxa_ep = &dev->ep[j]; ++ ++ if (i == 0) ++ virt_ep->pxa_ep = pxa_ep; ++ ++ pxa_ep->assigned = 1; ++ pxa_ep->ep_num = dev->ep_num; ++ pxa_ep->pxa_ep_num = j; ++ pxa_ep->usb_ep = &virt_ep->usb_ep; ++ pxa_ep->dev = dev; ++ pxa_ep->desc = desc; ++ pxa_ep->pio_irqs = pxa_ep->dma_irqs = 0; ++ pxa_ep->dma = -1; ++ ++ pxa_ep->fifo_size = fifo_size; ++ pxa_ep->dir_in = (desc->bEndpointAddress & USB_DIR_IN) ? 1 : 0; ++ pxa_ep->ep_type = desc->bmAttributes & USB_ENDPOINT_XFERTYPE_MASK; ++ pxa_ep->stopped = 1; ++ pxa_ep->dma_con = 0; ++ pxa_ep->config = epconfig->config; ++ pxa_ep->interface = epconfig->interface; ++ pxa_ep->aisn = epconfig->altinterface; ++ ++ pxa_ep->reg_udccsr = &UDCCSR0 + j; ++ pxa_ep->reg_udcbcr = &UDCBCR0 + j; ++ pxa_ep->reg_udcdr = &UDCDR0 + j ; ++ pxa_ep->reg_udccr = &UDCCRA - 1 + j; ++#ifdef USE_DMA ++ pxa_ep->reg_drcmr = &DRCMR24 + j; ++#endif ++ ++ /* Configure UDCCR */ ++ *pxa_ep->reg_udccr = ((pxa_ep->config << UDCCONR_CN_S) & UDCCONR_CN) ++ | ((pxa_ep->interface << UDCCONR_IN_S) & UDCCONR_IN) ++ | ((pxa_ep->aisn << UDCCONR_AISN_S) & UDCCONR_AISN) ++ | ((dev->ep_num << UDCCONR_EN_S) & UDCCONR_EN) ++ | ((pxa_ep->ep_type << UDCCONR_ET_S) & UDCCONR_ET) ++ | ((pxa_ep->dir_in) ? UDCCONR_ED : 0) ++ | ((min(pxa_ep->fifo_size, (unsigned)desc->wMaxPacketSize) << UDCCONR_MPS_S ) & UDCCONR_MPS) ++ | UDCCONR_EE; ++// | UDCCONR_DE | UDCCONR_EE; ++ ++ ++ ++#ifdef USE_DMA ++ /* Only BULK use DMA */ ++ if ((pxa_ep->ep_type & USB_ENDPOINT_XFERTYPE_MASK)\ ++ == USB_ENDPOINT_XFER_BULK) ++ *pxa_ep->reg_udccsr = UDCCSR_DME; ++#endif ++ ++ DMSG("UDCCR: 0x%p is 0x%x\n", pxa_ep->reg_udccr,*pxa_ep->reg_udccr); ++ ++ epconfig++; ++ } ++ ++ /* Fill ep name*/ ++ switch (desc->bmAttributes & USB_ENDPOINT_XFERTYPE_MASK) { ++ case USB_ENDPOINT_XFER_BULK: ++ sprintf(name, "ep%d%s-bulk", dev->ep_num, ++ ((desc->bEndpointAddress & USB_DIR_IN) ? "in":"out")); ++ break; ++ case USB_ENDPOINT_XFER_INT: ++ sprintf(name, "ep%d%s-intr", dev->ep_num, ++ ((desc->bEndpointAddress & USB_DIR_IN) ? "in":"out")); ++ break; ++ default: ++ sprintf(name, "ep%d%s", dev->ep_num, ++ ((desc->bEndpointAddress & USB_DIR_IN) ? "in":"out")); ++ break; ++ } ++ ++ virt_ep->desc = desc; ++ virt_ep->usb_ep.name = name; ++ virt_ep->usb_ep.ops = &pxa27x_ep_ops; ++ virt_ep->usb_ep.maxpacket = min((ushort)fifo_size, desc->wMaxPacketSize); ++ ++ list_add_tail(&virt_ep->usb_ep.ep_list, &gadget->ep_list); ++ ++ dev->ep_num++; ++ return &virt_ep->usb_ep; ++} ++ ++static int pxa27x_udc_get_frame(struct usb_gadget *_gadget) ++{ ++ return (UDCFNR & 0x7FF); ++} ++ ++static int pxa27x_udc_wakeup(struct usb_gadget *_gadget) ++{ ++ /* host may not have enabled remote wakeup */ ++ if ((UDCCR & UDCCR_DWRE) == 0) ++ return -EHOSTUNREACH; ++ udc_set_mask_UDCCR(UDCCR_UDR); ++ return 0; ++} ++ ++static const struct usb_gadget_ops pxa27x_udc_ops = { ++ .ep_alloc = pxa27x_ep_alloc, ++ .get_frame = pxa27x_udc_get_frame, ++ .wakeup = pxa27x_udc_wakeup, ++ // current versions must always be self-powered ++}; ++ ++ ++/*-------------------------------------------------------------------------*/ ++ ++#ifdef UDC_PROC_FILE ++ ++static const char proc_node_name [] = "driver/udc"; ++ ++static int ++udc_proc_read(char *page, char **start, off_t off, int count, ++ int *eof, void *_dev) ++{ ++ char *buf = page; ++ struct pxa27x_udc *dev = _dev; ++ char *next = buf; ++ unsigned size = count; ++ unsigned long flags; ++ int i, t; ++ u32 tmp; ++ ++ if (off != 0) ++ return 0; ++ ++ local_irq_save(flags); ++ ++ /* basic device status */ ++ t = scnprintf(next, size, DRIVER_DESC "\n" ++ "%s version: %s\nGadget driver: %s\n", ++ driver_name, DRIVER_VERSION DMASTR, ++ dev->driver ? dev->driver->driver.name : "(none)"); ++ size -= t; ++ next += t; ++ ++ /* registers for device and ep0 */ ++ t = scnprintf(next, size, ++ "uicr %02X.%02X, usir %02X.%02x, ufnr %02X\n", ++ UDCICR1, UDCICR0, UDCISR1, UDCISR0, UDCFNR); ++ size -= t; ++ next += t; ++ ++ tmp = UDCCR; ++ t = scnprintf(next, size,"udccr %02X =%s%s%s%s%s%s%s%s%s%s, con=%d,inter=%d,altinter=%d\n", tmp, ++ (tmp & UDCCR_OEN) ? " oen":"", ++ (tmp & UDCCR_AALTHNP) ? " aalthnp":"", ++ (tmp & UDCCR_AHNP) ? " rem" : "", ++ (tmp & UDCCR_BHNP) ? " rstir" : "", ++ (tmp & UDCCR_DWRE) ? " dwre" : "", ++ (tmp & UDCCR_SMAC) ? " smac" : "", ++ (tmp & UDCCR_EMCE) ? " emce" : "", ++ (tmp & UDCCR_UDR) ? " udr" : "", ++ (tmp & UDCCR_UDA) ? " uda" : "", ++ (tmp & UDCCR_UDE) ? " ude" : "", ++ (tmp & UDCCR_ACN) >> UDCCR_ACN_S, ++ (tmp & UDCCR_AIN) >> UDCCR_AIN_S, ++ (tmp & UDCCR_AAISN)>> UDCCR_AAISN_S ); ++ ++ size -= t; ++ next += t; ++ ++ tmp = UDCCSR0; ++ t = scnprintf(next, size, ++ "udccsr0 %02X =%s%s%s%s%s%s%s\n", tmp, ++ (tmp & UDCCSR0_SA) ? " sa" : "", ++ (tmp & UDCCSR0_RNE) ? " rne" : "", ++ (tmp & UDCCSR0_FST) ? " fst" : "", ++ (tmp & UDCCSR0_SST) ? " sst" : "", ++ (tmp & UDCCSR0_DME) ? " dme" : "", ++ (tmp & UDCCSR0_IPR) ? " ipr" : "", ++ (tmp & UDCCSR0_OPC) ? " opc" : ""); ++ size -= t; ++ next += t; ++ ++ if (!dev->driver) ++ goto done; ++ ++ t = scnprintf(next, size, "ep0 IN %lu/%lu, OUT %lu/%lu\nirqs %lu\n\n", ++ dev->stats.write.bytes, dev->stats.write.ops, ++ dev->stats.read.bytes, dev->stats.read.ops, ++ dev->stats.irqs); ++ size -= t; ++ next += t; ++ ++ /* dump endpoint queues */ ++ for (i = 0; i < UDC_EP_NUM; i++) { ++ struct pxa27x_ep *ep = &dev->ep [i]; ++ struct pxa27x_request *req; ++ int t; ++ ++ if (i != 0) { ++ const struct usb_endpoint_descriptor *d; ++ ++ d = ep->desc; ++ if (!d) ++ continue; ++ tmp = *dev->ep [i].reg_udccsr; ++ t = scnprintf(next, size, ++ "%d max %d %s udccs %02x udccr:0x%x\n", ++ i, le16_to_cpu (d->wMaxPacketSize), ++ (ep->dma >= 0) ? "dma" : "pio", tmp, ++ *dev->ep[i].reg_udccr); ++ /* TODO translate all five groups of udccs bits! */ ++ ++ } else /* ep0 should only have one transfer queued */ ++ t = scnprintf(next, size, "ep0 max 16 pio irqs %lu\n", ++ ep->pio_irqs); ++ if (t <= 0 || t > size) ++ goto done; ++ size -= t; ++ next += t; ++ ++ if (list_empty(&ep->queue)) { ++ t = scnprintf(next, size, "\t(nothing queued)\n"); ++ if (t <= 0 || t > size) ++ goto done; ++ size -= t; ++ next += t; ++ continue; ++ } ++ list_for_each_entry(req, &ep->queue, queue) { ++#ifdef USE_DMA ++ if (ep->dma >= 0 && req->queue.prev == &ep->queue) ++ t = scnprintf(next, size, "\treq %p len %d/%d " ++ "buf %p (dma%d dcmd %08x)\n", ++ &req->req, req->req.actual, ++ req->req.length, req->req.buf, ++ ep->dma, DCMD(ep->dma) ++ /* low 13 bits == bytes-to-go */); ++ else ++#endif ++ t = scnprintf(next, size, ++ "\treq %p len %d/%d buf %p\n", ++ &req->req, req->req.actual, ++ req->req.length, req->req.buf); ++ if (t <= 0 || t > size) ++ goto done; ++ size -= t; ++ next += t; ++ } ++ } ++ ++done: ++ local_irq_restore(flags); ++ *eof = 1; ++ return count - size; ++} ++ ++#define create_proc_files() \ ++ create_proc_read_entry(proc_node_name, 0, NULL, udc_proc_read, dev) ++#define remove_proc_files() \ ++ remove_proc_entry(proc_node_name, NULL) ++ ++#else /* !UDC_PROC_FILE */ ++#define create_proc_files() do {} while (0) ++#define remove_proc_files() do {} while (0) ++ ++#endif /* UDC_PROC_FILE */ ++ ++/* "function" sysfs attribute */ ++static ssize_t show_function(struct device *_dev, struct device_attribute *attr, char *buf) ++{ ++ struct pxa27x_udc *dev = dev_get_drvdata(_dev); ++ ++ if (!dev->driver || !dev->driver->function ++ || strlen(dev->driver->function) > PAGE_SIZE) ++ return 0; ++ return scnprintf(buf, PAGE_SIZE, "%s\n", dev->driver->function); ++} ++static DEVICE_ATTR(function, S_IRUGO, show_function, NULL); ++ ++/*-------------------------------------------------------------------------*/ ++ ++/* ++ * udc_disable - disable USB device controller ++ */ ++static void udc_disable(struct pxa27x_udc *dev) ++{ ++ UDCICR0 = UDCICR1 = 0x00000000; ++ ++ udc_clear_mask_UDCCR(UDCCR_UDE); ++ ++ /* Disable clock for USB device */ ++ pxa_set_cken(CKEN11_USB, 0); ++ ++ ep0_idle(dev); ++ dev->gadget.speed = USB_SPEED_UNKNOWN; ++ dev->mach->udc_command(PXA2XX_UDC_CMD_DISCONNECT); ++} ++ ++ ++/* ++ * udc_reinit - initialize software state ++ */ ++static void udc_reinit(struct pxa27x_udc *dev) ++{ ++ u32 i; ++ ++ dev->ep0state = EP0_IDLE; ++ ++ /* basic endpoint records init */ ++ for (i = 0; i < UDC_EP_NUM; i++) { ++ struct pxa27x_ep *ep = &dev->ep[i]; ++ ++ ep->stopped = 0; ++ ep->pio_irqs = ep->dma_irqs = 0; ++ } ++ dev->configuration = 0; ++ dev->interface = 0; ++ dev->alternate = 0; ++ /* the rest was statically initialized, and is read-only */ ++} ++ ++/* until it's enabled, this UDC should be completely invisible ++ * to any USB host. ++ */ ++static void udc_enable(struct pxa27x_udc *dev) ++{ ++ udc_clear_mask_UDCCR(UDCCR_UDE); ++ ++ /* Enable clock for USB device */ ++ pxa_set_cken(CKEN11_USB, 1); ++ ++ UDCICR0 = UDCICR1 = 0; ++ ++ ep0_idle(dev); ++ dev->gadget.speed = USB_SPEED_FULL; ++ dev->stats.irqs = 0; ++ ++ udc_set_mask_UDCCR(UDCCR_UDE); ++ udelay(2); ++ if (UDCCR & UDCCR_EMCE) ++ dev_err(dev->dev, "There are error in configuration, udc disabled\n"); ++ ++ /* caller must be able to sleep in order to cope ++ * with startup transients. ++ */ ++ msleep(100); ++ ++ /* enable suspend/resume and reset irqs */ ++ UDCICR1 = UDCICR1_IECC | UDCICR1_IERU | UDCICR1_IESU | UDCICR1_IERS; ++ ++ /* enable ep0 irqs */ ++ UDCICR0 = UDCICR_INT(0,UDCICR_INT_MASK); ++ ++ DMSG("Connecting\n"); ++ /* RPFIXME */ ++ UP2OCR = UP2OCR_HXOE | UP2OCR_DPPUE | UP2OCR_DPPUBE; ++ //dev->mach->udc_command(PXA2XX_UDC_CMD_CONNECT); ++} ++ ++ ++/* when a driver is successfully registered, it will receive ++ * control requests including set_configuration(), which enables ++ * non-control requests. then usb traffic follows until a ++ * disconnect is reported. then a host may connect again, or ++ * the driver might get unbound. ++ */ ++int usb_gadget_register_driver(struct usb_gadget_driver *driver) ++{ ++ struct pxa27x_udc *dev = the_controller; ++ int retval; ++ ++ if (!driver || driver->speed != USB_SPEED_FULL || !driver->bind ++ || !driver->unbind || !driver->disconnect || !driver->setup) ++ return -EINVAL; ++ if (!dev) ++ return -ENODEV; ++ if (dev->driver) ++ return -EBUSY; ++ ++ udc_disable(dev); ++ udc_init_ep(dev); ++ udc_reinit(dev); ++ ++ /* first hook up the driver ... */ ++ dev->driver = driver; ++ dev->gadget.dev.driver = &driver->driver; ++ dev->ep_num = 1; ++ ++ retval = device_add(&dev->gadget.dev); ++ if (retval) { ++ DMSG("device_add error %d\n", retval); ++ goto add_fail; ++ } ++ retval = driver->bind(&dev->gadget); ++ if (retval) { ++ DMSG("bind to driver %s --> error %d\n", ++ driver->driver.name, retval); ++ goto bind_fail; ++ } ++ retval = device_create_file(dev->dev, &dev_attr_function); ++ if (retval) { ++ DMSG("device_create_file failed: %d\n", retval); ++ goto create_file_fail; ++ } ++ ++ /* ... then enable host detection and ep0; and we're ready ++ * for set_configuration as well as eventual disconnect. ++ * NOTE: this shouldn't power up until later. ++ */ ++ DMSG("registered gadget driver '%s'\n", driver->driver.name); ++ udc_enable(dev); ++ dump_state(dev); ++ return 0; ++ ++create_file_fail: ++ driver->unbind(&dev->gadget); ++bind_fail: ++ device_del(&dev->gadget.dev); ++add_fail: ++ dev->driver = 0; ++ dev->gadget.dev.driver = 0; ++ return retval; ++} ++EXPORT_SYMBOL(usb_gadget_register_driver); ++ ++static void ++stop_activity(struct pxa27x_udc *dev, struct usb_gadget_driver *driver) ++{ ++ int i; ++ ++ DMSG("Trace path 1\n"); ++ /* don't disconnect drivers more than once */ ++ if (dev->gadget.speed == USB_SPEED_UNKNOWN) ++ driver = 0; ++ dev->gadget.speed = USB_SPEED_UNKNOWN; ++ ++ /* prevent new request submissions, kill any outstanding requests */ ++ for (i = 0; i < UDC_EP_NUM; i++) { ++ struct pxa27x_ep *ep = &dev->ep[i]; ++ ++ ep->stopped = 1; ++ nuke(ep, -ESHUTDOWN); ++ } ++ del_timer_sync(&dev->timer); ++ ++ /* report disconnect; the driver is already quiesced */ ++ if (driver) ++ driver->disconnect(&dev->gadget); ++ ++ /* re-init driver-visible data structures */ ++ udc_reinit(dev); ++} ++ ++int usb_gadget_unregister_driver(struct usb_gadget_driver *driver) ++{ ++ struct pxa27x_udc *dev = the_controller; ++ ++ if (!dev) ++ return -ENODEV; ++ if (!driver || driver != dev->driver) ++ return -EINVAL; ++ ++ local_irq_disable(); ++ udc_disable(dev); ++ stop_activity(dev, driver); ++ local_irq_enable(); ++ ++ driver->unbind(&dev->gadget); ++ pxa27x_ep_freeall(&dev->gadget); ++ dev->driver = 0; ++ ++ device_del(&dev->gadget.dev); ++ device_remove_file(dev->dev, &dev_attr_function); ++ ++ DMSG("unregistered gadget driver '%s'\n", driver->driver.name); ++ dump_state(dev); ++ return 0; ++} ++EXPORT_SYMBOL(usb_gadget_unregister_driver); ++ ++ ++/*-------------------------------------------------------------------------*/ ++ ++static inline void clear_ep_state(struct pxa27x_udc *dev) ++{ ++ unsigned i; ++ ++ /* hardware SET_{CONFIGURATION,INTERFACE} automagic resets endpoint ++ * fifos, and pending transactions mustn't be continued in any case. ++ */ ++ for (i = 1; i < UDC_EP_NUM; i++) ++ nuke(&dev->ep[i], -ECONNABORTED); ++} ++ ++static void udc_watchdog(unsigned long _dev) ++{ ++ struct pxa27x_udc *dev = (void *)_dev; ++ ++ local_irq_disable(); ++ if (dev->ep0state == EP0_STALL ++ && (UDCCSR0 & UDCCSR0_FST) == 0 ++ && (UDCCSR0 & UDCCSR0_SST) == 0) { ++ UDCCSR0 = UDCCSR0_FST|UDCCSR0_FTF; ++ DBG(DBG_VERBOSE, "ep0 re-stall\n"); ++ start_watchdog(dev); ++ } ++ local_irq_enable(); ++} ++ ++static void handle_ep0(struct pxa27x_udc *dev) ++{ ++ u32 udccsr0 = UDCCSR0; ++ struct pxa27x_ep *ep = &dev->ep[0]; ++ struct pxa27x_request *req; ++ union { ++ struct usb_ctrlrequest r; ++ u8 raw[8]; ++ u32 word[2]; ++ } u; ++ ++ if (list_empty(&ep->queue)) ++ req = 0; ++ else ++ req = list_entry(ep->queue.next, struct pxa27x_request, queue); ++ ++ /* clear stall status */ ++ if (udccsr0 & UDCCSR0_SST) { ++ nuke(ep, -EPIPE); ++ UDCCSR0 = UDCCSR0_SST; ++ del_timer(&dev->timer); ++ ep0_idle(dev); ++ } ++ ++ /* previous request unfinished? non-error iff back-to-back ... */ ++ if ((udccsr0 & UDCCSR0_SA) != 0 && dev->ep0state != EP0_IDLE) { ++ nuke(ep, 0); ++ del_timer(&dev->timer); ++ ep0_idle(dev); ++ } ++ ++ switch (dev->ep0state) { ++ case EP0_NO_ACTION: ++ dev_info(dev->dev, "%s: Busy\n", __FUNCTION__); ++ /*Fall through */ ++ case EP0_IDLE: ++ /* late-breaking status? */ ++ udccsr0 = UDCCSR0; ++ ++ /* start control request? */ ++ if (likely((udccsr0 & (UDCCSR0_OPC|UDCCSR0_SA|UDCCSR0_RNE)) ++ == (UDCCSR0_OPC|UDCCSR0_SA|UDCCSR0_RNE))) { ++ int i; ++ ++ nuke(ep, -EPROTO); ++ /* read SETUP packet */ ++ for (i = 0; i < 2; i++) { ++ if (unlikely(!(UDCCSR0 & UDCCSR0_RNE))) { ++bad_setup: ++ DMSG("SETUP %d!\n", i); ++ goto stall; ++ } ++ u.word [i] = UDCDR0; ++ } ++ if (unlikely((UDCCSR0 & UDCCSR0_RNE) != 0)) ++ goto bad_setup; ++ ++ le16_to_cpus(&u.r.wValue); ++ le16_to_cpus(&u.r.wIndex); ++ le16_to_cpus(&u.r.wLength); ++ ++ DBG(DBG_VERBOSE, "SETUP %02x.%02x v%04x i%04x l%04x\n", ++ u.r.bRequestType, u.r.bRequest, ++ u.r.wValue, u.r.wIndex, u.r.wLength); ++ /* cope with automagic for some standard requests. */ ++ dev->req_std = (u.r.bRequestType & USB_TYPE_MASK) ++ == USB_TYPE_STANDARD; ++ dev->req_config = 0; ++ dev->req_pending = 1; ++#if 0 ++ switch (u.r.bRequest) { ++ /* hardware was supposed to hide this */ ++ case USB_REQ_SET_CONFIGURATION: ++ case USB_REQ_SET_INTERFACE: ++ case USB_REQ_SET_ADDRESS: ++ dev_err(dev->dev, "Should not come here\n"); ++ break; ++ } ++ ++#endif ++ if (u.r.bRequestType & USB_DIR_IN) ++ dev->ep0state = EP0_IN_DATA_PHASE; ++ else ++ dev->ep0state = EP0_OUT_DATA_PHASE; ++ i = dev->driver->setup(&dev->gadget, &u.r); ++ ++ if (i < 0) { ++ /* hardware automagic preventing STALL... */ ++ if (dev->req_config) { ++ /* hardware sometimes neglects to tell ++ * tell us about config change events, ++ * so later ones may fail... ++ */ ++ WARN("config change %02x fail %d?\n", ++ u.r.bRequest, i); ++ return; ++ /* TODO experiment: if has_cfr, ++ * hardware didn't ACK; maybe we ++ * could actually STALL! ++ */ ++ } ++ DBG(DBG_VERBOSE, "protocol STALL, " ++ "%02x err %d\n", UDCCSR0, i); ++stall: ++ /* the watchdog timer helps deal with cases ++ * where udc seems to clear FST wrongly, and ++ * then NAKs instead of STALLing. ++ */ ++ ep0start(dev, UDCCSR0_FST|UDCCSR0_FTF, "stall"); ++ start_watchdog(dev); ++ dev->ep0state = EP0_STALL; ++ ++ /* deferred i/o == no response yet */ ++ } else if (dev->req_pending) { ++ if (likely(dev->ep0state == EP0_IN_DATA_PHASE ++ || dev->req_std || u.r.wLength)) ++ ep0start(dev, 0, "defer"); ++ else ++ ep0start(dev, UDCCSR0_IPR, "defer/IPR"); ++ } ++ ++ /* expect at least one data or status stage irq */ ++ return; ++ ++ } else { ++ /* some random early IRQ: ++ * - we acked FST ++ * - IPR cleared ++ * - OPC got set, without SA (likely status stage) ++ */ ++ UDCCSR0 = udccsr0 & (UDCCSR0_SA|UDCCSR0_OPC); ++ } ++ break; ++ case EP0_IN_DATA_PHASE: /* GET_DESCRIPTOR etc */ ++ if (udccsr0 & UDCCSR0_OPC) { ++ UDCCSR0 = UDCCSR0_OPC|UDCCSR0_FTF; ++ DBG(DBG_VERBOSE, "ep0in premature status\n"); ++ if (req) ++ done(ep, req, 0); ++ ep0_idle(dev); ++ } else /* irq was IPR clearing */ { ++ if (req) { ++ /* this IN packet might finish the request */ ++ (void) write_ep0_fifo(ep, req); ++ } /* else IN token before response was written */ ++ } ++ break; ++ case EP0_OUT_DATA_PHASE: /* SET_DESCRIPTOR etc */ ++ if (udccsr0 & UDCCSR0_OPC) { ++ if (req) { ++ /* this OUT packet might finish the request */ ++ if (read_ep0_fifo(ep, req)) ++ done(ep, req, 0); ++ /* else more OUT packets expected */ ++ } /* else OUT token before read was issued */ ++ } else /* irq was IPR clearing */ { ++ DBG(DBG_VERBOSE, "ep0out premature status\n"); ++ if (req) ++ done(ep, req, 0); ++ ep0_idle(dev); ++ } ++ break; ++ case EP0_STALL: ++ UDCCSR0 = UDCCSR0_FST; ++ break; ++ } ++ UDCISR0 = UDCISR_INT(0, UDCISR_INT_MASK); ++} ++ ++ ++static void handle_ep(struct pxa27x_ep *ep) ++{ ++ struct pxa27x_request *req; ++ int completed; ++ u32 udccsr=0; ++ ++ DMSG("%s is called\n", __FUNCTION__); ++ do { ++ completed = 0; ++ if (likely (!list_empty(&ep->queue))) { ++ req = list_entry(ep->queue.next, ++ struct pxa27x_request, queue); ++ } else ++ req = 0; ++ ++// udccsr = *ep->reg_udccsr; ++ DMSG("%s: req:%p, udcisr0:0x%x udccsr %p:0x%x\n", __FUNCTION__, ++ req, UDCISR0, ep->reg_udccsr, *ep->reg_udccsr); ++ if (unlikely(ep->dir_in)) { ++ udccsr = (UDCCSR_SST | UDCCSR_TRN) & *ep->reg_udccsr; ++ if (unlikely (udccsr)) ++ *ep->reg_udccsr = udccsr; ++ ++ if (req && likely ((*ep->reg_udccsr & UDCCSR_FS) != 0)) ++ completed = write_fifo(ep, req); ++ ++ } else { ++ udccsr = (UDCCSR_SST | UDCCSR_TRN) & *ep->reg_udccsr; ++ if (unlikely(udccsr)) ++ *ep->reg_udccsr = udccsr; ++ ++ /* fifos can hold packets, ready for reading... */ ++ if (likely(req)) { ++ completed = read_fifo(ep, req); ++ } else { ++ pio_irq_disable (ep->pxa_ep_num); ++ //*ep->reg_udccsr = UDCCSR_FEF; ++ DMSG("%s: no req for out data\n", ++ __FUNCTION__); ++ } ++ } ++ ep->pio_irqs++; ++ } while (completed); ++} ++ ++static void pxa27x_update_eps(struct pxa27x_udc *dev) ++{ ++ struct pxa27x_virt_ep *virt_ep; ++ int i; ++ ++ for (i = 1; i < UDC_EP_NUM; i++) { ++ if(!dev->ep[i].assigned || !dev->ep[i].usb_ep) ++ continue; ++ virt_ep = container_of(dev->ep[i].usb_ep, struct pxa27x_virt_ep, usb_ep); ++ ++ DMSG("%s, Updating eps %d:%d, %d:%d, %d:%d, %p,%p\n", __FUNCTION__, dev->ep[i].config, dev->configuration ++ ,dev->ep[i].interface, dev->interface, dev->ep[i].aisn, dev->alternate, virt_ep->pxa_ep, &dev->ep[i]); ++ ++ if(dev->ep[i].config == dev->configuration && virt_ep->pxa_ep != &dev->ep[i]) { ++ if ((dev->ep[i].interface == dev->interface && ++ dev->ep[i].aisn == dev->alternate) || virt_ep->pxa_ep->config != dev->configuration) { ++ ++ if (virt_ep->pxa_ep->desc) { ++ DMSG("%s, Changing end point to %d (en/dis)\n", __FUNCTION__, i); ++ pxa27x_ep_disable(&virt_ep->usb_ep); ++ virt_ep->pxa_ep = &dev->ep[i]; ++ pxa27x_ep_enable(&virt_ep->usb_ep, virt_ep->desc); ++ } else { ++ DMSG("%s, Changing end point to %d (no en/dis)\n", __FUNCTION__, i); ++ virt_ep->pxa_ep = &dev->ep[i]; ++ } ++ } ++ } ++ } ++} ++ ++static void pxa27x_change_configuration(struct pxa27x_udc *dev) ++{ ++ struct usb_ctrlrequest req ; ++ ++ pxa27x_update_eps(dev); ++ ++ req.bRequestType = 0; ++ req.bRequest = USB_REQ_SET_CONFIGURATION; ++ req.wValue = dev->configuration; ++ req.wIndex = 0; ++ req.wLength = 0; ++ ++ dev->ep0state = EP0_NO_ACTION; ++ dev->driver->setup(&dev->gadget, &req); ++} ++ ++static void pxa27x_change_interface(struct pxa27x_udc *dev) ++{ ++ struct usb_ctrlrequest req; ++ ++ pxa27x_update_eps(dev); ++ ++ req.bRequestType = USB_RECIP_INTERFACE; ++ req.bRequest = USB_REQ_SET_INTERFACE; ++ req.wValue = dev->alternate; ++ req.wIndex = dev->interface; ++ req.wLength = 0; ++ ++ dev->ep0state = EP0_NO_ACTION; ++ dev->driver->setup(&dev->gadget, &req); ++} ++ ++/* ++ * pxa27x_udc_irq - interrupt handler ++ * ++ * avoid delays in ep0 processing. the control handshaking isn't always ++ * under software control (pxa250c0 and the pxa255 are better), and delays ++ * could cause usb protocol errors. ++ */ ++static irqreturn_t pxa27x_udc_irq(int irq, void *_dev) ++{ ++ struct pxa27x_udc *dev = _dev; ++ int handled; ++ ++ dev->stats.irqs++; ++ ++ DBG(DBG_VERBOSE, "Interrupt, UDCISR0:0x%08x, UDCISR1:0x%08x, " ++ "UDCCR:0x%08x\n", UDCISR0, UDCISR1, UDCCR); ++ do { ++ u32 udcir = UDCISR1 & 0xF8000000; ++ ++ handled = 0; ++ ++ /* SUSpend Interrupt Request */ ++ if (unlikely(udcir & UDCISR1_IRSU)) { ++ UDCISR1 = UDCISR1_IRSU; ++ handled = 1; ++ DBG(DBG_VERBOSE, "USB suspend\n"); ++ if (dev->gadget.speed != USB_SPEED_UNKNOWN ++ && dev->driver ++ && dev->driver->suspend) ++ dev->driver->suspend(&dev->gadget); ++ ep0_idle(dev); ++ } ++ ++ /* RESume Interrupt Request */ ++ if (unlikely(udcir & UDCISR1_IRRU)) { ++ UDCISR1 = UDCISR1_IRRU; ++ handled = 1; ++ DBG(DBG_VERBOSE, "USB resume\n"); ++ ++ if (dev->gadget.speed != USB_SPEED_UNKNOWN ++ && dev->driver ++ && dev->driver->resume) ++ dev->driver->resume(&dev->gadget); ++ } ++ ++ if (unlikely(udcir & UDCISR1_IRCC)) { ++ unsigned config, interface, alternate; ++ ++ handled = 1; ++ DBG(DBG_VERBOSE, "USB SET_CONFIGURATION or " ++ "SET_INTERFACE command received\n"); ++ ++ config = (UDCCR & UDCCR_ACN) >> UDCCR_ACN_S; ++ ++ if (dev->configuration != config) { ++ dev->configuration = config; ++ pxa27x_change_configuration(dev) ; ++ } ++ ++ interface = (UDCCR & UDCCR_AIN) >> UDCCR_AIN_S; ++ alternate = (UDCCR & UDCCR_AAISN) >> UDCCR_AAISN_S; ++ ++ if ((dev->interface != interface) || (dev->alternate != alternate)) { ++ dev->interface = interface; ++ dev->alternate = alternate; ++ pxa27x_change_interface(dev); ++ } ++ ++ UDCCR |= UDCCR_SMAC; ++ ++ UDCISR1 = UDCISR1_IRCC; ++ DMSG("%s: con:%d,inter:%d,alt:%d\n", ++ __FUNCTION__, config,interface, alternate); ++ } ++ ++ /* ReSeT Interrupt Request - USB reset */ ++ if (unlikely(udcir & UDCISR1_IRRS)) { ++ UDCISR1 = UDCISR1_IRRS; ++ handled = 1; ++ ++ if ((UDCCR & UDCCR_UDA) == 0) { ++ DBG(DBG_VERBOSE, "USB reset start\n"); ++ ++ /* reset driver and endpoints, ++ * in case that's not yet done ++ */ ++ stop_activity(dev, dev->driver); ++ } ++ INFO("USB reset\n"); ++ dev->gadget.speed = USB_SPEED_FULL; ++ memset(&dev->stats, 0, sizeof dev->stats); ++ ++ } else { ++ u32 udcisr0 = UDCISR0 ; ++ u32 udcisr1 = UDCISR1 & 0xFFFF; ++ int i; ++ ++ if (unlikely (!udcisr0 && !udcisr1)) ++ continue; ++ ++ DBG(DBG_VERY_NOISY, "irq %02x.%02x\n", udcisr1,udcisr0); ++ ++ /* control traffic */ ++ if (udcisr0 & UDCISR0_IR0) { ++ dev->ep[0].pio_irqs++; ++ handle_ep0(dev); ++ handled = 1; ++ } ++ ++ udcisr0 >>= 2; ++ /* endpoint data transfers */ ++ for (i = 1; udcisr0!=0 && i < 16; udcisr0>>=2,i++) { ++ UDCISR0 = UDCISR_INT(i, UDCISR_INT_MASK); ++ ++ if (udcisr0 & UDC_INT_FIFOERROR) ++ dev_err(dev->dev, " Endpoint %d Fifo error\n", i); ++ if (udcisr0 & UDC_INT_PACKETCMP) { ++ handle_ep(&dev->ep[i]); ++ handled = 1; ++ } ++ ++ } ++ ++ for (i = 0; udcisr1!=0 && i < 8; udcisr1 >>= 2, i++) { ++ UDCISR1 = UDCISR_INT(i, UDCISR_INT_MASK); ++ ++ if (udcisr1 & UDC_INT_FIFOERROR) { ++ dev_err(dev->dev, "Endpoint %d fifo error\n", (i+16)); ++ } ++ ++ if (udcisr1 & UDC_INT_PACKETCMP) { ++ handle_ep(&dev->ep[i+16]); ++ handled = 1; ++ } ++ } ++ } ++ ++ /* we could also ask for 1 msec SOF (SIR) interrupts */ ++ ++ } while (handled); ++ return IRQ_HANDLED; ++} ++ ++int write_ep0_zlp(void) ++{ ++ UDCCSR0 = UDCCSR0_IPR; ++ return 0; ++} ++EXPORT_SYMBOL(write_ep0_zlp); ++ ++static void udc_init_ep(struct pxa27x_udc *dev) ++{ ++ int i; ++ ++ INIT_LIST_HEAD(&dev->gadget.ep_list); ++ INIT_LIST_HEAD(&dev->gadget.ep0->ep_list); ++ ++ for (i = 0; i < UDC_EP_NUM; i++) { ++ struct pxa27x_ep *ep = &dev->ep[i]; ++ ++ ep->dma = -1; ++ if (i != 0) { ++ memset(ep, 0, sizeof(*ep)); ++ } ++ INIT_LIST_HEAD(&ep->queue); ++ } ++} ++ ++/*-------------------------------------------------------------------------*/ ++ ++static void nop_release(struct device *dev) ++{ ++ DMSG("%s %s\n", __FUNCTION__, dev->bus_id); ++} ++ ++/* this uses load-time allocation and initialization (instead of ++ * doing it at run-time) to save code, eliminate fault paths, and ++ * be more obviously correct. ++ */ ++ ++static struct pxa27x_udc memory = { ++ .gadget = { ++ .ops = &pxa27x_udc_ops, ++ .ep0 = &memory.virt_ep0.usb_ep, ++ .name = driver_name, ++ .dev = { ++ .bus_id = "gadget", ++ .release = nop_release, ++ }, ++ }, ++ ++ /* control endpoint */ ++ .virt_ep0 = { ++ .pxa_ep = &memory.ep[0], ++ .usb_ep = { ++ .name = ep0name, ++ .ops = &pxa27x_ep_ops, ++ .maxpacket = EP0_FIFO_SIZE, ++ }, ++ }, ++ ++ .ep[0] = { ++ .usb_ep = &memory.virt_ep0.usb_ep, ++ .dev = &memory, ++ .reg_udccsr = &UDCCSR0, ++ .reg_udcdr = &UDCDR0, ++ }, ++}; ++ ++static int __init pxa27x_udc_probe(struct platform_device *_dev) ++{ ++ struct pxa27x_udc *dev = &memory; ++ int retval; ++ ++ /* other non-static parts of init */ ++ dev->dev = &_dev->dev; ++ dev->mach = _dev->dev.platform_data; ++ ++ /* RPFIXME */ ++ UP2OCR = UP2OCR_HXOE | UP2OCR_DPPUE | UP2OCR_DPPUBE; ++ ++ init_timer(&dev->timer); ++ dev->timer.function = udc_watchdog; ++ dev->timer.data = (unsigned long) dev; ++ ++ device_initialize(&dev->gadget.dev); ++ dev->gadget.dev.parent = &_dev->dev; ++ dev->gadget.dev.dma_mask = _dev->dev.dma_mask; ++ ++ the_controller = dev; ++ platform_set_drvdata(_dev, dev); ++ ++ udc_disable(dev); ++ udc_init_ep(dev); ++ udc_reinit(dev); ++ ++ /* irq setup after old hardware state is cleaned up */ ++ retval = request_irq(IRQ_USB, pxa27x_udc_irq, ++ SA_INTERRUPT, driver_name, dev); ++ if (retval != 0) { ++ dev_err(dev->dev, "%s: can't get irq %i, err %d\n", ++ driver_name, IRQ_USB, retval); ++ return -EBUSY; ++ } ++ dev->got_irq = 1; ++ ++ create_proc_files(); ++ ++ return 0; ++} ++ ++static int pxa27x_udc_remove(struct platform_device *_dev) ++{ ++ struct pxa27x_udc *dev = platform_get_drvdata(_dev); ++ ++ udc_disable(dev); ++ remove_proc_files(); ++ usb_gadget_unregister_driver(dev->driver); ++ ++ pxa27x_ep_freeall(&dev->gadget); ++ ++ if (dev->got_irq) { ++ free_irq(IRQ_USB, dev); ++ dev->got_irq = 0; ++ } ++ platform_set_drvdata(_dev, 0); ++ the_controller = 0; ++ return 0; ++} ++ ++#ifdef CONFIG_PM ++static void pxa27x_udc_shutdown(struct platform_device *_dev) ++{ ++ struct pxa27x_udc *dev = platform_get_drvdata(_dev); ++ ++ udc_disable(dev); ++} ++ ++static int pxa27x_udc_suspend(struct platform_device *_dev, pm_message_t state) ++{ ++ int i; ++ struct pxa27x_udc *dev = platform_get_drvdata(_dev); ++ ++ DMSG("%s is called\n", __FUNCTION__); ++ ++ dev->udccsr0 = UDCCSR0; ++ for(i=1; (i<UDC_EP_NUM); i++) { ++ if (dev->ep[i].assigned) { ++ struct pxa27x_ep *ep = &dev->ep[i]; ++ ep->udccsr_value = *ep->reg_udccsr; ++ ep->udccr_value = *ep->reg_udccr; ++ DMSG("EP%d, udccsr:0x%x, udccr:0x%x\n", ++ i, *ep->reg_udccsr, *ep->reg_udccr); ++ } ++ } ++ ++ udc_clear_mask_UDCCR(UDCCR_UDE); ++ pxa_set_cken(CKEN11_USB, 0); ++ ++ return 0; ++} ++ ++static int pxa27x_udc_resume(struct platform_device *_dev) ++{ ++ int i; ++ struct pxa27x_udc *dev = platform_get_drvdata(_dev); ++ ++ DMSG("%s is called\n", __FUNCTION__); ++ UDCCSR0 = dev->udccsr0 & (UDCCSR0_FST | UDCCSR0_DME); ++ for (i=1; i < UDC_EP_NUM; i++) { ++ if (dev->ep[i].assigned) { ++ struct pxa27x_ep *ep = &dev->ep[i]; ++ *ep->reg_udccsr = ep->udccsr_value; ++ *ep->reg_udccr = ep->udccr_value; ++ DMSG("EP%d, udccsr:0x%x, udccr:0x%x\n", ++ i, *ep->reg_udccsr, *ep->reg_udccr); ++ } ++ } ++ ++ udc_enable(dev); ++ ++ /* OTGPH bit is set when sleep mode is entered. ++ * it indicates that OTG pad is retaining its state. ++ * Upon exit from sleep mode and before clearing OTGPH, ++ * Software must configure the USB OTG pad, UDC, and UHC ++ * to the state they were in before entering sleep mode.*/ ++ PSSR |= PSSR_OTGPH; ++ ++ return 0; ++} ++#endif ++ ++/*-------------------------------------------------------------------------*/ ++ ++static struct platform_driver udc_driver = { ++ .driver = { ++ .name = "pxa2xx-udc", ++ }, ++ .probe = pxa27x_udc_probe, ++ .remove = pxa27x_udc_remove, ++#ifdef CONFIG_PM ++ .shutdown = pxa27x_udc_shutdown, ++ .suspend = pxa27x_udc_suspend, ++ .resume = pxa27x_udc_resume ++#endif ++}; ++ ++static int __init udc_init(void) ++{ ++ printk(KERN_INFO "%s: version %s\n", driver_name, DRIVER_VERSION); ++ return platform_driver_register(&udc_driver); ++} ++module_init(udc_init); ++ ++static void __exit udc_exit(void) ++{ ++ platform_driver_unregister(&udc_driver); ++} ++module_exit(udc_exit); ++ ++MODULE_DESCRIPTION(DRIVER_DESC); ++MODULE_AUTHOR("Frank Becker, Robert Schwebel, David Brownell"); ++MODULE_LICENSE("GPL"); +Index: git/drivers/usb/gadget/pxa27x_udc.h +=================================================================== +--- /dev/null 1970-01-01 00:00:00.000000000 +0000 ++++ git/drivers/usb/gadget/pxa27x_udc.h 2006-12-16 22:57:15.000000000 +0000 +@@ -0,0 +1,298 @@ ++/* ++ * linux/drivers/usb/gadget/pxa27x_udc.h ++ * Intel PXA27x on-chip full speed USB device controller ++ * ++ * Copyright (C) 2003 Robert Schwebel <r.schwebel@pengutronix.de>, Pengutronix ++ * Copyright (C) 2003 David Brownell ++ * Copyright (C) 2004 Intel Corporation ++ * ++ * 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; if not, write to the Free Software ++ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA ++ */ ++ ++#ifndef __LINUX_USB_GADGET_PXA27X_H ++#define __LINUX_USB_GADGET_PXA27X_H ++ ++#include <linux/types.h> ++ ++struct pxa27x_udc; ++ ++struct pxa27x_ep { ++ struct pxa27x_udc *dev; ++ struct usb_ep *usb_ep; ++ const struct usb_endpoint_descriptor *desc; ++ ++ struct list_head queue; ++ unsigned long pio_irqs; ++ unsigned long dma_irqs; ++ ++ unsigned pxa_ep_num; ++ int dma; ++ unsigned fifo_size; ++ unsigned ep_type; ++ ++ unsigned stopped : 1; ++ unsigned dma_con : 1; ++ unsigned dir_in : 1; ++ unsigned assigned : 1; ++ ++ unsigned ep_num; ++ unsigned config; ++ unsigned interface; ++ unsigned aisn; ++ /* UDCCSR = UDC Control/Status Register for this EP ++ * UBCR = UDC Byte Count Remaining (contents of OUT fifo) ++ * UDCDR = UDC Endpoint Data Register (the fifo) ++ * UDCCR = UDC Endpoint Configuration Registers ++ * DRCM = DMA Request Channel Map ++ */ ++ volatile u32 *reg_udccsr; ++ volatile u32 *reg_udcbcr; ++ volatile u32 *reg_udcdr; ++ volatile u32 *reg_udccr; ++#ifdef USE_DMA ++ volatile u32 *reg_drcmr; ++#define drcmr(n) .reg_drcmr = & DRCMR ## n , ++#else ++#define drcmr(n) ++#endif ++ ++#ifdef CONFIG_PM ++ unsigned udccsr_value; ++ unsigned udccr_value; ++#endif ++}; ++ ++struct pxa27x_virt_ep { ++ struct usb_ep usb_ep; ++ const struct usb_endpoint_descriptor *desc; ++ struct pxa27x_ep *pxa_ep; ++}; ++ ++struct pxa27x_request { ++ struct usb_request req; ++ struct list_head queue; ++}; ++ ++enum ep0_state { ++ EP0_IDLE, ++ EP0_IN_DATA_PHASE, ++ EP0_OUT_DATA_PHASE, ++// EP0_END_XFER, ++ EP0_STALL, ++ EP0_NO_ACTION ++}; ++ ++#define EP0_FIFO_SIZE ((unsigned)16) ++#define BULK_FIFO_SIZE ((unsigned)64) ++#define ISO_FIFO_SIZE ((unsigned)256) ++#define INT_FIFO_SIZE ((unsigned)8) ++ ++struct udc_stats { ++ struct ep0stats { ++ unsigned long ops; ++ unsigned long bytes; ++ } read, write; ++ unsigned long irqs; ++}; ++ ++#define UDC_EP_NUM 24 ++ ++ ++struct pxa27x_udc { ++ struct usb_gadget gadget; ++ struct usb_gadget_driver *driver; ++ ++ enum ep0_state ep0state; ++ struct udc_stats stats; ++ unsigned got_irq : 1, ++ has_cfr : 1, ++ req_pending : 1, ++ req_std : 1, ++ req_config : 1; ++ ++#define start_watchdog(dev) mod_timer(&dev->timer, jiffies + (HZ/200)) ++ struct timer_list timer; ++ ++ struct device *dev; ++ struct pxa2xx_udc_mach_info *mach; ++ u64 dma_mask; ++ struct pxa27x_virt_ep virt_ep0; ++ struct pxa27x_ep ep[UDC_EP_NUM]; ++ unsigned int ep_num; ++ ++ unsigned configuration, ++ interface, ++ alternate; ++#ifdef CONFIG_PM ++ unsigned udccsr0; ++#endif ++}; ++ ++static struct pxa27x_udc *the_controller; ++ ++#if 0 ++/*-------------------------------------------------------------------------*/ ++ ++ ++/* one GPIO should be used to detect host disconnect */ ++static inline int is_usb_connected(void) ++{ ++ if (!the_controller->mach->udc_is_connected) ++ return 1; ++ return the_controller->mach->udc_is_connected(); ++} ++ ++/* one GPIO should force the host to see this device (or not) */ ++static inline void make_usb_disappear(void) ++{ ++ if (!the_controller->mach->udc_command) ++ return; ++ the_controller->mach->udc_command(PXA27X_UDC_CMD_DISCONNECT); ++} ++ ++static inline void let_usb_appear(void) ++{ ++ if (!the_controller->mach->udc_command) ++ return; ++ the_controller->mach->udc_command(PXA2XX_UDC_CMD_CONNECT); ++} ++#endif ++ ++/*-------------------------------------------------------------------------*/ ++ ++/* ++ * Debugging support vanishes in non-debug builds. DBG_NORMAL should be ++ * mostly silent during normal use/testing, with no timing side-effects. ++ */ ++#define DBG_NORMAL 1 /* error paths, device state transitions */ ++#define DBG_VERBOSE 2 /* add some success path trace info */ ++#define DBG_NOISY 3 /* ... even more: request level */ ++#define DBG_VERY_NOISY 4 /* ... even more: packet level */ ++ ++#ifdef DEBUG ++static const char *state_name[] = { ++ "EP0_IDLE", ++ "EP0_IN_DATA_PHASE", "EP0_OUT_DATA_PHASE", ++ "EP0_END_XFER", "EP0_STALL" ++}; ++ ++#define DMSG(stuff...) printk(KERN_ERR "udc: " stuff) ++ ++#ifdef VERBOSE ++# define UDC_DEBUG DBG_VERBOSE ++#else ++# define UDC_DEBUG DBG_NORMAL ++#endif ++ ++static void __attribute__ ((__unused__)) ++dump_udccr(const char *label) ++{ ++ u32 udccr = UDCCR; ++ DMSG("%s 0x%08x =%s%s%s%s%s%s%s%s%s%s, con=%d,inter=%d,altinter=%d\n", ++ label, udccr, ++ (udccr & UDCCR_OEN) ? " oen":"", ++ (udccr & UDCCR_AALTHNP) ? " aalthnp":"", ++ (udccr & UDCCR_AHNP) ? " rem" : "", ++ (udccr & UDCCR_BHNP) ? " rstir" : "", ++ (udccr & UDCCR_DWRE) ? " dwre" : "", ++ (udccr & UDCCR_SMAC) ? " smac" : "", ++ (udccr & UDCCR_EMCE) ? " emce" : "", ++ (udccr & UDCCR_UDR) ? " udr" : "", ++ (udccr & UDCCR_UDA) ? " uda" : "", ++ (udccr & UDCCR_UDE) ? " ude" : "", ++ (udccr & UDCCR_ACN) >> UDCCR_ACN_S, ++ (udccr & UDCCR_AIN) >> UDCCR_AIN_S, ++ (udccr & UDCCR_AAISN)>> UDCCR_AAISN_S ); ++} ++ ++static void __attribute__ ((__unused__)) ++dump_udccsr0(const char *label) ++{ ++ u32 udccsr0 = UDCCSR0; ++ ++ DMSG("%s %s 0x%08x =%s%s%s%s%s%s%s\n", ++ label, state_name[the_controller->ep0state], udccsr0, ++ (udccsr0 & UDCCSR0_SA) ? " sa" : "", ++ (udccsr0 & UDCCSR0_RNE) ? " rne" : "", ++ (udccsr0 & UDCCSR0_FST) ? " fst" : "", ++ (udccsr0 & UDCCSR0_SST) ? " sst" : "", ++ (udccsr0 & UDCCSR0_DME) ? " dme" : "", ++ (udccsr0 & UDCCSR0_IPR) ? " ipr" : "", ++ (udccsr0 & UDCCSR0_OPC) ? " opr" : ""); ++} ++ ++static void __attribute__ ((__unused__)) ++dump_state(struct pxa27x_udc *dev) ++{ ++ unsigned i; ++ ++ DMSG("%s, udcicr %02X.%02X, udcsir %02X.%02x, udcfnr %02X\n", ++ state_name[dev->ep0state], ++ UDCICR1, UDCICR0, UDCISR1, UDCISR0, UDCFNR); ++ dump_udccr("udccr"); ++ ++ if (!dev->driver) { ++ DMSG("no gadget driver bound\n"); ++ return; ++ } else ++ DMSG("ep0 driver '%s'\n", dev->driver->driver.name); ++ ++ ++ dump_udccsr0 ("udccsr0"); ++ DMSG("ep0 IN %lu/%lu, OUT %lu/%lu\n", ++ dev->stats.write.bytes, dev->stats.write.ops, ++ dev->stats.read.bytes, dev->stats.read.ops); ++ ++ for (i = 1; i < UDC_EP_NUM; i++) { ++ if (dev->ep[i].assigned) ++ DMSG ("udccs%d = %02x\n", i, *dev->ep->reg_udccsr); ++ } ++} ++ ++#if 0 ++static void dump_regs(u8 ep) ++{ ++ DMSG("EP:%d UDCCSR:0x%08x UDCBCR:0x%08x\n UDCCR:0x%08x\n", ++ ep,UDCCSN(ep), UDCBCN(ep), UDCCN(ep)); ++} ++static void dump_req (struct pxa27x_request *req) ++{ ++ struct usb_request *r = &req->req; ++ ++ DMSG("%s: buf:0x%08x length:%d dma:0x%08x actual:%d\n", ++ __FUNCTION__, (unsigned)r->buf, r->length, ++ r->dma, r->actual); ++} ++#endif ++ ++#else ++ ++#define DMSG(stuff...) do{}while(0) ++ ++#define dump_udccr(x) do{}while(0) ++#define dump_udccsr0(x) do{}while(0) ++#define dump_state(x) do{}while(0) ++ ++#define UDC_DEBUG ((unsigned)4) ++ ++#endif ++ ++#define DBG(lvl, stuff...) do{if ((lvl) <= UDC_DEBUG) DMSG(stuff);}while(0) ++ ++#define WARN(stuff...) printk(KERN_WARNING "udc: " stuff) ++#define INFO(stuff...) printk(KERN_INFO "udc: " stuff) ++ ++ ++#endif /* __LINUX_USB_GADGET_PXA27X_H */ +Index: git/drivers/usb/gadget/pxa2xx_udc.h +=================================================================== +--- git.orig/drivers/usb/gadget/pxa2xx_udc.h 2006-12-16 22:57:10.000000000 +0000 ++++ git/drivers/usb/gadget/pxa2xx_udc.h 2006-12-16 22:57:15.000000000 +0000 +@@ -259,7 +259,8 @@ dump_state(struct pxa2xx_udc *dev) + unsigned i; + + DMSG("%s %s, uicr %02X.%02X, usir %02X.%02x, ufnr %02X.%02X\n", +- is_usb_connected() ? "host " : "disconnected", ++ //is_usb_connected() ? "host " : "disconnected", ++ "host ", + state_name[dev->ep0state], + UICR1, UICR0, USIR1, USIR0, UFNRH, UFNRL); + dump_udccr("udccr"); +@@ -276,8 +277,8 @@ dump_state(struct pxa2xx_udc *dev) + } else + DMSG("ep0 driver '%s'\n", dev->driver->driver.name); + +- if (!is_usb_connected()) +- return; ++ //if (!is_usb_connected()) ++ // return; + + dump_udccs0 ("udccs0"); + DMSG("ep0 IN %lu/%lu, OUT %lu/%lu\n", +Index: git/drivers/usb/gadget/ether.c +=================================================================== +--- git.orig/drivers/usb/gadget/ether.c 2006-12-16 22:57:14.000000000 +0000 ++++ git/drivers/usb/gadget/ether.c 2006-12-16 22:57:15.000000000 +0000 +@@ -1264,6 +1264,10 @@ + /* done sending after USB_CDC_GET_ENCAPSULATED_RESPONSE */ + } + ++#ifdef CONFIG_USB_GADGET_PXA27X ++int write_ep0_zlp(void); ++#endif ++ + static void rndis_command_complete (struct usb_ep *ep, struct usb_request *req) + { + struct eth_dev *dev = ep->driver_data; +@@ -1274,6 +1278,10 @@ + status = rndis_msg_parser (dev->rndis_config, (u8 *) req->buf); + if (status < 0) + ERROR(dev, "%s: rndis parse error %d\n", __FUNCTION__, status); ++ ++#ifdef CONFIG_USB_GADGET_PXA27X ++ write_ep0_zlp(); ++#endif + spin_unlock(&dev->lock); + } + +@@ -2206,6 +2214,9 @@ + #endif + #ifndef CONFIG_USB_ETH_RNDIS + rndis = 0; ++#ifdef CONFIG_USB_GADGET_PXA27X ++ ep_config[1].interface = 0; ++#endif + #endif + + /* Because most host side USB stacks handle CDC Ethernet, that diff --git a/packages/linux/logicpd-pxa270-2.6.19.2/xscale_cache_workaround-r1.patch b/packages/linux/logicpd-pxa270-2.6.19.2/xscale_cache_workaround-r1.patch new file mode 100644 index 0000000000..21f15d8287 --- /dev/null +++ b/packages/linux/logicpd-pxa270-2.6.19.2/xscale_cache_workaround-r1.patch @@ -0,0 +1,93 @@ +If PG_dcache_dirty is set for a page, we need to flush the source page +before performing any copypage operation using a different virtual address. + +This fixes the copypage implementations on XScale, StrongARM and ARMv6. + +This patch fixes segmentation faults seen in the dynamic linker under +the usage patterns in glibc 2.4/2.5. + +Signed-off-by: Richard Purdie <rpurdie@rpsys.net> + +--- + arch/arm/mm/copypage-v4mc.c | 6 ++++++ + arch/arm/mm/copypage-v6.c | 4 ++++ + arch/arm/mm/copypage-xscale.c | 6 ++++++ + include/asm-arm/cacheflush.h | 2 ++ + 4 files changed, 18 insertions(+) + +Index: git/arch/arm/mm/copypage-xscale.c +=================================================================== +--- git.orig/arch/arm/mm/copypage-xscale.c 2006-12-30 15:04:19.000000000 +0000 ++++ git/arch/arm/mm/copypage-xscale.c 2006-12-30 15:04:22.000000000 +0000 +@@ -19,6 +19,7 @@ + #include <asm/page.h> + #include <asm/pgtable.h> + #include <asm/tlbflush.h> ++#include <asm/cacheflush.h> + + #include "mm.h" + +@@ -91,6 +92,11 @@ mc_copy_user_page(void *from, void *to) + + void xscale_mc_copy_user_page(void *kto, const void *kfrom, unsigned long vaddr) + { ++ struct page *page = virt_to_page(kfrom); ++ ++ if (test_and_clear_bit(PG_dcache_dirty, &page->flags)) ++ __flush_dcache_page(page_mapping(page), page); ++ + spin_lock(&minicache_lock); + + set_pte_ext(TOP_PTE(COPYPAGE_MINICACHE), pfn_pte(__pa(kfrom) >> PAGE_SHIFT, minicache_pgprot), 0); +Index: git/arch/arm/mm/copypage-v4mc.c +=================================================================== +--- git.orig/arch/arm/mm/copypage-v4mc.c 2006-12-30 15:04:19.000000000 +0000 ++++ git/arch/arm/mm/copypage-v4mc.c 2006-12-30 15:04:22.000000000 +0000 +@@ -19,6 +19,7 @@ + #include <asm/page.h> + #include <asm/pgtable.h> + #include <asm/tlbflush.h> ++#include <asm/cacheflush.h> + + #include "mm.h" + +@@ -69,6 +70,11 @@ mc_copy_user_page(void *from, void *to) + + void v4_mc_copy_user_page(void *kto, const void *kfrom, unsigned long vaddr) + { ++ struct page *page = virt_to_page(kfrom); ++ ++ if (test_and_clear_bit(PG_dcache_dirty, &page->flags)) ++ __flush_dcache_page(page_mapping(page), page); ++ + spin_lock(&minicache_lock); + + set_pte_ext(TOP_PTE(0xffff8000), pfn_pte(__pa(kfrom) >> PAGE_SHIFT, minicache_pgprot), 0); +Index: git/arch/arm/mm/copypage-v6.c +=================================================================== +--- git.orig/arch/arm/mm/copypage-v6.c 2006-12-30 15:04:19.000000000 +0000 ++++ git/arch/arm/mm/copypage-v6.c 2006-12-30 15:04:22.000000000 +0000 +@@ -53,6 +53,10 @@ static void v6_copy_user_page_aliasing(v + { + unsigned int offset = CACHE_COLOUR(vaddr); + unsigned long from, to; ++ struct page *page = virt_to_page(kfrom); ++ ++ if (test_and_clear_bit(PG_dcache_dirty, &page->flags)) ++ __flush_dcache_page(page_mapping(page), page); + + /* + * Discard data in the kernel mapping for the new page. +Index: git/include/asm-arm/cacheflush.h +=================================================================== +--- git.orig/include/asm-arm/cacheflush.h 2006-12-30 15:04:19.000000000 +0000 ++++ git/include/asm-arm/cacheflush.h 2006-12-30 15:04:22.000000000 +0000 +@@ -355,6 +355,8 @@ extern void flush_ptrace_access(struct v + */ + extern void flush_dcache_page(struct page *); + ++extern void __flush_dcache_page(struct address_space *mapping, struct page *page); ++ + #define flush_dcache_mmap_lock(mapping) \ + write_lock_irq(&(mapping)->tree_lock) + #define flush_dcache_mmap_unlock(mapping) \ diff --git a/packages/linux/logicpd-pxa270_2.6.19.2.bb b/packages/linux/logicpd-pxa270_2.6.19.2.bb new file mode 100644 index 0000000000..283ebb1133 --- /dev/null +++ b/packages/linux/logicpd-pxa270_2.6.19.2.bb @@ -0,0 +1,60 @@ +SECTION = "kernel" +DESCRIPTION = "Linux kernel for the LogicPD Zoom(PXA270 ref design)" +LICENSE = "GPL" +PR = "r1" + +SRC_URI = "ftp://ftp.kernel.org/pub/linux/kernel/v2.6/linux-2.6.19.2.tar.bz2 \ + file://asoc-v0.12.4.patch;patch=1 \ + file://pxafb_fix_params-r2.patch;patch=1 \ + file://pxa_irda_susres_fix-r0.patch;patch=1 \ + file://pxa_keys-r5.patch;patch=1 \ + file://pxa_timerfix-r0.patch;patch=1 \ + file://input_power-r6.patch;patch=1 \ + file://pxa25x_cpufreq-r1.patch;patch=1 \ + file://pm_changes-r1.patch;patch=1 \ + file://usb_add_epalloc-r3.patch;patch=1 \ + file://usb_pxa27x_udc-r3.patch;patch=1 \ + file://kexec-arm-r3.patch;patch=1 \ + file://pxa27x_overlay-r4.patch;patch=1 \ + file://xscale_cache_workaround-r1.patch;patch=1 \ + file://ucb1400-touchscreen.patch;patch=1 \ + file://config-nr-tty-devices.patch;pnum=1;patch=1 \ + " + +SRC_URI_append_logicpd-pxa270 = "\ + file://logicpd-pxa270-cf-hack.patch;pnum=0;patch=1 \ + file://logicpd-pxa270-flash.patch;pnum=0;patch=1 \ + file://logicpd-pxa270-hardware-id-hack.patch;pnum=0;patch=1 \ + file://logicpd-pxa270-smc91x.patch;pnum=0;patch=1 \ + file://logicpd-pxa270-lcd-osd024ttea2.patch;pnum=0;patch=1 \ + file://defconfig-logicpd-pxa270 \ + " + +S = "${WORKDIR}/linux-2.6.19.2" + +COMPATIBLE_HOST = 'arm.*-linux' + +inherit kernel +inherit package + +ARCH = "arm" +KERNEL_IMAGETYPE = "zImage" + +FILES_kernel-image = "" + +do_configure_prepend() { +# install -m 0644 ${S}/arch/arm/configs/lpd270_defconfig ${S}/.config + install -m 0644 ${WORKDIR}/defconfig-logicpd-pxa270 ${S}/.config + +} + +do_deploy() { + install -d ${DEPLOY_DIR_IMAGE} + install -m 0644 arch/${ARCH}/boot/${KERNEL_IMAGETYPE} ${DEPLOY_DIR_IMAGE}/${KERNEL_IMAGETYPE}-${MACHINE}-${DATETIME}.bin +} + +do_deploy[dirs] = "${S}" + +addtask deploy before do_build after do_compile + +COMPATIBLE_MACHINE = "logicpd-pxa270" |