summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorLaura Lawrence <Laura.Lawrence@freescale.com>2008-01-22 15:30:16 -0600
committerDaniel Schaeffer <daniel.schaeffer@timesys.com>2008-08-25 15:20:36 -0400
commitd61adf321a13a299a644748a3148d8c350df20ca (patch)
treecde7b33d63eb0f2123c740d69232e37ff5a0140a
parent8e90502ae2a59747f9d044206577803c9a1dac90 (diff)
ENGR00060848 Add imx37 audio playback support
imx37 driver based on ASoC + Wolfson WM8350 AudioPlus support http://opensource.wolfsonmicro.com/node/8
-rw-r--r--arch/arm/configs/imx37_3stack_defconfig61
-rw-r--r--arch/arm/mach-mx37/Makefile5
-rw-r--r--arch/arm/mach-mx37/dma.c4
-rw-r--r--arch/arm/mach-mx37/mx37_3stack.c14
-rw-r--r--arch/arm/mach-mx37/mx37_3stack_pmic_wm8350.c496
-rw-r--r--drivers/pmic/Makefile5
-rw-r--r--drivers/pmic/pmic-wm8350-i2c.c167
-rw-r--r--sound/soc/Kconfig1
-rw-r--r--sound/soc/Makefile2
-rw-r--r--sound/soc/imx/Kconfig21
-rw-r--r--sound/soc/imx/Makefile10
-rw-r--r--sound/soc/imx/imx-pcm.c549
-rw-r--r--sound/soc/imx/imx-pcm.h67
-rw-r--r--sound/soc/imx/imx-ssi.c862
-rw-r--r--sound/soc/imx/imx-ssi.h220
-rw-r--r--sound/soc/imx/imx37-3stack-wm8350.c600
16 files changed, 3062 insertions, 22 deletions
diff --git a/arch/arm/configs/imx37_3stack_defconfig b/arch/arm/configs/imx37_3stack_defconfig
index bfa9a029f627..b6a63bb3b71e 100644
--- a/arch/arm/configs/imx37_3stack_defconfig
+++ b/arch/arm/configs/imx37_3stack_defconfig
@@ -381,6 +381,22 @@ CONFIG_FW_LOADER=m
#
CONFIG_CONNECTOR=y
CONFIG_PROC_EVENTS=y
+
+#
+# Power Management IC's
+#
+CONFIG_PMIC=y
+CONFIG_PMIC_DEBUG=n
+CONFIG_PMIC_WM8350=y
+
+#
+# WM8350 Config Mode
+#
+CONFIG_PMIC_WM8350_MODE_0=y
+# CONFIG_PMIC_WM8350_MODE_1 is not set
+# CONFIG_PMIC_WM8350_MODE_2 is not set
+# CONFIG_PMIC_WM8350_MODE_3 is not set
+
CONFIG_MTD=y
# CONFIG_MTD_DEBUG is not set
# CONFIG_MTD_CONCAT is not set
@@ -808,7 +824,50 @@ CONFIG_DUMMY_CONSOLE=y
#
# Sound
#
-# CONFIG_SOUND is not set
+CONFIG_SOUND=y
+
+#
+# Advanced Linux Sound Architecture
+#
+CONFIG_SND=y
+CONFIG_SND_TIMER=y
+CONFIG_SND_PCM=y
+# CONFIG_SND_SEQUENCER 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_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
+#
+
+#
+# System on Chip audio support
+#
+CONFIG_SND_SOC=y
+CONFIG_SND_MXC_SOC=y
+CONFIG_SND_MXC_SOC_SSI=y
+CONFIG_SND_SOC_IMX37_3STACK_WM8350=y
+CONFIG_SND_SOC_WM8350=y
+
+#
+# Open Sound System
+#
+# CONFIG_SOUND_PRIME is not set
#
# HID Devices
diff --git a/arch/arm/mach-mx37/Makefile b/arch/arm/mach-mx37/Makefile
index 15aba4791ee1..bfeaa9bf0db2 100644
--- a/arch/arm/mach-mx37/Makefile
+++ b/arch/arm/mach-mx37/Makefile
@@ -4,7 +4,8 @@
# Object file lists.
-obj-y := system.o iomux.o cpu.o mm.o clock.o devices.o serial.o dma.o
+obj-y := system.o iomux.o cpu.o mm.o clock.o devices.o serial.o dma.o
-obj-$(CONFIG_MACH_MX37_3DS) += mx37_3stack.o mx37_3stack_gpio.o
+obj-$(CONFIG_MACH_MX37_3DS) += mx37_3stack.o mx37_3stack_gpio.o
obj-$(CONFIG_SPI_MXC) += mx37_3stack_cpld.o
+obj-$(CONFIG_PMIC_WM8350) += mx37_3stack_pmic_wm8350.o
diff --git a/arch/arm/mach-mx37/dma.c b/arch/arm/mach-mx37/dma.c
index 548fb9c2ba29..1ceb0ca5001c 100644
--- a/arch/arm/mach-mx37/dma.c
+++ b/arch/arm/mach-mx37/dma.c
@@ -1,5 +1,5 @@
/*
- * Copyright 2007 Freescale Semiconductor, Inc. All Rights Reserved.
+ * Copyright 2008 Freescale Semiconductor, Inc. All Rights Reserved.
*/
/*
@@ -21,7 +21,7 @@
#define MXC_SSI_TX1_REG 0x4
#define MXC_SSI_RX0_REG 0x8
#define MXC_SSI_RX1_REG 0xC
-#define MXC_SSI_TXFIFO_WML 0x4
+#define MXC_SSI_TXFIFO_WML 0x2
#define MXC_SSI_RXFIFO_WML 0x6
typedef struct mxc_sdma_info_entry_s {
diff --git a/arch/arm/mach-mx37/mx37_3stack.c b/arch/arm/mach-mx37/mx37_3stack.c
index 66f9b37fc274..f6e24b2b20df 100644
--- a/arch/arm/mach-mx37/mx37_3stack.c
+++ b/arch/arm/mach-mx37/mx37_3stack.c
@@ -204,19 +204,6 @@ static void __init fixup_mxc_board(struct machine_desc *desc, struct tag *tags,
#endif
}
-/*! PMIC */
-static struct platform_device pmic_device = {
- .name = "wm8350-imx37-3stack",
- .id = 0,
-};
-
-static inline void mxc_init_pmic(void)
-{
- if (platform_device_register(&pmic_device) < 0)
- printk(KERN_ERR "Error: Registering the PMIC.\n");
-
-}
-
/*!
* Board specific initialization.
*/
@@ -229,7 +216,6 @@ static void __init mxc_board_init(void)
spi_register_board_info(mxc_spi_board_info,
ARRAY_SIZE(mxc_spi_board_info));
mxc_init_nand_mtd();
- mxc_init_pmic();
}
/*
diff --git a/arch/arm/mach-mx37/mx37_3stack_pmic_wm8350.c b/arch/arm/mach-mx37/mx37_3stack_pmic_wm8350.c
new file mode 100644
index 000000000000..22b3f8082755
--- /dev/null
+++ b/arch/arm/mach-mx37/mx37_3stack_pmic_wm8350.c
@@ -0,0 +1,496 @@
+/*
+ * mx37-3stack-pmic-wm8350.c -- i.MX37 3STACK Driver for Wolfson WM8350 PMIC
+ *
+ * Copyright 2007 Wolfson Microelectronics PLC.
+ * Copyright 2008 Freescale Semiconductor Inc.
+ *
+ * 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/init.h>
+#include <linux/delay.h>
+#include <linux/pm.h>
+#include <linux/bitops.h>
+#include <linux/platform_device.h>
+#include <linux/i2c.h>
+#include <linux/err.h>
+#include <linux/pmic/wm8350.h>
+#include <linux/interrupt.h>
+#include <linux/irq.h>
+#include <linux/leds.h>
+#include <linux/fb.h>
+#include <linux/pmic.h>
+
+#include <asm/arch/dma.h>
+#include <asm/arch/spba.h>
+#include <asm/arch/clock.h>
+#include "board-mx37_3stack.h"
+
+/*
+ * Set to 1 when testing battery that is connected otherwise spuriuos debug
+ */
+#define BATTERY 0
+
+extern const char imx37_3stack_audio[32];
+extern struct led_trigger *imx32ads_led_trigger;
+
+/* program WM8350 so board will boot from WM8350 supplies */
+static int program = 0;
+module_param(program, int, 0);
+MODULE_PARM_DESC(program, "program initial DCDC & LDO values");
+
+/*
+ * Program WM8350 with initial DCDC & LDO values.
+ */
+static int program_wm8350(struct wm8350 *wm8350)
+{
+ struct wm8350_pmic *pmic = &wm8350->pmic;
+ u16 val;
+
+ wm8350_reg_unlock(wm8350);
+
+ /* DCDC1 @ 1.6 V CPU */
+ wm8350_dcdc_set_voltage(pmic, WM8350_DCDC_1, 1600);
+ wm8350_dcdc_set_slot(pmic, WM8350_DCDC_1, 1, 0,
+ WM8350_DC1_ERRACT_SHUTDOWN_SYS);
+ wm8350_dcdc_enable(pmic, WM8350_DCDC_1, 1);
+
+ /* DCDC3 @ 2.8 V Peripherals & IO */
+ wm8350_dcdc_set_voltage(pmic, WM8350_DCDC_3, 2800);
+ wm8350_dcdc_set_slot(pmic, WM8350_DCDC_3, 1, 0, WM8350_DC1_ERRACT_NONE);
+ wm8350_dcdc_enable(pmic, WM8350_DCDC_3, 1);
+
+ /* DCDC4 @ 1.8 V Peripherals & IO */
+ wm8350_dcdc_set_voltage(pmic, WM8350_DCDC_4, 1800);
+ wm8350_dcdc_set_slot(pmic, WM8350_DCDC_4, 1, 0, WM8350_DC1_ERRACT_NONE);
+ wm8350_dcdc_enable(pmic, WM8350_DCDC_4, 1);
+
+ /* DCDC6 @ 1.8 V DDR Memory */
+ wm8350_dcdc_set_voltage(pmic, WM8350_DCDC_6, 1800);
+ wm8350_dcdc_set_slot(pmic, WM8350_DCDC_6, 1, 0, WM8350_DC1_ERRACT_NONE);
+ wm8350_dcdc_enable(pmic, WM8350_DCDC_6, 1);
+
+ /* DCDC2 @ OFF - 5.5V SW3 output */
+ wm8350_dcdc_set_slot(pmic, WM8350_DCDC_2, 1, 0,
+ WM8350_DC1_ERRACT_SHUTDOWN_CONV);
+ wm8350_dcdc_enable(pmic, WM8350_DCDC_2, 0);
+
+ /* DCDC5 @ OFF - White LEDS */
+ wm8350_dcdc_set_slot(pmic, WM8350_DCDC_5, 1, 1,
+ WM8350_DC1_ERRACT_SHUTDOWN_CONV);
+ wm8350_dcdc_enable(pmic, WM8350_DCDC_5, 0);
+
+ /* LDO1 @ 2.8 V Peripherals MMC & Camera */
+ wm8350_ldo_set_voltage(pmic, WM8350_LDO_1, 2800);
+ wm8350_ldo_set_slot(pmic, WM8350_LDO_1, 1, 0);
+ wm8350_ldo_enable(pmic, WM8350_LDO_1, 1);
+
+ /* LDO2 @ 3.3 V */
+ wm8350_ldo_set_voltage(pmic, WM8350_LDO_2, 3300);
+ wm8350_ldo_set_slot(pmic, WM8350_LDO_2, 1, 0);
+ wm8350_ldo_enable(pmic, WM8350_LDO_2, 1);
+
+ /* LDO3 @ 1.5 V VDIG, VGEN */
+ wm8350_ldo_set_voltage(pmic, WM8350_LDO_3, 1500);
+ wm8350_ldo_set_slot(pmic, WM8350_LDO_3, 1, 0);
+ wm8350_ldo_enable(pmic, WM8350_LDO_3, 1);
+
+ /* LDO4 @ 2.6 V Transceivers */
+ wm8350_ldo_set_voltage(pmic, WM8350_LDO_4, 2600);
+ wm8350_ldo_set_slot(pmic, WM8350_LDO_4, 1, 0);
+ wm8350_ldo_enable(pmic, WM8350_LDO_4, 1);
+
+ val = wm8350_reg_read(wm8350, WM8350_INTERFACE_CONTROL) &
+ ~(WM8350_RECONFIG_AT_ON | WM8350_USE_DEV_PINS |
+ WM8350_SPI_3WIRE | WM8350_SPI_4WIRE);
+ wm8350_reg_write(wm8350, WM8350_INTERFACE_CONTROL,
+ val | WM8350_CONFIG_DONE | WM8350_AUTOINC);
+
+ /* dummy read required when changing Sec -> Pri I2C */
+ printk("dummy read %x\n", wm8350_reg_read(wm8350, 176));
+ printk("now press button and then swap jumpers if leds work\n");
+ return 0;
+}
+
+#ifdef NOT_PORTED_TO_IMX37_3STACK_YET
+
+static void imx37_3stack_switch_handler(struct wm8350 *wm8350, int irq)
+{
+ printk("switch pressed %d\n", irq);
+}
+#endif
+static struct platform_device *imx_snd_device;
+
+/* i.MX CPU voltage scaling */
+struct pm_circuit imx_cpu_circuit = {
+ .name = "imx_cpu",
+ .regulator_id = WM8350_DCDC_1,
+ .load = PM_LOAD_CPU,
+ .mV_range = {
+ .dmin = 1400,
+ .dmax = 1600,
+ },
+};
+
+/* i.MX 3stack System IO */
+struct pm_circuit imx_io_hi_circuit = {
+ .name = "imx_io_hi",
+ .regulator_id = WM8350_DCDC_3,
+ .load = PM_LOAD_SYS_IO,
+ .mV_range = {
+ .sval = 2700,
+ },
+};
+
+/* i.MX 3stack System IO */
+struct pm_circuit imx_io_lo_circuit = {
+ .name = "imx_io_lo",
+ .regulator_id = WM8350_DCDC_4,
+ .load = PM_LOAD_SYS_IO,
+ .mV_range = {
+ .sval = 1800,
+ },
+};
+
+/* i.MX 3stack System IO */
+struct pm_circuit imx_io_gen_circuit = {
+ .name = "imx_io_gen",
+ .regulator_id = WM8350_LDO_3,
+ .load = PM_LOAD_SYS_IO,
+ .mV_range = {
+ .sval = 1500,
+ },
+};
+
+/* i.MX 3stack DDR RAM */
+struct pm_circuit imx_mem_circuit = {
+ .name = "imx_memory",
+ .regulator_id = WM8350_DCDC_6,
+ .load = PM_LOAD_MEM,
+ .mV_range = {
+ .sval = 1800,
+ },
+};
+
+/* i.MX 3stack & MMC & MStick */
+struct pm_circuit imx_vcam_circuit = {
+ .name = "imx_vcam",
+ .regulator_id = WM8350_LDO_1,
+ .load = PM_LOAD_MMC | PM_LOAD_MSTICK | PM_LOAD_CAMERA,
+ .mV_range = {
+ .sval = 2800,
+ },
+};
+
+/* i.MX 3stack SIM Card */
+struct pm_circuit imx_vsim_circuit = {
+ .name = "imx_vsim",
+ .regulator_id = WM8350_LDO_2,
+ .load = PM_LOAD_SIM | PM_LOAD_AUDIO,
+ .mV_range = {
+ .sval = 3300,
+ },
+};
+
+/* i.MX 3stack Tranceivers and other peripherals */
+struct pm_circuit imx_transceiver_circuit = {
+ .name = "imx_transceiver",
+ .regulator_id = WM8350_LDO_4,
+ .load = PM_LOAD_SERIAL | PM_LOAD_USB | PM_LOAD_IRDA | PM_LOAD_LCD |
+ PM_LOAD_TV | PM_LOAD_DISK | PM_LOAD_MMC,
+ .mV_range = {
+ .sval = 2800,
+ },
+};
+
+void wm8350_free(struct wm8350 *wm8350)
+{
+#if BATTERY
+ struct wm8350_power *power = &wm8350->power;
+#endif
+
+ pm_unregister_circuit(&imx_cpu_circuit);
+ pm_unregister_circuit(&imx_io_hi_circuit);
+ pm_unregister_circuit(&imx_io_lo_circuit);
+ pm_unregister_circuit(&imx_io_gen_circuit);
+ pm_unregister_circuit(&imx_mem_circuit);
+ pm_unregister_circuit(&imx_vcam_circuit);
+ pm_unregister_circuit(&imx_vsim_circuit);
+ pm_unregister_circuit(&imx_transceiver_circuit);
+#ifdef NOT_PORTED_TO_IMX37
+ /* make sure DCDC1 is back at 1.6 volts so we can safely reboot */
+ /* DCDC1 @ 1.6 V */
+ wm8350_dcdc_set_voltage(&wm8350->pmic, WM8350_DCDC_1, 1600);
+ wm8350_dcdc_set_slot(&wm8350->pmic, WM8350_DCDC_1, 1, 0,
+ WM8350_DC1_ERRACT_SHUTDOWN_SYS);
+ wm8350_dcdc_enable(&wm8350->pmic, WM8350_DCDC_1, 1);
+#endif
+ wm8350_mask_irq(wm8350, WM8350_IRQ_GPIO(7));
+ wm8350_free_irq(wm8350, WM8350_IRQ_GPIO(7));
+ wm8350_mask_irq(wm8350, WM8350_IRQ_WKUP_ONKEY);
+ wm8350_free_irq(wm8350, WM8350_IRQ_WKUP_ONKEY);
+#if BATTERY
+ wm8350_charger_enable(power, 0);
+ wm8350_fast_charger_enable(power, 0);
+#endif
+ if (wm8350->nirq)
+ free_irq(wm8350->nirq, wm8350);
+
+ flush_scheduled_work();
+#ifdef MXC_DPTC
+ mxc_pmic_unregister(&wm8350->pmic);
+#endif
+ if (wm8350->pmic.dev.is_registered)
+ device_unregister(&wm8350->pmic.dev);
+ if (wm8350->rtc.dev.is_registered)
+ device_unregister(&wm8350->rtc.dev);
+ if (wm8350->led.dev.is_registered)
+ device_unregister(&wm8350->led.dev);
+ if (wm8350->wdg.dev.is_registered)
+ device_unregister(&wm8350->wdg.dev);
+ if (wm8350->power.dev.is_registered)
+ device_unregister(&wm8350->power.dev);
+ if (wm8350->backlight.dev.is_registered)
+ device_unregister(&wm8350->backlight.dev);
+ platform_device_unregister(imx_snd_device);
+}
+
+#if BATTERY
+static int wm8350_init_battery(struct wm8350 *wm8350)
+{
+ struct wm8350_power *power = &wm8350->power;
+ struct wm8350_charger_policy *policy = &power->policy;
+
+ policy->eoc_mA = WM8350_CHG_EOC_mA(10);
+ policy->charge_mV = WM8350_CHG_4_05V;
+ policy->fast_limit_mA = WM8350_CHG_FAST_LIMIT_mA(400);
+ policy->charge_timeout = WM8350_CHG_TIME_MIN(60);
+ policy->trickle_start_mV = WM8350_CHG_TRICKLE_3_1V;
+ policy->trickle_charge_mA = WM8350_CHG_TRICKLE_50mA;
+
+ wm8350_charger_enable(power, 1);
+ wm8350_fast_charger_enable(power, 1);
+ return 0;
+}
+#endif
+
+#ifdef NOT_PORTED_TO_IMX37_3STACK_YET
+static int config_gpios(struct wm8350 *wm8350)
+{
+ /* power on */
+ wm8350_gpio_config(wm8350, 0, WM8350_GPIO_DIR_IN,
+ WM8350_GPIO0_PWR_ON_IN, WM8350_GPIO_ACTIVE_LOW,
+ WM8350_GPIO_PULL_UP, WM8350_GPIO_INVERT_OFF,
+ WM8350_GPIO_DEBOUNCE_ON);
+
+ /* Sw3 --> PWR_OFF_GPIO3 */
+ /* lg - TODO: GPIO1_0 to be pulled down */
+ wm8350_gpio_config(wm8350, 3, WM8350_GPIO_DIR_IN,
+ WM8350_GPIO3_PWR_OFF_IN, WM8350_GPIO_ACTIVE_HIGH,
+ WM8350_GPIO_PULL_DOWN, WM8350_GPIO_INVERT_OFF,
+ WM8350_GPIO_DEBOUNCE_ON);
+
+ /* MR or MEMRST ????? */
+ wm8350_gpio_config(wm8350, 4, WM8350_GPIO_DIR_IN,
+ WM8350_GPIO4_MR_IN, WM8350_GPIO_ACTIVE_HIGH,
+ WM8350_GPIO_PULL_DOWN, WM8350_GPIO_INVERT_OFF,
+ WM8350_GPIO_DEBOUNCE_OFF);
+
+ /* Hibernate -- needs level but GPIO5 is edge so we
+ * will only wake with PWR_ON switch atm */
+ wm8350_gpio_config(wm8350, 5, WM8350_GPIO_DIR_IN,
+ WM8350_GPIO5_HIBERNATE_IN, WM8350_GPIO_ACTIVE_HIGH,
+ WM8350_GPIO_PULL_DOWN, WM8350_GPIO_INVERT_OFF,
+ WM8350_GPIO_DEBOUNCE_OFF);
+
+ /* SDOUT */
+ wm8350_gpio_config(wm8350, 6, WM8350_GPIO_DIR_OUT,
+ WM8350_GPIO6_SDOUT_OUT, WM8350_GPIO_ACTIVE_HIGH,
+ WM8350_GPIO_PULL_NONE, WM8350_GPIO_INVERT_OFF,
+ WM8350_GPIO_DEBOUNCE_OFF);
+
+ /* GPIO switch SW2 */
+ wm8350_gpio_config(wm8350, 7, WM8350_GPIO_DIR_IN, WM8350_GPIO7_GPIO_IN,
+ WM8350_GPIO_ACTIVE_HIGH, WM8350_GPIO_PULL_DOWN,
+ WM8350_GPIO_INVERT_OFF, WM8350_GPIO_DEBOUNCE_ON);
+ wm8350_register_irq(wm8350, WM8350_IRQ_GPIO(7),
+ imx37_3stack_switch_handler);
+ wm8350_unmask_irq(wm8350, WM8350_IRQ_GPIO(7));
+
+ /* PWR_FAIL */
+ wm8350_gpio_config(wm8350, 8, WM8350_GPIO_DIR_OUT,
+ WM8350_GPIO8_VCC_FAULT_OUT, WM8350_GPIO_ACTIVE_LOW,
+ WM8350_GPIO_PULL_NONE, WM8350_GPIO_INVERT_OFF,
+ WM8350_GPIO_DEBOUNCE_OFF);
+
+ /* BATT Fault */
+ wm8350_gpio_config(wm8350, 9, WM8350_GPIO_DIR_OUT,
+ WM8350_GPIO9_BATT_FAULT_OUT, WM8350_GPIO_ACTIVE_LOW,
+ WM8350_GPIO_PULL_NONE, WM8350_GPIO_INVERT_OFF,
+ WM8350_GPIO_DEBOUNCE_OFF);
+
+ return 0;
+}
+
+static int config_hibernate(struct wm8350 *wm8350)
+{
+ struct wm8350_pmic *pmic = &wm8350->pmic;
+
+ /* dont assert RTS when hibernating */
+ wm8350_set_bits(wm8350, WM8350_SYSTEM_HIBERNATE, WM8350_RST_HIB_MODE);
+
+ /* set up hibernate voltages -- needs refining */
+ wm8350_dcdc_set_image_voltage(pmic, WM8350_DCDC_1, 1400, /*1200, *//* 1.0v for mx32 */
+ WM8350_DCDC_HIB_MODE_IMAGE,
+ WM8350_DCDC_HIB_SIG_REG);
+ wm8350_dcdc_set_image_voltage(pmic, WM8350_DCDC_3, 2800,
+ WM8350_DCDC_HIB_MODE_IMAGE,
+ WM8350_DCDC_HIB_SIG_REG);
+ wm8350_dcdc_set_image_voltage(pmic, WM8350_DCDC_4, 1800,
+ WM8350_DCDC_HIB_MODE_IMAGE,
+ WM8350_DCDC_HIB_SIG_REG);
+ wm8350_dcdc_set_image_voltage(pmic, WM8350_DCDC_6, 1800,
+ WM8350_DCDC_HIB_MODE_IMAGE,
+ WM8350_DCDC_HIB_SIG_REG);
+ wm8350_ldo_set_image_voltage(pmic, WM8350_LDO_1, 2800,
+ WM8350_LDO_HIB_MODE_IMAGE,
+ WM8350_LDO_HIB_SIG_REG);
+ wm8350_ldo_set_image_voltage(pmic, WM8350_LDO_2, 3300,
+ WM8350_LDO_HIB_MODE_IMAGE,
+ WM8350_LDO_HIB_SIG_REG);
+ wm8350_ldo_set_image_voltage(pmic, WM8350_LDO_3, 1500,
+ WM8350_LDO_HIB_MODE_IMAGE,
+ WM8350_LDO_HIB_SIG_REG);
+ wm8350_ldo_set_image_voltage(pmic, WM8350_LDO_4, 2500,
+ WM8350_LDO_HIB_MODE_IMAGE,
+ WM8350_LDO_HIB_MODE_DIS);
+ return 0;
+}
+#endif //NOT_PORTED_TO_IMX37_3STACK_YET
+
+int wm8350_init(struct wm8350 *wm8350)
+{
+ int ret;
+
+ if (program) {
+ /* once only...
+ * or if VRTC batt discharges and there is no PIC */
+ program_wm8350(wm8350);
+ return -EBUSY;
+ }
+
+ /* configure system power circuits */
+ pm_register_circuit(&imx_cpu_circuit);
+ pm_register_circuit(&imx_io_hi_circuit);
+ pm_register_circuit(&imx_io_lo_circuit);
+ pm_register_circuit(&imx_io_gen_circuit);
+ pm_register_circuit(&imx_mem_circuit);
+ pm_register_circuit(&imx_vcam_circuit);
+ pm_register_circuit(&imx_vsim_circuit);
+ pm_register_circuit(&imx_transceiver_circuit);
+
+ /* backlight properties */
+ wm8350->backlight.props.max_brightness = 10;
+ wm8350->backlight.props.power = FB_BLANK_UNBLANK;
+ wm8350->backlight.props.brightness = 5;
+ wm8350->backlight.dcdc = WM8350_DCDC_2;
+ wm8350->backlight.isink = WM8350_ISINK_B;
+ wm8350->backlight.retries = 5;
+
+ /* LED properties */
+ wm8350->led.dcdc = WM8350_DCDC_5;
+ wm8350->led.isink = WM8350_ISINK_A;
+ wm8350->led.retries = 5;
+ wm8350->led.half_value = 986376;
+ wm8350->led.full_value = 1972752;
+
+ /* RTC periodic irq :-
+ * If we had a dedicated RTC PER IRQ pin we could add it here. */
+ wm8350->rtc.per_irq = 0;
+
+ /* now register all i.MX37 3STACK WM8350 client devices */
+ wm8350_device_register_pmic(wm8350);
+#ifdef NOT_PORTED_TO_IMX37
+ wm8350_device_register_rtc(wm8350);
+ wm8350_device_register_wdg(wm8350);
+ wm8350_device_register_led(wm8350);
+ wm8350_device_register_power(wm8350);
+ wm8350_device_register_backlight(wm8350);
+ /* make sure DCDC1 is back at 1.6 volts so we can safely work */
+ /* DCDC1 @ 1.6 V */
+ wm8350_dcdc_set_voltage(&wm8350->pmic, WM8350_DCDC_1, 1600);
+ wm8350_dcdc_set_slot(&wm8350->pmic, WM8350_DCDC_1, 1, 0,
+ WM8350_DC1_ERRACT_SHUTDOWN_SYS);
+ wm8350_dcdc_enable(&wm8350->pmic, WM8350_DCDC_1, 1);
+#endif
+ /* register sound */
+ printk("Registering imx37_snd_device");
+ imx_snd_device = platform_device_alloc(imx37_3stack_audio, -1);
+ if (!imx_snd_device) {
+ ret = -ENOMEM;
+ goto err;
+ }
+ platform_set_drvdata(imx_snd_device, &wm8350->audio);
+ ret = platform_device_add(imx_snd_device);
+ if (ret)
+ goto snd_err;
+
+ /* set up PMIC IRQ (active high) to i.MX32ADS */
+#ifdef NOT_PORTED_TO_IMX37
+ printk("Registering PMIC INT");
+ INIT_WORK(&wm8350->work, wm8350_irq_work);
+ wm8350_reg_unlock(wm8350);
+ wm8350_set_bits(wm8350, WM8350_SYSTEM_CONTROL_1, WM8350_IRQ_POL);
+ wm8350_reg_lock(wm8350);
+ set_irq_type(MXC_PMIC_INT_LINE, IRQT_RISING);
+ ret = request_irq(MXC_PMIC_INT_LINE, wm8350_irq_handler,
+ IRQF_DISABLED, "wm8350-pmic", wm8350);
+ if (ret != 0) {
+ printk(KERN_ERR "wm8350: cant request irq %d\n",
+ MXC_PMIC_INT_LINE);
+ goto err;
+ }
+ wm8350->nirq = MXC_PMIC_INT_LINE;
+ printk("Configuring WM8350 GPIOS");
+ config_gpios(wm8350);
+ config_hibernate(wm8350);
+
+ /* Sw1 --> PWR_ON */
+ printk("Registering and unmasking the WM8350 wakeup key");
+ wm8350_register_irq(wm8350, WM8350_IRQ_WKUP_ONKEY,
+ imx37_3stack_switch_handler);
+ wm8350_unmask_irq(wm8350, WM8350_IRQ_WKUP_ONKEY);
+
+ /* unmask all & clear sticky */
+ printk("Unmasking WM8350 local interrupts");
+ wm8350_reg_write(wm8350, WM8350_SYSTEM_INTERRUPTS_MASK, 0x0);
+ schedule_work(&wm8350->work);
+#endif
+
+#ifdef MXC_DPTC
+ /* DPTC reg */
+ mxc_pmic_register(&wm8350->pmic, wm8350_dcdc_set_voltage);
+#endif
+
+#if BATTERY
+ /* not much use without a battery atm */
+ wm8350_init_battery(wm8350);
+#endif
+ printk("Exiting normally from wm8350_init()");
+ return ret;
+ snd_err:
+ platform_device_put(imx_snd_device);
+ err:
+ printk("wm8350_init() FAILED");
+ kfree(wm8350->reg_cache);
+ return ret;
+}
diff --git a/drivers/pmic/Makefile b/drivers/pmic/Makefile
index 9b556e115b45..cab9157a938a 100644
--- a/drivers/pmic/Makefile
+++ b/drivers/pmic/Makefile
@@ -3,11 +3,12 @@
#
ifeq ($(CONFIG_PMIC_DEBUG),y)
- EXTRA_CFLAGS += -DDEBUG
+ EXTRA_CFLAGS += -DDEBUG
endif
obj-$(CONFIG_PMIC) += pmic.o
obj-$(CONFIG_PMIC_WM8350) += \
- pmic-wm8350.o pmic-wm8350-bus.o
+ pmic-wm8350.o pmic-wm8350-bus.o pmic-wm8350-i2c.o
+
diff --git a/drivers/pmic/pmic-wm8350-i2c.c b/drivers/pmic/pmic-wm8350-i2c.c
new file mode 100644
index 000000000000..9851c159a38f
--- /dev/null
+++ b/drivers/pmic/pmic-wm8350-i2c.c
@@ -0,0 +1,167 @@
+#include <linux/module.h>
+#include <linux/init.h>
+#include <linux/delay.h>
+#include <linux/pm.h>
+#include <linux/bitops.h>
+#include <linux/platform_device.h>
+#include <linux/i2c.h>
+#include <linux/err.h>
+#include <linux/interrupt.h>
+#include <linux/irq.h>
+#include <linux/leds.h>
+#include <linux/fb.h>
+#include <linux/pmic.h>
+#include <linux/pmic/wm8350.h>
+
+#define WM8350_I2C_ADDR (0x34 >> 1)
+
+extern void wm8350_free(struct wm8350 *wm8350);
+extern int wm8350_init(struct wm8350* wm8350);
+
+
+static int wm8350_i2c_detach(struct i2c_client *client);
+
+void wm8350_irq_work(struct work_struct *work)
+{
+ wm8350_irq_worker(work);
+}
+
+irqreturn_t wm8350_irq_handler(int irq, void *data)
+{
+ struct wm8350 *wm8350 = (struct wm8350*)data;
+ schedule_work(&wm8350->work);
+ return IRQ_HANDLED;
+}
+
+/*
+ * WM8350 2 wire address
+ */
+static unsigned short normal_i2c[] = { WM8350_I2C_ADDR, I2C_CLIENT_END };
+
+/* Magic definition of all other variables and things */
+I2C_CLIENT_INSMOD;
+
+static int wm8350_pmic_i2c_detect(struct i2c_adapter *adap, int addr, int kind)
+{
+ struct wm8350* wm8350;
+ struct i2c_client *i2c;
+ int ret = 0;
+
+ if (addr != WM8350_I2C_ADDR)
+ return -ENODEV;
+
+ i2c = kzalloc(sizeof(struct i2c_client), GFP_KERNEL);
+ if (i2c == NULL)
+ return -ENOMEM;
+
+ wm8350 = kzalloc(sizeof(struct wm8350), GFP_KERNEL);
+ if (wm8350 == NULL) {
+ kfree(i2c);
+ return -ENOMEM;
+ }
+
+ i2c->addr = addr;
+ i2c->adapter = adap;
+
+ mutex_init(&wm8350->work_mutex);
+ i2c_set_clientdata(i2c, wm8350);
+ wm8350->i2c_client = i2c;
+ wm8350_set_io(wm8350, WM8350_IO_I2C, NULL, NULL);
+
+ ret = i2c_attach_client(i2c);
+ if (ret < 0) {
+ printk(KERN_ERR "wm8350: failed to attach device at addr 0x%x\n", addr);
+ goto err;
+ }
+
+ ret = wm8350_create_cache(wm8350);
+ if (ret < 0) {
+ printk(KERN_ERR "wm8350: failed to create register cache\n");
+ goto err;
+ }
+
+ if (wm8350_reg_read(wm8350, 0) == 0x0)
+ printk("wm8350: found Rev C device\n");
+ else if (wm8350_reg_read(wm8350, 0) == 0x6143)
+ printk("wm8350: found Rev E device\n");
+ else {
+ printk(KERN_ERR "wm8350: device is not a WM8350\n");
+ ret = -ENODEV;
+ goto err;
+ }
+
+ ret = wm8350_init(wm8350);
+ if (ret == 0)
+ return ret;
+
+err:
+ wm8350_i2c_detach(i2c);
+ return ret;
+}
+
+static int wm8350_i2c_attach(struct i2c_adapter *adap)
+{
+ return i2c_probe(adap, &addr_data, wm8350_pmic_i2c_detect);
+}
+
+static int wm8350_i2c_detach(struct i2c_client *client)
+{
+ struct wm8350 *wm8350 = i2c_get_clientdata(client);
+
+ wm8350_free(wm8350);
+ i2c_detach_client(client);
+ kfree(client);
+ if (wm8350->reg_cache)
+ kfree(wm8350->reg_cache);
+ kfree(wm8350);
+
+ return 0;
+}
+
+#ifdef CONFIG_PM
+static int wm8350_i2c_suspend(struct i2c_client *client, pm_message_t state)
+{
+ int ret = 0;
+ return ret;
+}
+
+static int wm8350_i2c_resume(struct i2c_client *client)
+{
+ int ret = 0;
+ return ret;
+}
+
+#else
+#define wm8350_i2c_suspend NULL
+#define wm8350_i2c_resume NULL
+#endif
+
+static struct i2c_driver wm8350_i2c_driver = {
+ .driver = {
+ .name = "WM8350",
+ },
+ .attach_adapter = wm8350_i2c_attach,
+ .detach_client = wm8350_i2c_detach,
+ .suspend = wm8350_i2c_suspend,
+ .resume = wm8350_i2c_resume,
+ .command = NULL,
+};
+
+
+static int __init wm8350_pmic_init(void)
+{
+ return i2c_add_driver(&wm8350_i2c_driver);
+}
+
+static void __exit wm8350_pmic_exit(void)
+{
+ i2c_del_driver(&wm8350_i2c_driver);
+}
+
+module_init(wm8350_pmic_init);
+module_exit(wm8350_pmic_exit);
+
+MODULE_AUTHOR("Liam Girdwood");
+MODULE_DESCRIPTION("PMIC WM8350 Driver");
+MODULE_LICENSE("GPL");
+
diff --git a/sound/soc/Kconfig b/sound/soc/Kconfig
index 97b255233175..8578f7a6785f 100644
--- a/sound/soc/Kconfig
+++ b/sound/soc/Kconfig
@@ -28,6 +28,7 @@ source "sound/soc/at91/Kconfig"
source "sound/soc/pxa/Kconfig"
source "sound/soc/s3c24xx/Kconfig"
source "sound/soc/sh/Kconfig"
+source "sound/soc/imx/Kconfig"
# Supported codecs
source "sound/soc/codecs/Kconfig"
diff --git a/sound/soc/Makefile b/sound/soc/Makefile
index 304140377632..94e96a78a7e6 100644
--- a/sound/soc/Makefile
+++ b/sound/soc/Makefile
@@ -1,4 +1,4 @@
snd-soc-core-objs := soc-core.o soc-dapm.o
obj-$(CONFIG_SND_SOC) += snd-soc-core.o
-obj-$(CONFIG_SND_SOC) += codecs/ at91/ pxa/ s3c24xx/ sh/
+obj-$(CONFIG_SND_SOC) += codecs/ at91/ pxa/ s3c24xx/ sh/ imx/
diff --git a/sound/soc/imx/Kconfig b/sound/soc/imx/Kconfig
new file mode 100644
index 000000000000..ed3da5c4e8f6
--- /dev/null
+++ b/sound/soc/imx/Kconfig
@@ -0,0 +1,21 @@
+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 I2S or SSP interface. You will also need
+ to select the audio interfaces to support below.
+
+config SND_MXC_SOC_SSI
+ tristate
+
+config SND_SOC_IMX37_3STACK_WM8350
+ tristate "SoC Audio support for MX37 - WM8350"
+ select SND_MXC_SOC_SSI
+ select SND_SOC_WM8350
+ select PMIC_WM8350
+ help
+ Say Y if you want to add support for SoC audio on IMX37 3STACK
+ with the WM8350.
+
diff --git a/sound/soc/imx/Makefile b/sound/soc/imx/Makefile
new file mode 100644
index 000000000000..d84f385cf921
--- /dev/null
+++ b/sound/soc/imx/Makefile
@@ -0,0 +1,10 @@
+# i.MX Platform Support
+snd-soc-imx-objs := imx-pcm.o
+snd-soc-imx-ssi-objs := imx-ssi.o
+
+obj-$(CONFIG_SND_MXC_SOC) += snd-soc-imx.o
+obj-$(CONFIG_SND_MXC_SOC_SSI) += snd-soc-imx-ssi.o
+
+# i.MX Machine Support
+snd-soc-imx37-3stack-wm8350-objs := imx37-3stack-wm8350.o
+obj-$(CONFIG_SND_SOC_IMX37_3STACK_WM8350) += snd-soc-imx37-3stack-wm8350.o
diff --git a/sound/soc/imx/imx-pcm.c b/sound/soc/imx/imx-pcm.c
new file mode 100644
index 000000000000..e39ced46c935
--- /dev/null
+++ b/sound/soc/imx/imx-pcm.c
@@ -0,0 +1,549 @@
+/*
+ * imx-pcm.c -- ALSA SoC interface for the Freescale i.MX3 CPU's
+ *
+ * Copyright 2006 Wolfson Microelectronics PLC.
+ * Author: Liam Girdwood
+ * liam.girdwood@wolfsonmicro.com or linux@wolfsonmicro.com
+ *
+ * Based on imx31-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.
+ *
+ *
+ */
+
+#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/dma.h>
+#include <asm/arch/spba.h>
+#include <asm/arch/clock.h>
+#include <asm/mach-types.h>
+#include <asm/hardware.h>
+
+#include "imx-pcm.h"
+
+/* debug */
+#define IMX_PCM_DEBUG 0
+#if IMX_PCM_DEBUG
+#define dbg(format, arg...) printk(format, ## arg)
+#else
+#define dbg(format, arg...)
+#endif
+
+/*
+ * Coherent DMA memory is used by default, although Freescale have used
+ * bounce buffers in all their drivers for i.MX31 to date. If you have any
+ * issues, please select bounce buffers.
+ */
+#define IMX31_DMA_BOUNCE 0
+
+static const struct snd_pcm_hardware imx_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 imx_pcm_dma_param *dma_params;
+ spinlock_t dma_lock;
+ int active, period, periods;
+ int dma_wchannel;
+ int dma_active;
+ int old_offset;
+ int dma_alloc;
+};
+
+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 long flags;
+#if IMX31_DMA_BOUNCE
+ unsigned int dma_size;
+ unsigned int offset;
+
+ dma_size = frames_to_bytes(runtime, runtime->period_size);
+ offset = dma_size * prtd->periods;
+#endif
+
+ /* stops the dma channel and clears the buffer ptrs */
+ spin_lock_irqsave(&prtd->dma_lock, flags);
+ prtd->active = 0;
+ prtd->period = 0;
+ prtd->periods = 0;
+ mxc_dma_disable(prtd->dma_wchannel);
+
+#if IMX31_DMA_BOUNCE
+ if (substream->stream == 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);
+#endif
+ spin_unlock_irqrestore(&prtd->dma_lock, flags);
+}
+
+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 = frames_to_bytes(runtime, runtime->period_size);
+ unsigned int offset = dma_size * prtd->period;
+ int ret = 0;
+ mxc_dma_requestbuf_t sdma_request;
+
+ if (!prtd->active)
+ return 0;
+ //printk(KERN_EMERG"In func %s \n",__FUNCTION__);
+ memset(&sdma_request, 0, sizeof(mxc_dma_requestbuf_t));
+
+ dbg("period pos ALSA %x DMA %x\n", runtime->periods, prtd->period);
+ dbg("period size ALSA %x DMA %x Offset %x dmasize %x\n",
+ (unsigned int)runtime->period_size, runtime->dma_bytes,
+ offset, dma_size);
+ dbg("DMA addr %x\n", runtime->dma_addr + offset);
+
+#if IMX31_DMA_BOUNCE
+ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)
+ sdma_request.src_addr = (dma_addr_t) (dma_map_single(NULL,
+ runtime->
+ dma_area +
+ offset,
+ dma_size,
+ DMA_TO_DEVICE));
+ else
+ sdma_request.dst_addr = (dma_addr_t) (dma_map_single(NULL,
+ runtime->
+ dma_area +
+ offset,
+ dma_size,
+ DMA_FROM_DEVICE));
+#else
+ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)
+ sdma_request.src_addr =
+ (dma_addr_t) (runtime->dma_addr + offset);
+ else
+ sdma_request.dst_addr =
+ (dma_addr_t) (runtime->dma_addr + offset);
+
+#endif
+ sdma_request.num_of_bytes = dma_size;
+
+ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
+ mxc_dma_config(prtd->dma_wchannel, &sdma_request, 1,
+ MXC_DMA_MODE_WRITE);
+ ret = mxc_dma_enable(prtd->dma_wchannel);
+ } else {
+
+ mxc_dma_config(prtd->dma_wchannel, &sdma_request, 1,
+ MXC_DMA_MODE_READ);
+ ret = mxc_dma_enable(prtd->dma_wchannel);
+ }
+ prtd->dma_active = 1;
+ prtd->period++;
+ prtd->period %= runtime->periods;
+
+ return ret;
+}
+
+static void audio_dma_irq(void *data)
+{
+ struct snd_pcm_substream *substream = (struct snd_pcm_substream *)data;
+ struct snd_pcm_runtime *runtime = substream->runtime;
+ struct mxc_runtime_data *prtd = runtime->private_data;
+#if IMX31_DMA_BOUNCE
+ unsigned int dma_size = frames_to_bytes(runtime, runtime->period_size);
+ unsigned int offset = dma_size * prtd->periods;
+#endif
+
+ //printk(KERN_EMERG"In func %s \n",__FUNCTION__);
+ prtd->dma_active = 0;
+ prtd->periods++;
+ prtd->periods %= runtime->periods;
+
+ dbg("irq per %d offset %x\n", prtd->periods,
+ frames_to_bytes(runtime, runtime->period_size) * prtd->periods);
+#if IMX31_DMA_BOUNCE
+ if (substream->stream == 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);
+
+#endif
+
+ if (prtd->active)
+ snd_pcm_period_elapsed(substream);
+ dma_new_period(substream);
+}
+
+static int imx_pcm_prepare(struct snd_pcm_substream *substream)
+{
+ struct snd_pcm_runtime *runtime = substream->runtime;
+ struct mxc_runtime_data *prtd = runtime->private_data;
+
+ prtd->period = 0;
+ prtd->periods = 0;
+ return 0;
+}
+
+static int imx_pcm_hw_params(struct snd_pcm_substream *substream,
+ struct snd_pcm_hw_params *params)
+{
+ struct snd_pcm_runtime *runtime = substream->runtime;
+ struct mxc_runtime_data *prtd = runtime->private_data;
+ struct snd_soc_pcm_link *pcm_link = substream->private_data;
+ struct snd_soc_dai *cpu_dai = pcm_link->cpu_dai;
+ struct mxc_pcm_dma_params *dma = cpu_dai->dma_data;
+ int ret = 0, channel = 0;
+
+ /* only allocate the DMA chn once */
+ if (!prtd->dma_alloc) {
+ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
+
+ channel =
+ mxc_dma_request(MXC_DMA_SSI1_16BIT_TX0,
+ "ALSA TX SDMA");
+ if (channel < 0) {
+ printk(KERN_ERR
+ "imx-pcm: error requesting a write dma channel\n");
+ return channel;
+ }
+ ret = mxc_dma_callback_set(channel, (mxc_dma_callback_t)
+ audio_dma_irq,
+ (void *)substream);
+
+ } else {
+ channel =
+ mxc_dma_request(MXC_DMA_SSI1_16BIT_RX0,
+ "ALSA RX SDMA");
+ if (channel < 0) {
+ printk(KERN_ERR
+ "imx-pcm: error requesting a read dma channel\n");
+ return channel;
+ }
+ }
+ prtd->dma_wchannel = channel;
+ prtd->dma_alloc = 1;
+
+ /* set up chn with params */
+ // dma->params.callback = audio_dma_irq;
+ dma->params.arg = substream;
+
+ switch (params_format(params)) {
+ case SNDRV_PCM_FORMAT_S16_LE:
+ dma->params.word_size = TRANSFER_16BIT;
+ break;
+ case SNDRV_PCM_FORMAT_S20_3LE:
+ case SNDRV_PCM_FORMAT_S24_LE:
+ dma->params.word_size = TRANSFER_24BIT;
+ break;
+ }
+
+ }
+#if IMX31_DMA_BOUNCE
+ ret = snd_pcm_lib_malloc_pages(substream, params_buffer_bytes(params));
+ if (ret < 0) {
+ printk(KERN_ERR "imx-pcm: failed to malloc pcm pages\n");
+ if (channel)
+ mxc_dma_free(channel);
+ return ret;
+ }
+ runtime->dma_addr = virt_to_phys(runtime->dma_area);
+#else
+ snd_pcm_set_runtime_buffer(substream, &substream->dma_buffer);
+#endif
+ return ret;
+}
+
+static int imx_pcm_hw_free(struct snd_pcm_substream *substream)
+{
+ struct snd_pcm_runtime *runtime = substream->runtime;
+ struct mxc_runtime_data *prtd = runtime->private_data;
+
+ if (prtd->dma_wchannel) {
+ mxc_dma_free(prtd->dma_wchannel);
+ prtd->dma_wchannel = 0;
+ prtd->dma_alloc = 0;
+ }
+#if IMX31_DMA_BOUNCE
+ snd_pcm_lib_free_pages(substream);
+#endif
+ return 0;
+}
+
+static int imx_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->dma_active = 0;
+ /* requested stream startup */
+ prtd->active = 1;
+ ret = dma_new_period(substream);
+ break;
+ case SNDRV_PCM_TRIGGER_STOP:
+ /* requested stream shutdown */
+ 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->dma_active = 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->dma_active = 0;
+ ret = dma_new_period(substream);
+ }
+ break;
+ default:
+ ret = -EINVAL;
+ break;
+ }
+
+ return ret;
+}
+
+static snd_pcm_uframes_t imx_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;
+
+ /* is a transfer active ? */
+ if (prtd->dma_active) {
+ 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 * (prtd->periods));
+ if (offset >= runtime->buffer_size)
+ offset = 0;
+ }
+ dbg("pointer offset %x\n", offset);
+
+ return offset;
+}
+
+static int imx_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, &imx_pcm_hardware);
+
+ ret = snd_pcm_hw_constraint_integer(runtime,
+ SNDRV_PCM_HW_PARAM_PERIODS);
+ if (ret < 0)
+ return ret;
+
+ prtd = kzalloc(sizeof(struct mxc_runtime_data), GFP_KERNEL);
+ if (prtd == NULL)
+ return -ENOMEM;
+
+ runtime->private_data = prtd;
+ return 0;
+}
+
+static int imx_pcm_close(struct snd_pcm_substream *substream)
+{
+ struct snd_pcm_runtime *runtime = substream->runtime;
+ struct mxc_runtime_data *prtd = runtime->private_data;
+
+ kfree(prtd);
+ return 0;
+}
+
+static int
+imx_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 imx_pcm_ops = {
+ .open = imx_pcm_open,
+ .close = imx_pcm_close,
+ .ioctl = snd_pcm_lib_ioctl,
+ .hw_params = imx_pcm_hw_params,
+ .hw_free = imx_pcm_hw_free,
+ .prepare = imx_pcm_prepare,
+ .trigger = imx_pcm_trigger,
+ .pointer = imx_pcm_pointer,
+ .mmap = imx_pcm_mmap,
+};
+
+static int imx_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 = imx_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 imx_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 imx_pcm_dmamask = 0xffffffff;
+
+static int imx_pcm_new(struct snd_soc_platform *platform,
+ struct snd_card *card, int playback, int capture,
+ struct snd_pcm *pcm)
+{
+ int ret = 0;
+
+ if (!card->dev->dma_mask)
+ card->dev->dma_mask = &imx_pcm_dmamask;
+ if (!card->dev->coherent_dma_mask)
+ card->dev->coherent_dma_mask = 0xffffffff;
+#if IMX31_DMA_BOUNCE
+ ret = snd_pcm_lib_preallocate_pages_for_all(pcm,
+ SNDRV_DMA_TYPE_CONTINUOUS,
+ snd_dma_continuous_data
+ (GFP_KERNEL),
+ imx_pcm_hardware.
+ buffer_bytes_max * 2,
+ imx_pcm_hardware.
+ buffer_bytes_max * 2);
+ if (ret < 0) {
+ printk(KERN_ERR "imx-pcm: failed to preallocate pages\n");
+ goto out;
+ }
+#else
+ if (playback) {
+ ret = imx_pcm_preallocate_dma_buffer(pcm,
+ SNDRV_PCM_STREAM_PLAYBACK);
+ if (ret)
+ goto out;
+ }
+
+ if (capture) {
+ ret = imx_pcm_preallocate_dma_buffer(pcm,
+ SNDRV_PCM_STREAM_CAPTURE);
+ if (ret)
+ goto out;
+ }
+#endif
+ out:
+ return ret;
+}
+
+static const struct snd_soc_platform_ops imx_platform_ops = {
+ .pcm_new = imx_pcm_new,
+#if IMX31_DMA_BOUNCE
+ .pcm_free = NULL,
+#else
+ .pcm_free = imx_pcm_free_dma_buffers,
+#endif
+};
+
+static int imx_pcm_probe(struct device *dev)
+{
+ struct snd_soc_platform *platform = to_snd_soc_platform(dev);
+
+ platform->pcm_ops = &imx_pcm_ops;
+ platform->platform_ops = &imx_platform_ops;
+ snd_soc_register_platform(platform);
+ return 0;
+}
+
+static int imx_pcm_remove(struct device *dev)
+{
+ return 0;
+}
+
+const char imx_pcm[SND_SOC_PLATFORM_NAME_SIZE] = "imx-pcm";
+EXPORT_SYMBOL_GPL(imx_pcm);
+
+static struct snd_soc_device_driver imx_pcm_driver = {
+ .type = SND_SOC_BUS_TYPE_DMA,
+ .driver = {
+ .name = imx_pcm,
+ .owner = THIS_MODULE,
+ .bus = &asoc_bus_type,
+ .probe = imx_pcm_probe,
+ .remove = __devexit_p(imx_pcm_remove),
+ },
+};
+
+static __init int imx_pcm_init(void)
+{
+ return driver_register(&imx_pcm_driver.driver);
+}
+
+static __exit void imx_pcm_exit(void)
+{
+ driver_unregister(&imx_pcm_driver.driver);
+}
+
+module_init(imx_pcm_init);
+module_exit(imx_pcm_exit);
+
+MODULE_AUTHOR("Liam Girdwood");
+MODULE_DESCRIPTION("Freescale i.MX3x PCM DMA module");
+MODULE_LICENSE("GPL");
diff --git a/sound/soc/imx/imx-pcm.h b/sound/soc/imx/imx-pcm.h
new file mode 100644
index 000000000000..7cf7b75ed714
--- /dev/null
+++ b/sound/soc/imx/imx-pcm.h
@@ -0,0 +1,67 @@
+/*
+ * imx-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
+
+#include <asm/arch/dma.h>
+
+/* AUDMUX regs definition */
+#define AUDMUX_IO_BASE_ADDR IO_ADDRESS(AUDMUX_BASE_ADDR)
+
+#define DAM_PTCR1 (*((volatile u32 *)(AUDMUX_IO_BASE_ADDR + 0x00)))
+#define DAM_PDCR1 (*((volatile u32 *)(AUDMUX_IO_BASE_ADDR + 0x04)))
+#define DAM_PTCR2 (*((volatile u32 *)(AUDMUX_IO_BASE_ADDR + 0x08)))
+#define DAM_PDCR2 (*((volatile u32 *)(AUDMUX_IO_BASE_ADDR + 0x0C)))
+#define DAM_PTCR3 (*((volatile u32 *)(AUDMUX_IO_BASE_ADDR + 0x10)))
+#define DAM_PDCR3 (*((volatile u32 *)(AUDMUX_IO_BASE_ADDR + 0x14)))
+#define DAM_PTCR4 (*((volatile u32 *)(AUDMUX_IO_BASE_ADDR + 0x18)))
+#define DAM_PDCR4 (*((volatile u32 *)(AUDMUX_IO_BASE_ADDR + 0x1C)))
+#define DAM_PTCR5 (*((volatile u32 *)(AUDMUX_IO_BASE_ADDR + 0x20)))
+#define DAM_PDCR5 (*((volatile u32 *)(AUDMUX_IO_BASE_ADDR + 0x24)))
+#define DAM_PTCR6 (*((volatile u32 *)(AUDMUX_IO_BASE_ADDR + 0x28)))
+#define DAM_PDCR6 (*((volatile u32 *)(AUDMUX_IO_BASE_ADDR + 0x2C)))
+#define DAM_PTCR7 (*((volatile u32 *)(AUDMUX_IO_BASE_ADDR + 0x30)))
+#define DAM_PDCR7 (*((volatile u32 *)(AUDMUX_IO_BASE_ADDR + 0x34)))
+#define DAM_CNMCR (*((volatile u32 *)(AUDMUX_IO_BASE_ADDR + 0x38)))
+#define DAM_PTCR(a) (*((volatile u32 *)(AUDMUX_IO_BASE_ADDR + a*8)))
+#define DAM_PDCR(a) (*((volatile u32 *)(AUDMUX_IO_BASE_ADDR + 4 + a*8)))
+
+#define AUDMUX_PTCR_TFSDIR (1 << 31)
+#define AUDMUX_PTCR_TFSSEL(x, y) ((x << 30) | (((y - 1) & 0x7) << 27))
+#define AUDMUX_PTCR_TCLKDIR (1 << 26)
+#define AUDMUX_PTCR_TCSEL(x, y) ((x << 25) | (((y - 1) & 0x7) << 22))
+#define AUDMUX_PTCR_RFSDIR (1 << 21)
+#define AUDMUX_PTCR_RFSSEL(x, y) ((x << 20) | (((y - 1) & 0x7) << 17))
+#define AUDMUX_PTCR_RCLKDIR (1 << 16)
+#define AUDMUX_PTCR_RCSEL(x, y) ((x << 15) | (((y - 1) & 0x7) << 12))
+#define AUDMUX_PTCR_SYN (1 << 11)
+
+#define AUDMUX_FROM_TXFS 0
+#define AUDMUX_FROM_RXFS 1
+
+#define AUDMUX_PDCR_RXDSEL(x) (((x - 1) & 0x7) << 13)
+#define AUDMUX_PDCR_TXDXEN (1 << 12)
+#define AUDMUX_PDCR_MODE(x) (((x) & 0x3) << 8)
+#define AUDMUX_PDCR_INNMASK(x) (((x) & 0xff) << 0)
+
+#define AUDMUX_CNMCR_CEN (1 << 18)
+#define AUDMUX_CNMCR_FSPOL (1 << 17)
+#define AUDMUX_CNMCR_CLKPOL (1 << 16)
+#define AUDMUX_CNMCR_CNTHI(x) (((x) & 0xff) << 8)
+#define AUDMUX_CNMCR_CNTLOW(x) (((x) & 0xff) << 0)
+
+struct mxc_pcm_dma_params {
+ char *name; /* stream identifier */
+ dma_channel_params params;
+};
+
+/* platform data */
+extern const char imx_pcm[SND_SOC_PLATFORM_NAME_SIZE];
+
+#endif
diff --git a/sound/soc/imx/imx-ssi.c b/sound/soc/imx/imx-ssi.c
new file mode 100644
index 000000000000..cf63305d0ca5
--- /dev/null
+++ b/sound/soc/imx/imx-ssi.c
@@ -0,0 +1,862 @@
+/*
+ * 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.
+ *
+ * TODO:
+ * Need to rework SSI register defs when new defs go into mainline.
+ * Add support for TDM and FIFO 1.
+ *
+ */
+
+#include <linux/module.h>
+#include <linux/init.h>
+#include <linux/platform_device.h>
+#include <linux/slab.h>
+#include <linux/dma-mapping.h>
+#include <linux/clk.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/dma.h>
+#include <asm/arch/clock.h>
+#include <asm/mach-types.h>
+#include <asm/hardware.h>
+
+#include "imx-ssi.h"
+#include "imx-pcm.h"
+
+/* debug */
+#define IMX_SSI_DEBUG 0
+#if IMX_SSI_DEBUG
+#define dbg(format, arg...) printk(format, ## arg)
+#else
+#define dbg(format, arg...)
+#endif
+
+#define IMX_SSI_DUMP 0
+#if IMX_SSI_DUMP
+#define SSI_DUMP() \
+ printk("dump @ %s\n", __FUNCTION__); \
+ printk("scr %x\n", SSI1_SCR); \
+ printk("sisr %x\n", SSI1_SISR); \
+ printk("stcr %x\n", SSI1_STCR); \
+ printk("srcr %x\n", SSI1_SRCR); \
+ printk("stccr %x\n", SSI1_STCCR); \
+ printk("srccr %x\n", SSI1_SRCCR); \
+ printk("sfcsr %x\n", SSI1_SFCSR); \
+ printk("stmsk %x\n", SSI1_STMSK); \
+ printk("srmsk %x\n", SSI1_SRMSK); \
+ printk("sier %x\n", SSI1_SIER);
+#else
+#define SSI_DUMP()
+#endif
+
+#define SSI1_PORT 0
+#define SSI2_PORT 1
+
+static int ssi_active[2] = { 0, 0 };
+
+static struct mxc_pcm_dma_params imx_ssi1_pcm_stereo_out0 = {
+ .name = "SSI1 PCM Stereo out 0",
+ .params = {
+ .bd_number = 32,
+ .word_size = TRANSFER_16BIT,
+ .transfer_type = emi_2_per,
+ .watermark_level = SDMA_TXFIFO_WATERMARK,
+ .per_address = SSI1_BASE_ADDR,
+ .event_id = DMA_REQ_SSI1_TX1,
+ .peripheral_type = SSI,
+ },
+};
+
+static struct mxc_pcm_dma_params imx_ssi1_pcm_stereo_out1 = {
+ .name = "SSI1 PCM Stereo out 1",
+ .params = {
+ .bd_number = 32,
+ .word_size = TRANSFER_16BIT,
+ .transfer_type = emi_2_per,
+ .watermark_level = SDMA_TXFIFO_WATERMARK,
+ .per_address = SSI1_BASE_ADDR + 0x4,
+ .event_id = DMA_REQ_SSI1_TX2,
+ .peripheral_type = SSI,
+ },
+};
+
+static struct mxc_pcm_dma_params imx_ssi1_pcm_stereo_in0 = {
+ .name = "SSI1 PCM Stereo in 0",
+ .params = {
+ .bd_number = 1,
+ .transfer_type = per_2_emi,
+ .watermark_level = SDMA_RXFIFO_WATERMARK,
+ .per_address = SSI1_BASE_ADDR + 0x8,
+ .event_id = DMA_REQ_SSI1_RX1,
+ .peripheral_type = SSI,
+ },
+};
+
+static struct mxc_pcm_dma_params imx_ssi1_pcm_stereo_in1 = {
+ .name = "SSI1 PCM Stereo in 1",
+ .params = {
+ .bd_number = 1,
+ .transfer_type = per_2_emi,
+ .watermark_level = SDMA_RXFIFO_WATERMARK,
+ .per_address = SSI1_BASE_ADDR + 0xc,
+ .event_id = DMA_REQ_SSI1_RX2,
+ .peripheral_type = SSI,
+ },
+};
+
+static struct mxc_pcm_dma_params imx_ssi2_pcm_stereo_out0 = {
+ .name = "SSI2 PCM Stereo out 0",
+ .params = {
+ .bd_number = 1,
+ .transfer_type = emi_2_per,
+ .watermark_level = SDMA_TXFIFO_WATERMARK,
+ .per_address = SSI2_BASE_ADDR,
+ .event_id = DMA_REQ_SSI2_TX1,
+ .peripheral_type = SSI,
+ },
+};
+
+static struct mxc_pcm_dma_params imx_ssi2_pcm_stereo_out1 = {
+ .name = "SSI2 PCM Stereo out 1",
+ .params = {
+ .bd_number = 1,
+ .transfer_type = emi_2_per,
+ .watermark_level = SDMA_TXFIFO_WATERMARK,
+ .per_address = SSI2_BASE_ADDR + 0x4,
+ .event_id = DMA_REQ_SSI2_TX2,
+ .peripheral_type = SSI,
+ },
+};
+
+static struct mxc_pcm_dma_params imx_ssi2_pcm_stereo_in0 = {
+ .name = "SSI2 PCM Stereo in 0",
+ .params = {
+ .bd_number = 1,
+ .transfer_type = per_2_emi,
+ .watermark_level = SDMA_RXFIFO_WATERMARK,
+ .per_address = SSI2_BASE_ADDR + 0x8,
+ .event_id = DMA_REQ_SSI2_RX1,
+ .peripheral_type = SSI,
+ },
+};
+
+static struct mxc_pcm_dma_params imx_ssi2_pcm_stereo_in1 = {
+ .name = "SSI2 PCM Stereo in 1",
+ .params = {
+ .bd_number = 1,
+ .transfer_type = per_2_emi,
+ .watermark_level = SDMA_RXFIFO_WATERMARK,
+ .per_address = SSI2_BASE_ADDR + 0xc,
+ .event_id = DMA_REQ_SSI2_RX2,
+ .peripheral_type = SSI,
+ },
+};
+
+/*
+ * SSI system clock configuration.
+ * Should only be called when port is inactive (i.e. SSIEN = 0).
+ */
+static int imx_ssi_set_dai_sysclk(struct snd_soc_dai *cpu_dai,
+ int clk_id, unsigned int freq, int dir)
+{
+ u32 scr;
+
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2)
+ scr = SSI1_SCR;
+ else
+ scr = SSI2_SCR;
+
+ if (scr & SSI_SCR_SSIEN)
+ return 0;
+
+ switch (clk_id) {
+ case IMX_SSP_SYS_CLK:
+ if (dir == SND_SOC_CLOCK_OUT)
+ scr |= SSI_SCR_SYS_CLK_EN;
+ else
+ scr &= ~SSI_SCR_SYS_CLK_EN;
+ break;
+ default:
+ return -EINVAL;
+ }
+
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2)
+ SSI1_SCR = scr;
+ else
+ SSI2_SCR = scr;
+
+ return 0;
+}
+
+/*
+ * SSI Clock dividers
+ * Should only be called when port is inactive (i.e. SSIEN = 0).
+ */
+static int imx_ssi_set_dai_clkdiv(struct snd_soc_dai *cpu_dai,
+ int div_id, int div)
+{
+ u32 stccr, srccr;
+
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2) {
+ if (SSI1_SCR & SSI_SCR_SSIEN)
+ return 0;
+
+ srccr = SSI1_STCCR;
+ stccr = SSI1_STCCR;
+ } else {
+ if (SSI2_SCR & SSI_SCR_SSIEN)
+ return 0;
+
+ srccr = SSI2_STCCR;
+ stccr = SSI2_STCCR;
+ }
+
+ switch (div_id) {
+ case IMX_SSI_TX_DIV_2:
+ stccr &= ~SSI_STCCR_DIV2;
+ stccr |= div;
+ break;
+ case IMX_SSI_TX_DIV_PSR:
+ stccr &= ~SSI_STCCR_PSR;
+ stccr |= div;
+ break;
+ case IMX_SSI_TX_DIV_PM:
+ stccr &= ~0xff;
+ stccr |= SSI_STCCR_PM(div);
+ break;
+ case IMX_SSI_RX_DIV_2:
+ stccr &= ~SSI_STCCR_DIV2;
+ stccr |= div;
+ break;
+ case IMX_SSI_RX_DIV_PSR:
+ stccr &= ~SSI_STCCR_PSR;
+ stccr |= div;
+ break;
+ case IMX_SSI_RX_DIV_PM:
+ stccr &= ~0xff;
+ stccr |= SSI_STCCR_PM(div);
+ break;
+ default:
+ return -EINVAL;
+ }
+
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2) {
+ SSI1_STCCR = stccr;
+ SSI1_SRCCR = srccr;
+ } else {
+ SSI2_STCCR = stccr;
+ SSI2_SRCCR = srccr;
+ }
+ return 0;
+}
+
+/*
+ * SSI Network Mode or TDM slots configuration.
+ * Should only be called when port is inactive (i.e. SSIEN = 0).
+ */
+static int imx_ssi_set_dai_tdm_slot(struct snd_soc_dai *cpu_dai,
+ unsigned int mask, int slots)
+{
+ u32 stmsk, srmsk, stccr;
+
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2) {
+ if (SSI1_SCR & SSI_SCR_SSIEN)
+ return 0;
+ stccr = SSI1_STCCR;
+ } else {
+ if (SSI2_SCR & SSI_SCR_SSIEN)
+ return 0;
+ stccr = SSI2_STCCR;
+ }
+
+ stmsk = srmsk = mask;
+ stccr &= ~SSI_STCCR_DC_MASK;
+ stccr |= SSI_STCCR_DC(slots - 1);
+
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2) {
+ SSI1_STMSK = stmsk;
+ SSI1_SRMSK = srmsk;
+ SSI1_SRCCR = SSI1_STCCR = stccr;
+ } else {
+ SSI2_STMSK = stmsk;
+ SSI2_SRMSK = srmsk;
+ SSI2_SRCCR = SSI2_STCCR = stccr;
+ }
+
+ return 0;
+}
+
+/*
+ * SSI DAI format configuration.
+ * Should only be called when port is inactive (i.e. SSIEN = 0).
+ * Note: We don't use the I2S modes but instead manually configure the
+ * SSI for I2S.
+ */
+static int imx_ssi_set_dai_fmt(struct snd_soc_dai *cpu_dai, unsigned int fmt)
+{
+ u32 stcr = 0, srcr = 0, scr;
+
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2)
+ scr = SSI1_SCR & ~(SSI_SCR_SYN | SSI_SCR_NET);
+ else
+ scr = SSI2_SCR & ~(SSI_SCR_SYN | SSI_SCR_NET);
+
+ if (scr & SSI_SCR_SSIEN)
+ return 0;
+
+ /* DAI mode */
+ switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
+ case SND_SOC_DAIFMT_I2S:
+ /* data on rising edge of bclk, frame low 1clk before data */
+ stcr |= SSI_STCR_TFSI | SSI_STCR_TEFS | SSI_STCR_TXBIT0;
+ srcr |= SSI_SRCR_RFSI | SSI_SRCR_REFS | SSI_SRCR_RXBIT0;
+ break;
+ case SND_SOC_DAIFMT_LEFT_J:
+ /* data on rising edge of bclk, frame high with data */
+ stcr |= SSI_STCR_TXBIT0;
+ srcr |= SSI_SRCR_RXBIT0;
+ break;
+ case SND_SOC_DAIFMT_DSP_B:
+ /* data on rising edge of bclk, frame high with data */
+ stcr |= SSI_STCR_TFSL;
+ srcr |= SSI_SRCR_RFSL;
+ break;
+ case SND_SOC_DAIFMT_DSP_A:
+ /* data on rising edge of bclk, frame high 1clk before data */
+ stcr |= SSI_STCR_TFSL | SSI_STCR_TEFS;
+ srcr |= SSI_SRCR_RFSL | SSI_SRCR_REFS;
+ break;
+ }
+
+ /* DAI clock inversion */
+ switch (fmt & SND_SOC_DAIFMT_INV_MASK) {
+ case SND_SOC_DAIFMT_IB_IF:
+ stcr |= SSI_STCR_TFSI;
+ stcr &= ~SSI_STCR_TSCKP;
+ srcr |= SSI_SRCR_RFSI;
+ srcr &= ~SSI_SRCR_RSCKP;
+ break;
+ case SND_SOC_DAIFMT_IB_NF:
+ stcr &= ~(SSI_STCR_TSCKP | SSI_STCR_TFSI);
+ srcr &= ~(SSI_SRCR_RSCKP | SSI_SRCR_RFSI);
+ break;
+ case SND_SOC_DAIFMT_NB_IF:
+ stcr |= SSI_STCR_TFSI | SSI_STCR_TSCKP;
+ srcr |= SSI_SRCR_RFSI | SSI_SRCR_RSCKP;
+ break;
+ case SND_SOC_DAIFMT_NB_NF:
+ stcr &= ~SSI_STCR_TFSI;
+ stcr |= SSI_STCR_TSCKP;
+ srcr &= ~SSI_SRCR_RFSI;
+ srcr |= SSI_SRCR_RSCKP;
+ break;
+ }
+
+ /* DAI clock master masks */
+ switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) {
+ case SND_SOC_DAIFMT_CBS_CFS:
+ stcr |= SSI_STCR_TFDIR | SSI_STCR_TXDIR;
+ srcr |= SSI_SRCR_RFDIR | SSI_SRCR_RXDIR;
+ break;
+ case SND_SOC_DAIFMT_CBM_CFS:
+ stcr |= SSI_STCR_TFDIR;
+ srcr |= SSI_SRCR_RFDIR;
+ break;
+ case SND_SOC_DAIFMT_CBS_CFM:
+ stcr |= SSI_STCR_TXDIR;
+ srcr |= SSI_SRCR_RXDIR;
+ break;
+ }
+
+ /* sync */
+ if (!(fmt & SND_SOC_DAIFMT_ASYNC))
+ scr |= SSI_SCR_SYN;
+
+ /* tdm - only for stereo atm */
+ if (fmt & SND_SOC_DAIFMT_TDM)
+ scr |= SSI_SCR_NET;
+
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2) {
+ SSI1_STCR = stcr;
+ SSI1_SRCR = srcr;
+ SSI1_SCR = scr;
+ } else {
+ SSI2_STCR = stcr;
+ SSI2_SRCR = srcr;
+ SSI1_SCR = scr;
+ }
+ SSI_DUMP();
+ return 0;
+}
+
+static struct clk *ssi_clk;
+
+static int imx_ssi_startup(struct snd_pcm_substream *substream)
+{
+ struct snd_soc_pcm_link *pcm_link = substream->private_data;
+ struct snd_soc_dai *cpu_dai = pcm_link->cpu_dai;
+
+ /* we cant really change any SSI values after SSI is enabled
+ * need to fix in software for max flexibility - lrg */
+ if (cpu_dai->active)
+ return 0;
+
+ /* reset the SSI port - Sect 45.4.4 */
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2) {
+
+ if (ssi_active[SSI1_PORT]++)
+ return 0;
+
+ SSI1_SCR = 0;
+ ssi_clk = clk_get(NULL, "ssi_clk.0");
+ clk_enable(ssi_clk);
+
+ /* BIG FAT WARNING
+ * SDMA FIFO watermark must == SSI FIFO watermark for
+ * best results.
+ */
+ SSI1_SFCSR = SSI_SFCSR_RFWM1(SSI_RXFIFO_WATERMARK) |
+ SSI_SFCSR_RFWM0(SSI_RXFIFO_WATERMARK) |
+ SSI_SFCSR_TFWM1(SSI_TXFIFO_WATERMARK) |
+ SSI_SFCSR_TFWM0(SSI_TXFIFO_WATERMARK);
+ } else {
+
+ if (ssi_active[SSI2_PORT]++)
+ return 0;
+
+ SSI2_SCR = 0;
+
+ /* above warning applies here too */
+ SSI2_SFCSR = SSI_SFCSR_RFWM1(SSI_RXFIFO_WATERMARK) |
+ SSI_SFCSR_RFWM0(SSI_RXFIFO_WATERMARK) |
+ SSI_SFCSR_TFWM1(SSI_TXFIFO_WATERMARK) |
+ SSI_SFCSR_TFWM0(SSI_TXFIFO_WATERMARK);
+ }
+
+ SSI_DUMP();
+ return 0;
+}
+
+static int imx_ssi_hw_tx_params(struct snd_pcm_substream *substream,
+ struct snd_pcm_hw_params *params)
+{
+ struct snd_soc_pcm_link *pcm_link = substream->private_data;
+ struct snd_soc_dai *cpu_dai = pcm_link->cpu_dai;
+ u32 stccr, stcr, sier;
+
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2) {
+ stccr = SSI1_STCCR & ~SSI_STCCR_WL_MASK;
+ stcr = SSI1_STCR;
+ sier = SSI1_SIER;
+ } else {
+ stccr = SSI2_STCCR & ~SSI_STCCR_WL_MASK;
+ stcr = SSI2_STCR;
+ sier = SSI2_SIER;
+ }
+
+ /* DAI data (word) size */
+ switch (params_format(params)) {
+ case SNDRV_PCM_FORMAT_S16_LE:
+ stccr |= SSI_STCCR_WL(16);
+ break;
+ case SNDRV_PCM_FORMAT_S20_3LE:
+ stccr |= SSI_STCCR_WL(20);
+ break;
+ case SNDRV_PCM_FORMAT_S24_LE:
+ stccr |= SSI_STCCR_WL(24);
+ break;
+ }
+
+ /* enable interrupts */
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2)
+ stcr |= SSI_STCR_TFEN0;
+ else
+ stcr |= SSI_STCR_TFEN1;
+ sier |= SSI_SIER_TDMAE | SSI_SIER_TFE0_EN | SSI_SIER_TFE1_EN |
+ SSI_SIER_TUE0_EN | SSI_SIER_TUE1_EN;
+
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2) {
+ SSI1_STCR = stcr;
+ SSI1_STCCR = stccr;
+ SSI1_SIER = sier;
+ } else {
+ SSI2_STCR = stcr;
+ SSI2_STCCR = stccr;
+ SSI2_SIER = sier;
+ }
+
+ return 0;
+}
+
+static int imx_ssi_hw_rx_params(struct snd_pcm_substream *substream,
+ struct snd_pcm_hw_params *params)
+{
+ struct snd_soc_pcm_link *pcm_link = substream->private_data;
+ struct snd_soc_dai *cpu_dai = pcm_link->cpu_dai;
+ u32 srccr, srcr, sier;
+
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2) {
+ srccr = SSI1_SRCCR & ~SSI_SRCCR_WL_MASK;
+ srcr = SSI1_SRCR;
+ sier = SSI1_SIER;
+ } else {
+ srccr = SSI2_SRCCR & ~SSI_SRCCR_WL_MASK;
+ srcr = SSI2_SRCR;
+ sier = SSI2_SIER;
+ }
+
+ /* DAI data (word) size */
+ switch (params_format(params)) {
+ case SNDRV_PCM_FORMAT_S16_LE:
+ srccr |= SSI_SRCCR_WL(16);
+ break;
+ case SNDRV_PCM_FORMAT_S20_3LE:
+ srccr |= SSI_SRCCR_WL(20);
+ break;
+ case SNDRV_PCM_FORMAT_S24_LE:
+ srccr |= SSI_SRCCR_WL(24);
+ break;
+ }
+
+ /* enable interrupts */
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2)
+ srcr |= SSI_SRCR_RFEN0;
+ else
+ srcr |= SSI_SRCR_RFEN1;
+ sier |= SSI_SIER_RDMAE | SSI_SIER_RFF0_EN | SSI_SIER_RFF1_EN |
+ SSI_SIER_ROE0_EN | SSI_SIER_ROE1_EN;;
+
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2) {
+ SSI1_SRCR = srcr;
+ SSI1_SRCCR = srccr;
+ SSI1_SIER = sier;
+ } else {
+ SSI2_SRCR = srcr;
+ SSI2_SRCCR = srccr;
+ SSI2_SIER = sier;
+ }
+ return 0;
+}
+
+/*
+ * Should only be called when port is inactive (i.e. SSIEN = 0),
+ * although can be called multiple times by upper layers.
+ */
+static int imx_ssi_hw_params(struct snd_pcm_substream *substream,
+ struct snd_pcm_hw_params *params)
+{
+ struct snd_soc_pcm_link *pcm_link = substream->private_data;
+ struct snd_soc_dai *cpu_dai = pcm_link->cpu_dai;
+
+ /* Tx/Rx config */
+ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
+ /* set up DMA params */
+ switch (cpu_dai->id) {
+ case IMX_DAI_SSI0:
+ cpu_dai->dma_data = &imx_ssi1_pcm_stereo_out0;
+ break;
+ case IMX_DAI_SSI1:
+ cpu_dai->dma_data = &imx_ssi1_pcm_stereo_out1;
+ break;
+ case IMX_DAI_SSI2:
+ cpu_dai->dma_data = &imx_ssi2_pcm_stereo_out0;
+ break;
+ case IMX_DAI_SSI3:
+ cpu_dai->dma_data = &imx_ssi2_pcm_stereo_out1;
+ break;
+ }
+
+ /* cant change any parameters when SSI is running */
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2) {
+ if (SSI1_SCR & SSI_SCR_SSIEN)
+ return 0;
+ } else {
+ if (SSI2_SCR & SSI_SCR_SSIEN)
+ return 0;
+ }
+ return imx_ssi_hw_tx_params(substream, params);
+ } else {
+ /* set up DMA params */
+ switch (cpu_dai->id) {
+ case IMX_DAI_SSI0:
+ cpu_dai->dma_data = &imx_ssi1_pcm_stereo_in0;
+ break;
+ case IMX_DAI_SSI1:
+ cpu_dai->dma_data = &imx_ssi1_pcm_stereo_in1;
+ break;
+ case IMX_DAI_SSI2:
+ cpu_dai->dma_data = &imx_ssi2_pcm_stereo_in0;
+ break;
+ case IMX_DAI_SSI3:
+ cpu_dai->dma_data = &imx_ssi2_pcm_stereo_in1;
+ break;
+ }
+
+ /* cant change any parameters when SSI is running */
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2) {
+ if (SSI1_SCR & SSI_SCR_SSIEN)
+ return 0;
+ } else {
+ if (SSI2_SCR & SSI_SCR_SSIEN)
+ return 0;
+ }
+ return imx_ssi_hw_rx_params(substream, params);
+ }
+}
+
+static int imx_ssi_prepare(struct snd_pcm_substream *substream)
+{
+ struct snd_soc_pcm_link *pcm_link = substream->private_data;
+ struct snd_soc_dai *cpu_dai = pcm_link->cpu_dai;
+ u32 scr;
+
+ /* enable the SSI port, note that no other port config
+ * should happen after SSIEN is set */
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2) {
+ scr = SSI1_SCR;
+ SSI1_SCR = scr | SSI_SCR_SSIEN;
+ } else {
+ scr = SSI2_SCR;
+ SSI2_SCR = scr | SSI_SCR_SSIEN;
+ }
+ SSI_DUMP();
+ return 0;
+}
+
+static int imx_ssi_trigger(struct snd_pcm_substream *substream, int cmd)
+{
+ struct snd_soc_pcm_link *pcm_link = substream->private_data;
+ struct snd_soc_dai *cpu_dai = pcm_link->cpu_dai;
+ u32 scr;
+
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2)
+ scr = SSI1_SCR;
+ else
+ scr = SSI2_SCR;
+
+ switch (cmd) {
+ case SNDRV_PCM_TRIGGER_START:
+ case SNDRV_PCM_TRIGGER_RESUME:
+ case SNDRV_PCM_TRIGGER_PAUSE_RELEASE:
+ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)
+ scr |= SSI_SCR_TE;
+ else
+ scr |= SSI_SCR_RE;
+ break;
+ case SNDRV_PCM_TRIGGER_SUSPEND:
+ case SNDRV_PCM_TRIGGER_STOP:
+ case SNDRV_PCM_TRIGGER_PAUSE_PUSH:
+ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)
+ scr &= ~SSI_SCR_TE;
+ else
+ scr &= ~SSI_SCR_RE;
+ break;
+ default:
+ return -EINVAL;
+ }
+
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2)
+ SSI1_SCR = scr;
+ else
+ SSI2_SCR = scr;
+
+ SSI_DUMP();
+ return 0;
+}
+
+static void imx_ssi_shutdown(struct snd_pcm_substream *substream)
+{
+ struct snd_soc_pcm_link *pcm_link = substream->private_data;
+ struct snd_soc_dai *cpu_dai = pcm_link->cpu_dai;
+
+ /* shutdown SSI if neither Tx or Rx is active */
+ if (!cpu_dai->active) {
+
+ if (cpu_dai->id == IMX_DAI_SSI0 || cpu_dai->id == IMX_DAI_SSI2) {
+
+ if (--ssi_active[SSI1_PORT] > 1)
+ return;
+
+ SSI1_SCR = 0;
+
+ clk_put(ssi_clk);
+
+ } else {
+ if (--ssi_active[SSI2_PORT])
+ return;
+ SSI2_SCR = 0;
+
+ }
+ }
+}
+
+#ifdef CONFIG_PM
+static int imx_ssi_suspend(struct device *dev, pm_message_t state)
+{
+ struct snd_soc_dai *dai = to_snd_soc_dai(dev);
+
+ if (!dai->active)
+ return 0;
+
+ // do we need to disable any clocks
+
+ return 0;
+}
+
+static int imx_ssi_resume(struct device *dev)
+{
+ struct snd_soc_dai *dai = to_snd_soc_dai(dev);
+
+ if (!dai->active)
+ return 0;
+
+ // do we need to enable any clocks
+ return 0;
+}
+
+#else
+#define imx_ssi_suspend NULL
+#define imx_ssi_resume NULL
+#endif
+
+#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 const struct snd_soc_pcm_stream imx_ssi_playback = {
+ .stream_name = "Playback",
+ .channels_min = 1,
+ .channels_max = 2,
+ .rates = IMX_SSI_RATES,
+ .formats = IMX_SSI_BITS,
+};
+
+static const struct snd_soc_pcm_stream imx_ssi_capture = {
+ .stream_name = "Capture",
+ .channels_min = 1,
+ .channels_max = 2,
+ .rates = IMX_SSI_RATES,
+ .formats = IMX_SSI_BITS,
+};
+
+/* dai ops, called by machine drivers */
+static const struct snd_soc_dai_ops imx_ssi_dai_ops = {
+ .set_sysclk = imx_ssi_set_dai_sysclk,
+ .set_clkdiv = imx_ssi_set_dai_clkdiv,
+ .set_fmt = imx_ssi_set_dai_fmt,
+ .set_tdm_slot = imx_ssi_set_dai_tdm_slot,
+};
+
+/* audio ops, called by alsa */
+static const struct snd_soc_ops imx_ssi_audio_ops = {
+ .startup = imx_ssi_startup,
+ .shutdown = imx_ssi_shutdown,
+ .trigger = imx_ssi_trigger,
+ .prepare = imx_ssi_prepare,
+ .hw_params = imx_ssi_hw_params,
+};
+
+const char imx_ssi_1[SND_SOC_DAI_NAME_SIZE] = {
+ "imx-ssi-1"
+};
+
+EXPORT_SYMBOL_GPL(imx_ssi_1);
+
+const char imx_ssi_2[SND_SOC_DAI_NAME_SIZE] = {
+ "imx-ssi-2"
+};
+
+EXPORT_SYMBOL_GPL(imx_ssi_2);
+
+const char imx_ssi_3[SND_SOC_DAI_NAME_SIZE] = {
+ "imx-ssi-3"
+};
+
+EXPORT_SYMBOL_GPL(imx_ssi_3);
+
+const char imx_ssi_4[SND_SOC_DAI_NAME_SIZE] = {
+ "imx-ssi-4"
+};
+
+EXPORT_SYMBOL_GPL(imx_ssi_4);
+
+static int imx_ssi_probe(struct device *dev)
+{
+ struct snd_soc_dai *dai = to_snd_soc_dai(dev);
+
+ if (!strcmp(imx_ssi_1, dai->name))
+ dai->id = IMX_DAI_SSI0;
+ else if (!strcmp(imx_ssi_2, dai->name))
+ dai->id = IMX_DAI_SSI1;
+ else if (!strcmp(imx_ssi_3, dai->name))
+ dai->id = IMX_DAI_SSI2;
+ else if (!strcmp(imx_ssi_4, dai->name))
+ dai->id = IMX_DAI_SSI3;
+ else {
+ printk(KERN_ERR "%s: invalid device %s\n", __func__, dai->name);
+ return -ENODEV;
+ }
+
+ dai->type = SND_SOC_DAI_PCM;
+ dai->ops = &imx_ssi_dai_ops;
+ dai->audio_ops = &imx_ssi_audio_ops;
+ dai->capture = &imx_ssi_capture;
+ dai->playback = &imx_ssi_playback;
+ snd_soc_register_cpu_dai(dai);
+ return 0;
+}
+
+static struct snd_soc_device_driver imx_ssi_driver = {
+ .type = SND_SOC_BUS_TYPE_DAI,
+ .driver = {
+ .name = "imx-ssi",
+ .owner = THIS_MODULE,
+ .bus = &asoc_bus_type,
+ .probe = imx_ssi_probe,
+ .suspend = imx_ssi_suspend,
+ .resume = imx_ssi_resume,
+ },
+};
+
+static int __init imx_ssi_init(void)
+{
+ return driver_register(&imx_ssi_driver.driver);
+}
+
+static void __exit imx_ssi_exit(void)
+{
+ driver_unregister(&imx_ssi_driver.driver);
+}
+
+module_init(imx_ssi_init);
+module_exit(imx_ssi_exit);
+
+/* Module information */
+MODULE_AUTHOR
+ ("Liam Girdwood, liam.girdwood@wolfsonmicro.com, www.wolfsonmicro.com");
+MODULE_DESCRIPTION("i.MX ASoC I2S driver");
+MODULE_LICENSE("GPL");
diff --git a/sound/soc/imx/imx-ssi.h b/sound/soc/imx/imx-ssi.h
new file mode 100644
index 000000000000..8f4d12c4ca7c
--- /dev/null
+++ b/sound/soc/imx/imx-ssi.h
@@ -0,0 +1,220 @@
+/*
+ * 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 _IMX_SSI_H
+#define _IMX_SSI_H
+
+#include <asm/arch/hardware.h>
+
+/* SSI regs definition */
+#define SSI1_IO_BASE_ADDR IO_ADDRESS(SSI1_BASE_ADDR)
+#define SSI2_IO_BASE_ADDR IO_ADDRESS(SSI2_BASE_ADDR)
+
+#define SSI1_STX0 *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x00))
+#define SSI1_STX1 *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x04))
+#define SSI1_SRX0 *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x08))
+#define SSI1_SRX1 *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x0c))
+#define SSI1_SCR *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x10))
+#define SSI1_SISR *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x14))
+#define SSI1_SIER *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x18))
+#define SSI1_STCR *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x1c))
+#define SSI1_SRCR *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x20))
+#define SSI1_STCCR *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x24))
+#define SSI1_SRCCR *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x28))
+#define SSI1_SFCSR *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x2c))
+#define SSI1_STR *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x30))
+#define SSI1_SOR *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x34))
+#define SSI1_SACNT *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x38))
+#define SSI1_SACADD *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x3c))
+#define SSI1_SACDAT *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x40))
+#define SSI1_SATAG *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x44))
+#define SSI1_STMSK *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x48))
+#define SSI1_SRMSK *((volatile u32 *)(SSI1_IO_BASE_ADDR + 0x4c))
+
+#define SSI2_STX0 *((volatile u32 *)(SSI2_IO_BASE_ADDR + 0x00))
+#define SSI2_STX1 *((volatile u32 *)(SSI2_IO_BASE_ADDR + 0x04))
+#define SSI2_SRX0 *((volatile u32 *)(SSI2_IO_BASE_ADDR + 0x08))
+#define SSI2_SRX1 *((volatile u32 *)(SSI2_IO_BASE_ADDR + 0x0c))
+#define SSI2_SCR *((volatile u32 *)(SSI2_IO_BASE_ADDR + 0x10))
+#define SSI2_SISR *((volatile u32 *)(SSI2_IO_BASE_ADDR + 0x14))
+#define SSI2_SIER *((volatile u32 *)(SSI2_IO_BASE_ADDR + 0x18))
+#define SSI2_STCR *((volatile u32 *)(SSI2_IO_BASE_ADDR + 0x1c))
+#define SSI2_SRCR *((volatile u32 *)(SSI2_IO_BASE_ADDR + 0x20))
+#define SSI2_STCCR *((volatile u32 *)(SSI2_IO_BASE_ADDR + 0x24))
+#define SSI2_SRCCR *((volatile u32 *)(SSI2_IO_BASE_ADDR + 0x28))
+#define SSI2_SFCSR *((volatile u32 *)(SSI2_IO_BASE_ADDR + 0x2c))
+#define SSI2_STR *((volatile u32 *)(SSI2_IO_BASE_ADDR + 0x30))
+#define SSI2_SOR *((volatile u32 *)(SSI2_IO_BASE_ADDR + 0x34))
+#define SSI2_SACNT *((volatile u32 *)(SSI2_IO_BASE_ADDR + 0x38))
+#define SSI2_SACADD *((volatile u32 *)(SSI2_IO_BASE_ADDR + 0x3c))
+#define SSI2_SACDAT *((volatile u32 *)(SSI2_IO_BASE_ADDR + 0x40))
+#define SSI2_SATAG *((volatile u32 *)(SSI2_IO_BASE_ADDR + 0x44))
+#define SSI2_STMSK *((volatile u32 *)(SSI2_IO_BASE_ADDR + 0x48))
+#define SSI2_SRMSK *((volatile u32 *)(SSI2_IO_BASE_ADDR + 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_STCCR_WL_MASK (0xf << 13)
+#define SSI_STCCR_DC_MASK (0x1f << 8)
+#define SSI_STCCR_PM_MASK (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_SRCCR_WL_MASK (0xf << 13)
+#define SSI_SRCCR_DC_MASK (0x1f << 8)
+#define SSI_SRCCR_PM_MASK (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_AC97EN (x << 0)
+
+/* SDMA & SSI watermarks for FIFO's */
+#define SDMA_TXFIFO_WATERMARK 0x2
+#define SDMA_RXFIFO_WATERMARK 0x6
+#define SSI_TXFIFO_WATERMARK 0x2
+#define SSI_RXFIFO_WATERMARK 0x6
+
+/* i.MX DAI SSP ID's */
+#define IMX_DAI_SSI0 0 /* SSI1 FIFO 0 */
+#define IMX_DAI_SSI1 1 /* SSI1 FIFO 1 */
+#define IMX_DAI_SSI2 2 /* SSI2 FIFO 0 */
+#define IMX_DAI_SSI3 3 /* SSI2 FIFO 1 */
+
+/* SSI clock sources */
+#define IMX_SSP_SYS_CLK 0
+
+/* SSI audio dividers */
+#define IMX_SSI_TX_DIV_2 0
+#define IMX_SSI_TX_DIV_PSR 1
+#define IMX_SSI_TX_DIV_PM 2
+#define IMX_SSI_RX_DIV_2 3
+#define IMX_SSI_RX_DIV_PSR 4
+#define IMX_SSI_RX_DIV_PM 5
+
+
+/* SSI Div 2 */
+#define IMX_SSI_DIV_2_OFF ~SSI_STCCR_DIV2
+#define IMX_SSI_DIV_2_ON SSI_STCCR_DIV2
+
+extern const char imx_ssi_1[SND_SOC_DAI_NAME_SIZE];
+extern const char imx_ssi_2[SND_SOC_DAI_NAME_SIZE];
+extern const char imx_ssi_3[SND_SOC_DAI_NAME_SIZE];
+extern const char imx_ssi_4[SND_SOC_DAI_NAME_SIZE];
+
+#endif
diff --git a/sound/soc/imx/imx37-3stack-wm8350.c b/sound/soc/imx/imx37-3stack-wm8350.c
new file mode 100644
index 000000000000..461e6beecf68
--- /dev/null
+++ b/sound/soc/imx/imx37-3stack-wm8350.c
@@ -0,0 +1,600 @@
+/*
+ * imx37-3stack-wm8350.c -- i.MX37 3Stack Driver for Wolfson WM8350 Codec
+ *
+ * Copyright 2007 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
+ * 19th Jun 2007 Initial version.
+ *
+ */
+
+#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/i2c.h>
+#include <linux/err.h>
+#include <linux/pmic/wm8350.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 <asm/arch/dma.h>
+#include <asm/arch/spba.h>
+#include <asm/arch/clock.h>
+
+#include "imx-ssi.h"
+#include "imx-pcm.h"
+#include "../sound/soc/codecs/wm8350.h"
+
+extern void gpio_activate_audio_ports(void);
+
+/* SSI BCLK and LRC master */
+#define WM8350_SSI_MASTER 1
+
+#define IMX37_3STACK_AUDIO_VERSION "0.3"
+
+struct imx37_3stack_pcm_state {
+ int lr_clk_active;
+};
+
+struct _wm8350_audio {
+ unsigned int channels;
+ snd_pcm_format_t format;
+ unsigned int rate;
+ unsigned int sysclk;
+ unsigned int bclkdiv;
+ unsigned int clkdiv;
+ unsigned int lr_rate;
+};
+
+/* in order of power consumption per rate (lowest first) */
+static const struct _wm8350_audio wm8350_audio[] = {
+ /* 16bit mono modes */
+ {1, SNDRV_PCM_FORMAT_S16_LE, 8000, 12288000 >> 1,
+ WM8350_BCLK_DIV_48, WM8350_DACDIV_3, 16,},
+
+ /* 16 bit stereo modes */
+ {2, SNDRV_PCM_FORMAT_S16_LE, 8000, 12288000,
+ WM8350_BCLK_DIV_48, WM8350_DACDIV_6, 32,},
+ {2, SNDRV_PCM_FORMAT_S16_LE, 16000, 12288000,
+ WM8350_BCLK_DIV_24, WM8350_DACDIV_3, 32,},
+ {2, SNDRV_PCM_FORMAT_S16_LE, 32000, 12288000,
+ WM8350_BCLK_DIV_12, WM8350_DACDIV_1_5, 32,},
+ {2, SNDRV_PCM_FORMAT_S16_LE, 48000, 12288000,
+ WM8350_BCLK_DIV_8, WM8350_DACDIV_1, 32,},
+ {2, SNDRV_PCM_FORMAT_S16_LE, 96000, 24576000,
+ WM8350_BCLK_DIV_8, WM8350_DACDIV_1, 32,},
+ {2, SNDRV_PCM_FORMAT_S16_LE, 11025, 11289600,
+ WM8350_BCLK_DIV_32, WM8350_DACDIV_4, 32,},
+ {2, SNDRV_PCM_FORMAT_S16_LE, 22050, 11289600,
+ WM8350_BCLK_DIV_16, WM8350_DACDIV_2, 32,},
+ {2, SNDRV_PCM_FORMAT_S16_LE, 44100, 11289600,
+ WM8350_BCLK_DIV_8, WM8350_DACDIV_1, 32,},
+ {2, SNDRV_PCM_FORMAT_S16_LE, 88200, 22579200,
+ WM8350_BCLK_DIV_8, WM8350_DACDIV_1, 32,},
+
+ /* 24bit stereo modes */
+ {2, SNDRV_PCM_FORMAT_S24_LE, 48000, 12288000,
+ WM8350_BCLK_DIV_4, WM8350_DACDIV_1, 64,},
+ {2, SNDRV_PCM_FORMAT_S24_LE, 96000, 24576000,
+ WM8350_BCLK_DIV_4, WM8350_DACDIV_1, 64,},
+ {2, SNDRV_PCM_FORMAT_S24_LE, 44100, 11289600,
+ WM8350_BCLK_DIV_4, WM8350_DACDIV_1, 64,},
+ {2, SNDRV_PCM_FORMAT_S24_LE, 88200, 22579200,
+ WM8350_BCLK_DIV_4, WM8350_DACDIV_1, 64,},
+};
+
+#if WM8350_SSI_MASTER
+static int imx37_3stack_startup(struct snd_pcm_substream *substream)
+{
+ struct snd_soc_pcm_link *pcm_link = substream->private_data;
+ struct snd_soc_codec *codec = pcm_link->codec;
+ struct wm8350* wm8350 = codec->control_data;
+ struct imx37_3stack_pcm_state *state = pcm_link->private_data;
+
+ /* In master mode the LR clock can come from either the DAC or ADC.
+ * We use the LR clock from whatever stream is enabled first.
+ */
+
+ if (!state->lr_clk_active) {
+ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)
+ wm8350_clear_bits(wm8350, WM8350_CLOCK_CONTROL_2,
+ WM8350_LRC_ADC_SEL);
+ else
+ wm8350_set_bits(wm8350, WM8350_CLOCK_CONTROL_2,
+ WM8350_LRC_ADC_SEL);
+ }
+ state->lr_clk_active++;
+ return 0;
+}
+#else
+#define imx37_3stack_startup NULL
+#endif
+
+static int imx37_3stack_hifi_hw_params(struct snd_pcm_substream *substream,
+ struct snd_pcm_hw_params *params)
+{
+ struct snd_soc_pcm_link *pcm_link = substream->private_data;
+ struct snd_soc_dai *cpu_dai = pcm_link->cpu_dai;
+ struct snd_soc_dai *codec_dai = pcm_link->codec_dai;
+ struct imx37_3stack_pcm_state *state = pcm_link->private_data;
+ int i, found = 0;
+ snd_pcm_format_t format = params_format(params);
+ unsigned int rate = params_rate(params);
+ unsigned int channels = params_channels(params);
+ u32 dai_format;
+
+ /* only need to do this once as capture and playback are sync */
+ if (state->lr_clk_active > 1)
+ return 0;
+
+ /* find the correct audio parameters */
+ for (i = 0; i < ARRAY_SIZE(wm8350_audio); i++) {
+ if (rate == wm8350_audio[i].rate &&
+ format == wm8350_audio[i].format &&
+ channels == wm8350_audio[i].channels) {
+ found = 1;
+ break;
+ }
+ }
+ if (!found)
+ return -EINVAL;
+
+#if WM8350_SSI_MASTER
+ dai_format = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF |
+ SND_SOC_DAIFMT_CBM_CFM | SND_SOC_DAIFMT_SYNC;
+ if (channels == 2)
+ dai_format |= SND_SOC_DAIFMT_TDM;
+
+ /* set codec DAI configuration */
+ codec_dai->ops->set_fmt(codec_dai, dai_format);
+
+ /* set cpu DAI configuration */
+ cpu_dai->ops->set_fmt(cpu_dai, dai_format);
+
+ /* set 32KHZ as the codec system clock for DAC and ADC */
+ codec_dai->ops->set_sysclk(codec_dai, WM3850_MCLK_SEL_PLL_32K,
+ wm8350_audio[i].sysclk, SND_SOC_CLOCK_IN);
+
+
+#else
+ dai_format = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF |
+ SND_SOC_DAIFMT_CBS_CFS | SND_SOC_DAIFMT_SYNC;
+ if (channels == 2)
+ format |= SND_SOC_DAIFMT_TDM;
+
+ /* set codec DAI configuration */
+ codec_dai->ops->set_fmt(codec_dai, dai_format);
+
+ /* set cpu DAI configuration */
+ cpu_dai->ops->set_fmt(cpu_dai, dai_format);
+
+ /* set DAC LRC as the codec system clock for DAC and ADC */
+ codec_dai->ops->set_sysclk(codec_dai, WM3850_MCLK_SEL_PLL_DAC,
+ wm8350_audio[i].sysclk, SND_SOC_CLOCK_IN);
+#endif
+
+ /* set i.MX active slot mask */
+ cpu_dai->ops->set_tdm_slot(cpu_dai,
+ channels == 1 ? 0xfffffffe : 0xfffffffc, channels);
+
+ /* set the SSI system clock as input (unused) */
+ cpu_dai->ops->set_sysclk(cpu_dai, IMX_SSP_SYS_CLK, 0,
+ SND_SOC_CLOCK_IN);
+
+ /* set codec BCLK division for sample rate */
+ codec_dai->ops->set_clkdiv(codec_dai, WM8350_BCLK_CLKDIV,
+ wm8350_audio[i].bclkdiv);
+
+ /* DAI is synchronous and clocked with DAC LRCLK */
+ codec_dai->ops->set_clkdiv(codec_dai,
+ WM8350_DACLR_CLKDIV, wm8350_audio[i].lr_rate);
+
+ /* now configure DAC and ADC clocks */
+ codec_dai->ops->set_clkdiv(codec_dai,
+ WM8350_DAC_CLKDIV, wm8350_audio[i].clkdiv);
+
+ codec_dai->ops->set_clkdiv(codec_dai,
+ WM8350_ADC_CLKDIV, wm8350_audio[i].clkdiv);
+
+#if WM8350_SSI_MASTER
+ /* codec FLL input is 32768 kHz from MCLK */
+ codec_dai->ops->set_pll(codec_dai, 0, 32768,
+ wm8350_audio[i].sysclk);
+#else
+ /* codec FLL input is rate from DAC LRC */
+ codec_dai->ops->set_pll(codec_dai, 0, rate,
+ wm8350_audio[i].sysclk);
+#endif
+
+ return 0;
+}
+
+static void imx37_3stack_shutdown(struct snd_pcm_substream *substream)
+{
+ struct snd_soc_pcm_link *pcm_link = substream->private_data;
+ struct snd_soc_dai *codec_dai = pcm_link->codec_dai;
+ struct imx37_3stack_pcm_state *state = pcm_link->private_data;
+
+ /* disable the PLL if there are no active Tx or Rx channels */
+ if (!codec_dai->active)
+ codec_dai->ops->set_pll(codec_dai, 0, 0, 0);
+ state->lr_clk_active--;
+}
+
+/*
+ * imx37_3stack WM8350 HiFi DAI opserations.
+ */
+static struct snd_soc_ops imx37_3stack_hifi_ops = {
+ .startup = imx37_3stack_startup,
+ .shutdown = imx37_3stack_shutdown,
+ .hw_params = imx37_3stack_hifi_hw_params,
+};
+
+/* need to refine these */
+static struct wm8350_platform_data imx37_3stack_wm8350_setup = {
+ .vmid_discharge_msecs = 1000,
+ .drain_msecs = 30,
+ .cap_discharge_msecs = 700,
+ .vmid_charge_msecs = 700,
+ .vmid_s_curve = WM8350_S_CURVE_SLOW,
+ .dis_out4 = WM8350_DISCHARGE_SLOW,
+ .dis_out3 = WM8350_DISCHARGE_SLOW,
+ .dis_out2 = WM8350_DISCHARGE_SLOW,
+ .dis_out1 = WM8350_DISCHARGE_SLOW,
+ .vroi_out4 = WM8350_TIE_OFF_500R,
+ .vroi_out3 = WM8350_TIE_OFF_500R,
+ .vroi_out2 = WM8350_TIE_OFF_500R,
+ .vroi_out1 = WM8350_TIE_OFF_500R,
+ .vroi_enable = 0,
+ .codec_current_d0 = WM8350_CODEC_ISEL_1_0,
+ .codec_current_d3 = WM8350_CODEC_ISEL_0_5,
+ .codec_current_charge = WM8350_CODEC_ISEL_1_5,
+};
+
+static int imx37_3stack_pcm_new(struct snd_soc_pcm_link *pcm_link)
+{
+ struct imx37_3stack_pcm_state *state;
+ int ret;
+
+ state = kzalloc(sizeof(struct imx37_3stack_pcm_state), GFP_KERNEL);
+ if (state == NULL)
+ return -ENOMEM;
+
+ pcm_link->audio_ops = &imx37_3stack_hifi_ops;
+ pcm_link->private_data = state;
+
+ ret = snd_soc_pcm_new(pcm_link, 1, 1);
+ if (ret < 0) {
+ printk(KERN_ERR "%s: failed to create hifi pcm\n", __func__);
+ kfree(state);
+ return ret;
+ }
+
+ printk(KERN_INFO "i.MX37 WM8350 Audio version %s\n",
+ IMX37_3STACK_AUDIO_VERSION);
+
+ gpio_activate_audio_ports();
+
+ /* WM8350 uses SSI1 via AUDMUX port 5 for audio */
+
+ /* reset port 1 & 5*/
+ DAM_PTCR1 = 0;
+ DAM_PDCR1 = 0;
+ DAM_PTCR5 = 0;
+ DAM_PDCR5 = 0;
+
+ /* set to synchronous */
+ DAM_PTCR1 |= AUDMUX_PTCR_SYN;
+ DAM_PTCR5 |= AUDMUX_PTCR_SYN;
+
+#if WM8350_SSI_MASTER
+ /* set Rx sources 1 <--> 5 */
+ DAM_PDCR1 |= AUDMUX_PDCR_RXDSEL(5);
+ DAM_PDCR5 |= AUDMUX_PDCR_RXDSEL(1);
+
+ /* set Tx frame direction and source 5--> 1 output */
+ DAM_PTCR1 |= AUDMUX_PTCR_TFSDIR;
+ DAM_PTCR1 |= AUDMUX_PTCR_TFSSEL(AUDMUX_FROM_TXFS, 5);
+
+ /* set Tx Clock direction and source 5--> 1 output */
+ DAM_PTCR1 |= AUDMUX_PTCR_TCLKDIR;
+ DAM_PTCR1 |= AUDMUX_PTCR_TCSEL(AUDMUX_FROM_TXFS, 5);
+#else
+ /* set Rx sources 1 <--> 5 */
+ DAM_PDCR1 |= AUDMUX_PDCR_RXDSEL(5);
+ DAM_PDCR5 |= AUDMUX_PDCR_RXDSEL(1);
+
+ /* set Tx frame direction and source 1 --> 5 output */
+ DAM_PTCR5 |= AUDMUX_PTCR_TFSDIR;
+ DAM_PTCR5 |= AUDMUX_PTCR_TFSSEL(AUDMUX_FROM_TXFS, 1);
+
+ /* set Tx Clock direction and source 1--> 5 output */
+ DAM_PTCR5 |= AUDMUX_PTCR_TCLKDIR;
+ DAM_PTCR5 |= AUDMUX_PTCR_TCSEL(AUDMUX_FROM_TXFS, 1);
+#endif
+
+ return 0;
+}
+
+static int imx37_3stack_pcm_free(struct snd_soc_pcm_link *pcm_link)
+{
+ kfree(pcm_link->private_data);
+ return 0;
+}
+
+static const struct snd_soc_pcm_link_ops imx37_3stack_pcm_ops = {
+ .new = imx37_3stack_pcm_new,
+ .free = imx37_3stack_pcm_free,
+};
+
+/* imx37_3stack machine dapm widgets */
+static const struct snd_soc_dapm_widget imx37_3stack_dapm_widgets[] = {
+ SND_SOC_DAPM_MIC("SiMIC", NULL),
+ SND_SOC_DAPM_MIC("Mic1 Jack", NULL),
+ SND_SOC_DAPM_MIC("Mic2 Jack", NULL),
+ SND_SOC_DAPM_LINE("Line In Jack", NULL),
+ SND_SOC_DAPM_LINE("Line Out Jack", NULL),
+ SND_SOC_DAPM_HP("Headphone Jack", NULL),
+};
+
+/* imx37_3stack machine audio map */
+static const char* audio_map[][3] = {
+
+ /* SiMIC --> IN1LN (with automatic bias) via SP1 */
+ {"IN1LN", NULL, "Mic Bias"},
+ {"Mic Bias", NULL, "SiMIC"},
+
+ /* Mic 1 Jack --> IN1LN and IN1LP (with automatic bias) */
+ {"IN1LN", NULL, "Mic Bias"},
+ {"IN1LP", NULL, "Mic1 Jack"},
+ {"Mic Bias", NULL, "Mic1 Jack"},
+
+ /* Mic 2 Jack --> IN1RN and IN1RP (with automatic bias) */
+ {"IN1RN", NULL, "Mic Bias"},
+ {"IN1RP", NULL, "Mic1 Jack"},
+ {"Mic Bias", NULL, "Mic1 Jack"},
+
+ /* Line in Jack --> AUX (L+R) */
+ {"IN3R", NULL, "Line In Jack"},
+ {"IN3L", NULL, "Line In Jack"},
+
+ /* Out1 --> Headphone Jack */
+ {"Headphone Jack", NULL, "OUT1R"},
+ {"Headphone Jack", NULL, "OUT1L"},
+
+ /* Out1 --> Line Out Jack */
+ {"Line Out Jack", NULL, "OUT2R"},
+ {"Line Out Jack", NULL, "OUT2L"},
+
+ {NULL, NULL, NULL},
+};
+
+#ifdef CONFIG_PM
+static int imx37_3stack_wm8350_audio_suspend(struct platform_device *dev,
+ pm_message_t state)
+{
+// struct snd_soc_machine *machine = pdev->dev.driver_data;
+ int ret = 0;
+
+ return ret;
+}
+
+static int imx37_3stack_wm8350_audio_resume(struct platform_device *dev)
+{
+// struct snd_soc_machine *machine = pdev->dev.driver_data;
+ int ret = 0;
+
+ return ret;
+}
+
+#else
+#define imx37_3stack_wm8350_audio_suspend NULL
+#define imx37_3stack_wm8350_audio_resume NULL
+#endif
+
+static void imx37_3stack_jack_handler(struct wm8350 *wm8350, int irq)
+{
+ struct snd_soc_machine *machine = &wm8350->audio;
+ u16 reg;
+
+ /* debounce for 200ms */
+ schedule_timeout_interruptible(msecs_to_jiffies(200));
+ reg = wm8350_reg_read(wm8350, WM8350_JACK_PIN_STATUS);
+ if (reg & WM8350_JACK_R_LVL) {
+ snd_soc_dapm_set_endpoint(machine, "Line Out Jack", 0);
+ snd_soc_dapm_set_endpoint(machine, "Headphone Jack", 1);
+ } else {
+ snd_soc_dapm_set_endpoint(machine, "Headphone Jack", 0);
+ snd_soc_dapm_set_endpoint(machine, "Line Out Jack", 1);
+ }
+ snd_soc_dapm_sync_endpoints(machine);
+}
+
+int mach_probe(struct snd_soc_machine *machine)
+{
+ struct snd_soc_codec *codec;
+ struct snd_soc_pcm_link *pcm_link;
+ struct wm8350* wm8350 = to_wm8350_from_audio(machine);
+ int i, ret;
+ u16 reg;
+
+ pcm_link = list_first_entry(&machine->active_list,
+ struct snd_soc_pcm_link, active_list);
+
+ codec = pcm_link->codec;
+ codec->control_data = wm8350;
+ codec->platform_data = &imx37_3stack_wm8350_setup;
+ codec->ops->io_probe(codec, machine);
+
+ /* set unused imx37_3stack WM8350 codec pins */
+ snd_soc_dapm_set_endpoint(machine, "OUT3", 0);
+ snd_soc_dapm_set_endpoint(machine, "OUT4", 0);
+ snd_soc_dapm_set_endpoint(machine, "IN2R", 0);
+ snd_soc_dapm_set_endpoint(machine, "IN2L", 0);
+ snd_soc_dapm_set_endpoint(machine, "OUT2L", 0);
+ snd_soc_dapm_set_endpoint(machine, "OUT2R", 0);
+
+#if 0
+ /* add imx37_3stack specific controls */
+ for (i = 0; i < ARRAY_SIZE(imx37_3stack_wm8350_audio_controls); i++) {
+ if ((err = snd_ctl_add(machine->card,
+ snd_soc_cnew(&imx37_3stack_wm8350_audio_controls[i],
+ codec, NULL))) < 0)
+ return err;
+ }
+#endif
+
+ /* Add imx37_3stack specific widgets */
+ for(i = 0; i < ARRAY_SIZE(imx37_3stack_dapm_widgets); i++) {
+ snd_soc_dapm_new_control(machine, codec,
+ &imx37_3stack_dapm_widgets[i]);
+ }
+
+ /* set up imx37_3stack specific audio path audio map */
+ for(i = 0; audio_map[i][0] != NULL; i++) {
+ snd_soc_dapm_connect_input(machine, audio_map[i][0],
+ audio_map[i][1], audio_map[i][2]);
+ }
+
+ /* connect and enable all imx37_3stack WM8350 jacks (for now) */
+ snd_soc_dapm_set_endpoint(machine, "SiMIC", 1);
+ snd_soc_dapm_set_endpoint(machine, "Mic1 Jack", 1);
+ snd_soc_dapm_set_endpoint(machine, "Mic2 Jack", 1);
+ snd_soc_dapm_set_endpoint(machine, "Line In Jack", 1);
+
+ snd_soc_dapm_set_policy(machine, SND_SOC_DAPM_POLICY_STREAM);
+ snd_soc_dapm_sync_endpoints(machine);
+
+ /* enable slow clock gen for jack detect */
+ reg = wm8350_reg_read(wm8350, WM8350_POWER_MGMT_4);
+ wm8350_reg_write(wm8350, WM8350_POWER_MGMT_4,
+ reg | WM8350_TOCLK_ENA);
+ /* enable jack detect */
+ reg = wm8350_reg_read(wm8350, WM8350_JACK_DETECT);
+ wm8350_reg_write(wm8350, WM8350_JACK_DETECT,
+ reg | WM8350_JDR_ENA);
+ wm8350_register_irq(wm8350, WM8350_IRQ_CODEC_JCK_DET_R,
+ imx37_3stack_jack_handler);
+ wm8350_unmask_irq(wm8350, WM8350_IRQ_CODEC_JCK_DET_R);
+
+ /* register card with ALSA upper layers */
+ ret = snd_soc_register_card(machine);
+ if (ret < 0) {
+ printk(KERN_ERR "%s: failed to register sound card\n",
+ __FUNCTION__);
+ goto link_err;
+ }
+ return 0;
+link_err:
+ snd_soc_machine_free(machine);
+ return ret;
+}
+
+struct snd_soc_machine_ops machine_ops = {
+ .mach_probe = mach_probe,
+};
+
+static int __devinit imx37_3stack_wm8350_audio_probe(struct platform_device *pdev)
+{
+ struct snd_soc_machine *machine = platform_get_drvdata(pdev);
+ struct snd_soc_pcm_link *hifi;
+ int ret;
+
+ machine->owner = THIS_MODULE;
+ machine->pdev = pdev;
+ machine->name = "i.MX32ADS";
+ machine->longname = "WM8350";
+ machine->ops = &machine_ops;
+
+ /* register card */
+ ret = snd_soc_new_card(machine, 1, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1);
+ if (ret < 0) {
+ printk(KERN_ERR "%s: failed to create pcms\n", __func__);
+ return ret;
+ }
+
+ /* i.MX32ADS WM8350 hifi interface */
+ ret = -ENODEV;
+ hifi = snd_soc_pcm_link_new(machine, "imx37_3stack-hifi", &imx37_3stack_pcm_ops,
+ imx_pcm, wm8350_codec, wm8350_hifi_dai, imx_ssi_1);
+ if (hifi == NULL) {
+ printk("failed to create HiFi PCM link\n");
+ goto link_err;
+ }
+ ret = snd_soc_pcm_link_attach(hifi);
+ if (ret < 0) {
+ printk(KERN_ERR "%s: failed to attach hifi pcm\n", __func__);
+ goto link_err;
+ }
+ return ret;
+
+link_err:
+ snd_soc_machine_free(machine);
+ return ret;
+}
+
+static int __devexit imx37_3stack_wm8350_audio_remove(struct platform_device *pdev)
+{
+ struct snd_soc_machine *machine = platform_get_drvdata(pdev);
+ struct wm8350* wm8350 = to_wm8350_from_audio(machine);
+ struct snd_soc_codec *codec;
+ struct snd_soc_pcm_link *pcm_link;
+
+ pcm_link = list_first_entry(&machine->active_list,
+ struct snd_soc_pcm_link, active_list);
+
+ codec = pcm_link->codec;
+ codec->ops->io_remove(codec, machine);
+ wm8350_mask_irq(wm8350, WM8350_IRQ_CODEC_JCK_DET_R);
+ wm8350_free_irq(wm8350, WM8350_IRQ_CODEC_JCK_DET_R);
+
+ snd_soc_machine_free(machine);
+ return 0;
+}
+
+const char imx37_3stack_audio[32] = "wm8350-imx37-3stack-audio";
+EXPORT_SYMBOL_GPL(imx37_3stack_audio);
+
+static struct platform_driver imx37_3stack_wm8350_audio_driver = {
+ .probe = imx37_3stack_wm8350_audio_probe,
+ .remove = __devexit_p(imx37_3stack_wm8350_audio_remove),
+ .suspend = imx37_3stack_wm8350_audio_suspend,
+ .resume = imx37_3stack_wm8350_audio_resume,
+ .driver = {
+ .name = imx37_3stack_audio,
+ },
+};
+
+static int __init imx37_3stack_wm8350_audio_init(void)
+{
+ return platform_driver_register(&imx37_3stack_wm8350_audio_driver);
+}
+
+static void __exit imx37_3stack_wm8350_audio_exit(void)
+{
+ platform_driver_unregister(&imx37_3stack_wm8350_audio_driver);
+}
+
+module_init(imx37_3stack_wm8350_audio_init);
+module_exit(imx37_3stack_wm8350_audio_exit);
+
+MODULE_AUTHOR("Liam Girdwood");
+MODULE_DESCRIPTION("PMIC WM8350 Driver for i.MX37");
+MODULE_LICENSE("GPL");