summaryrefslogtreecommitdiff
path: root/recipes/linux/linux-2.6.28/tx27/linux-2.6.28-karo4.diff
diff options
context:
space:
mode:
Diffstat (limited to 'recipes/linux/linux-2.6.28/tx27/linux-2.6.28-karo4.diff')
-rw-r--r--recipes/linux/linux-2.6.28/tx27/linux-2.6.28-karo4.diff40137
1 files changed, 40137 insertions, 0 deletions
diff --git a/recipes/linux/linux-2.6.28/tx27/linux-2.6.28-karo4.diff b/recipes/linux/linux-2.6.28/tx27/linux-2.6.28-karo4.diff
new file mode 100644
index 0000000000..a1be6cabb4
--- /dev/null
+++ b/recipes/linux/linux-2.6.28/tx27/linux-2.6.28-karo4.diff
@@ -0,0 +1,40137 @@
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/Kconfig linux-2.6.28-karo/arch/arm/Kconfig
+--- linux-2.6.28/arch/arm/Kconfig 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/Kconfig 2009-03-11 13:16:24.000000000 +0100
+@@ -1268,6 +1268,8 @@ source "drivers/rtc/Kconfig"
+
+ source "drivers/dma/Kconfig"
+
++source "drivers/mxc/Kconfig"
++
+ source "drivers/dca/Kconfig"
+
+ source "drivers/auxdisplay/Kconfig"
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/configs/karo_tx27_defconfig linux-2.6.28-karo/arch/arm/configs/karo_tx27_defconfig
+--- linux-2.6.28/arch/arm/configs/karo_tx27_defconfig 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/configs/karo_tx27_defconfig 2009-03-12 16:52:26.000000000 +0100
+@@ -0,0 +1,1799 @@
++#
++# Automatically generated make config: don't edit
++# Linux kernel version: 2.6.28
++# Thu Mar 12 15:05:31 2009
++#
++CONFIG_ARM=y
++CONFIG_SYS_SUPPORTS_APM_EMULATION=y
++CONFIG_GENERIC_GPIO=y
++CONFIG_GENERIC_TIME=y
++CONFIG_GENERIC_CLOCKEVENTS=y
++CONFIG_MMU=y
++# CONFIG_NO_IOPORT is not set
++CONFIG_GENERIC_HARDIRQS=y
++CONFIG_STACKTRACE_SUPPORT=y
++CONFIG_HAVE_LATENCYTOP_SUPPORT=y
++CONFIG_LOCKDEP_SUPPORT=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_GENERIC_HARDIRQS_NO__DO_IRQ=y
++CONFIG_VECTORS_BASE=0xffff0000
++CONFIG_DEFCONFIG_LIST="/lib/modules/$UNAME_RELEASE/.config"
++
++#
++# General setup
++#
++CONFIG_EXPERIMENTAL=y
++CONFIG_BROKEN_ON_SMP=y
++CONFIG_LOCK_KERNEL=y
++CONFIG_INIT_ENV_ARG_LIMIT=32
++CONFIG_LOCALVERSION=""
++# CONFIG_LOCALVERSION_AUTO is not set
++# CONFIG_SWAP is not set
++CONFIG_SYSVIPC=y
++CONFIG_SYSVIPC_SYSCTL=y
++CONFIG_POSIX_MQUEUE=y
++# CONFIG_BSD_PROCESS_ACCT is not set
++# CONFIG_TASKSTATS is not set
++# CONFIG_AUDIT is not set
++# CONFIG_IKCONFIG is not set
++CONFIG_LOG_BUF_SHIFT=17
++# CONFIG_CGROUPS is not set
++# CONFIG_GROUP_SCHED is not set
++# CONFIG_SYSFS_DEPRECATED_V2 is not set
++# CONFIG_RELAY is not set
++# CONFIG_NAMESPACES is not set
++# CONFIG_BLK_DEV_INITRD is not set
++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=y
++# CONFIG_KALLSYMS_EXTRA_PASS is not set
++CONFIG_HOTPLUG=y
++CONFIG_PRINTK=y
++CONFIG_BUG=y
++# CONFIG_ELF_CORE is not set
++CONFIG_COMPAT_BRK=y
++CONFIG_BASE_FULL=y
++CONFIG_FUTEX=y
++CONFIG_ANON_INODES=y
++CONFIG_EPOLL=y
++CONFIG_SIGNALFD=y
++CONFIG_TIMERFD=y
++CONFIG_EVENTFD=y
++CONFIG_SHMEM=y
++# CONFIG_AIO is not set
++# CONFIG_VM_EVENT_COUNTERS is not set
++CONFIG_SLAB=y
++# CONFIG_SLUB is not set
++# CONFIG_SLOB is not set
++# CONFIG_PROFILING is not set
++# CONFIG_MARKERS is not set
++CONFIG_HAVE_OPROFILE=y
++# CONFIG_KPROBES is not set
++CONFIG_HAVE_KPROBES=y
++CONFIG_HAVE_KRETPROBES=y
++CONFIG_HAVE_GENERIC_DMA_COHERENT=y
++CONFIG_SLABINFO=y
++CONFIG_RT_MUTEXES=y
++# CONFIG_TINY_SHMEM is not set
++CONFIG_BASE_SMALL=0
++CONFIG_MODULES=y
++# CONFIG_MODULE_FORCE_LOAD is not set
++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
++CONFIG_BLOCK=y
++# CONFIG_LBD is not set
++# CONFIG_BLK_DEV_IO_TRACE is not set
++# CONFIG_LSF is not set
++# CONFIG_BLK_DEV_BSG is not set
++# CONFIG_BLK_DEV_INTEGRITY is not set
++
++#
++# IO Schedulers
++#
++CONFIG_IOSCHED_NOOP=y
++CONFIG_IOSCHED_AS=y
++CONFIG_IOSCHED_DEADLINE=y
++CONFIG_IOSCHED_CFQ=y
++# CONFIG_DEFAULT_AS is not set
++# CONFIG_DEFAULT_DEADLINE is not set
++CONFIG_DEFAULT_CFQ=y
++# CONFIG_DEFAULT_NOOP is not set
++CONFIG_DEFAULT_IOSCHED="cfq"
++CONFIG_CLASSIC_RCU=y
++CONFIG_FREEZER=y
++
++#
++# 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_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_IOP13XX is not set
++# CONFIG_ARCH_IOP32X is not set
++# CONFIG_ARCH_IOP33X is not set
++# CONFIG_ARCH_IXP23XX is not set
++# CONFIG_ARCH_IXP2000 is not set
++# CONFIG_ARCH_IXP4XX is not set
++# CONFIG_ARCH_L7200 is not set
++# CONFIG_ARCH_KIRKWOOD is not set
++# CONFIG_ARCH_KS8695 is not set
++# CONFIG_ARCH_NS9XXX is not set
++# CONFIG_ARCH_LOKI is not set
++# CONFIG_ARCH_MV78XX0 is not set
++CONFIG_ARCH_MXC=y
++# CONFIG_ARCH_ORION5X is not set
++# CONFIG_ARCH_PNX4008 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_DAVINCI is not set
++# CONFIG_ARCH_OMAP is not set
++# CONFIG_ARCH_MSM is not set
++
++#
++# Boot options
++#
++
++#
++# Power management
++#
++
++#
++# Freescale MXC Implementations
++#
++CONFIG_ARCH_MX2=y
++# CONFIG_ARCH_MX3 is not set
++
++#
++# MX2 family CPU support
++#
++CONFIG_MACH_MX27=y
++
++#
++# MX2 Platforms
++#
++# CONFIG_MACH_MX27ADS is not set
++# CONFIG_MACH_PCM038 is not set
++CONFIG_MACH_TX27=y
++CONFIG_BASE_CLK_26MHz=y
++# CONFIG_KARO_DEBUG is not set
++CONFIG_MXC_EMMA=y
++# CONFIG_MXC_IRQ_PRIOR is not set
++CONFIG_MXC_ULPI=y
++
++#
++# Processor Type
++#
++CONFIG_CPU_32=y
++CONFIG_CPU_ARM926T=y
++CONFIG_CPU_32v5=y
++CONFIG_CPU_ABRT_EV5TJ=y
++CONFIG_CPU_PABRT_NOIFAR=y
++CONFIG_CPU_CACHE_VIVT=y
++CONFIG_CPU_COPY_V4WB=y
++CONFIG_CPU_TLB_V4WBI=y
++CONFIG_CPU_CP15=y
++CONFIG_CPU_CP15_MMU=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_OUTER_CACHE is not set
++
++#
++# Bus support
++#
++# CONFIG_PCI_SYSCALL is not set
++# CONFIG_ARCH_SUPPORTS_MSI is not set
++# CONFIG_PCCARD is not set
++
++#
++# Kernel Features
++#
++CONFIG_TICK_ONESHOT=y
++CONFIG_NO_HZ=y
++CONFIG_HIGH_RES_TIMERS=y
++CONFIG_GENERIC_CLOCKEVENTS_BUILD=y
++CONFIG_VMSPLIT_3G=y
++# CONFIG_VMSPLIT_2G is not set
++# CONFIG_VMSPLIT_1G is not set
++CONFIG_PAGE_OFFSET=0xC0000000
++CONFIG_PREEMPT=y
++CONFIG_HZ=100
++CONFIG_AEABI=y
++CONFIG_OABI_COMPAT=y
++CONFIG_ARCH_FLATMEM_HAS_HOLES=y
++# CONFIG_ARCH_SPARSEMEM_DEFAULT is not set
++# CONFIG_ARCH_SELECT_MEMORY_MODEL 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_PAGEFLAGS_EXTENDED=y
++CONFIG_SPLIT_PTLOCK_CPUS=4096
++# CONFIG_RESOURCES_64BIT is not set
++# CONFIG_PHYS_ADDR_T_64BIT is not set
++CONFIG_ZONE_DMA_FLAG=0
++CONFIG_VIRT_TO_BUS=y
++CONFIG_UNEVICTABLE_LRU=y
++CONFIG_ALIGNMENT_TRAP=y
++
++#
++# Boot options
++#
++CONFIG_ZBOOT_ROM_TEXT=0x0
++CONFIG_ZBOOT_ROM_BSS=0x0
++CONFIG_CMDLINE="root=/dev/mtdblock1 rootfstype=jffs2 console=ttymxc0,115200 ro panic=5"
++# CONFIG_XIP_KERNEL is not set
++# CONFIG_KEXEC is not set
++
++#
++# CPU Power Management
++#
++# CONFIG_CPU_IDLE 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 is not set
++
++#
++# Userspace binary formats
++#
++CONFIG_BINFMT_ELF=y
++# CONFIG_CORE_DUMP_DEFAULT_ELF_HEADERS is not set
++CONFIG_HAVE_AOUT=y
++# CONFIG_BINFMT_AOUT is not set
++# CONFIG_BINFMT_MISC is not set
++
++#
++# Power management options
++#
++CONFIG_PM=y
++# CONFIG_PM_DEBUG is not set
++CONFIG_PM_SLEEP=y
++CONFIG_SUSPEND=y
++CONFIG_SUSPEND_FREEZER=y
++CONFIG_APM_EMULATION=y
++CONFIG_ARCH_SUSPEND_POSSIBLE=y
++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=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_XFRM_MODE_TRANSPORT is not set
++# CONFIG_INET_XFRM_MODE_TUNNEL is not set
++# CONFIG_INET_XFRM_MODE_BEET is not set
++# CONFIG_INET_LRO is not set
++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"
++# CONFIG_TCP_MD5SIG is not set
++# CONFIG_IPV6 is not set
++# CONFIG_NETWORK_SECMARK is not set
++# CONFIG_NETFILTER is not set
++# CONFIG_IP_DCCP is not set
++# CONFIG_IP_SCTP is not set
++# CONFIG_TIPC is not set
++# CONFIG_ATM is not set
++# CONFIG_BRIDGE is not set
++# CONFIG_NET_DSA 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
++# CONFIG_NET_SCHED is not set
++
++#
++# Network testing
++#
++# CONFIG_NET_PKTGEN is not set
++# CONFIG_HAMRADIO is not set
++# CONFIG_CAN is not set
++# CONFIG_IRDA is not set
++# CONFIG_BT is not set
++# CONFIG_AF_RXRPC is not set
++# CONFIG_PHONET is not set
++CONFIG_WIRELESS=y
++CONFIG_CFG80211=m
++CONFIG_NL80211=y
++CONFIG_WIRELESS_OLD_REGULATORY=y
++CONFIG_WIRELESS_EXT=y
++CONFIG_WIRELESS_EXT_SYSFS=y
++CONFIG_MAC80211=m
++
++#
++# Rate control algorithm selection
++#
++CONFIG_MAC80211_RC_PID=y
++# CONFIG_MAC80211_RC_MINSTREL is not set
++CONFIG_MAC80211_RC_DEFAULT_PID=y
++# CONFIG_MAC80211_RC_DEFAULT_MINSTREL is not set
++CONFIG_MAC80211_RC_DEFAULT="pid"
++# CONFIG_MAC80211_MESH is not set
++# CONFIG_MAC80211_LEDS is not set
++# CONFIG_MAC80211_DEBUG_MENU is not set
++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_RFKILL is not set
++# CONFIG_NET_9P is not set
++
++#
++# Device Drivers
++#
++
++#
++# Generic Driver Options
++#
++CONFIG_UEVENT_HELPER_PATH="/sbin/hotplug"
++CONFIG_STANDALONE=y
++CONFIG_PREVENT_FIRMWARE_BUILD=y
++CONFIG_FW_LOADER=m
++CONFIG_FIRMWARE_IN_KERNEL=y
++CONFIG_EXTRA_FIRMWARE=""
++# CONFIG_DEBUG_DRIVER is not set
++# CONFIG_DEBUG_DEVRES is not set
++# CONFIG_SYS_HYPERVISOR is not set
++CONFIG_CONNECTOR=y
++CONFIG_PROC_EVENTS=y
++CONFIG_MTD=y
++# CONFIG_MTD_DEBUG is not set
++# CONFIG_MTD_CONCAT is not set
++CONFIG_MTD_PARTITIONS=y
++CONFIG_MTD_REDBOOT_PARTS=y
++CONFIG_MTD_REDBOOT_DIRECTORY_BLOCK=-5
++CONFIG_MTD_REDBOOT_PARTS_UNALLOCATED=y
++CONFIG_MTD_REDBOOT_PARTS_READONLY=y
++CONFIG_MTD_CMDLINE_PARTS=y
++# CONFIG_MTD_AFS_PARTS is not set
++# CONFIG_MTD_AR7_PARTS is not set
++
++#
++# User Modules And Translation Layers
++#
++CONFIG_MTD_CHAR=y
++CONFIG_MTD_BLKDEVS=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
++# CONFIG_MTD_OOPS 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 is not set
++# CONFIG_MTD_ABSENT is not set
++
++#
++# Mapping drivers for chip access
++#
++# CONFIG_MTD_COMPLEX_MAPPINGS is not set
++# CONFIG_MTD_PLATRAM is not set
++
++#
++# Self-contained MTD device drivers
++#
++# CONFIG_MTD_DATAFLASH is not set
++# CONFIG_MTD_M25P80 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
++CONFIG_MTD_NAND=y
++# CONFIG_MTD_NAND_VERIFY_WRITE is not set
++# CONFIG_MTD_NAND_ECC_SMC is not set
++# CONFIG_MTD_NAND_MUSEUM_IDS is not set
++# CONFIG_MTD_NAND_GPIO is not set
++CONFIG_MTD_NAND_IDS=y
++# CONFIG_MTD_NAND_DISKONCHIP is not set
++CONFIG_MTD_NAND_MXC_FLASH_BBT=y
++# CONFIG_MTD_NAND_NANDSIM is not set
++# CONFIG_MTD_NAND_PLATFORM is not set
++# CONFIG_MTD_ALAUDA is not set
++CONFIG_MTD_NAND_MXC=y
++# CONFIG_MTD_ONENAND is not set
++
++#
++# UBI - Unsorted block images
++#
++# CONFIG_MTD_UBI is not set
++# CONFIG_PARPORT is not set
++CONFIG_BLK_DEV=y
++# 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=y
++CONFIG_BLK_DEV_RAM_COUNT=16
++CONFIG_BLK_DEV_RAM_SIZE=16384
++# CONFIG_BLK_DEV_XIP is not set
++# CONFIG_CDROM_PKTCDVD is not set
++# CONFIG_ATA_OVER_ETH is not set
++CONFIG_MISC_DEVICES=y
++# CONFIG_EEPROM_93CX6 is not set
++# CONFIG_ICS932S401 is not set
++# CONFIG_ENCLOSURE_SERVICES is not set
++# CONFIG_C2PORT is not set
++CONFIG_HAVE_IDE=y
++# CONFIG_IDE is not set
++
++#
++# SCSI device support
++#
++# CONFIG_RAID_ATTRS is not set
++CONFIG_SCSI=m
++CONFIG_SCSI_DMA=y
++# CONFIG_SCSI_TGT is not set
++# CONFIG_SCSI_NETLINK is not set
++CONFIG_SCSI_PROC_FS=y
++
++#
++# SCSI support type (disk, tape, CD-ROM)
++#
++CONFIG_BLK_DEV_SD=m
++# 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=y
++# CONFIG_SCSI_CONSTANTS is not set
++# CONFIG_SCSI_LOGGING is not set
++# CONFIG_SCSI_SCAN_ASYNC is not set
++CONFIG_SCSI_WAIT_SCAN=m
++
++#
++# 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_LIBSAS is not set
++# CONFIG_SCSI_SRP_ATTRS is not set
++CONFIG_SCSI_LOWLEVEL=y
++# CONFIG_ISCSI_TCP is not set
++# CONFIG_SCSI_DEBUG is not set
++# CONFIG_SCSI_DH is not set
++# CONFIG_ATA is not set
++# CONFIG_MD is not set
++CONFIG_NETDEVICES=y
++# CONFIG_DUMMY is not set
++# CONFIG_BONDING is not set
++# CONFIG_MACVLAN is not set
++# CONFIG_EQUALIZER is not set
++# CONFIG_TUN is not set
++# CONFIG_VETH is not set
++CONFIG_PHYLIB=y
++
++#
++# MII PHY device drivers
++#
++# CONFIG_MARVELL_PHY is not set
++# CONFIG_DAVICOM_PHY is not set
++# CONFIG_QSEMI_PHY is not set
++# CONFIG_LXT_PHY is not set
++# CONFIG_CICADA_PHY is not set
++# CONFIG_VITESSE_PHY is not set
++CONFIG_SMSC_PHY=y
++# CONFIG_BROADCOM_PHY is not set
++# CONFIG_ICPLUS_PHY is not set
++# CONFIG_REALTEK_PHY is not set
++# CONFIG_FIXED_PHY is not set
++# CONFIG_MDIO_BITBANG is not set
++CONFIG_NET_ETHERNET=y
++CONFIG_MII=y
++# CONFIG_AX88796 is not set
++# CONFIG_SMC91X is not set
++# CONFIG_DM9000 is not set
++# CONFIG_ENC28J60 is not set
++# CONFIG_SMC911X is not set
++# CONFIG_IBM_NEW_EMAC_ZMII is not set
++# CONFIG_IBM_NEW_EMAC_RGMII is not set
++# CONFIG_IBM_NEW_EMAC_TAH is not set
++# CONFIG_IBM_NEW_EMAC_EMAC4 is not set
++# CONFIG_IBM_NEW_EMAC_NO_FLOW_CTRL is not set
++# CONFIG_IBM_NEW_EMAC_MAL_CLR_ICINTSTAT is not set
++# CONFIG_IBM_NEW_EMAC_MAL_COMMON_ERR is not set
++# CONFIG_B44 is not set
++CONFIG_FEC=y
++# CONFIG_FEC2 is not set
++# CONFIG_NETDEV_1000 is not set
++# CONFIG_NETDEV_10000 is not set
++
++#
++# Wireless LAN
++#
++# CONFIG_WLAN_PRE80211 is not set
++CONFIG_WLAN_80211=y
++CONFIG_LIBERTAS=m
++CONFIG_LIBERTAS_USB=m
++# CONFIG_LIBERTAS_SDIO is not set
++# CONFIG_LIBERTAS_DEBUG is not set
++# CONFIG_LIBERTAS_THINFIRM is not set
++# CONFIG_USB_ZD1201 is not set
++# CONFIG_USB_NET_RNDIS_WLAN is not set
++# CONFIG_RTL8187 is not set
++# CONFIG_MAC80211_HWSIM is not set
++# CONFIG_P54_COMMON is not set
++# CONFIG_IWLWIFI_LEDS is not set
++# CONFIG_HOSTAP is not set
++# CONFIG_B43 is not set
++# CONFIG_B43LEGACY is not set
++# CONFIG_ZD1211RW is not set
++# CONFIG_RT2X00 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 is not set
++CONFIG_USB_NET_CDCETHER=m
++# CONFIG_USB_NET_DM9601 is not set
++# CONFIG_USB_NET_SMSC95XX is not set
++# CONFIG_USB_NET_GL620A is not set
++# CONFIG_USB_NET_NET1080 is not set
++# 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=m
++# CONFIG_USB_ALI_M5632 is not set
++# CONFIG_USB_AN2720 is not set
++# CONFIG_USB_BELKIN is not set
++CONFIG_USB_ARMLINUX=y
++# CONFIG_USB_EPSON2888 is not set
++# CONFIG_USB_KC2190 is not set
++# CONFIG_USB_NET_ZAURUS is not set
++# CONFIG_WAN is not set
++# CONFIG_PPP is not set
++# CONFIG_SLIP is not set
++# CONFIG_NETCONSOLE is not set
++# CONFIG_NETPOLL is not set
++# CONFIG_NET_POLL_CONTROLLER is not set
++# CONFIG_ISDN is not set
++
++#
++# Input device support
++#
++CONFIG_INPUT=y
++# CONFIG_INPUT_FF_MEMLESS is not set
++# CONFIG_INPUT_POLLDEV is not set
++
++#
++# Userland interfaces
++#
++# CONFIG_INPUT_MOUSEDEV is not set
++# CONFIG_INPUT_JOYDEV is not set
++CONFIG_INPUT_EVDEV=y
++CONFIG_INPUT_EVBUG=m
++# CONFIG_INPUT_APMPOWER is not set
++
++#
++# 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_STOWAWAY is not set
++CONFIG_KEYBOARD_GPIO=m
++# CONFIG_INPUT_MOUSE is not set
++# CONFIG_INPUT_JOYSTICK is not set
++# CONFIG_INPUT_TABLET is not set
++CONFIG_INPUT_TOUCHSCREEN=y
++# CONFIG_TOUCHSCREEN_ADS7846 is not set
++# CONFIG_TOUCHSCREEN_FUJITSU is not set
++# CONFIG_TOUCHSCREEN_GUNZE is not set
++# CONFIG_TOUCHSCREEN_ELO is not set
++# CONFIG_TOUCHSCREEN_MTOUCH is not set
++# CONFIG_TOUCHSCREEN_INEXIO 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_USB_COMPOSITE=m
++CONFIG_TOUCHSCREEN_USB_EGALAX=y
++# CONFIG_TOUCHSCREEN_USB_PANJIT is not set
++# CONFIG_TOUCHSCREEN_USB_3M is not set
++# CONFIG_TOUCHSCREEN_USB_ITM is not set
++# CONFIG_TOUCHSCREEN_USB_ETURBO is not set
++# CONFIG_TOUCHSCREEN_USB_GUNZE is not set
++# CONFIG_TOUCHSCREEN_USB_DMC_TSC10 is not set
++CONFIG_TOUCHSCREEN_USB_IRTOUCH=y
++CONFIG_TOUCHSCREEN_USB_IDEALTEK=y
++CONFIG_TOUCHSCREEN_USB_GENERAL_TOUCH=y
++CONFIG_TOUCHSCREEN_USB_GOTOP=y
++# CONFIG_TOUCHSCREEN_TOUCHIT213 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_CONSOLE_TRANSLATIONS is not set
++CONFIG_VT_CONSOLE=y
++CONFIG_HW_CONSOLE=y
++CONFIG_VT_HW_CONSOLE_BINDING=y
++# CONFIG_DEVKMEM is not set
++# CONFIG_SERIAL_NONSTANDARD is not set
++
++#
++# Serial drivers
++#
++# CONFIG_SERIAL_8250 is not set
++
++#
++# Non-8250 serial port support
++#
++CONFIG_SERIAL_IMX=y
++CONFIG_SERIAL_IMX_CONSOLE=y
++CONFIG_SERIAL_CORE=y
++CONFIG_SERIAL_CORE_CONSOLE=y
++CONFIG_UNIX98_PTYS=y
++CONFIG_LEGACY_PTYS=y
++CONFIG_LEGACY_PTY_COUNT=16
++# CONFIG_IPMI_HANDLER is not set
++# CONFIG_HW_RANDOM is not set
++# CONFIG_NVRAM is not set
++# CONFIG_R3964 is not set
++# CONFIG_RAW_DRIVER is not set
++# CONFIG_TCG_TPM is not set
++CONFIG_I2C=y
++CONFIG_I2C_BOARDINFO=y
++CONFIG_I2C_CHARDEV=m
++CONFIG_I2C_HELPER_AUTO=y
++CONFIG_I2C_ALGOBIT=m
++
++#
++# I2C Hardware Bus support
++#
++
++#
++# I2C system bus drivers (mostly embedded / system-on-chip)
++#
++CONFIG_I2C_GPIO=m
++CONFIG_I2C_MXC=y
++# CONFIG_I2C_OCORES is not set
++# CONFIG_I2C_SIMTEC is not set
++
++#
++# External I2C/SMBus adapter drivers
++#
++# CONFIG_I2C_PARPORT_LIGHT is not set
++# CONFIG_I2C_TAOS_EVM is not set
++# CONFIG_I2C_TINY_USB is not set
++
++#
++# Other I2C/SMBus bus drivers
++#
++# CONFIG_I2C_PCA_PLATFORM is not set
++# CONFIG_I2C_STUB is not set
++
++#
++# Miscellaneous I2C Chip support
++#
++# CONFIG_DS1682 is not set
++CONFIG_AT24=m
++# CONFIG_SENSORS_EEPROM is not set
++# CONFIG_SENSORS_PCF8574 is not set
++# CONFIG_PCF8575 is not set
++# CONFIG_SENSORS_PCA9539 is not set
++# CONFIG_SENSORS_PCF8591 is not set
++# CONFIG_TPS65010 is not set
++# CONFIG_SENSORS_MAX6875 is not set
++# CONFIG_SENSORS_TSL2550 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
++CONFIG_SPI=y
++# CONFIG_SPI_DEBUG is not set
++CONFIG_SPI_MASTER=y
++
++#
++# SPI Master Controller Drivers
++#
++CONFIG_SPI_BITBANG=m
++CONFIG_SPI_MXC=m
++# CONFIG_SPI_MXC_TEST_LOOPBACK is not set
++CONFIG_SPI_MXC_SELECT1=y
++# CONFIG_SPI_MXC_SELECT2 is not set
++# CONFIG_SPI_MXC_SELECT3 is not set
++CONFIG_SPI_MXC_REV0=y
++# CONFIG_SPI_MXC_REV4 is not set
++# CONFIG_SPI_MXC_REV5 is not set
++# CONFIG_SPI_MXC_REV7 is not set
++
++#
++# SPI Protocol Masters
++#
++CONFIG_SPI_AT25=m
++CONFIG_SPI_SPIDEV=m
++# CONFIG_SPI_TLE62X0 is not set
++CONFIG_ARCH_REQUIRE_GPIOLIB=y
++CONFIG_GPIOLIB=y
++# CONFIG_DEBUG_GPIO is not set
++CONFIG_GPIO_SYSFS=y
++
++#
++# Memory mapped GPIO expanders:
++#
++
++#
++# I2C GPIO expanders:
++#
++# CONFIG_GPIO_MAX732X is not set
++# CONFIG_GPIO_PCA953X is not set
++# CONFIG_GPIO_PCF857X is not set
++
++#
++# PCI GPIO expanders:
++#
++
++#
++# SPI GPIO expanders:
++#
++# CONFIG_GPIO_MAX7301 is not set
++# CONFIG_GPIO_MCP23S08 is not set
++CONFIG_W1=m
++CONFIG_W1_CON=y
++
++#
++# 1-wire Bus Masters
++#
++# CONFIG_W1_MASTER_DS2490 is not set
++# CONFIG_W1_MASTER_DS2482 is not set
++CONFIG_W1_MASTER_MXC=m
++# CONFIG_W1_MASTER_GPIO is not set
++
++#
++# 1-wire Slaves
++#
++# CONFIG_W1_SLAVE_THERM is not set
++CONFIG_W1_SLAVE_SMEM=m
++CONFIG_W1_SLAVE_DS2433=m
++CONFIG_W1_SLAVE_DS2433_CRC=y
++# CONFIG_W1_SLAVE_DS2760 is not set
++# CONFIG_W1_SLAVE_BQ27000 is not set
++CONFIG_POWER_SUPPLY=m
++# CONFIG_POWER_SUPPLY_DEBUG is not set
++# CONFIG_PDA_POWER is not set
++# CONFIG_APM_POWER is not set
++# CONFIG_BATTERY_DS2760 is not set
++# CONFIG_BATTERY_BQ27x00 is not set
++CONFIG_LP3972=m
++CONFIG_HWMON=m
++# CONFIG_HWMON_VID is not set
++# CONFIG_SENSORS_AD7414 is not set
++# CONFIG_SENSORS_AD7418 is not set
++# CONFIG_SENSORS_ADCXX is not set
++# CONFIG_SENSORS_ADM1021 is not set
++# CONFIG_SENSORS_ADM1025 is not set
++# CONFIG_SENSORS_ADM1026 is not set
++# CONFIG_SENSORS_ADM1029 is not set
++# CONFIG_SENSORS_ADM1031 is not set
++# CONFIG_SENSORS_ADM9240 is not set
++# CONFIG_SENSORS_ADT7462 is not set
++# CONFIG_SENSORS_ADT7470 is not set
++# CONFIG_SENSORS_ADT7473 is not set
++# CONFIG_SENSORS_ATXP1 is not set
++# CONFIG_SENSORS_DS1621 is not set
++# CONFIG_SENSORS_F71805F is not set
++# CONFIG_SENSORS_F71882FG is not set
++# CONFIG_SENSORS_F75375S is not set
++# CONFIG_SENSORS_GL518SM is not set
++# CONFIG_SENSORS_GL520SM is not set
++# CONFIG_SENSORS_IT87 is not set
++# CONFIG_SENSORS_LM63 is not set
++# CONFIG_SENSORS_LM70 is not set
++# CONFIG_SENSORS_LM75 is not set
++CONFIG_SENSORS_LM77=m
++# CONFIG_SENSORS_LM78 is not set
++# CONFIG_SENSORS_LM80 is not set
++# CONFIG_SENSORS_LM83 is not set
++# CONFIG_SENSORS_LM85 is not set
++# CONFIG_SENSORS_LM87 is not set
++# CONFIG_SENSORS_LM90 is not set
++# CONFIG_SENSORS_LM92 is not set
++# CONFIG_SENSORS_LM93 is not set
++# CONFIG_SENSORS_MAX1111 is not set
++# CONFIG_SENSORS_MAX1619 is not set
++# CONFIG_SENSORS_MAX6650 is not set
++# CONFIG_SENSORS_PC87360 is not set
++# CONFIG_SENSORS_PC87427 is not set
++# CONFIG_SENSORS_DME1737 is not set
++# CONFIG_SENSORS_SMSC47M1 is not set
++# CONFIG_SENSORS_SMSC47M192 is not set
++# CONFIG_SENSORS_SMSC47B397 is not set
++# CONFIG_SENSORS_ADS7828 is not set
++# CONFIG_SENSORS_THMC50 is not set
++# CONFIG_SENSORS_VT1211 is not set
++# 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_W83L786NG is not set
++# CONFIG_SENSORS_W83627HF is not set
++# CONFIG_SENSORS_W83627EHF is not set
++# CONFIG_HWMON_DEBUG_CHIP is not set
++# CONFIG_THERMAL is not set
++CONFIG_WATCHDOG=y
++CONFIG_WATCHDOG_NOWAYOUT=y
++
++#
++# Watchdog Device Drivers
++#
++# CONFIG_SOFT_WATCHDOG is not set
++
++#
++# USB-based Watchdog Cards
++#
++# CONFIG_USBPCWATCHDOG is not set
++CONFIG_SSB_POSSIBLE=y
++
++#
++# Sonics Silicon Backplane
++#
++# CONFIG_SSB is not set
++
++#
++# Multifunction device drivers
++#
++# CONFIG_MFD_CORE is not set
++# CONFIG_MFD_SM501 is not set
++# CONFIG_MFD_ASIC3 is not set
++# CONFIG_HTC_EGPIO is not set
++# CONFIG_HTC_PASIC3 is not set
++# CONFIG_MFD_TMIO is not set
++# CONFIG_MFD_TC6393XB is not set
++# CONFIG_PMIC_DA903X is not set
++# CONFIG_MFD_WM8400 is not set
++# CONFIG_MFD_WM8350_I2C is not set
++
++#
++# Multimedia devices
++#
++
++#
++# Multimedia core support
++#
++CONFIG_VIDEO_DEV=m
++CONFIG_VIDEO_V4L2_COMMON=m
++# CONFIG_VIDEO_ALLOW_V4L1 is not set
++# CONFIG_VIDEO_V4L1_COMPAT is not set
++# CONFIG_DVB_CORE is not set
++CONFIG_VIDEO_MEDIA=m
++
++#
++# Multimedia drivers
++#
++# CONFIG_MEDIA_ATTACH is not set
++CONFIG_MEDIA_TUNER=m
++CONFIG_MEDIA_TUNER_CUSTOMIZE=y
++# CONFIG_MEDIA_TUNER_SIMPLE is not set
++# CONFIG_MEDIA_TUNER_TDA8290 is not set
++# CONFIG_MEDIA_TUNER_TDA827X is not set
++# CONFIG_MEDIA_TUNER_TDA18271 is not set
++# CONFIG_MEDIA_TUNER_TDA9887 is not set
++# CONFIG_MEDIA_TUNER_TEA5761 is not set
++# CONFIG_MEDIA_TUNER_TEA5767 is not set
++# CONFIG_MEDIA_TUNER_MT20XX is not set
++# CONFIG_MEDIA_TUNER_MT2060 is not set
++# CONFIG_MEDIA_TUNER_MT2266 is not set
++# CONFIG_MEDIA_TUNER_MT2131 is not set
++# CONFIG_MEDIA_TUNER_QT1010 is not set
++# CONFIG_MEDIA_TUNER_XC2028 is not set
++# CONFIG_MEDIA_TUNER_XC5000 is not set
++# CONFIG_MEDIA_TUNER_MXL5005S is not set
++# CONFIG_MEDIA_TUNER_MXL5007T is not set
++CONFIG_VIDEO_V4L2=m
++CONFIG_VIDEOBUF_GEN=m
++CONFIG_VIDEOBUF_DMA_CONTIG=m
++CONFIG_VIDEO_CAPTURE_DRIVERS=y
++# CONFIG_VIDEO_ADV_DEBUG is not set
++# CONFIG_VIDEO_FIXED_MINOR_RANGES is not set
++# CONFIG_VIDEO_HELPER_CHIPS_AUTO is not set
++
++#
++# Encoders/decoders and other helper chips
++#
++
++#
++# Audio decoders
++#
++# CONFIG_VIDEO_TVAUDIO is not set
++# CONFIG_VIDEO_TDA7432 is not set
++# CONFIG_VIDEO_TDA9840 is not set
++# CONFIG_VIDEO_TDA9875 is not set
++# CONFIG_VIDEO_TEA6415C is not set
++# CONFIG_VIDEO_TEA6420 is not set
++# CONFIG_VIDEO_MSP3400 is not set
++# CONFIG_VIDEO_CS5345 is not set
++# CONFIG_VIDEO_CS53L32A is not set
++# CONFIG_VIDEO_M52790 is not set
++# CONFIG_VIDEO_TLV320AIC23B is not set
++# CONFIG_VIDEO_WM8775 is not set
++# CONFIG_VIDEO_WM8739 is not set
++# CONFIG_VIDEO_VP27SMPX is not set
++
++#
++# Video decoders
++#
++# CONFIG_VIDEO_OV7670 is not set
++# CONFIG_VIDEO_TCM825X is not set
++# CONFIG_VIDEO_SAA711X is not set
++# CONFIG_VIDEO_SAA717X is not set
++# CONFIG_VIDEO_TVP5150 is not set
++
++#
++# Video and audio decoders
++#
++# CONFIG_VIDEO_CX25840 is not set
++
++#
++# MPEG video encoders
++#
++# CONFIG_VIDEO_CX2341X is not set
++
++#
++# Video encoders
++#
++# CONFIG_VIDEO_SAA7127 is not set
++
++#
++# Video improvement chips
++#
++# CONFIG_VIDEO_UPD64031A is not set
++# CONFIG_VIDEO_UPD64083 is not set
++# CONFIG_VIDEO_VIVI is not set
++CONFIG_VIDEO_MXC_CAMERA=m
++
++#
++# MXC Camera/V4L2 PRP Features support
++#
++# CONFIG_MXC_CAMERA_MICRON111 is not set
++# CONFIG_MXC_CAMERA_MC521DA is not set
++# CONFIG_MXC_CAMERA_OV2640 is not set
++CONFIG_VIDEO_MXC_OUTPUT=m
++# CONFIG_VIDEO_MXC_OUTPUT_DEBUG is not set
++CONFIG_VIDEO_MXC_EMMA_OUTPUT=m
++# CONFIG_VIDEO_MXC_OPL is not set
++# CONFIG_VIDEO_SAA5246A is not set
++# CONFIG_VIDEO_SAA5249 is not set
++CONFIG_SOC_CAMERA=m
++# CONFIG_SOC_CAMERA_MT9M001 is not set
++# CONFIG_SOC_CAMERA_MT9M111 is not set
++# CONFIG_SOC_CAMERA_MT9V022 is not set
++CONFIG_SOC_CAMERA_PLATFORM=m
++# CONFIG_VIDEO_SH_MOBILE_CEU is not set
++# CONFIG_V4L_USB_DRIVERS is not set
++CONFIG_VIDEO_MX27=m
++# CONFIG_RADIO_ADAPTERS is not set
++# CONFIG_DAB is not set
++
++#
++# Graphics support
++#
++# CONFIG_VGASTATE is not set
++# CONFIG_VIDEO_OUTPUT_CONTROL is not set
++CONFIG_FB=y
++# CONFIG_FIRMWARE_EDID is not set
++# CONFIG_FB_DDC is not set
++# CONFIG_FB_BOOT_VESA_SUPPORT is not set
++CONFIG_FB_CFB_FILLRECT=y
++CONFIG_FB_CFB_COPYAREA=y
++CONFIG_FB_CFB_IMAGEBLIT=y
++# CONFIG_FB_CFB_REV_PIXELS_IN_BYTE is not set
++# CONFIG_FB_SYS_FILLRECT is not set
++# CONFIG_FB_SYS_COPYAREA is not set
++# CONFIG_FB_SYS_IMAGEBLIT is not set
++# CONFIG_FB_FOREIGN_ENDIAN is not set
++# CONFIG_FB_SYS_FOPS is not set
++# CONFIG_FB_SVGALIB is not set
++# CONFIG_FB_MACMODES is not set
++# CONFIG_FB_BACKLIGHT is not set
++CONFIG_FB_MODE_HELPERS=y
++CONFIG_FB_TILEBLITTING=y
++
++#
++# Frame buffer hardware drivers
++#
++CONFIG_FB_IMX=y
++# CONFIG_FB_UVESA is not set
++# CONFIG_FB_S1D13XXX is not set
++# CONFIG_FB_VIRTUAL is not set
++# CONFIG_FB_METRONOME is not set
++# CONFIG_FB_MB862XX is not set
++# CONFIG_BACKLIGHT_LCD_SUPPORT is not set
++
++#
++# Display device support
++#
++# CONFIG_DISPLAY_SUPPORT is not set
++
++#
++# Console display driver support
++#
++# CONFIG_VGA_CONSOLE is not set
++CONFIG_DUMMY_CONSOLE=y
++CONFIG_FRAMEBUFFER_CONSOLE=y
++# CONFIG_FRAMEBUFFER_CONSOLE_DETECT_PRIMARY is not set
++CONFIG_FRAMEBUFFER_CONSOLE_ROTATION=y
++# CONFIG_FONTS is not set
++CONFIG_FONT_8x8=y
++CONFIG_FONT_8x16=y
++CONFIG_LOGO=y
++CONFIG_LOGO_LINUX_MONO=y
++CONFIG_LOGO_LINUX_VGA16=y
++CONFIG_LOGO_LINUX_CLUT224=y
++# CONFIG_SOUND is not set
++CONFIG_HID_SUPPORT=y
++CONFIG_HID=m
++# CONFIG_HID_DEBUG is not set
++# CONFIG_HIDRAW is not set
++
++#
++# USB Input Devices
++#
++CONFIG_USB_HID=m
++# CONFIG_HID_PID 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
++
++#
++# Special HID drivers
++#
++# CONFIG_HID_COMPAT is not set
++# CONFIG_HID_A4TECH is not set
++# CONFIG_HID_APPLE is not set
++# CONFIG_HID_BELKIN is not set
++# CONFIG_HID_BRIGHT is not set
++# CONFIG_HID_CHERRY is not set
++# CONFIG_HID_CHICONY is not set
++# CONFIG_HID_CYPRESS is not set
++# CONFIG_HID_DELL is not set
++# CONFIG_HID_EZKEY is not set
++# CONFIG_HID_GYRATION is not set
++# CONFIG_HID_LOGITECH is not set
++# CONFIG_HID_MICROSOFT is not set
++# CONFIG_HID_MONTEREY is not set
++# CONFIG_HID_PANTHERLORD is not set
++# CONFIG_HID_PETALYNX is not set
++# CONFIG_HID_SAMSUNG is not set
++# CONFIG_HID_SONY is not set
++# CONFIG_HID_SUNPLUS is not set
++# CONFIG_THRUSTMASTER_FF is not set
++# CONFIG_ZEROPLUS_FF is not set
++CONFIG_USB_SUPPORT=y
++CONFIG_USB_ARCH_HAS_HCD=y
++# CONFIG_USB_ARCH_HAS_OHCI is not set
++CONFIG_USB_ARCH_HAS_EHCI=y
++CONFIG_USB=m
++# CONFIG_USB_DEBUG is not set
++# CONFIG_USB_ANNOUNCE_NEW_DEVICES is not set
++
++#
++# Miscellaneous USB options
++#
++CONFIG_USB_DEVICEFS=y
++CONFIG_USB_DEVICE_CLASS=y
++# CONFIG_USB_DYNAMIC_MINORS is not set
++CONFIG_USB_SUSPEND=y
++# CONFIG_USB_OTG is not set
++# CONFIG_USB_OTG_WHITELIST is not set
++# CONFIG_USB_OTG_BLACKLIST_HUB is not set
++# CONFIG_USB_MON is not set
++# CONFIG_USB_WUSB is not set
++# CONFIG_USB_WUSB_CBAF is not set
++
++#
++# USB Host Controller Drivers
++#
++# CONFIG_USB_C67X00_HCD is not set
++CONFIG_USB_EHCI_HCD=m
++CONFIG_USB_EHCI_ROOT_HUB_TT=y
++# CONFIG_USB_EHCI_TT_NEWSCHED is not set
++CONFIG_USB_EHCI_MXC=y
++# CONFIG_USB_ISP116X_HCD is not set
++# CONFIG_USB_SL811_HCD is not set
++# CONFIG_USB_R8A66597_HCD is not set
++# CONFIG_USB_HWA_HCD is not set
++
++#
++# Enable Host or Gadget support to see Inventra options
++#
++
++#
++# USB Device Class drivers
++#
++# CONFIG_USB_ACM is not set
++# CONFIG_USB_PRINTER is not set
++# CONFIG_USB_WDM is not set
++# CONFIG_USB_TMC is not set
++
++#
++# NOTE: USB_STORAGE depends on SCSI but BLK_DEV_SD 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_STORAGE_ONETOUCH is not set
++# CONFIG_USB_STORAGE_KARMA is not set
++# CONFIG_USB_STORAGE_CYPRESS_ATACB is not set
++# CONFIG_USB_LIBUSUAL is not set
++
++#
++# USB Imaging devices
++#
++# CONFIG_USB_MDC800 is not set
++# CONFIG_USB_MICROTEK is not set
++
++#
++# USB port drivers
++#
++CONFIG_USB_SERIAL=m
++# CONFIG_USB_EZUSB is not set
++CONFIG_USB_SERIAL_GENERIC=y
++# CONFIG_USB_SERIAL_AIRCABLE is not set
++# CONFIG_USB_SERIAL_ARK3116 is not set
++CONFIG_USB_SERIAL_BELKIN=m
++# CONFIG_USB_SERIAL_CH341 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_IUU 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_MOTOROLA is not set
++# CONFIG_USB_SERIAL_NAVMAN is not set
++CONFIG_USB_SERIAL_PL2303=m
++# CONFIG_USB_SERIAL_OTI6858 is not set
++# CONFIG_USB_SERIAL_SPCP8X5 is not set
++# 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
++# CONFIG_USB_SERIAL_DEBUG 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_SEVSEG is not set
++# CONFIG_USB_RIO500 is not set
++# CONFIG_USB_LEGOTOWER is not set
++# CONFIG_USB_LCD is not set
++# CONFIG_USB_BERRY_CHARGE 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_SISUSBVGA is not set
++# CONFIG_USB_LD is not set
++# CONFIG_USB_TRANCEVIBRATOR is not set
++# CONFIG_USB_IOWARRIOR is not set
++CONFIG_USB_TEST=m
++# CONFIG_USB_ISIGHTFW is not set
++# CONFIG_USB_VST is not set
++# CONFIG_USB_GADGET is not set
++CONFIG_MMC=m
++# CONFIG_MMC_DEBUG is not set
++# CONFIG_MMC_UNSAFE_RESUME is not set
++
++#
++# MMC/SD/SDIO Card Drivers
++#
++CONFIG_MMC_BLOCK=m
++CONFIG_MMC_BLOCK_BOUNCE=y
++# CONFIG_SDIO_UART is not set
++# CONFIG_MMC_TEST is not set
++
++#
++# MMC/SD/SDIO Host Controller Drivers
++#
++# CONFIG_MMC_SDHCI is not set
++CONFIG_MMC_MXC=m
++# CONFIG_MMC_SPI is not set
++# CONFIG_MEMSTICK is not set
++# CONFIG_ACCESSIBILITY is not set
++CONFIG_NEW_LEDS=y
++CONFIG_LEDS_CLASS=y
++
++#
++# LED drivers
++#
++# CONFIG_LEDS_PCA9532 is not set
++CONFIG_LEDS_GPIO=y
++# CONFIG_LEDS_PCA955X is not set
++
++#
++# LED Triggers
++#
++CONFIG_LEDS_TRIGGERS=y
++CONFIG_LEDS_TRIGGER_TIMER=m
++CONFIG_LEDS_TRIGGER_HEARTBEAT=y
++# CONFIG_LEDS_TRIGGER_BACKLIGHT is not set
++# CONFIG_LEDS_TRIGGER_DEFAULT_ON is not set
++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
++# CONFIG_RTC_DRV_TEST is not set
++
++#
++# I2C RTC drivers
++#
++# CONFIG_RTC_DRV_DS1307 is not set
++CONFIG_RTC_DRV_DS13XX=y
++# CONFIG_RTC_DRV_DS1374 is not set
++# CONFIG_RTC_DRV_DS1672 is not set
++# CONFIG_RTC_DRV_MAX6900 is not set
++# CONFIG_RTC_DRV_RS5C372 is not set
++# CONFIG_RTC_DRV_ISL1208 is not set
++# CONFIG_RTC_DRV_X1205 is not set
++# CONFIG_RTC_DRV_PCF8563 is not set
++# CONFIG_RTC_DRV_PCF8583 is not set
++# CONFIG_RTC_DRV_M41T80 is not set
++# CONFIG_RTC_DRV_S35390A is not set
++# CONFIG_RTC_DRV_FM3130 is not set
++# CONFIG_RTC_DRV_RX8581 is not set
++
++#
++# SPI RTC drivers
++#
++# CONFIG_RTC_DRV_M41T94 is not set
++# CONFIG_RTC_DRV_DS1305 is not set
++# CONFIG_RTC_DRV_DS1390 is not set
++# CONFIG_RTC_DRV_MAX6902 is not set
++# CONFIG_RTC_DRV_R9701 is not set
++# CONFIG_RTC_DRV_RS5C348 is not set
++# CONFIG_RTC_DRV_DS3234 is not set
++
++#
++# Platform RTC drivers
++#
++# CONFIG_RTC_DRV_CMOS is not set
++# CONFIG_RTC_DRV_DS1286 is not set
++# CONFIG_RTC_DRV_DS1511 is not set
++# CONFIG_RTC_DRV_DS1553 is not set
++# CONFIG_RTC_DRV_DS1742 is not set
++# CONFIG_RTC_DRV_STK17TA8 is not set
++# CONFIG_RTC_DRV_M48T86 is not set
++# CONFIG_RTC_DRV_M48T35 is not set
++# CONFIG_RTC_DRV_M48T59 is not set
++# CONFIG_RTC_DRV_BQ4802 is not set
++# CONFIG_RTC_DRV_V3020 is not set
++
++#
++# on-CPU RTC drivers
++#
++CONFIG_RTC_MXC=y
++# CONFIG_DMADEVICES is not set
++CONFIG_DRIVERS_MXC=y
++
++#
++# MXC VPU(Video Processing Unit) support
++#
++CONFIG_MXC_VPU=m
++# CONFIG_MXC_VPU_DEBUG is not set
++# CONFIG_REGULATOR is not set
++# CONFIG_UIO 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=y
++# CONFIG_EXT3_FS_POSIX_ACL is not set
++# CONFIG_EXT3_FS_SECURITY is not set
++# CONFIG_EXT4_FS is not set
++CONFIG_JBD=m
++CONFIG_FS_MBCACHE=m
++# CONFIG_REISERFS_FS is not set
++# CONFIG_JFS_FS is not set
++# CONFIG_FS_POSIX_ACL is not set
++CONFIG_FILE_LOCKING=y
++# CONFIG_XFS_FS is not set
++# CONFIG_OCFS2_FS is not set
++CONFIG_DNOTIFY=y
++CONFIG_INOTIFY=y
++CONFIG_INOTIFY_USER=y
++# CONFIG_QUOTA is not set
++# CONFIG_AUTOFS_FS is not set
++CONFIG_AUTOFS4_FS=m
++# 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=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_PROC_SYSCTL=y
++CONFIG_PROC_PAGE_MONITOR=y
++CONFIG_SYSFS=y
++CONFIG_TMPFS=y
++# CONFIG_TMPFS_POSIX_ACL is not set
++# CONFIG_HUGETLB_PAGE is not set
++CONFIG_CONFIGFS_FS=m
++
++#
++# 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_FS_WBUF_VERIFY is not set
++CONFIG_JFFS2_SUMMARY=y
++# CONFIG_JFFS2_FS_XATTR is not set
++# CONFIG_JFFS2_COMPRESSION_OPTIONS is not set
++CONFIG_JFFS2_ZLIB=y
++# CONFIG_JFFS2_LZO is not set
++CONFIG_JFFS2_RTIME=y
++# CONFIG_JFFS2_RUBIN is not set
++CONFIG_CRAMFS=y
++# CONFIG_VXFS_FS is not set
++# CONFIG_MINIX_FS is not set
++# CONFIG_OMFS_FS is not set
++# CONFIG_HPFS_FS is not set
++# CONFIG_QNX4FS_FS is not set
++# CONFIG_ROMFS_FS is not set
++# CONFIG_SYSV_FS is not set
++# CONFIG_UFS_FS is not set
++CONFIG_NETWORK_FILESYSTEMS=y
++CONFIG_NFS_FS=y
++CONFIG_NFS_V3=y
++# CONFIG_NFS_V3_ACL is not set
++# CONFIG_NFS_V4 is not set
++CONFIG_ROOT_NFS=y
++# CONFIG_NFSD is not set
++CONFIG_LOCKD=y
++CONFIG_LOCKD_V4=y
++CONFIG_NFS_COMMON=y
++CONFIG_SUNRPC=y
++# CONFIG_SUNRPC_REGISTER_V4 is not set
++# 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
++
++#
++# 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
++# CONFIG_SYSV68_PARTITION is not set
++CONFIG_NLS=m
++CONFIG_NLS_DEFAULT="iso8859-1"
++CONFIG_NLS_CODEPAGE_437=m
++# CONFIG_NLS_CODEPAGE_737 is not set
++# CONFIG_NLS_CODEPAGE_775 is not set
++CONFIG_NLS_CODEPAGE_850=m
++# 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=m
++CONFIG_NLS_ISO8859_1=m
++# 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=m
++# CONFIG_NLS_KOI8_R is not set
++# CONFIG_NLS_KOI8_U is not set
++CONFIG_NLS_UTF8=m
++# CONFIG_DLM is not set
++
++#
++# Kernel hacking
++#
++# CONFIG_PRINTK_TIME is not set
++# CONFIG_ENABLE_WARN_DEPRECATED is not set
++# CONFIG_ENABLE_MUST_CHECK is not set
++CONFIG_FRAME_WARN=1024
++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=y
++# CONFIG_DEBUG_SHIRQ is not set
++# CONFIG_DETECT_SOFTLOCKUP is not set
++# CONFIG_SCHED_DEBUG is not set
++# CONFIG_SCHEDSTATS is not set
++# CONFIG_TIMER_STATS is not set
++# CONFIG_DEBUG_OBJECTS is not set
++# CONFIG_DEBUG_SLAB is not set
++# CONFIG_DEBUG_PREEMPT 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_LOCK_ALLOC is not set
++# CONFIG_PROVE_LOCKING is not set
++# CONFIG_LOCK_STAT 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=y
++# CONFIG_DEBUG_INFO is not set
++# CONFIG_DEBUG_VM is not set
++# CONFIG_DEBUG_WRITECOUNT is not set
++# CONFIG_DEBUG_MEMORY_INIT is not set
++# CONFIG_DEBUG_LIST is not set
++# CONFIG_DEBUG_SG is not set
++CONFIG_FRAME_POINTER=y
++# CONFIG_BOOT_PRINTK_DELAY is not set
++# CONFIG_RCU_TORTURE_TEST is not set
++# CONFIG_RCU_CPU_STALL_DETECTOR is not set
++# CONFIG_BACKTRACE_SELF_TEST is not set
++# CONFIG_DEBUG_BLOCK_EXT_DEVT is not set
++# CONFIG_FAULT_INJECTION is not set
++# CONFIG_LATENCYTOP is not set
++# CONFIG_SYSCTL_SYSCALL_CHECK is not set
++CONFIG_HAVE_FUNCTION_TRACER=y
++
++#
++# Tracers
++#
++# CONFIG_FUNCTION_TRACER is not set
++# CONFIG_IRQSOFF_TRACER is not set
++# CONFIG_PREEMPT_TRACER is not set
++# CONFIG_SCHED_TRACER is not set
++# CONFIG_CONTEXT_SWITCH_TRACER is not set
++# CONFIG_BOOT_TRACER is not set
++# CONFIG_STACK_TRACER is not set
++# CONFIG_DYNAMIC_PRINTK_DEBUG is not set
++# CONFIG_SAMPLES is not set
++CONFIG_HAVE_ARCH_KGDB=y
++# CONFIG_KGDB is not set
++CONFIG_DEBUG_USER=y
++CONFIG_DEBUG_ERRORS=y
++# CONFIG_DEBUG_STACK_USAGE is not set
++# CONFIG_DEBUG_LL is not set
++
++#
++# Security options
++#
++# CONFIG_KEYS is not set
++# CONFIG_SECURITY is not set
++# CONFIG_SECURITYFS is not set
++# CONFIG_SECURITY_FILE_CAPABILITIES is not set
++CONFIG_CRYPTO=m
++
++#
++# Crypto core or helper
++#
++# CONFIG_CRYPTO_FIPS is not set
++CONFIG_CRYPTO_ALGAPI=m
++CONFIG_CRYPTO_ALGAPI2=m
++CONFIG_CRYPTO_AEAD2=m
++CONFIG_CRYPTO_BLKCIPHER=m
++CONFIG_CRYPTO_BLKCIPHER2=m
++CONFIG_CRYPTO_HASH=m
++CONFIG_CRYPTO_HASH2=m
++CONFIG_CRYPTO_RNG2=m
++CONFIG_CRYPTO_MANAGER=m
++CONFIG_CRYPTO_MANAGER2=m
++# CONFIG_CRYPTO_GF128MUL is not set
++# CONFIG_CRYPTO_NULL is not set
++# CONFIG_CRYPTO_CRYPTD is not set
++# CONFIG_CRYPTO_AUTHENC is not set
++# CONFIG_CRYPTO_TEST is not set
++
++#
++# Authenticated Encryption with Associated Data
++#
++# CONFIG_CRYPTO_CCM is not set
++# CONFIG_CRYPTO_GCM is not set
++# CONFIG_CRYPTO_SEQIV is not set
++
++#
++# Block modes
++#
++# CONFIG_CRYPTO_CBC is not set
++# CONFIG_CRYPTO_CTR is not set
++# CONFIG_CRYPTO_CTS is not set
++CONFIG_CRYPTO_ECB=m
++# CONFIG_CRYPTO_LRW is not set
++# CONFIG_CRYPTO_PCBC is not set
++# CONFIG_CRYPTO_XTS is not set
++
++#
++# Hash modes
++#
++CONFIG_CRYPTO_HMAC=m
++# CONFIG_CRYPTO_XCBC is not set
++
++#
++# Digest
++#
++# CONFIG_CRYPTO_CRC32C is not set
++# CONFIG_CRYPTO_MD4 is not set
++CONFIG_CRYPTO_MD5=m
++CONFIG_CRYPTO_MICHAEL_MIC=m
++# CONFIG_CRYPTO_RMD128 is not set
++# CONFIG_CRYPTO_RMD160 is not set
++# CONFIG_CRYPTO_RMD256 is not set
++# CONFIG_CRYPTO_RMD320 is not set
++# CONFIG_CRYPTO_SHA1 is not set
++# CONFIG_CRYPTO_SHA256 is not set
++# CONFIG_CRYPTO_SHA512 is not set
++# CONFIG_CRYPTO_TGR192 is not set
++# CONFIG_CRYPTO_WP512 is not set
++
++#
++# Ciphers
++#
++CONFIG_CRYPTO_AES=m
++# CONFIG_CRYPTO_ANUBIS is not set
++CONFIG_CRYPTO_ARC4=m
++# CONFIG_CRYPTO_BLOWFISH is not set
++# CONFIG_CRYPTO_CAMELLIA is not set
++# CONFIG_CRYPTO_CAST5 is not set
++# CONFIG_CRYPTO_CAST6 is not set
++# CONFIG_CRYPTO_DES is not set
++# CONFIG_CRYPTO_FCRYPT is not set
++# CONFIG_CRYPTO_KHAZAD is not set
++# CONFIG_CRYPTO_SALSA20 is not set
++# CONFIG_CRYPTO_SEED is not set
++# CONFIG_CRYPTO_SERPENT is not set
++# CONFIG_CRYPTO_TEA is not set
++# CONFIG_CRYPTO_TWOFISH is not set
++
++#
++# Compression
++#
++# CONFIG_CRYPTO_DEFLATE is not set
++# CONFIG_CRYPTO_LZO is not set
++
++#
++# Random Number Generation
++#
++# CONFIG_CRYPTO_ANSI_CPRNG is not set
++CONFIG_CRYPTO_HW=y
++
++#
++# Library routines
++#
++CONFIG_BITREVERSE=y
++CONFIG_CRC_CCITT=m
++CONFIG_CRC16=m
++# CONFIG_CRC_T10DIF is not set
++# CONFIG_CRC_ITU_T is not set
++CONFIG_CRC32=y
++# CONFIG_CRC7 is not set
++# CONFIG_LIBCRC32C is not set
++CONFIG_ZLIB_INFLATE=y
++CONFIG_ZLIB_DEFLATE=y
++CONFIG_PLIST=y
++CONFIG_HAS_IOMEM=y
++CONFIG_HAS_IOPORT=y
++CONFIG_HAS_DMA=y
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/mach-imx/include/mach/imx-regs.h linux-2.6.28-karo/arch/arm/mach-imx/include/mach/imx-regs.h
+--- linux-2.6.28/arch/arm/mach-imx/include/mach/imx-regs.h 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/mach-imx/include/mach/imx-regs.h 2009-03-11 13:16:24.000000000 +0100
+@@ -416,7 +416,8 @@
+ #define PCR_BPIX_4 (2<<25)
+ #define PCR_BPIX_8 (3<<25)
+ #define PCR_BPIX_12 (4<<25)
+-#define PCR_BPIX_16 (4<<25)
++#define PCR_BPIX_16 (5<<25)
++#define PCR_BPIX_18 (6<<25)
+ #define PCR_PIXPOL (1<<24)
+ #define PCR_FLMPOL (1<<23)
+ #define PCR_LPPOL (1<<22)
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/mach-imx/include/mach/imxfb.h linux-2.6.28-karo/arch/arm/mach-imx/include/mach/imxfb.h
+--- linux-2.6.28/arch/arm/mach-imx/include/mach/imxfb.h 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/mach-imx/include/mach/imxfb.h 1970-01-01 01:00:00.000000000 +0100
+@@ -1,37 +0,0 @@
+-/*
+- * This structure describes the machine which we are running on.
+- */
+-struct imxfb_mach_info {
+- u_long pixclock;
+-
+- u_short xres;
+- u_short yres;
+-
+- u_int nonstd;
+- u_char bpp;
+- u_char hsync_len;
+- u_char left_margin;
+- u_char right_margin;
+-
+- u_char vsync_len;
+- u_char upper_margin;
+- u_char lower_margin;
+- u_char sync;
+-
+- u_int cmap_greyscale:1,
+- cmap_inverse:1,
+- cmap_static:1,
+- unused:29;
+-
+- u_int pcr;
+- u_int pwmr;
+- u_int lscr1;
+- u_int dmacr;
+-
+- u_char * fixed_screen_cpu;
+- dma_addr_t fixed_screen_dma;
+-
+- void (*lcd_power)(int);
+- void (*backlight_power)(int);
+-};
+-void set_imx_fb_info(struct imxfb_mach_info *hard_imx_fb_info);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/mach-mx2/Kconfig linux-2.6.28-karo/arch/arm/mach-mx2/Kconfig
+--- linux-2.6.28/arch/arm/mach-mx2/Kconfig 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/mach-mx2/Kconfig 2009-03-11 13:16:24.000000000 +0100
+@@ -24,6 +24,28 @@ config MACH_PCM038
+ Include support for phyCORE-i.MX27 (aka pcm038) platform. This
+ includes specific configurations for the module and its peripherals.
+
++config MACH_TX27
++ bool "Support Ka-Ro electronics TX27 module"
++ depends on MACH_MX27
++ select MXC_ULPI
++ help
++ Include support for Ka-Ro TX27 processor module
++
++config BASE_CLK_26MHz
++ bool "Use external 26MHz oscillator"
++ depends on MACH_TX27
++ default yes
++ help
++ Unselect this option to switch off the external 26MHz oscillator
++ and use the 32.768kHz reference and i.MX27 internal FPM
++
++config KARO_DEBUG
++ bool "Enable Ka-Ro specific debug messages"
++ depends on MACH_TX27
++ help
++ Compile the architecture specific files with -DDEBUG to enable
++ additional debug messages
++
+ choice
+ prompt "Baseboard"
+ depends on MACH_PCM038
+@@ -32,6 +54,7 @@ choice
+ config MACH_PCM970_BASEBOARD
+ prompt "PHYTEC PCM970 development board"
+ bool
++ select MXC_ULPI
+ help
+ This adds board specific devices that can be found on Phytec's
+ PCM970 evaluation board.
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/mach-mx2/Makefile linux-2.6.28-karo/arch/arm/mach-mx2/Makefile
+--- linux-2.6.28/arch/arm/mach-mx2/Makefile 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/mach-mx2/Makefile 2009-03-11 13:16:24.000000000 +0100
+@@ -2,13 +2,19 @@
+ # Makefile for the linux kernel.
+ #
+
++ifeq ($(CONFIG_KARO_DEBUG),y)
++ EXTRA_CFLAGS += -DDEBUG
++endif
++
+ # Object file lists.
+
+ obj-y := system.o generic.o devices.o serial.o
+
+ obj-$(CONFIG_MACH_MX27) += cpu_imx27.o
+ obj-$(CONFIG_MACH_MX27) += clock_imx27.o
++obj-$(CONFIG_PM) += pm.o mxc_pm.o
+
+-obj-$(CONFIG_MACH_MX27ADS) += mx27ads.o
+-obj-$(CONFIG_MACH_PCM038) += pcm038.o
+-obj-$(CONFIG_MACH_PCM970_BASEBOARD) += pcm970-baseboard.o
++obj-$(CONFIG_MACH_MX27ADS) += mx27ads.o
++obj-$(CONFIG_MACH_PCM038) += pcm038.o
++obj-$(CONFIG_MACH_PCM970_BASEBOARD) += pcm970-baseboard.o
++obj-$(CONFIG_MACH_TX27) += karo-tx27.o tx27_gpio.o
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/mach-mx2/clock_imx27.c linux-2.6.28-karo/arch/arm/mach-mx2/clock_imx27.c
+--- linux-2.6.28/arch/arm/mach-mx2/clock_imx27.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/mach-mx2/clock_imx27.c 2009-03-11 13:16:24.000000000 +0100
+@@ -21,6 +21,7 @@
+ #include <linux/io.h>
+ #include <linux/module.h>
+ #include <linux/spinlock.h>
++#include <linux/delay.h>
+
+ #include <mach/clock.h>
+ #include <mach/common.h>
+@@ -233,9 +234,23 @@ static void _clk_mstick1_disable(struct
+ _clk_pccr10_disable(CCM_PCCR1_MSHC_BAUD_MASK, CCM_PCCR0_MSHC_MASK);
+ }
+
+-#define CSCR() (__raw_readl(CCM_CSCR))
+-#define PCDR0() (__raw_readl(CCM_PCDR0))
+-#define PCDR1() (__raw_readl(CCM_PCDR1))
++static int _clk_wdog_enable(struct clk *clk)
++{
++ u32 cscr;
++
++ cscr = __raw_readl(CCM_CSCR);
++ if (!(cscr & CCM_CSCR_FPM)) {
++ cscr |= CCM_CSCR_FPM;
++ __raw_writel(cscr, CCM_CSCR);
++ /* wait for FPM startup */
++ udelay(90);
++ }
++ return _clk_enable(clk);
++}
++
++#define CSCR() __raw_readl(CCM_CSCR)
++#define PCDR0() __raw_readl(CCM_PCDR0)
++#define PCDR1() __raw_readl(CCM_PCDR1)
+
+ static int _clk_cpu_set_parent(struct clk *clk, struct clk *parent)
+ {
+@@ -288,7 +303,7 @@ static int _clk_cpu_set_rate(struct clk
+
+ div = parent_rate / rate;
+
+- if (div > 4 || div < 1 || ((parent_rate / div) != rate))
++ if (div > 4 || div < 1 || ((parent_rate / div / 100) != rate / 100))
+ return -EINVAL;
+
+ div--;
+@@ -297,10 +312,13 @@ static int _clk_cpu_set_rate(struct clk
+ if (mx27_revision() >= CHIP_REV_2_0) {
+ reg &= ~CCM_CSCR_ARM_MASK;
+ reg |= div << CCM_CSCR_ARM_OFFSET;
++#if 0
++ // FIXME: What's this? This breaks _clk_spll_enable()!
+ reg &= ~0x06;
++#endif
+ __raw_writel(reg | 0x80000000, CCM_CSCR);
+ } else {
+- printk(KERN_ERR "Cant set CPU frequency!\n");
++ printk(KERN_ERR "Can't set CPU frequency!\n");
+ }
+
+ return 0;
+@@ -361,6 +379,24 @@ static unsigned long _clk_usb_recalc(str
+ return parent_rate / (usb_pdf + 1U);
+ }
+
++static int _clk_usb_set_rate(struct clk *clk, unsigned long rate)
++{
++ u32 div, reg;
++ unsigned long parent_rate;
++
++ parent_rate = clk_get_rate(clk->parent);
++ div = parent_rate / rate;
++
++ if (div > 8 || div < 1 || ((parent_rate / div) != rate)) {
++ return -EINVAL;
++ }
++ div--;
++
++ reg = (CSCR() & ~CCM_CSCR_USB_MASK) | (div << CCM_CSCR_USB_OFFSET);
++ __raw_writel(reg, CCM_CSCR);
++ return 0;
++}
++
+ static unsigned long _clk_ssi1_recalc(struct clk *clk)
+ {
+ unsigned long ssi1_pdf;
+@@ -488,10 +524,13 @@ static unsigned long get_mpll_clk(struct
+ {
+ uint32_t reg;
+ unsigned long ref_clk;
+- unsigned long mfi = 0, mfn = 0, mfd = 0, pdf = 0;
++ int mfi, mfn, mfd, pdf;
+ unsigned long long temp;
++ int sign = 1;
+
+ ref_clk = clk_get_rate(clk->parent);
++ if (clk->parent == &ckil_clk)
++ ref_clk *= 1024;
+
+ reg = __raw_readl(CCM_MPCTL0);
+ pdf = (reg & CCM_MPCTL0_PD_MASK) >> CCM_MPCTL0_PD_OFFSET;
+@@ -500,9 +539,13 @@ static unsigned long get_mpll_clk(struct
+ mfn = (reg & CCM_MPCTL0_MFN_MASK) >> CCM_MPCTL0_MFN_OFFSET;
+
+ mfi = (mfi <= 5) ? 5 : mfi;
++ if (mfn >= 512) {
++ mfn = 1024 - mfn;
++ sign = -1;
++ }
+ temp = 2LL * ref_clk * mfn;
+ do_div(temp, mfd + 1);
+- temp = 2LL * ref_clk * mfi + temp;
++ temp = temp * sign + 2LL * ref_clk * mfi;
+ do_div(temp, pdf + 1);
+
+ return (unsigned long)temp;
+@@ -555,10 +598,13 @@ static unsigned long get_spll_clk(struct
+ {
+ uint32_t reg;
+ unsigned long ref_clk;
+- unsigned long mfi = 0, mfn = 0, mfd = 0, pdf = 0;
+- unsigned long long temp;
++ int mfi, mfn, mfd, pdf;
++ u64 temp;
++ int sign = 1;
+
+ ref_clk = clk_get_rate(clk->parent);
++ if (clk->parent == &ckil_clk)
++ ref_clk *= 1024;
+
+ reg = __raw_readl(CCM_SPCTL0);
+ /*TODO: This is TO2 Bug */
+@@ -571,9 +617,13 @@ static unsigned long get_spll_clk(struct
+ mfn = (reg & CCM_SPCTL0_MFN_MASK) >> CCM_SPCTL0_MFN_OFFSET;
+
+ mfi = (mfi <= 5) ? 5 : mfi;
++ if (mfn >= 512) {
++ sign = -1;
++ mfn = 1024 - mfn;
++ }
+ temp = 2LL * ref_clk * mfn;
+ do_div(temp, mfd + 1);
+- temp = 2LL * ref_clk * mfi + temp;
++ temp = temp * sign + 2LL * ref_clk * mfi;
+ do_div(temp, pdf + 1);
+
+ return (unsigned long)temp;
+@@ -1079,6 +1129,7 @@ static struct clk usb_clk[] = {
+ .name = "usb_clk",
+ .parent = &spll_clk,
+ .get_rate = _clk_usb_recalc,
++ .set_rate = _clk_usb_set_rate,
+ .enable = _clk_enable,
+ .enable_reg = CCM_PCCR1,
+ .enable_shift = CCM_PCCR1_USBOTG_OFFSET,
+@@ -1234,7 +1285,7 @@ static struct clk mstick1_clk = {
+ static struct clk wdog_clk = {
+ .name = "wdog_clk",
+ .parent = &ipg_clk,
+- .enable = _clk_enable,
++ .enable = _clk_wdog_enable,
+ .enable_reg = CCM_PCCR1,
+ .enable_shift = CCM_PCCR1_WDT_OFFSET,
+ .disable = _clk_disable,
+@@ -1537,26 +1588,22 @@ void __init change_external_low_referenc
+ external_low_reference = new_ref;
+ }
+
+-unsigned long __init clk_early_get_timer_rate(void)
+-{
+- return clk_get_rate(&per_clk[0]);
+-}
+-
+ static void __init probe_mxc_clocks(void)
+ {
+ int i;
++ u32 cscr = CSCR();
+
+ if (mx27_revision() >= CHIP_REV_2_0) {
+- if (CSCR() & 0x8000)
++ if (cscr & 0x8000)
+ cpu_clk.parent = &mpll_main_clk[0];
+
+- if (!(CSCR() & 0x00800000))
++ if (!(cscr & 0x00800000))
+ ssi2_clk[0].parent = &spll_clk;
+
+- if (!(CSCR() & 0x00400000))
++ if (!(cscr & 0x00400000))
+ ssi1_clk[0].parent = &spll_clk;
+
+- if (!(CSCR() & 0x00200000))
++ if (!(cscr & 0x00200000))
+ vpu_clk.parent = &spll_clk;
+ } else {
+ cpu_clk.parent = &mpll_clk;
+@@ -1618,7 +1665,7 @@ int __init mxc_clocks_init(unsigned long
+ clk_enable(&gpio_clk);
+ clk_enable(&iim_clk);
+ clk_enable(&gpt1_clk[0]);
+-#ifdef CONFIG_DEBUG_LL_CONSOLE
++#ifdef CONFIG_DEBUG_LL
+ clk_enable(&uart1_clk[0]);
+ #endif
+ return 0;
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/mach-mx2/crm_regs.h linux-2.6.28-karo/arch/arm/mach-mx2/crm_regs.h
+--- linux-2.6.28/arch/arm/mach-mx2/crm_regs.h 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/mach-mx2/crm_regs.h 2009-03-11 13:16:24.000000000 +0100
+@@ -22,252 +22,276 @@
+
+ #include <mach/hardware.h>
+
++#define SYSCTRL_BASE IO_ADDRESS(SYSCTRL_BASE_ADDR)
++
+ /* Register offsets */
+-#define CCM_CSCR (IO_ADDRESS(CCM_BASE_ADDR) + 0x0)
+-#define CCM_MPCTL0 (IO_ADDRESS(CCM_BASE_ADDR) + 0x4)
+-#define CCM_MPCTL1 (IO_ADDRESS(CCM_BASE_ADDR) + 0x8)
+-#define CCM_SPCTL0 (IO_ADDRESS(CCM_BASE_ADDR) + 0xC)
+-#define CCM_SPCTL1 (IO_ADDRESS(CCM_BASE_ADDR) + 0x10)
+-#define CCM_OSC26MCTL (IO_ADDRESS(CCM_BASE_ADDR) + 0x14)
+-#define CCM_PCDR0 (IO_ADDRESS(CCM_BASE_ADDR) + 0x18)
+-#define CCM_PCDR1 (IO_ADDRESS(CCM_BASE_ADDR) + 0x1c)
+-#define CCM_PCCR0 (IO_ADDRESS(CCM_BASE_ADDR) + 0x20)
+-#define CCM_PCCR1 (IO_ADDRESS(CCM_BASE_ADDR) + 0x24)
+-#define CCM_CCSR (IO_ADDRESS(CCM_BASE_ADDR) + 0x28)
+-#define CCM_PMCTL (IO_ADDRESS(CCM_BASE_ADDR) + 0x2c)
+-#define CCM_PMCOUNT (IO_ADDRESS(CCM_BASE_ADDR) + 0x30)
+-#define CCM_WKGDCTL (IO_ADDRESS(CCM_BASE_ADDR) + 0x34)
+-
+-#define CCM_CSCR_USB_OFFSET 28
+-#define CCM_CSCR_USB_MASK (0x7 << 28)
+-#define CCM_CSCR_SD_OFFSET 24
+-#define CCM_CSCR_SD_MASK (0x3 << 24)
+-#define CCM_CSCR_SSI2 (1 << 23)
+-#define CCM_CSCR_SSI2_OFFSET 23
+-#define CCM_CSCR_SSI1 (1 << 22)
+-#define CCM_CSCR_SSI1_OFFSET 22
+-#define CCM_CSCR_VPU (1 << 21)
+-#define CCM_CSCR_VPU_OFFSET 21
+-#define CCM_CSCR_MSHC (1 << 20)
+-#define CCM_CSCR_SPLLRES (1 << 19)
+-#define CCM_CSCR_MPLLRES (1 << 18)
+-#define CCM_CSCR_SP (1 << 17)
+-#define CCM_CSCR_MCU (1 << 16)
++#define CCM_CSCR (IO_ADDRESS(CCM_BASE_ADDR) + 0x0)
++#define CCM_MPCTL0 (IO_ADDRESS(CCM_BASE_ADDR) + 0x4)
++#define CCM_MPCTL1 (IO_ADDRESS(CCM_BASE_ADDR) + 0x8)
++#define CCM_SPCTL0 (IO_ADDRESS(CCM_BASE_ADDR) + 0xC)
++#define CCM_SPCTL1 (IO_ADDRESS(CCM_BASE_ADDR) + 0x10)
++#define CCM_OSC26MCTL (IO_ADDRESS(CCM_BASE_ADDR) + 0x14)
++#define CCM_PCDR0 (IO_ADDRESS(CCM_BASE_ADDR) + 0x18)
++#define CCM_PCDR1 (IO_ADDRESS(CCM_BASE_ADDR) + 0x1c)
++#define CCM_PCCR0 (IO_ADDRESS(CCM_BASE_ADDR) + 0x20)
++#define CCM_PCCR1 (IO_ADDRESS(CCM_BASE_ADDR) + 0x24)
++#define CCM_CCSR (IO_ADDRESS(CCM_BASE_ADDR) + 0x28)
++#define CCM_PMCTL (IO_ADDRESS(CCM_BASE_ADDR) + 0x2c)
++#define CCM_PMCOUNT (IO_ADDRESS(CCM_BASE_ADDR) + 0x30)
++#define CCM_WKGDCTL (IO_ADDRESS(CCM_BASE_ADDR) + 0x34)
++#define MXC_CCM_PMCR0 (SYSCTRL_BASE + 0x60)
++#define MXC_CCM_DCVR0 (SYSCTRL_BASE + 0x64)
++#define MXC_CCM_DCVR1 (SYSCTRL_BASE + 0x68)
++#define MXC_CCM_DCVR2 (SYSCTRL_BASE + 0x72)
++#define MXC_CCM_DCVR3 (SYSCTRL_BASE + 0x76)
++#define MXC_CCM_PMCR0_DPTEN 0x00000001
++#define MXC_CCM_DIE 0x00000002
++#define MXC_CCM_DIM 0x0000000C
++#define MXC_CCM_DCR 0x00000200
++#define MXC_CCM_PMCR0_DRCE0 0x00000010
++#define MXC_CCM_PMCR0_DRCE1 0x00000020
++#define MXC_CCM_PMCR0_DRCE2 0x00000040
++#define MXC_CCM_PMCR0_DRCE3 0x00000080
++#define MXC_CCM_PMCR0_PTVAIM MXC_CCM_DIM
++
++#define CCM_CSCR_USB_OFFSET 28
++#define CCM_CSCR_USB_MASK (0x7 << CCM_CSCR_USB_OFFSET)
++#define CCM_CSCR_SD_OFFSET 24
++#define CCM_CSCR_SD_MASK (0x3 << CCM_CSCR_SD_OFFSET)
++#define CCM_CSCR_SSI2_OFFSET 23
++#define CCM_CSCR_SSI2 (1 << CCM_CSCR_SSI2_OFFSET)
++#define CCM_CSCR_SSI1_OFFSET 22
++#define CCM_CSCR_SSI1 (1 << CCM_CSCR_SSI1_OFFSET)
++#define CCM_CSCR_VPU_OFFSET 21
++#define CCM_CSCR_VPU (1 << CCM_CSCR_VPU_OFFSET)
++#define CCM_CSCR_MSHC (1 << 20)
++#define CCM_CSCR_SPLLRES (1 << 19)
++#define CCM_CSCR_MPLLRES (1 << 18)
++#define CCM_CSCR_SP (1 << 17)
++#define CCM_CSCR_MCU (1 << 16)
+ /* CCM_CSCR_ARM_xxx just be avaliable on i.MX27 TO2*/
+-#define CCM_CSCR_ARM_SRC (1 << 15)
+-#define CCM_CSCR_ARM_OFFSET 12
+-#define CCM_CSCR_ARM_MASK (0x3 << 12)
++#define CCM_CSCR_ARM_SRC (1 << 15)
++#define CCM_CSCR_ARM_OFFSET 12
++#define CCM_CSCR_ARM_MASK (0x3 << CCM_CSCR_ARM_OFFSET)
+ /* CCM_CSCR_ARM_xxx just be avaliable on i.MX27 TO2*/
+-#define CCM_CSCR_PRESC_OFFSET 13
+-#define CCM_CSCR_PRESC_MASK (0x7 << 13)
+-#define CCM_CSCR_BCLK_OFFSET 9
+-#define CCM_CSCR_BCLK_MASK (0xf << 9)
+-#define CCM_CSCR_IPDIV_OFFSET 8
+-#define CCM_CSCR_IPDIV (1 << 8)
++#define CCM_CSCR_PRESC_OFFSET 13
++#define CCM_CSCR_PRESC_MASK (0x7 << CCM_CSCR_PRESC_OFFSET)
++#define CCM_CSCR_BCLK_OFFSET 9
++#define CCM_CSCR_BCLK_MASK (0xf << CCM_CSCR_BCLK_OFFSET)
++#define CCM_CSCR_IPDIV_OFFSET 8
++#define CCM_CSCR_IPDIV (1 << CCM_CSCR_IPDIV_OFFSET)
+ /* CCM_CSCR_AHB_xxx just be avaliable on i.MX27 TO2*/
+-#define CCM_CSCR_AHB_OFFSET 8
+-#define CCM_CSCR_AHB_MASK (0x3 << 8)
++#define CCM_CSCR_AHB_OFFSET 8
++#define CCM_CSCR_AHB_MASK (0x3 << CCM_CSCR_AHB_OFFSET)
+ /* CCM_CSCR_AHB_xxx just be avaliable on i.MX27 TO2*/
+-#define CCM_CSCR_OSC26MDIV (1 << 4)
+-#define CCM_CSCR_OSC26M (1 << 3)
+-#define CCM_CSCR_FPM (1 << 2)
+-#define CCM_CSCR_SPEN (1 << 1)
+-#define CCM_CSCR_MPEN 1
+-
+-#define CCM_MPCTL0_CPLM (1 << 31)
+-#define CCM_MPCTL0_PD_OFFSET 26
+-#define CCM_MPCTL0_PD_MASK (0xf << 26)
+-#define CCM_MPCTL0_MFD_OFFSET 16
+-#define CCM_MPCTL0_MFD_MASK (0x3ff << 16)
+-#define CCM_MPCTL0_MFI_OFFSET 10
+-#define CCM_MPCTL0_MFI_MASK (0xf << 10)
+-#define CCM_MPCTL0_MFN_OFFSET 0
+-#define CCM_MPCTL0_MFN_MASK 0x3ff
+-
+-#define CCM_MPCTL1_LF (1 << 15)
+-#define CCM_MPCTL1_BRMO (1 << 6)
+-
+-#define CCM_SPCTL0_CPLM (1 << 31)
+-#define CCM_SPCTL0_PD_OFFSET 26
+-#define CCM_SPCTL0_PD_MASK (0xf << 26)
+-#define CCM_SPCTL0_MFD_OFFSET 16
+-#define CCM_SPCTL0_MFD_MASK (0x3ff << 16)
+-#define CCM_SPCTL0_MFI_OFFSET 10
+-#define CCM_SPCTL0_MFI_MASK (0xf << 10)
+-#define CCM_SPCTL0_MFN_OFFSET 0
+-#define CCM_SPCTL0_MFN_MASK 0x3ff
+-
+-#define CCM_SPCTL1_LF (1 << 15)
+-#define CCM_SPCTL1_BRMO (1 << 6)
+-
+-#define CCM_OSC26MCTL_PEAK_OFFSET 16
+-#define CCM_OSC26MCTL_PEAK_MASK (0x3 << 16)
+-#define CCM_OSC26MCTL_AGC_OFFSET 8
+-#define CCM_OSC26MCTL_AGC_MASK (0x3f << 8)
+-#define CCM_OSC26MCTL_ANATEST_OFFSET 0
+-#define CCM_OSC26MCTL_ANATEST_MASK 0x3f
+-
+-#define CCM_PCDR0_SSI2BAUDDIV_OFFSET 26
+-#define CCM_PCDR0_SSI2BAUDDIV_MASK (0x3f << 26)
+-#define CCM_PCDR0_CLKO_EN 25
+-#define CCM_PCDR0_CLKODIV_OFFSET 22
+-#define CCM_PCDR0_CLKODIV_MASK (0x7 << 22)
+-#define CCM_PCDR0_SSI1BAUDDIV_OFFSET 16
+-#define CCM_PCDR0_SSI1BAUDDIV_MASK (0x3f << 16)
++#define CCM_CSCR_OSC26MDIV (1 << 4)
++#define CCM_CSCR_OSC26M (1 << 3)
++#define CCM_CSCR_FPM (1 << 2)
++#define CCM_CSCR_SPEN (1 << 1)
++#define CCM_CSCR_MPEN 1
++
++#define CCM_MPCTL0_CPLM (1 << 31)
++#define CCM_MPCTL0_PD_OFFSET 26
++#define CCM_MPCTL0_PD_MASK (0xf << CCM_MPCTL0_PD_OFFSET)
++#define CCM_MPCTL0_PD_VAL(n) (((n) << CCM_MPCTL0_PD_OFFSET) & CCM_MPCTL0_PD_MASK)
++#define CCM_MPCTL0_MFD_OFFSET 16
++#define CCM_MPCTL0_MFD_MASK (0x3ff << CCM_MPCTL0_MFD_OFFSET)
++#define CCM_MPCTL0_MFD_VAL(n) (((n) << CCM_MPCTL0_MFD_OFFSET) & CCM_MPCTL0_MFD_MASK)
++#define CCM_MPCTL0_MFI_OFFSET 10
++#define CCM_MPCTL0_MFI_MASK (0xf << CCM_MPCTL0_MFI_OFFSET)
++#define CCM_MPCTL0_MFI_VAL(n) (((n) << CCM_MPCTL0_MFI_OFFSET) & CCM_MPCTL0_MFI_MASK)
++#define CCM_MPCTL0_MFN_OFFSET 0
++#define CCM_MPCTL0_MFN_MASK (0x3ff << CCM_MPCTL0_MFN_OFFSET)
++#define CCM_MPCTL0_MFN_VAL(n) (((n) << CCM_MPCTL0_MFN_OFFSET) & CCM_MPCTL0_MFN_MASK)
++
++#define CCM_MPCTL1_LF (1 << 15)
++#define CCM_MPCTL1_BRMO (1 << 6)
++
++#define CCM_SPCTL0_CPLM (1 << 31)
++#define CCM_SPCTL0_PD_OFFSET 26
++#define CCM_SPCTL0_PD_MASK (0xf << CCM_SPCTL0_PD_OFFSET)
++#define CCM_SPCTL0_PD_VAL(n) (((n) << CCM_SPCTL0_PD_OFFSET) & CCM_SPCTL0_PD_MASK)
++#define CCM_SPCTL0_MFD_OFFSET 16
++#define CCM_SPCTL0_MFD_MASK (0x3ff << CCM_SPCTL0_MFD_OFFSET)
++#define CCM_SPCTL0_MFD_VAL(n) (((n) << CCM_SPCTL0_MFD_OFFSET) & CCM_SPCTL0_MFD_MASK)
++#define CCM_SPCTL0_MFI_OFFSET 10
++#define CCM_SPCTL0_MFI_MASK (0xf << CCM_SPCTL0_MFI_OFFSET)
++#define CCM_SPCTL0_MFI_VAL(n) (((n) << CCM_SPCTL0_MFI_OFFSET) & CCM_SPCTL0_MFI_MASK)
++#define CCM_SPCTL0_MFN_OFFSET 0
++#define CCM_SPCTL0_MFN_MASK (0x3ff << CCM_SPCTL0_MFN_OFFSET)
++#define CCM_SPCTL0_MFN_VAL(n) (((n) << CCM_SPCTL0_MFN_OFFSET) & CCM_SPCTL0_MFN_MASK)
++
++#define CCM_SPCTL1_LF (1 << 15)
++#define CCM_SPCTL1_BRMO (1 << 6)
++
++#define CCM_OSC26MCTL_PEAK_OFFSET 16
++#define CCM_OSC26MCTL_PEAK_MASK (0x3 << CCM_OSC26MCTL_PEAK_OFFSET)
++#define CCM_OSC26MCTL_AGC_OFFSET 8
++#define CCM_OSC26MCTL_AGC_MASK (0x3f << CCM_OSC26MCTL_AGC_OFFSET)
++#define CCM_OSC26MCTL_ANATEST_OFFSET 0
++#define CCM_OSC26MCTL_ANATEST_MASK 0x3f
++
++#define CCM_PCDR0_SSI2BAUDDIV_OFFSET 26
++#define CCM_PCDR0_SSI2BAUDDIV_MASK (0x3f << CCM_PCDR0_SSI2BAUDDIV_OFFSET)
++#define CCM_PCDR0_CLKO_EN 25
++#define CCM_PCDR0_CLKODIV_OFFSET 22
++#define CCM_PCDR0_CLKODIV_MASK (0x7 << CCM_PCDR0_CLKODIV_OFFSET)
++#define CCM_PCDR0_SSI1BAUDDIV_OFFSET 16
++#define CCM_PCDR0_SSI1BAUDDIV_MASK (0x3f << CCM_PCDR0_SSI1BAUDDIV_OFFSET)
+ /*The difinition for i.MX27 TO2*/
+-#define CCM_PCDR0_VPUDIV2_OFFSET 10
+-#define CCM_PCDR0_VPUDIV2_MASK (0x3f << 10)
+-#define CCM_PCDR0_NFCDIV2_OFFSET 6
+-#define CCM_PCDR0_NFCDIV2_MASK (0xf << 6)
+-#define CCM_PCDR0_MSHCDIV2_MASK 0x3f
++#define CCM_PCDR0_VPUDIV2_OFFSET 10
++#define CCM_PCDR0_VPUDIV2_MASK (0x3f << CCM_PCDR0_VPUDIV2_OFFSET)
++#define CCM_PCDR0_NFCDIV2_OFFSET 6
++#define CCM_PCDR0_NFCDIV2_MASK (0xf << CCM_PCDR0_NFCDIV2_OFFSET)
++#define CCM_PCDR0_MSHCDIV2_MASK 0x3f
+ /*The difinition for i.MX27 TO2*/
+-#define CCM_PCDR0_NFCDIV_OFFSET 12
+-#define CCM_PCDR0_NFCDIV_MASK (0xf << 12)
+-#define CCM_PCDR0_VPUDIV_OFFSET 8
+-#define CCM_PCDR0_VPUDIV_MASK (0xf << 8)
+-#define CCM_PCDR0_MSHCDIV_OFFSET 0
+-#define CCM_PCDR0_MSHCDIV_MASK 0x1f
+-
+-#define CCM_PCDR1_PERDIV4_OFFSET 24
+-#define CCM_PCDR1_PERDIV4_MASK (0x3f << 24)
+-#define CCM_PCDR1_PERDIV3_OFFSET 16
+-#define CCM_PCDR1_PERDIV3_MASK (0x3f << 16)
+-#define CCM_PCDR1_PERDIV2_OFFSET 8
+-#define CCM_PCDR1_PERDIV2_MASK (0x3f << 8)
+-#define CCM_PCDR1_PERDIV1_OFFSET 0
+-#define CCM_PCDR1_PERDIV1_MASK 0x3f
+-
+-#define CCM_PCCR0_CSPI1_OFFSET 31
+-#define CCM_PCCR0_CSPI1_MASK (1 << 31)
+-#define CCM_PCCR0_CSPI2_OFFSET 30
+-#define CCM_PCCR0_CSPI2_MASK (1 << 30)
+-#define CCM_PCCR0_CSPI3_OFFSET 29
+-#define CCM_PCCR0_CSPI3_MASK (1 << 29)
+-#define CCM_PCCR0_DMA_OFFSET 28
+-#define CCM_PCCR0_DMA_MASK (1 << 28)
+-#define CCM_PCCR0_EMMA_OFFSET 27
+-#define CCM_PCCR0_EMMA_MASK (1 << 27)
+-#define CCM_PCCR0_FEC_OFFSET 26
+-#define CCM_PCCR0_FEC_MASK (1 << 26)
+-#define CCM_PCCR0_GPIO_OFFSET 25
+-#define CCM_PCCR0_GPIO_MASK (1 << 25)
+-#define CCM_PCCR0_GPT1_OFFSET 24
+-#define CCM_PCCR0_GPT1_MASK (1 << 24)
+-#define CCM_PCCR0_GPT2_OFFSET 23
+-#define CCM_PCCR0_GPT2_MASK (1 << 23)
+-#define CCM_PCCR0_GPT3_OFFSET 22
+-#define CCM_PCCR0_GPT3_MASK (1 << 22)
+-#define CCM_PCCR0_GPT4_OFFSET 21
+-#define CCM_PCCR0_GPT4_MASK (1 << 21)
+-#define CCM_PCCR0_GPT5_OFFSET 20
+-#define CCM_PCCR0_GPT5_MASK (1 << 20)
+-#define CCM_PCCR0_GPT6_OFFSET 19
+-#define CCM_PCCR0_GPT6_MASK (1 << 19)
+-#define CCM_PCCR0_I2C1_OFFSET 18
+-#define CCM_PCCR0_I2C1_MASK (1 << 18)
+-#define CCM_PCCR0_I2C2_OFFSET 17
+-#define CCM_PCCR0_I2C2_MASK (1 << 17)
+-#define CCM_PCCR0_IIM_OFFSET 16
+-#define CCM_PCCR0_IIM_MASK (1 << 16)
+-#define CCM_PCCR0_KPP_OFFSET 15
+-#define CCM_PCCR0_KPP_MASK (1 << 15)
+-#define CCM_PCCR0_LCDC_OFFSET 14
+-#define CCM_PCCR0_LCDC_MASK (1 << 14)
+-#define CCM_PCCR0_MSHC_OFFSET 13
+-#define CCM_PCCR0_MSHC_MASK (1 << 13)
+-#define CCM_PCCR0_OWIRE_OFFSET 12
+-#define CCM_PCCR0_OWIRE_MASK (1 << 12)
+-#define CCM_PCCR0_PWM_OFFSET 11
+-#define CCM_PCCR0_PWM_MASK (1 << 11)
+-#define CCM_PCCR0_RTC_OFFSET 9
+-#define CCM_PCCR0_RTC_MASK (1 << 9)
+-#define CCM_PCCR0_RTIC_OFFSET 8
+-#define CCM_PCCR0_RTIC_MASK (1 << 8)
+-#define CCM_PCCR0_SAHARA_OFFSET 7
+-#define CCM_PCCR0_SAHARA_MASK (1 << 7)
+-#define CCM_PCCR0_SCC_OFFSET 6
+-#define CCM_PCCR0_SCC_MASK (1 << 6)
+-#define CCM_PCCR0_SDHC1_OFFSET 5
+-#define CCM_PCCR0_SDHC1_MASK (1 << 5)
+-#define CCM_PCCR0_SDHC2_OFFSET 4
+-#define CCM_PCCR0_SDHC2_MASK (1 << 4)
+-#define CCM_PCCR0_SDHC3_OFFSET 3
+-#define CCM_PCCR0_SDHC3_MASK (1 << 3)
+-#define CCM_PCCR0_SLCDC_OFFSET 2
+-#define CCM_PCCR0_SLCDC_MASK (1 << 2)
+-#define CCM_PCCR0_SSI1_IPG_OFFSET 1
+-#define CCM_PCCR0_SSI1_IPG_MASK (1 << 1)
+-#define CCM_PCCR0_SSI2_IPG_OFFSET 0
+-#define CCM_PCCR0_SSI2_IPG_MASK (1 << 0)
+-
+-#define CCM_PCCR1_UART1_OFFSET 31
+-#define CCM_PCCR1_UART1_MASK (1 << 31)
+-#define CCM_PCCR1_UART2_OFFSET 30
+-#define CCM_PCCR1_UART2_MASK (1 << 30)
+-#define CCM_PCCR1_UART3_OFFSET 29
+-#define CCM_PCCR1_UART3_MASK (1 << 29)
+-#define CCM_PCCR1_UART4_OFFSET 28
+-#define CCM_PCCR1_UART4_MASK (1 << 28)
+-#define CCM_PCCR1_UART5_OFFSET 27
+-#define CCM_PCCR1_UART5_MASK (1 << 27)
+-#define CCM_PCCR1_UART6_OFFSET 26
+-#define CCM_PCCR1_UART6_MASK (1 << 26)
+-#define CCM_PCCR1_USBOTG_OFFSET 25
+-#define CCM_PCCR1_USBOTG_MASK (1 << 25)
+-#define CCM_PCCR1_WDT_OFFSET 24
+-#define CCM_PCCR1_WDT_MASK (1 << 24)
+-#define CCM_PCCR1_HCLK_ATA_OFFSET 23
+-#define CCM_PCCR1_HCLK_ATA_MASK (1 << 23)
+-#define CCM_PCCR1_HCLK_BROM_OFFSET 22
+-#define CCM_PCCR1_HCLK_BROM_MASK (1 << 22)
+-#define CCM_PCCR1_HCLK_CSI_OFFSET 21
+-#define CCM_PCCR1_HCLK_CSI_MASK (1 << 21)
+-#define CCM_PCCR1_HCLK_DMA_OFFSET 20
+-#define CCM_PCCR1_HCLK_DMA_MASK (1 << 20)
+-#define CCM_PCCR1_HCLK_EMI_OFFSET 19
+-#define CCM_PCCR1_HCLK_EMI_MASK (1 << 19)
+-#define CCM_PCCR1_HCLK_EMMA_OFFSET 18
+-#define CCM_PCCR1_HCLK_EMMA_MASK (1 << 18)
+-#define CCM_PCCR1_HCLK_FEC_OFFSET 17
+-#define CCM_PCCR1_HCLK_FEC_MASK (1 << 17)
+-#define CCM_PCCR1_HCLK_VPU_OFFSET 16
+-#define CCM_PCCR1_HCLK_VPU_MASK (1 << 16)
+-#define CCM_PCCR1_HCLK_LCDC_OFFSET 15
+-#define CCM_PCCR1_HCLK_LCDC_MASK (1 << 15)
+-#define CCM_PCCR1_HCLK_RTIC_OFFSET 14
+-#define CCM_PCCR1_HCLK_RTIC_MASK (1 << 14)
+-#define CCM_PCCR1_HCLK_SAHARA_OFFSET 13
+-#define CCM_PCCR1_HCLK_SAHARA_MASK (1 << 13)
+-#define CCM_PCCR1_HCLK_SLCDC_OFFSET 12
+-#define CCM_PCCR1_HCLK_SLCDC_MASK (1 << 12)
+-#define CCM_PCCR1_HCLK_USBOTG_OFFSET 11
+-#define CCM_PCCR1_HCLK_USBOTG_MASK (1 << 11)
+-#define CCM_PCCR1_PERCLK1_OFFSET 10
+-#define CCM_PCCR1_PERCLK1_MASK (1 << 10)
+-#define CCM_PCCR1_PERCLK2_OFFSET 9
+-#define CCM_PCCR1_PERCLK2_MASK (1 << 9)
+-#define CCM_PCCR1_PERCLK3_OFFSET 8
+-#define CCM_PCCR1_PERCLK3_MASK (1 << 8)
+-#define CCM_PCCR1_PERCLK4_OFFSET 7
+-#define CCM_PCCR1_PERCLK4_MASK (1 << 7)
+-#define CCM_PCCR1_VPU_BAUD_OFFSET 6
+-#define CCM_PCCR1_VPU_BAUD_MASK (1 << 6)
+-#define CCM_PCCR1_SSI1_BAUD_OFFSET 5
+-#define CCM_PCCR1_SSI1_BAUD_MASK (1 << 5)
+-#define CCM_PCCR1_SSI2_BAUD_OFFSET 4
+-#define CCM_PCCR1_SSI2_BAUD_MASK (1 << 4)
+-#define CCM_PCCR1_NFC_BAUD_OFFSET 3
+-#define CCM_PCCR1_NFC_BAUD_MASK (1 << 3)
+-#define CCM_PCCR1_MSHC_BAUD_OFFSET 2
+-#define CCM_PCCR1_MSHC_BAUD_MASK (1 << 2)
+-
+-#define CCM_CCSR_32KSR (1 << 15)
+-#define CCM_CCSR_CLKMODE1 (1 << 9)
+-#define CCM_CCSR_CLKMODE0 (1 << 8)
+-#define CCM_CCSR_CLKOSEL_OFFSET 0
+-#define CCM_CCSR_CLKOSEL_MASK 0x1f
++#define CCM_PCDR0_NFCDIV_OFFSET 12
++#define CCM_PCDR0_NFCDIV_MASK (0xf << CCM_PCDR0_NFCDIV_OFFSET)
++#define CCM_PCDR0_VPUDIV_OFFSET 8
++#define CCM_PCDR0_VPUDIV_MASK (0xf << CCM_PCDR0_VPUDIV_OFFSET)
++#define CCM_PCDR0_MSHCDIV_OFFSET 0
++#define CCM_PCDR0_MSHCDIV_MASK (0x1f << CCM_PCDR0_MSHCDIV_OFFSET)
++
++#define CCM_PCDR1_PERDIV4_OFFSET 24
++#define CCM_PCDR1_PERDIV4_MASK (0x3f << CCM_PCDR1_PERDIV4_OFFSET)
++#define CCM_PCDR1_PERDIV3_OFFSET 16
++#define CCM_PCDR1_PERDIV3_MASK (0x3f << CCM_PCDR1_PERDIV3_OFFSET)
++#define CCM_PCDR1_PERDIV2_OFFSET 8
++#define CCM_PCDR1_PERDIV2_MASK (0x3f << CCM_PCDR1_PERDIV2_OFFSET)
++#define CCM_PCDR1_PERDIV1_OFFSET 0
++#define CCM_PCDR1_PERDIV1_MASK (0x3f << CCM_PCDR1_PERDIV1_OFFSET)
++
++#define CCM_PCCR0_CSPI1_OFFSET 31
++#define CCM_PCCR0_CSPI1_MASK (1 << CCM_PCCR0_CSPI1_OFFSET)
++#define CCM_PCCR0_CSPI2_OFFSET 30
++#define CCM_PCCR0_CSPI2_MASK (1 << CCM_PCCR0_CSPI2_OFFSET)
++#define CCM_PCCR0_CSPI3_OFFSET 29
++#define CCM_PCCR0_CSPI3_MASK (1 << CCM_PCCR0_CSPI3_OFFSET)
++#define CCM_PCCR0_DMA_OFFSET 28
++#define CCM_PCCR0_DMA_MASK (1 << CCM_PCCR0_DMA_OFFSET)
++#define CCM_PCCR0_EMMA_OFFSET 27
++#define CCM_PCCR0_EMMA_MASK (1 << CCM_PCCR0_EMMA_OFFSET)
++#define CCM_PCCR0_FEC_OFFSET 26
++#define CCM_PCCR0_FEC_MASK (1 << CCM_PCCR0_FEC_OFFSET)
++#define CCM_PCCR0_GPIO_OFFSET 25
++#define CCM_PCCR0_GPIO_MASK (1 << CCM_PCCR0_GPIO_OFFSET)
++#define CCM_PCCR0_GPT1_OFFSET 24
++#define CCM_PCCR0_GPT1_MASK (1 << CCM_PCCR0_GPT1_OFFSET)
++#define CCM_PCCR0_GPT2_OFFSET 23
++#define CCM_PCCR0_GPT2_MASK (1 << CCM_PCCR0_GPT2_OFFSET)
++#define CCM_PCCR0_GPT3_OFFSET 22
++#define CCM_PCCR0_GPT3_MASK (1 << CCM_PCCR0_GPT3_OFFSET)
++#define CCM_PCCR0_GPT4_OFFSET 21
++#define CCM_PCCR0_GPT4_MASK (1 << CCM_PCCR0_GPT4_OFFSET)
++#define CCM_PCCR0_GPT5_OFFSET 20
++#define CCM_PCCR0_GPT5_MASK (1 << CCM_PCCR0_GPT5_OFFSET)
++#define CCM_PCCR0_GPT6_OFFSET 19
++#define CCM_PCCR0_GPT6_MASK (1 << CCM_PCCR0_GPT6_OFFSET)
++#define CCM_PCCR0_I2C1_OFFSET 18
++#define CCM_PCCR0_I2C1_MASK (1 << CCM_PCCR0_I2C1_OFFSET)
++#define CCM_PCCR0_I2C2_OFFSET 17
++#define CCM_PCCR0_I2C2_MASK (1 << CCM_PCCR0_I2C2_OFFSET)
++#define CCM_PCCR0_IIM_OFFSET 16
++#define CCM_PCCR0_IIM_MASK (1 << CCM_PCCR0_IIM_OFFSET)
++#define CCM_PCCR0_KPP_OFFSET 15
++#define CCM_PCCR0_KPP_MASK (1 << CCM_PCCR0_KPP_OFFSET)
++#define CCM_PCCR0_LCDC_OFFSET 14
++#define CCM_PCCR0_LCDC_MASK (1 << CCM_PCCR0_LCDC_OFFSET)
++#define CCM_PCCR0_MSHC_OFFSET 13
++#define CCM_PCCR0_MSHC_MASK (1 << CCM_PCCR0_MSHC_OFFSET)
++#define CCM_PCCR0_OWIRE_OFFSET 12
++#define CCM_PCCR0_OWIRE_MASK (1 << CCM_PCCR0_OWIRE_OFFSET)
++#define CCM_PCCR0_PWM_OFFSET 11
++#define CCM_PCCR0_PWM_MASK (1 << CCM_PCCR0_PWM_OFFSET)
++#define CCM_PCCR0_RTC_OFFSET 9
++#define CCM_PCCR0_RTC_MASK (1 << CCM_PCCR0_RTC_OFFSET)
++#define CCM_PCCR0_RTIC_OFFSET 8
++#define CCM_PCCR0_RTIC_MASK (1 << CCM_PCCR0_RTIC_OFFSET)
++#define CCM_PCCR0_SAHARA_OFFSET 7
++#define CCM_PCCR0_SAHARA_MASK (1 << CCM_PCCR0_SAHARA_OFFSET)
++#define CCM_PCCR0_SCC_OFFSET 6
++#define CCM_PCCR0_SCC_MASK (1 << CCM_PCCR0_SCC_OFFSET)
++#define CCM_PCCR0_SDHC1_OFFSET 5
++#define CCM_PCCR0_SDHC1_MASK (1 << CCM_PCCR0_SDHC1_OFFSET)
++#define CCM_PCCR0_SDHC2_OFFSET 4
++#define CCM_PCCR0_SDHC2_MASK (1 << CCM_PCCR0_SDHC2_OFFSET)
++#define CCM_PCCR0_SDHC3_OFFSET 3
++#define CCM_PCCR0_SDHC3_MASK (1 << CCM_PCCR0_SDHC3_OFFSET)
++#define CCM_PCCR0_SLCDC_OFFSET 2
++#define CCM_PCCR0_SLCDC_MASK (1 << CCM_PCCR0_SLCDC_OFFSET)
++#define CCM_PCCR0_SSI1_IPG_OFFSET 1
++#define CCM_PCCR0_SSI1_IPG_MASK (1 << CCM_PCCR0_SSI1_IPG_OFFSET)
++#define CCM_PCCR0_SSI2_IPG_OFFSET 0
++#define CCM_PCCR0_SSI2_IPG_MASK (1 << CCM_PCCR0_SSI2_IPG_OFFSET)
++
++#define CCM_PCCR1_UART1_OFFSET 31
++#define CCM_PCCR1_UART1_MASK (1 << CCM_PCCR1_UART1_OFFSET)
++#define CCM_PCCR1_UART2_OFFSET 30
++#define CCM_PCCR1_UART2_MASK (1 << CCM_PCCR1_UART2_OFFSET)
++#define CCM_PCCR1_UART3_OFFSET 29
++#define CCM_PCCR1_UART3_MASK (1 << CCM_PCCR1_UART3_OFFSET)
++#define CCM_PCCR1_UART4_OFFSET 28
++#define CCM_PCCR1_UART4_MASK (1 << CCM_PCCR1_UART4_OFFSET)
++#define CCM_PCCR1_UART5_OFFSET 27
++#define CCM_PCCR1_UART5_MASK (1 << CCM_PCCR1_UART5_OFFSET)
++#define CCM_PCCR1_UART6_OFFSET 26
++#define CCM_PCCR1_UART6_MASK (1 << CCM_PCCR1_UART6_OFFSET)
++#define CCM_PCCR1_USBOTG_OFFSET 25
++#define CCM_PCCR1_USBOTG_MASK (1 << CCM_PCCR1_USBOTG_OFFSET)
++#define CCM_PCCR1_WDT_OFFSET 24
++#define CCM_PCCR1_WDT_MASK (1 << CCM_PCCR1_WDT_OFFSET)
++#define CCM_PCCR1_HCLK_ATA_OFFSET 23
++#define CCM_PCCR1_HCLK_ATA_MASK (1 << CCM_PCCR1_HCLK_ATA_OFFSET)
++#define CCM_PCCR1_HCLK_BROM_OFFSET 22
++#define CCM_PCCR1_HCLK_BROM_MASK (1 << CCM_PCCR1_HCLK_BROM_OFFSET)
++#define CCM_PCCR1_HCLK_CSI_OFFSET 21
++#define CCM_PCCR1_HCLK_CSI_MASK (1 << CCM_PCCR1_HCLK_CSI_OFFSET)
++#define CCM_PCCR1_HCLK_DMA_OFFSET 20
++#define CCM_PCCR1_HCLK_DMA_MASK (1 << CCM_PCCR1_HCLK_DMA_OFFSET)
++#define CCM_PCCR1_HCLK_EMI_OFFSET 19
++#define CCM_PCCR1_HCLK_EMI_MASK (1 << CCM_PCCR1_HCLK_EMI_OFFSET)
++#define CCM_PCCR1_HCLK_EMMA_OFFSET 18
++#define CCM_PCCR1_HCLK_EMMA_MASK (1 << CCM_PCCR1_HCLK_EMMA_OFFSET)
++#define CCM_PCCR1_HCLK_FEC_OFFSET 17
++#define CCM_PCCR1_HCLK_FEC_MASK (1 << CCM_PCCR1_HCLK_FEC_OFFSET)
++#define CCM_PCCR1_HCLK_VPU_OFFSET 16
++#define CCM_PCCR1_HCLK_VPU_MASK (1 << CCM_PCCR1_HCLK_VPU_OFFSET)
++#define CCM_PCCR1_HCLK_LCDC_OFFSET 15
++#define CCM_PCCR1_HCLK_LCDC_MASK (1 << CCM_PCCR1_HCLK_LCDC_OFFSET)
++#define CCM_PCCR1_HCLK_RTIC_OFFSET 14
++#define CCM_PCCR1_HCLK_RTIC_MASK (1 << CCM_PCCR1_HCLK_RTIC_OFFSET)
++#define CCM_PCCR1_HCLK_SAHARA_OFFSET 13
++#define CCM_PCCR1_HCLK_SAHARA_MASK (1 << CCM_PCCR1_HCLK_SAHARA_OFFSET)
++#define CCM_PCCR1_HCLK_SLCDC_OFFSET 12
++#define CCM_PCCR1_HCLK_SLCDC_MASK (1 << CCM_PCCR1_HCLK_SLCDC_OFFSET)
++#define CCM_PCCR1_HCLK_USBOTG_OFFSET 11
++#define CCM_PCCR1_HCLK_USBOTG_MASK (1 << CCM_PCCR1_HCLK_USBOTG_OFFSET)
++#define CCM_PCCR1_PERCLK1_OFFSET 10
++#define CCM_PCCR1_PERCLK1_MASK (1 << CCM_PCCR1_PERCLK1_OFFSET)
++#define CCM_PCCR1_PERCLK2_OFFSET 9
++#define CCM_PCCR1_PERCLK2_MASK (1 << CCM_PCCR1_PERCLK2_OFFSET)
++#define CCM_PCCR1_PERCLK3_OFFSET 8
++#define CCM_PCCR1_PERCLK3_MASK (1 << CCM_PCCR1_PERCLK3_OFFSET)
++#define CCM_PCCR1_PERCLK4_OFFSET 7
++#define CCM_PCCR1_PERCLK4_MASK (1 << CCM_PCCR1_PERCLK4_OFFSET)
++#define CCM_PCCR1_VPU_BAUD_OFFSET 6
++#define CCM_PCCR1_VPU_BAUD_MASK (1 << CCM_PCCR1_VPU_BAUD_OFFSET)
++#define CCM_PCCR1_SSI1_BAUD_OFFSET 5
++#define CCM_PCCR1_SSI1_BAUD_MASK (1 << CCM_PCCR1_SSI1_BAUD_OFFSET)
++#define CCM_PCCR1_SSI2_BAUD_OFFSET 4
++#define CCM_PCCR1_SSI2_BAUD_MASK (1 << CCM_PCCR1_SSI2_BAUD_OFFSET)
++#define CCM_PCCR1_NFC_BAUD_OFFSET 3
++#define CCM_PCCR1_NFC_BAUD_MASK (1 << CCM_PCCR1_NFC_BAUD_OFFSET)
++#define CCM_PCCR1_MSHC_BAUD_OFFSET 2
++#define CCM_PCCR1_MSHC_BAUD_MASK (1 << CCM_PCCR1_MSHC_BAUD_OFFSET)
++
++#define CCM_CCSR_32KSR (1 << 15)
++#define CCM_CCSR_CLKMODE1 (1 << 9)
++#define CCM_CCSR_CLKMODE0 (1 << 8)
++#define CCM_CCSR_CLKOSEL_OFFSET 0
++#define CCM_CCSR_CLKOSEL_MASK (0x1f << CCM_CCSR_CLKOSEL_OFFSET)
+
+-#define SYS_FMCR 0x14 /* Functional Muxing Control Reg */
+-#define SYS_CHIP_ID 0x00 /* The offset of CHIP ID register */
++#define SYS_FMCR 0x14 /* Functional Muxing Control Reg */
++#define SYS_CHIP_ID 0x00 /* The offset of CHIP ID register */
+
+ #endif /* __ARCH_ARM_MACH_MX2_CRM_REGS_H__ */
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/mach-mx2/devices.c linux-2.6.28-karo/arch/arm/mach-mx2/devices.c
+--- linux-2.6.28/arch/arm/mach-mx2/devices.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/mach-mx2/devices.c 2009-03-11 13:16:24.000000000 +0100
+@@ -33,6 +33,7 @@
+ #include <linux/gpio.h>
+
+ #include <mach/hardware.h>
++#include <mach/mmc.h>
+
+ /*
+ * Resource definition for the MXC IrDA
+@@ -58,6 +59,118 @@ struct platform_device mxc_irda_device =
+ .resource = mxc_irda_resources,
+ };
+
++static u64 mxc_vpu_dmamask = 0xffffffffUL;
++
++/* Platform Data for MXC VPU */
++struct platform_device mxc_vpu_device = {
++ .name = "mxc_vpu",
++ .id = 0,
++ .dev = {
++ .dma_mask = &mxc_vpu_dmamask,
++ .coherent_dma_mask = ~0UL,
++ },
++};
++
++#ifdef CONFIG_MACH_MX27
++static struct resource mx27_camera_resources[] = {
++ {
++ .start = CSI_BASE_ADDR,
++ .end = CSI_BASE_ADDR + 0x1f,
++ .flags = IORESOURCE_MEM,
++ }, {
++ .start = EMMA_PRP_BASE_ADDR,
++ .end = EMMA_PRP_BASE_ADDR + 0x1f,
++ .flags = IORESOURCE_MEM,
++ }, {
++ .start = MXC_INT_CSI,
++ .end = MXC_INT_CSI,
++ .flags = IORESOURCE_IRQ,
++ }, {
++ .start = MXC_INT_EMMAPRP,
++ .end = MXC_INT_EMMAPRP,
++ .flags = IORESOURCE_IRQ,
++ },
++};
++
++struct platform_device mx27_camera_device = {
++ .name = "mx27-camera",
++ .id = 0,
++ .num_resources = ARRAY_SIZE(mx27_camera_resources),
++ .resource = mx27_camera_resources,
++};
++#endif
++
++/*
++ * SPI master controller
++ *
++ * - i.MX1: 2 channel (slighly different register setting)
++ * - i.MX21: 2 channel
++ * - i.MX27: 3 channel
++ */
++static struct resource mxc_spi_resources0[] = {
++ [0] = {
++ .start = CSPI1_BASE_ADDR,
++ .end = CSPI1_BASE_ADDR + 0x1F,
++ .flags = IORESOURCE_MEM,
++ },
++ [1] = {
++ .start = MXC_INT_CSPI1,
++ .end = MXC_INT_CSPI1,
++ .flags = IORESOURCE_IRQ,
++ },
++};
++
++static struct resource mxc_spi_resources1[] = {
++ [0] = {
++ .start = CSPI2_BASE_ADDR,
++ .end = CSPI2_BASE_ADDR + SZ_4K - 1,
++ .flags = IORESOURCE_MEM,
++ },
++ [1] = {
++ .start = MXC_INT_CSPI2,
++ .end = MXC_INT_CSPI2,
++ .flags = IORESOURCE_IRQ,
++ },
++};
++
++#ifdef CONFIG_MACH_MX27
++static struct resource mxc_spi_resources2[] = {
++ [0] = {
++ .start = CSPI3_BASE_ADDR,
++ .end = CSPI3_BASE_ADDR + SZ_4K - 1,
++ .flags = IORESOURCE_MEM,
++ },
++ [1] = {
++ .start = MXC_INT_CSPI3,
++ .end = MXC_INT_CSPI3,
++ .flags = IORESOURCE_IRQ,
++ },
++};
++#endif
++
++struct platform_device mxc_spi_device0 = {
++ .name = "mxc_spi",
++ .id = 0,
++ .num_resources = ARRAY_SIZE(mxc_spi_resources0),
++ .resource = mxc_spi_resources0,
++};
++
++struct platform_device mxc_spi_device1 = {
++ .name = "mxc_spi",
++ .id = 1,
++ .num_resources = ARRAY_SIZE(mxc_spi_resources1),
++ .resource = mxc_spi_resources1,
++};
++
++#ifdef CONFIG_MACH_MX27
++struct platform_device mxc_spi_device2 = {
++ .name = "mxc_spi",
++ .id = 2,
++ .num_resources = ARRAY_SIZE(mxc_spi_resources2),
++ .resource = mxc_spi_resources2,
++};
++#endif
++
+ /*
+ * General Purpose Timer
+ * - i.MX1: 2 timer (slighly different register handling)
+@@ -169,6 +282,72 @@ struct platform_device mxc_gpt5 = {
+ };
+ #endif
+
++/* I2C channel #1 */
++static struct resource imx_i2c_1_resources[] = {
++ [0] = {
++ .start = I2C_BASE_ADDR,
++ .end = I2C_BASE_ADDR + 0x0F,
++ .flags = IORESOURCE_MEM
++ },
++ [1] = {
++ .start = MXC_INT_I2C,
++ .end = MXC_INT_I2C,
++ .flags = IORESOURCE_IRQ
++ }
++};
++
++#ifdef CONFIG_MACH_MX27
++static struct resource imx_i2c_2_resources[] = {
++ [0] = {
++ .start = I2C2_BASE_ADDR,
++ .end = I2C2_BASE_ADDR + 0x0F,
++ .flags = IORESOURCE_MEM
++ },
++ [1] = {
++ .start = MXC_INT_I2C2,
++ .end = MXC_INT_I2C2,
++ .flags = IORESOURCE_IRQ
++ }
++};
++#endif
++
++struct platform_device imx_i2c_device0 = {
++ .name = "imx_i2c",
++ .id = 0,
++ .num_resources = ARRAY_SIZE(imx_i2c_1_resources),
++ .resource = imx_i2c_1_resources
++};
++
++#ifdef CONFIG_MACH_MX27
++struct platform_device imx_i2c_device1 = {
++ .name = "imx_i2c",
++ .id = 1,
++ .num_resources = ARRAY_SIZE(imx_i2c_2_resources),
++ .resource = imx_i2c_2_resources
++};
++#endif
++
++#if defined(CONFIG_RTC_MXC) || defined(CONFIG_RTC_MXC_MODULE)
++static struct resource rtc_resources[] = {
++ {
++ .start = RTC_BASE_ADDR,
++ .end = RTC_BASE_ADDR + 0x30,
++ .flags = IORESOURCE_MEM,
++ },
++ {
++ .start = MXC_INT_RTC,
++ .flags = IORESOURCE_IRQ,
++ },
++};
++
++struct platform_device mxc_rtc_device = {
++ .name = "mxc_rtc",
++ .id = 0,
++ .num_resources = ARRAY_SIZE(rtc_resources),
++ .resource = rtc_resources,
++};
++#endif
++
+ /*
+ * Watchdog:
+ * - i.MX1
+@@ -190,6 +369,291 @@ struct platform_device mxc_wdt = {
+ .resource = mxc_wdt_resources,
+ };
+
++static struct resource mxc_nand_resources[] = {
++ {
++ .start = NFC_BASE_ADDR,
++ .end = NFC_BASE_ADDR + 0xfff,
++ .flags = IORESOURCE_MEM
++ }, {
++ .start = MXC_INT_NANDFC,
++ .end = MXC_INT_NANDFC,
++ .flags = IORESOURCE_IRQ
++ },
++};
++
++struct platform_device mxc_nand_device = {
++ .name = "mxc_nand",
++ .id = 0,
++ .num_resources = ARRAY_SIZE(mxc_nand_resources),
++ .resource = mxc_nand_resources,
++};
++
++static struct resource mxc_w1_master_resources[] = {
++ {
++ .start = OWIRE_BASE_ADDR,
++ .end = OWIRE_BASE_ADDR + SZ_4K - 1,
++ .flags = IORESOURCE_MEM,
++ },
++};
++
++struct platform_device mxc_w1_master_device = {
++ .name = "mxc_w1",
++ .id = 0,
++ .num_resources = ARRAY_SIZE(mxc_w1_master_resources),
++ .resource = mxc_w1_master_resources,
++};
++
++/*
++ * Resource definition for the MXC SDHC
++ */
++static struct resource mxc_sdhc1_resources[] = {
++ [0] = {
++ .start = SDHC1_BASE_ADDR,
++ .end = SDHC1_BASE_ADDR + SZ_4K - 1,
++ .flags = IORESOURCE_MEM,
++ },
++ [1] = {
++ .start = MXC_INT_SDHC1,
++ .end = MXC_INT_SDHC1,
++ .flags = IORESOURCE_IRQ,
++ },
++ [2] = {
++ .name = "sdhc1",
++ .start = DMA_REQ_SDHC1,
++ .end = DMA_REQ_SDHC1,
++ .flags = IORESOURCE_DMA
++ },
++};
++
++static u64 mxc_sdhc1_dmamask = 0xffffffffUL;
++
++struct platform_device mxc_sdhc_device0 = {
++ .name = "imx-mmc",
++ .id = 0,
++ .dev = {
++ .dma_mask = &mxc_sdhc1_dmamask,
++ .coherent_dma_mask = 0xffffffff,
++ },
++ .num_resources = ARRAY_SIZE(mxc_sdhc1_resources),
++ .resource = mxc_sdhc1_resources,
++};
++
++static struct resource mxc_sdhc2_resources[] = {
++ [0] = {
++ .start = SDHC2_BASE_ADDR,
++ .end = SDHC2_BASE_ADDR + SZ_4K - 1,
++ .flags = IORESOURCE_MEM,
++ },
++ [1] = {
++ .start = MXC_INT_SDHC2,
++ .end = MXC_INT_SDHC2,
++ .flags = IORESOURCE_IRQ,
++ },
++ [2] = {
++ .name = "sdhc2",
++ .start = DMA_REQ_SDHC2,
++ .end = DMA_REQ_SDHC2,
++ .flags = IORESOURCE_DMA
++ },
++};
++
++static u64 mxc_sdhc2_dmamask = 0xffffffffUL;
++
++struct platform_device mxc_sdhc_device1 = {
++ .name = "imx-mmc",
++ .id = 1,
++ .dev = {
++ .dma_mask = &mxc_sdhc2_dmamask,
++ .coherent_dma_mask = 0xffffffff,
++ },
++ .num_resources = ARRAY_SIZE(mxc_sdhc2_resources),
++ .resource = mxc_sdhc2_resources,
++};
++
++static u64 otg_dmamask = ~(u32) 0;
++
++static struct resource mxc_otg_resources[] = {
++ {
++ .start = OTG_BASE_ADDR + 0x000,
++ .end = OTG_BASE_ADDR + 0x1ff,
++ .flags = IORESOURCE_MEM,
++ }, {
++ .start = MXC_INT_USB1,
++ .flags = IORESOURCE_IRQ,
++ },
++};
++
++/*
++ * lcdc:
++ * - i.MX1: the basic controller
++ * - i.MX21: to be checked
++ * - i.MX27: like i.MX1, with slightly variations
++ */
++static struct resource mxc_fb[] = {
++ {
++ .start = LCDC_BASE_ADDR,
++ .end = LCDC_BASE_ADDR + 0xFFF,
++ .flags = IORESOURCE_MEM,
++ }, {
++ .start = MXC_INT_LCDC,
++ .end = MXC_INT_LCDC,
++ .flags = IORESOURCE_IRQ,
++ },
++};
++
++/* mxc lcd driver */
++struct platform_device mxc_fb_device = {
++ .name = "imx-fb",
++ .id = 0,
++ .num_resources = ARRAY_SIZE(mxc_fb),
++ .resource = mxc_fb,
++ .dev = {
++ .coherent_dma_mask = 0xFFFFFFFF,
++ },
++};
++
++struct platform_device mxc_otg = {
++ .name = "mxc-ehci",
++ .id = 0,
++ .dev = {
++ .coherent_dma_mask = 0xffffffff,
++ .dma_mask = &otg_dmamask,
++ },
++ .resource = mxc_otg_resources,
++ .num_resources = ARRAY_SIZE(mxc_otg_resources),
++};
++
++static struct resource mxc_ehci2_resources[] = {
++ {
++ .start = OTG_BASE_ADDR + 0x400,
++ .end = OTG_BASE_ADDR + 0x5ff,
++ .flags = IORESOURCE_MEM,
++ }, {
++ .start = MXC_INT_USB2,
++ .flags = IORESOURCE_IRQ,
++ },
++};
++
++static u64 ehci2_dmamask = ~(u32) 0;
++
++struct platform_device mxc_ehci2 = {
++ .name = "mxc-ehci",
++ .id = 1,
++ .dev = {
++ .coherent_dma_mask = 0xffffffff,
++ .dma_mask = &ehci2_dmamask,
++ },
++ .num_resources = ARRAY_SIZE(mxc_ehci2_resources),
++ .resource = mxc_ehci2_resources,
++};
++
++/*
++ * SSI bus:
++ * - i.MX1: 2 channels
++ * - i.MX21: 2 channels
++ * - i.MX27: 2 channels
++ */
++static struct resource imx_ssi_resources0[] = {
++ [0] = {
++ .start = SSI1_BASE_ADDR,
++ .end = SSI1_BASE_ADDR + 0x6F,
++ .flags = IORESOURCE_MEM
++ },
++ [1] = {
++ .start = MXC_INT_SSI1,
++ .end = MXC_INT_SSI1,
++ .flags = IORESOURCE_IRQ
++ },
++ [2] = {
++ .name = "tx0",
++ .start = DMA_REQ_SSI1_TX0,
++ .end = DMA_REQ_SSI1_TX0,
++ .flags = IORESOURCE_DMA
++ },
++ [3] = {
++ .name = "rx0",
++ .start = DMA_REQ_SSI1_RX0,
++ .end = DMA_REQ_SSI1_RX0,
++ .flags = IORESOURCE_DMA
++ },
++ [4] = {
++ .name = "tx1",
++ .start = DMA_REQ_SSI1_TX1,
++ .end = DMA_REQ_SSI1_TX1,
++ .flags = IORESOURCE_DMA
++ },
++ [5] = {
++ .name = "rx1",
++ .start = DMA_REQ_SSI1_RX1,
++ .end = DMA_REQ_SSI1_RX1,
++ .flags = IORESOURCE_DMA
++ }
++};
++
++static struct resource imx_ssi_resources1[] = {
++ [0] = {
++ .start = SSI2_BASE_ADDR,
++ .end = SSI2_BASE_ADDR + 0x6F,
++ .flags = IORESOURCE_MEM
++ },
++ [1] = {
++ .start = MXC_INT_SSI2,
++ .end = MXC_INT_SSI2,
++ .flags = IORESOURCE_IRQ
++ },
++ [2] = {
++ .name = "tx0",
++ .start = DMA_REQ_SSI2_TX0,
++ .end = DMA_REQ_SSI2_TX0,
++ .flags = IORESOURCE_DMA
++ },
++ [3] = {
++ .name = "rx0",
++ .start = DMA_REQ_SSI2_RX0,
++ .end = DMA_REQ_SSI2_RX0,
++ .flags = IORESOURCE_DMA
++ },
++ [4] = {
++ .name = "tx1",
++ .start = DMA_REQ_SSI2_TX1,
++ .end = DMA_REQ_SSI2_TX1,
++ .flags = IORESOURCE_DMA
++ },
++ [5] = {
++ .name = "rx1",
++ .start = DMA_REQ_SSI2_RX1,
++ .end = DMA_REQ_SSI2_RX1,
++ .flags = IORESOURCE_DMA
++ }
++};
++
++struct platform_device imx_ssi_device0 = {
++ .name = "mxc-ssi",
++ .id = 0,
++ .num_resources = ARRAY_SIZE(imx_ssi_resources0),
++ .resource = imx_ssi_resources0
++};
++
++struct platform_device imx_ssi_device1 = {
++ .name = "mxc-ssi",
++ .id = 1,
++ .num_resources = ARRAY_SIZE(imx_ssi_resources1),
++ .resource = imx_ssi_resources1
++};
++
++static struct resource mxc_dam_resources = {
++ .start = AUDMUX_BASE_ADDR,
++ .end = AUDMUX_BASE_ADDR + 0x1F,
++ .flags = IORESOURCE_MEM
++};
++
++struct platform_device mxc_dam_device = {
++ .name = "mxc-dam",
++ .id = 0,
++ .num_resources = 1,
++ .resource = &mxc_dam_resources
++};
++
+ /* GPIO port description */
+ static struct mxc_gpio_port imx_gpio_ports[] = {
+ [0] = {
+@@ -229,3 +693,4 @@ int __init mxc_register_gpios(void)
+ {
+ return mxc_gpio_init(imx_gpio_ports, ARRAY_SIZE(imx_gpio_ports));
+ }
++//arch_initcall(mxc_register_gpios);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/mach-mx2/devices.h linux-2.6.28-karo/arch/arm/mach-mx2/devices.h
+--- linux-2.6.28/arch/arm/mach-mx2/devices.h 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/mach-mx2/devices.h 2009-03-11 13:16:24.000000000 +0100
+@@ -4,6 +4,7 @@ extern struct platform_device mxc_gpt2;
+ extern struct platform_device mxc_gpt3;
+ extern struct platform_device mxc_gpt4;
+ extern struct platform_device mxc_gpt5;
++extern struct platform_device mxc_rtc_device;
+ extern struct platform_device mxc_wdt;
+ extern struct platform_device mxc_irda_device;
+ extern struct platform_device mxc_uart_device0;
+@@ -12,4 +13,20 @@ extern struct platform_device mxc_uart_d
+ extern struct platform_device mxc_uart_device3;
+ extern struct platform_device mxc_uart_device4;
+ extern struct platform_device mxc_uart_device5;
+-
++extern struct platform_device mxc_nand_device;
++extern struct platform_device mxc_w1_master_device;
++extern struct platform_device mxc_sdhc_device0;
++extern struct platform_device mxc_sdhc_device1;
++extern struct platform_device mxc_otg;
++extern struct platform_device mxc_ehci2;
++extern struct platform_device mxc_spi_device0;
++extern struct platform_device mxc_spi_device1;
++extern struct platform_device mxc_spi_device2;
++extern struct platform_device imx_ssi_device0;
++extern struct platform_device imx_ssi_device1;
++extern struct platform_device mxc_dam_device;
++extern struct platform_device imx_i2c_device0;
++extern struct platform_device imx_i2c_device1;
++extern struct platform_device mx27_camera_device;
++extern struct platform_device mxc_fb_device;
++extern struct platform_device mxc_vpu_device;
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/mach-mx2/generic.c linux-2.6.28-karo/arch/arm/mach-mx2/generic.c
+--- linux-2.6.28/arch/arm/mach-mx2/generic.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/mach-mx2/generic.c 2009-03-11 13:16:24.000000000 +0100
+@@ -35,7 +35,7 @@ static struct map_desc mxc_io_desc[] __i
+ * - and some reserved space
+ */
+ {
+- .virtual = AIPI_BASE_ADDR_VIRT,
++ .virtual = (unsigned long)AIPI_BASE_ADDR_VIRT,
+ .pfn = __phys_to_pfn(AIPI_BASE_ADDR),
+ .length = AIPI_SIZE,
+ .type = MT_DEVICE
+@@ -46,7 +46,7 @@ static struct map_desc mxc_io_desc[] __i
+ * - ATA
+ */
+ {
+- .virtual = SAHB1_BASE_ADDR_VIRT,
++ .virtual = (unsigned long)SAHB1_BASE_ADDR_VIRT,
+ .pfn = __phys_to_pfn(SAHB1_BASE_ADDR),
+ .length = SAHB1_SIZE,
+ .type = MT_DEVICE
+@@ -56,7 +56,7 @@ static struct map_desc mxc_io_desc[] __i
+ * - EMI
+ */
+ {
+- .virtual = X_MEMC_BASE_ADDR_VIRT,
++ .virtual = (unsigned long)X_MEMC_BASE_ADDR_VIRT,
+ .pfn = __phys_to_pfn(X_MEMC_BASE_ADDR),
+ .length = X_MEMC_SIZE,
+ .type = MT_DEVICE
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/mach-mx2/include/mach/board-tx27.h linux-2.6.28-karo/arch/arm/mach-mx2/include/mach/board-tx27.h
+--- linux-2.6.28/arch/arm/mach-mx2/include/mach/board-tx27.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/mach-mx2/include/mach/board-tx27.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,94 @@
++/*
++ * arch/arm/mach-mx27/board-tx27.h
++ *
++ * Copyright 2008 <LW@KARO-electronics.de>
++ * based on arch/arm/mach-mx27/board-mx27ads.h by eescale Semiconductor, Inc.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++//#define FEC_MII_IRQ
++
++#define MXC_LL_UART_PADDR UART1_BASE_ADDR
++#define MXC_LL_UART_VADDR AIPI_IO_ADDRESS(UART1_BASE_ADDR)
++
++/*!
++ * These defines are used to enable or disable a particular UART port. If
++ * disabled, the UART will not be registered in the file system and the user
++ * will not be able to access it.
++ * Specify a value of 1 to enable the UART and 0 to disable it.
++ */
++#define UART1_ENABLED 1
++/* UART 2 configuration */
++#define UART2_ENABLED 1
++/* UART 3 configuration */
++#define UART3_ENABLED 1
++/* UART 4 configuration */
++#define UART4_ENABLED 0 /* Disable UART 4 as its pins are shared with ATA */
++#define UART5_ENABLED 0
++#define UART6_ENABLED 0
++
++#ifndef __ASSEMBLY__
++#include <mach/iomux.h>
++
++#define MXC_BD_LED1 0
++#define MXC_BD_LED2 (1 << 5)
++static inline void MXC_BD_LED_ON(unsigned int led)
++{
++ if (led & MXC_BD_LED2) {
++ __gpio_set_value(GPIO_PORTF | 13, 1);
++ }
++}
++static inline void MXC_BD_LED_OFF(unsigned int led)
++{
++ if (led & MXC_BD_LED2) {
++ __gpio_set_value(GPIO_PORTF | 13, 0);
++ }
++}
++
++extern int gpio_fec_active(void);
++extern void gpio_fec_inactive(void);
++extern int gpio_sdhc_active(int module);
++extern int gpio_sdhc_inactive(int module);
++extern int gpio_usbh2_active(void);
++extern void gpio_usbh2_inactive(void);
++#if 0
++extern int gpio_uart_active(int port, int irda);
++extern int gpio_uart_inactive(int port);
++extern int config_uartdma_event(int port);
++extern int gpio_usbh1_active(void);
++extern void gpio_usbh1_inactive(void);
++extern int gpio_usbotg_fs_active(void);
++extern void gpio_usbotg_fs_inactive(void);
++extern int gpio_i2c_active(int i2c_num);
++extern int gpio_i2c_inactive(int i2c_num);
++extern int gpio_spi_active(int cspi_mod);
++extern int gpio_spi_inactive(int cspi_mod);
++extern int gpio_nand_active(void);
++extern void gpio_nand_inactive(void);
++extern int gpio_sensor_active(void);
++extern void gpio_sensor_inactive(void);
++extern int gpio_keypad_active(void);
++extern void gpio_keypad_inactive(void);
++extern int gpio_ata_active(void);
++extern void gpio_ata_inactive(void);
++extern int gpio_slcdc_active(int type);
++extern int gpio_slcdc_inactive(int type);
++extern int gpio_ssi_active(int ssi_num);
++extern int gpio_ssi_inactive(int ssi_num);
++extern int gpio_owire_active(void);
++extern void gpio_owire_inactive(void);
++#endif
++extern int gpio_usbotg_hs_active(void);
++extern void gpio_usbotg_hs_inactive(void);
++extern int gpio_ac97_active(void);
++extern void gpio_ac97_inactive(void);
++
++#endif // __ASSEMBLY__
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/mach-mx2/include/mach/iomux.h linux-2.6.28-karo/arch/arm/mach-mx2/include/mach/iomux.h
+--- linux-2.6.28/arch/arm/mach-mx2/include/mach/iomux.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/mach-mx2/include/mach/iomux.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,460 @@
++/*
++ * arch/arm/mach-mx1/include/mach/iomux.h
++ *
++ * Copyright (C) 2008 by Lothar Wassmann <LW@KARO-electronics.de>
++ * derived from: arch/arm/plat-mxc/include/mach/iomux-mx1-mx2.h
++ * Copyright (C) 2008 by Sascha Hauer <kernel@pengutronix.de>
++ *
++ * 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., 51 Franklin Street, Fifth Floor, Boston,
++ * MA 02110-1301, USA.
++ *
++ * Pin definitions for the Freescale MX2 architecture
++ */
++
++#ifndef _MXC_GPIO_MX2_H
++#define _MXC_GPIO_MX2_H
++
++#define GPIO_PORT_MAX 5
++
++#include <mach/iomux-mx1-mx2.h>
++
++enum {
++ MXC_DEFINE_PIN(A, 0, PF, USBH2_CLK, 0),
++ MXC_DEFINE_PIN(A, 1, PF, USBH2_DIR, 0),
++ MXC_DEFINE_PIN(A, 2, PF, USBH2_DATA7, 0),
++ MXC_DEFINE_PIN(A, 3, PF, USBH2_NXT, 0),
++ MXC_DEFINE_PIN(A, 4, PF, USBH2_STP, 0),
++
++ MXC_DEFINE_PIN(A, 5, PF, LSCLK, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 6, PF, LD0, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 7, PF, LD1, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 8, PF, LD2, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 9, PF, LD3, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 10, PF, LD4, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 11, PF, LD5, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 12, PF, LD6, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 13, PF, LD7, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 14, PF, LD8, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 15, PF, LD9, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 16, PF, LD10, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 17, PF, LD11, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 18, PF, LD12, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 19, PF, LD13, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 20, PF, LD14, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 21, PF, LD15, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 22, PF, LD16, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 23, PF, LD17, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 24, PF, REV, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 25, PF, CLS, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 26, PF, PS, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 27, PF, SPL_SPR, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 28, PF, HSYNC, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 29, PF, VSYNC, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 30, PF, CONTRAST, GPIO_OUT),
++ MXC_DEFINE_PIN(A, 31, PF, OE_ACD, GPIO_OUT),
++
++ MXC_DEFINE_PIN(B, 10, PF, CSI_D0, GPIO_OUT),
++ MXC_DEFINE_PIN(B, 11, PF, CSI_D1, GPIO_OUT),
++ MXC_DEFINE_PIN(B, 12, PF, CSI_D2, GPIO_OUT),
++ MXC_DEFINE_PIN(B, 13, PF, CSI_D3, GPIO_OUT),
++ MXC_DEFINE_PIN(B, 14, PF, CSI_D4, GPIO_OUT),
++ MXC_DEFINE_PIN(B, 15, PF, CSI_MCLK, GPIO_OUT),
++ MXC_DEFINE_PIN(B, 16, PF, CSI_PIXCLK, GPIO_OUT),
++ MXC_DEFINE_PIN(B, 17, PF, CSI_D5, GPIO_OUT),
++ MXC_DEFINE_PIN(B, 18, PF, CSI_D6, GPIO_OUT),
++
++ MXC_DEFINE_PIN(B, 10, AF, UART6_TXD, GPIO_OUT),
++ MXC_DEFINE_PIN(B, 11, AF, UART6_RXD, GPIO_IN),
++ MXC_DEFINE_PIN(B, 12, AF, UART6_CTS, GPIO_OUT),
++ MXC_DEFINE_PIN(B, 13, AF, UART6_RTS, GPIO_IN),
++
++ MXC_DEFINE_PIN(B, 18, AF, UART5_TXD, GPIO_OUT),
++ MXC_DEFINE_PIN(B, 19, AF, UART5_RXD, GPIO_IN),
++ MXC_DEFINE_PIN(B, 20, AF, UART5_CTS, GPIO_OUT),
++ MXC_DEFINE_PIN(B, 21, AF, UART5_RTS, GPIO_IN),
++
++ MXC_DEFINE_PIN(B, 19, PF, CSI_D7, GPIO_OUT),
++ MXC_DEFINE_PIN(B, 20, PF, CSI_VSYNC, GPIO_OUT),
++ MXC_DEFINE_PIN(B, 21, PF, CSI_HSYNC, GPIO_OUT),
++
++ MXC_DEFINE_PIN(B, 22, PF, USBH1_SUSP, 0),
++ MXC_DEFINE_PIN(B, 23, PF, USB_PWR, 0),
++ MXC_DEFINE_PIN(B, 24, PF, USB_OC_B, 0),
++ MXC_DEFINE_PIN(B, 25, PF, USBH1_RCV, 0),
++ MXC_DEFINE_PIN(B, 26, PF, USBH1_FS, 0),
++ MXC_DEFINE_PIN(B, 27, PF, USBH1_OE_B, 0),
++ MXC_DEFINE_PIN(B, 28, PF, USBH1_TXDM, 0),
++ MXC_DEFINE_PIN(B, 29, PF, USBH1_TXDP, 0),
++ MXC_DEFINE_PIN(B, 30, PF, USBH1_RXDM, 0),
++ MXC_DEFINE_PIN(B, 31, PF, USBH1_RXDP, 0),
++
++ MXC_DEFINE_PIN(B, 26, AF, UART4_RTS, GPIO_IN),
++ MXC_DEFINE_PIN(B, 28, AF, UART4_TXD, GPIO_OUT),
++ MXC_DEFINE_PIN(B, 29, AF, UART4_CTS, GPIO_OUT),
++ MXC_DEFINE_PIN(B, 31, AF, UART4_RXD, GPIO_IN),
++
++ MXC_DEFINE_PIN(C, 14, PF, TOUT, GPIO_OUT),
++ MXC_DEFINE_PIN(C, 15, PF, TIN, GPIO_IN),
++
++ MXC_DEFINE_PIN(C, 5, PF, I2C2_SDA, GPIO_IN),
++ MXC_DEFINE_PIN(C, 6, PF, I2C2_SCL, GPIO_IN),
++
++ MXC_DEFINE_PIN(C, 7, PF, USBOTG_DATA5, GPIO_OUT),
++ MXC_DEFINE_PIN(C, 8, PF, USBOTG_DATA6, GPIO_OUT),
++ MXC_DEFINE_PIN(C, 9, PF, USBOTG_DATA0, GPIO_OUT),
++ MXC_DEFINE_PIN(C, 10, PF, USBOTG_DATA2, GPIO_OUT),
++ MXC_DEFINE_PIN(C, 11, PF, USBOTG_DATA1, GPIO_OUT),
++ MXC_DEFINE_PIN(C, 12, PF, USBOTG_DATA4, GPIO_OUT),
++ MXC_DEFINE_PIN(C, 13, PF, USBOTG_DATA3, GPIO_OUT),
++ MXC_DEFINE_PIN(E, 0, PF, USBOTG_NXT, GPIO_OUT),
++ MXC_DEFINE_PIN(E, 1, PF, USBOTG_STP, GPIO_OUT),
++ MXC_DEFINE_PIN(E, 2, PF, USBOTG_DIR, GPIO_OUT),
++ MXC_DEFINE_PIN(E, 24, PF, USBOTG_CLK, GPIO_OUT),
++ MXC_DEFINE_PIN(E, 25, PF, USBOTG_DATA7, GPIO_OUT),
++
++ MXC_DEFINE_PIN(C, 20, PF, SSI1_FS, GPIO_IN),
++ MXC_DEFINE_PIN(C, 21, PF, SSI1_RXD, GPIO_IN),
++ MXC_DEFINE_PIN(C, 22, PF, SSI1_TXD, GPIO_IN),
++ MXC_DEFINE_PIN(C, 23, PF, SSI1_CLK, GPIO_IN),
++
++ MXC_DEFINE_PIN(C, 24, PF, SSI2_FS, GPIO_IN),
++ MXC_DEFINE_PIN(C, 25, PF, SSI2_RXD, GPIO_IN),
++ MXC_DEFINE_PIN(C, 26, PF, SSI2_TXD, GPIO_IN),
++ MXC_DEFINE_PIN(C, 27, PF, SSI2_CLK, GPIO_IN),
++
++ MXC_DEFINE_PIN(C, 28, PF, SSI3_FS, GPIO_IN),
++ MXC_DEFINE_PIN(C, 29, PF, SSI3_RXD, GPIO_IN),
++ MXC_DEFINE_PIN(C, 30, PF, SSI3_TXD, GPIO_IN),
++ MXC_DEFINE_PIN(C, 31, PF, SSI3_CLK, GPIO_IN),
++
++ MXC_DEFINE_PIN(C, 16, PF, SSI4_FS, GPIO_IN),
++ MXC_DEFINE_PIN(C, 17, PF, SSI4_RXD, GPIO_IN),
++ MXC_DEFINE_PIN(C, 18, PF, SSI4_TXD, GPIO_IN),
++ MXC_DEFINE_PIN(C, 19, PF, SSI4_CLK, GPIO_IN),
++
++ MXC_DEFINE_PIN(B, 5, AIN, SLCDC1_CLK, 0),
++ MXC_DEFINE_PIN(B, 6, AIN, SLCDC1_D0, 0),
++ MXC_DEFINE_PIN(B, 7, AIN, SLCDC1_RS, 0),
++ MXC_DEFINE_PIN(B, 8, AIN, SLCDC1_CS, 0),
++
++ MXC_DEFINE_PIN(C, 28, AIN, SLCDC2_D0, GPIO_OUT),
++ MXC_DEFINE_PIN(C, 29, AIN, SLCDC2_RS, GPIO_OUT),
++ MXC_DEFINE_PIN(C, 30, AIN, SLCDC2_CS, GPIO_OUT),
++ MXC_DEFINE_PIN(C, 31, AIN, SLCDC2_CLK, GPIO_OUT),
++
++ MXC_DEFINE_PIN(D, 2, PF, ATA_DATA0, 0),
++ MXC_DEFINE_PIN(D, 3, PF, ATA_DATA1, 0),
++ MXC_DEFINE_PIN(D, 4, PF, ATA_DATA2, 0),
++ MXC_DEFINE_PIN(D, 5, PF, ATA_DATA3, 0),
++ MXC_DEFINE_PIN(D, 6, PF, ATA_DATA4, 0),
++ MXC_DEFINE_PIN(D, 7, PF, ATA_DATA5, 0),
++ MXC_DEFINE_PIN(D, 8, PF, ATA_DATA6, 0),
++ MXC_DEFINE_PIN(D, 9, PF, ATA_DATA7, 0),
++ MXC_DEFINE_PIN(D, 10, PF, ATA_DATA8, 0),
++ MXC_DEFINE_PIN(D, 11, PF, ATA_DATA9, 0),
++ MXC_DEFINE_PIN(D, 12, PF, ATA_DATA10, 0),
++ MXC_DEFINE_PIN(D, 13, PF, ATA_DATA11, 0),
++ MXC_DEFINE_PIN(D, 14, PF, ATA_DATA12, 0),
++ MXC_DEFINE_PIN(D, 15, PF, ATA_DATA13, 0),
++ MXC_DEFINE_PIN(D, 16, PF, ATA_DATA14, 0),
++ MXC_DEFINE_PIN(F, 23, PF, ATA_DATA15, 0),
++
++ MXC_DEFINE_PIN(D, 0, AIN, FEC_TXD0, GPIO_OUT),
++ MXC_DEFINE_PIN(D, 1, AIN, FEC_TXD1, GPIO_OUT),
++ MXC_DEFINE_PIN(D, 2, AIN, FEC_TXD2, GPIO_OUT),
++ MXC_DEFINE_PIN(D, 3, AIN, FEC_TXD3, GPIO_OUT),
++ MXC_DEFINE_PIN(D, 4, AOUT, FEC_RX_ER, GPIO_IN),
++ MXC_DEFINE_PIN(D, 5, AOUT, FEC_RXD1, GPIO_IN),
++ MXC_DEFINE_PIN(D, 6, AOUT, FEC_RXD2, GPIO_IN),
++ MXC_DEFINE_PIN(D, 7, AOUT, FEC_RXD3, GPIO_IN),
++ MXC_DEFINE_PIN(D, 8, AF, FEC_MDIO, GPIO_IN),
++ MXC_DEFINE_PIN(D, 9, AIN, FEC_MDC, GPIO_OUT),
++ MXC_DEFINE_PIN(D, 10, AOUT, FEC_CRS, GPIO_IN),
++ MXC_DEFINE_PIN(D, 11, AOUT, FEC_TX_CLK, GPIO_IN),
++ MXC_DEFINE_PIN(D, 12, AOUT, FEC_RXD0, GPIO_IN),
++ MXC_DEFINE_PIN(D, 13, AOUT, FEC_RX_DV, GPIO_IN),
++ MXC_DEFINE_PIN(D, 14, AOUT, FEC_RX_CLK, GPIO_IN),
++ MXC_DEFINE_PIN(D, 15, AOUT, FEC_COL, GPIO_IN),
++ MXC_DEFINE_PIN(D, 16, AIN, FEC_TX_ER, GPIO_OUT),
++
++ MXC_DEFINE_PIN(D, 17, PF, I2C_DATA, GPIO_OUT),
++ MXC_DEFINE_PIN(D, 18, PF, I2C_CLK, GPIO_OUT),
++
++ MXC_DEFINE_PIN(D, 19, PF, CSPI2_SS2, 0),
++ MXC_DEFINE_PIN(D, 20, PF, CSPI2_SS1, 0),
++ MXC_DEFINE_PIN(D, 21, PF, CSPI2_SS0, 0),
++ MXC_DEFINE_PIN(D, 22, PF, CSPI2_SCLK, 0),
++ MXC_DEFINE_PIN(D, 23, PF, CSPI2_MISO, 0),
++ MXC_DEFINE_PIN(D, 24, PF, CSPI2_MOSI, 0),
++
++ MXC_DEFINE_PIN(D, 19, AF, USBH2_DATA4, 0),
++ MXC_DEFINE_PIN(D, 20, AF, USBH2_DATA3, 0),
++ MXC_DEFINE_PIN(D, 21, AF, USBH2_DATA6, 0),
++ MXC_DEFINE_PIN(D, 22, AF, USBH2_DATA0, 0),
++ MXC_DEFINE_PIN(D, 23, AF, USBH2_DATA2, 0),
++ MXC_DEFINE_PIN(D, 24, AF, USBH2_DATA1, 0),
++ MXC_DEFINE_PIN(D, 26, AF, USBH2_DATA5, 0),
++
++ MXC_DEFINE_PIN(D, 25, PF, CSPI1_RDY, GPIO_OUT),
++ MXC_DEFINE_PIN(D, 26, PF, CSPI1_SS2, GPIO_OUT),
++ MXC_DEFINE_PIN(D, 27, PF, CSPI1_SS1, GPIO_OUT),
++ MXC_DEFINE_PIN(D, 28, PF, CSPI1_SS0, GPIO_OUT),
++ MXC_DEFINE_PIN(D, 29, PF, CSPI1_SCLK, GPIO_OUT),
++ MXC_DEFINE_PIN(D, 30, PF, CSPI1_MISO, GPIO_IN),
++ MXC_DEFINE_PIN(D, 31, PF, CSPI1_MOSI, GPIO_OUT),
++
++ MXC_DEFINE_PIN(E, 3, PF, UART2_CTS, GPIO_OUT),
++ MXC_DEFINE_PIN(E, 4, PF, UART2_RTS, GPIO_IN),
++ MXC_DEFINE_PIN(E, 6, PF, UART2_TXD, GPIO_OUT),
++ MXC_DEFINE_PIN(E, 7, PF, UART2_RXD, GPIO_IN),
++
++ MXC_DEFINE_PIN(E, 8, PF, UART3_TXD, GPIO_OUT),
++ MXC_DEFINE_PIN(E, 9, PF, UART3_RXD, GPIO_IN),
++ MXC_DEFINE_PIN(E, 10, PF, UART3_CTS, GPIO_OUT),
++ MXC_DEFINE_PIN(E, 11, PF, UART3_RTS, GPIO_IN),
++
++ MXC_DEFINE_PIN(E, 12, PF, UART1_TXD, GPIO_OUT),
++ MXC_DEFINE_PIN(E, 13, PF, UART1_RXD, GPIO_IN),
++ MXC_DEFINE_PIN(E, 14, PF, UART1_CTS, GPIO_OUT),
++ MXC_DEFINE_PIN(E, 15, PF, UART1_RTS, GPIO_IN),
++
++ MXC_DEFINE_PIN(E, 16, AF, OWIRE, GPIO_OUT),
++ MXC_DEFINE_PIN(E, 16, PF, RTCK, GPIO_OUT),
++
++ MXC_DEFINE_PIN(E, 18, PF, SD1_D0, 0),
++ MXC_DEFINE_PIN(E, 19, PF, SD1_D1, 0),
++ MXC_DEFINE_PIN(E, 20, PF, SD1_D2, 0),
++ MXC_DEFINE_PIN(E, 21, PF, SD1_D3, 0),
++ MXC_DEFINE_PIN(E, 22, PF, SD1_CMD, 0),
++ MXC_DEFINE_PIN(E, 23, PF, SD1_CLK, 0),
++
++ MXC_DEFINE_PIN(B, 4, PF, SD2_D0, 0),
++ MXC_DEFINE_PIN(B, 5, PF, SD2_D1, 0),
++ MXC_DEFINE_PIN(B, 6, PF, SD2_D2, 0),
++ MXC_DEFINE_PIN(B, 7, PF, SD2_D3, 0),
++ MXC_DEFINE_PIN(B, 8, PF, SD2_CMD, 0),
++ MXC_DEFINE_PIN(B, 9, PF, SD2_CLK, 0),
++
++ MXC_DEFINE_PIN(D, 0, PF, SD3_CMD, GPIO_OUT),
++ MXC_DEFINE_PIN(D, 1, PF, SD3_CLK, GPIO_OUT),
++ MXC_DEFINE_PIN(D, 2, AF, SD3_D0, GPIO_OUT),
++ MXC_DEFINE_PIN(D, 3, AF, SD3_D1, GPIO_OUT),
++ MXC_DEFINE_PIN(D, 4, AF, SD3_D2, GPIO_OUT),
++ MXC_DEFINE_PIN(D, 5, AF, SD3_D3, GPIO_OUT),
++
++ MXC_DEFINE_PIN(E, 18, AF, CSPI3_MISO, GPIO_IN),
++ MXC_DEFINE_PIN(E, 21, AF, CSPI3_SS, GPIO_OUT),
++ MXC_DEFINE_PIN(E, 22, AF, CSPI3_MOSI, GPIO_OUT),
++ MXC_DEFINE_PIN(E, 23, AF, CSPI3_SCLK, GPIO_OUT),
++
++ MXC_DEFINE_PIN(F, 0, PF, NFRB, GPIO_IN),
++ MXC_DEFINE_PIN(F, 1, PF, NFCLE, GPIO_OUT),
++ MXC_DEFINE_PIN(F, 2, PF, NFWP_B, GPIO_OUT),
++ MXC_DEFINE_PIN(F, 3, PF, NFCE_B, GPIO_OUT),
++ MXC_DEFINE_PIN(F, 4, PF, NFALE, GPIO_OUT),
++ MXC_DEFINE_PIN(F, 5, PF, NFRE_B, GPIO_OUT),
++ MXC_DEFINE_PIN(F, 6, PF, NFWE_B, GPIO_OUT),
++
++ MXC_DEFINE_PIN(F, 7, AF, PC_POE, GPIO_OUT),
++ MXC_DEFINE_PIN(F, 8, AF, PC_RW_B, GPIO_OUT),
++ MXC_DEFINE_PIN(F, 9, AF, IOIS16, GPIO_IN),
++ MXC_DEFINE_PIN(F, 10, AF, PC_RST, GPIO_OUT),
++ MXC_DEFINE_PIN(F, 11, AF, PC_BVD2, GPIO_IN),
++ MXC_DEFINE_PIN(F, 12, AF, PC_BVD1, GPIO_IN),
++ MXC_DEFINE_PIN(F, 13, AF, PC_VS2, GPIO_IN),
++ MXC_DEFINE_PIN(F, 14, AF, PC_VS1, GPIO_IN),
++ MXC_DEFINE_PIN(F, 15, PF, CLKO, GPIO_OUT),
++ MXC_DEFINE_PIN(F, 16, AF, PC_PWRON, GPIO_IN),
++ MXC_DEFINE_PIN(F, 17, AF, PC_READY, GPIO_IN),
++ MXC_DEFINE_PIN(F, 18, AF, PC_WAIT_B, GPIO_IN),
++ MXC_DEFINE_PIN(F, 19, AF, PC_CD2_B, GPIO_IN),
++ MXC_DEFINE_PIN(F, 20, AF, PC_CD1_B, GPIO_IN),
++
++ MXC_DEFINE_PIN(F, 23, AIN, FEC_TX_EN, GPIO_OUT),
++};
++
++#define _PIN_NAME(v) \
++ case v: \
++ name = #v; \
++ break
++
++static inline const char *MX27_PIN_NAME(int iomux)
++{
++ const char *name = "<noname>";
++ switch (iomux) {
++ _PIN_NAME(PA0_PF_USBH2_CLK);
++ _PIN_NAME(PA1_PF_USBH2_DIR);
++ _PIN_NAME(PA2_PF_USBH2_DATA7);
++ _PIN_NAME(PA3_PF_USBH2_NXT);
++ _PIN_NAME(PA4_PF_USBH2_STP);
++ _PIN_NAME(PA5_PF_LSCLK);
++ _PIN_NAME(PA6_PF_LD0);
++ _PIN_NAME(PA7_PF_LD1);
++ _PIN_NAME(PA8_PF_LD2);
++ _PIN_NAME(PA9_PF_LD3);
++ _PIN_NAME(PA10_PF_LD4);
++ _PIN_NAME(PA11_PF_LD5);
++ _PIN_NAME(PA12_PF_LD6);
++ _PIN_NAME(PA13_PF_LD7);
++ _PIN_NAME(PA14_PF_LD8);
++ _PIN_NAME(PA15_PF_LD9);
++ _PIN_NAME(PA16_PF_LD10);
++ _PIN_NAME(PA17_PF_LD11);
++ _PIN_NAME(PA18_PF_LD12);
++ _PIN_NAME(PA19_PF_LD13);
++ _PIN_NAME(PA20_PF_LD14);
++ _PIN_NAME(PA21_PF_LD15);
++ _PIN_NAME(PA22_PF_LD16);
++ _PIN_NAME(PA23_PF_LD17);
++ _PIN_NAME(PA24_PF_REV);
++ _PIN_NAME(PA25_PF_CLS);
++ _PIN_NAME(PA26_PF_PS);
++ _PIN_NAME(PA27_PF_SPL_SPR);
++ _PIN_NAME(PA28_PF_HSYNC);
++ _PIN_NAME(PA29_PF_VSYNC);
++ _PIN_NAME(PA30_PF_CONTRAST);
++ _PIN_NAME(PA31_PF_OE_ACD);
++ _PIN_NAME(PB4_PF_SD2_D0);
++ _PIN_NAME(PB5_PF_SD2_D1);
++ _PIN_NAME(PB6_PF_SD2_D2);
++ _PIN_NAME(PB7_PF_SD2_D3);
++ _PIN_NAME(PB8_PF_SD2_CMD);
++ _PIN_NAME(PB9_PF_SD2_CLK);
++ _PIN_NAME(PB10_PF_CSI_D0);
++ _PIN_NAME(PB10_AF_UART6_TXD);
++ _PIN_NAME(PB11_PF_CSI_D1);
++ _PIN_NAME(PB11_AF_UART6_RXD);
++ _PIN_NAME(PB12_PF_CSI_D2);
++ _PIN_NAME(PB12_AF_UART6_CTS);
++ _PIN_NAME(PB13_PF_CSI_D3);
++ _PIN_NAME(PB13_AF_UART6_RTS);
++ _PIN_NAME(PB14_PF_CSI_D4);
++ _PIN_NAME(PB15_PF_CSI_MCLK);
++ _PIN_NAME(PB16_PF_CSI_PIXCLK);
++ _PIN_NAME(PB17_PF_CSI_D5);
++ _PIN_NAME(PB18_PF_CSI_D6);
++ _PIN_NAME(PB18_AF_UART5_TXD);
++ _PIN_NAME(PB19_PF_CSI_D7);
++ _PIN_NAME(PB19_AF_UART5_RXD);
++ _PIN_NAME(PB20_PF_CSI_VSYNC);
++ _PIN_NAME(PB20_AF_UART5_CTS);
++ _PIN_NAME(PB21_PF_CSI_HSYNC);
++ _PIN_NAME(PB21_AF_UART5_RTS);
++ _PIN_NAME(PB22_PF_USBH1_SUSP);
++ _PIN_NAME(PB23_PF_USB_PWR);
++ _PIN_NAME(PB24_PF_USB_OC_B);
++ _PIN_NAME(PB25_PF_USBH1_RCV);
++ _PIN_NAME(PB26_PF_USBH1_FS);
++ _PIN_NAME(PB27_PF_USBH1_OE_B);
++ _PIN_NAME(PB28_PF_USBH1_TXDM);
++ _PIN_NAME(PB29_PF_USBH1_TXDP);
++ _PIN_NAME(PB30_PF_USBH1_RXDM);
++ _PIN_NAME(PB31_PF_USBH1_RXDP);
++ _PIN_NAME(PB26_AF_UART4_RTS);
++ _PIN_NAME(PB28_AF_UART4_TXD);
++ _PIN_NAME(PB29_AF_UART4_CTS);
++ _PIN_NAME(PB31_AF_UART4_RXD);
++ _PIN_NAME(PC5_PF_I2C2_SDA);
++ _PIN_NAME(PC6_PF_I2C2_SCL);
++ _PIN_NAME(PC7_PF_USBOTG_DATA5);
++ _PIN_NAME(PC8_PF_USBOTG_DATA6);
++ _PIN_NAME(PC9_PF_USBOTG_DATA0);
++ _PIN_NAME(PC10_PF_USBOTG_DATA2);
++ _PIN_NAME(PC11_PF_USBOTG_DATA1);
++ _PIN_NAME(PC12_PF_USBOTG_DATA4);
++ _PIN_NAME(PC13_PF_USBOTG_DATA3);
++ _PIN_NAME(PC16_PF_SSI4_FS);
++ _PIN_NAME(PC17_PF_SSI4_RXD);
++ _PIN_NAME(PC18_PF_SSI4_TXD);
++ _PIN_NAME(PC19_PF_SSI4_CLK);
++ _PIN_NAME(PC20_PF_SSI1_FS);
++ _PIN_NAME(PC21_PF_SSI1_RXD);
++ _PIN_NAME(PC22_PF_SSI1_TXD);
++ _PIN_NAME(PC23_PF_SSI1_CLK);
++ _PIN_NAME(PC24_PF_SSI2_FS);
++ _PIN_NAME(PC25_PF_SSI2_RXD);
++ _PIN_NAME(PC26_PF_SSI2_TXD);
++ _PIN_NAME(PC27_PF_SSI2_CLK);
++ _PIN_NAME(PC28_PF_SSI3_FS);
++ _PIN_NAME(PC29_PF_SSI3_RXD);
++ _PIN_NAME(PC30_PF_SSI3_TXD);
++ _PIN_NAME(PC31_PF_SSI3_CLK);
++ _PIN_NAME(PD0_AIN_FEC_TXD0);
++ _PIN_NAME(PD1_AIN_FEC_TXD1);
++ _PIN_NAME(PD2_AIN_FEC_TXD2);
++ _PIN_NAME(PD3_AIN_FEC_TXD3);
++ _PIN_NAME(PD4_AOUT_FEC_RX_ER);
++ _PIN_NAME(PD5_AOUT_FEC_RXD1);
++ _PIN_NAME(PD6_AOUT_FEC_RXD2);
++ _PIN_NAME(PD7_AOUT_FEC_RXD3);
++ _PIN_NAME(PD8_AF_FEC_MDIO);
++ _PIN_NAME(PD9_AIN_FEC_MDC);
++ _PIN_NAME(PD10_AOUT_FEC_CRS);
++ _PIN_NAME(PD11_AOUT_FEC_TX_CLK);
++ _PIN_NAME(PD12_AOUT_FEC_RXD0);
++ _PIN_NAME(PD13_AOUT_FEC_RX_DV);
++ _PIN_NAME(PD14_AOUT_FEC_RX_CLK);
++ _PIN_NAME(PD15_AOUT_FEC_COL);
++ _PIN_NAME(PD16_AIN_FEC_TX_ER);
++ _PIN_NAME(PD17_PF_I2C_DATA);
++ _PIN_NAME(PD18_PF_I2C_CLK);
++ _PIN_NAME(PD19_AF_USBH2_DATA4);
++ _PIN_NAME(PD20_AF_USBH2_DATA3);
++ _PIN_NAME(PD21_AF_USBH2_DATA6);
++ _PIN_NAME(PD22_AF_USBH2_DATA0);
++ _PIN_NAME(PD23_AF_USBH2_DATA2);
++ _PIN_NAME(PD24_AF_USBH2_DATA1);
++ _PIN_NAME(PD25_PF_CSPI1_RDY);
++ _PIN_NAME(PD26_PF_CSPI1_SS2);
++ _PIN_NAME(PD26_AF_USBH2_DATA5);
++ _PIN_NAME(PD27_PF_CSPI1_SS1);
++ _PIN_NAME(PD28_PF_CSPI1_SS0);
++ _PIN_NAME(PD29_PF_CSPI1_SCLK);
++ _PIN_NAME(PD30_PF_CSPI1_MISO);
++ _PIN_NAME(PD31_PF_CSPI1_MOSI);
++ _PIN_NAME(PF23_AIN_FEC_TX_EN);
++ _PIN_NAME(PE0_PF_USBOTG_NXT);
++ _PIN_NAME(PE1_PF_USBOTG_STP);
++ _PIN_NAME(PE2_PF_USBOTG_DIR);
++ _PIN_NAME(PE3_PF_UART2_CTS);
++ _PIN_NAME(PE4_PF_UART2_RTS);
++ _PIN_NAME(PE6_PF_UART2_TXD);
++ _PIN_NAME(PE7_PF_UART2_RXD);
++ _PIN_NAME(PE8_PF_UART3_TXD);
++ _PIN_NAME(PE9_PF_UART3_RXD);
++ _PIN_NAME(PE10_PF_UART3_CTS);
++ _PIN_NAME(PE11_PF_UART3_RTS);
++ _PIN_NAME(PE12_PF_UART1_TXD);
++ _PIN_NAME(PE13_PF_UART1_RXD);
++ _PIN_NAME(PE14_PF_UART1_CTS);
++ _PIN_NAME(PE15_PF_UART1_RTS);
++ _PIN_NAME(PE16_AF_OWIRE);
++ _PIN_NAME(PE16_PF_RTCK);
++ _PIN_NAME(PE18_PF_SD1_D0);
++ _PIN_NAME(PE18_AF_CSPI3_MISO);
++ _PIN_NAME(PE19_PF_SD1_D1);
++ _PIN_NAME(PE20_PF_SD1_D2);
++ _PIN_NAME(PE21_PF_SD1_D3);
++ _PIN_NAME(PE21_AF_CSPI3_SS);
++ _PIN_NAME(PE22_PF_SD1_CMD);
++ _PIN_NAME(PE22_AF_CSPI3_MOSI);
++ _PIN_NAME(PE23_PF_SD1_CLK);
++ _PIN_NAME(PE23_AF_CSPI3_SCLK);
++ _PIN_NAME(PE24_PF_USBOTG_CLK);
++ _PIN_NAME(PE25_PF_USBOTG_DATA7);
++ }
++ return name;
++}
++
++#endif /* _MXC_GPIO_MX2_H */
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/mach-mx2/include/mach/mxc_pm.h linux-2.6.28-karo/arch/arm/mach-mx2/include/mach/mxc_pm.h
+--- linux-2.6.28/arch/arm/mach-mx2/include/mach/mxc_pm.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/mach-mx2/include/mach/mxc_pm.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,229 @@
++
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @defgroup LPMD Low-Level Power Management Driver
++ */
++
++/*!
++ * @file arch-mxc/mxc_pm.h
++ *
++ * @brief This file contains the chip level configuration details and
++ * public API declarations for CRM_AP module
++ *
++ * @ingroup LPMD
++ */
++
++#ifndef __ASM_ARCH_MXC_PM_H__
++#define __ASM_ARCH_MXC_PM_H__
++
++#define WAIT_MODE 111
++#define DOZE_MODE 112
++#define STOP_MODE 113
++#define DSM_MODE 114
++
++#define GATE_STOP_WAIT 9
++#define GATE_STOP 10
++
++/*
++ * Used for MHz conversion
++ */
++#define MEGA_HERTZ 1000000
++
++/*
++ * If invalid frequency value other than the following
++ * CORE_133 - ARM desired to run @133MHz, LoV (1.2V)
++ * CORE_266 - ARM desired to run @266MHz, LoV (1.2V)
++ * CORE_399 - ARM desired to run @399MHz, LoV (1.2V)
++ * CORE_532 - ARM desired to run @133MHz, HiV (1.6V)
++ * are passed then this error is returned,
++ */
++#define ERR_FREQ_INVALID 1
++
++/*
++ * If PLL freq is less than desired ARM frequency during Integer
++ * DVFS, then return this error
++ */
++#define PLL_LESS_ARM_ERR 2
++
++/*
++ * Frequency change within the same-lo voltage is not approved.
++ * Inorder to do Integer DFS, move to the high voltage range and
++ * then set LFDF and move to the low voltage range
++ */
++#define INT_DFS_LOW_NOT_ALLOW 3
++
++/*
++ * If the desired AHB or IPG exceeds 133MHz or 66.5MHz respectively,
++ * then return this error
++ */
++#define AHB_IPG_EXCEED_LIMIT 4
++
++/*
++ * If the desired ARM frequency is too low to get by PLL scaling
++ * and the mxc_pm_pllscale API is called, return this error:
++ */
++#define PLL_DVFS_FREQ_TOO_LOW 5
++
++/*
++ * Invalid frequencies requested
++ */
++#define MXC_PM_INVALID_PARAM 6
++
++/*
++ * If AHB and/or IPG frequencies are greater than maximum allowed
++ */
++#define FREQ_OUT_OF_RANGE 2
++
++/*
++ * If AHB and/or IPG frequencies are other than 100 or 50Mhz
++ */
++#define BUS_FREQ_INVALID 2
++
++/*
++ * If MAX_PDF is greater than max value (8) then return this error
++ */
++#define AHB_MAX_DIV_ERR 3
++
++/*
++ * If IPG_PDF is greater than max value (2) then return this error
++ */
++#define IPG_MAX_DIV_ERR 4
++
++/*
++ * If ARM freq is out of range i.e., less than 133 or greater than
++ * 399 then return this error
++ */
++#define INVALID_ARM_FREQ 5
++
++/*
++ * This file includes all platform APIs. Some of the APIs are not
++ * appicable to some platforms. So, this error is used to indicate
++ * that a particular API is not available
++ */
++#define MXC_PM_API_NOT_SUPPORTED 6
++
++/*!
++ * Additional define for stop mode
++ */
++#define PM_SUSPEND_STOP ((__force suspend_state_t) 2)
++
++/*!
++ * CKOH pins configuration
++ */
++#define CKOH_AP_SEL 1
++#define CKOH_AHB_SEL 2
++#define CKOH_IP_SEL 3
++
++/*!
++ * Defines for Stop and DSM mode acknowledgements
++ */
++#define MXC_PM_LOWPWR_ACK_SDMA 0x01
++#define MXC_PM_LOWPWR_ACK_IPU 0x02
++#define MXC_PM_LOWPWR_ACK_MAX 0x04
++#define MXC_PM_LOWPWR_ACK_MQSPI 0x08
++#define MXC_PM_LOWPWR_ACK_USB 0x10
++#define MXC_PM_LOWPWR_ACK_RTIC 0x20
++
++/*
++ * PMIC configuration
++ */
++#define MXC_PMIC_1_2_VOLT 0xC
++#define MXC_PMIC_1_6_VOLT 0x1C
++#define MXC_PMIC_1_0_VOLT 0x4
++#define MXC_PMIC_DVS_SPEED 0x3
++
++/*!
++ * Implementing Level 1 CRM Gate Control. Level 2 gate control
++ * is provided at module level using LPMD registers
++ *
++ * @param group The desired clock gate control register bits.
++ * Possible values are 0 through 6
++ * @param opt The desired option requesting clock to run during stop
++ * and wait modes or just during the stop mode. Possible
++ * values are GATE_STOP_WAIT and GATE_STOP.
++ *
++ */
++void mxc_pm_clockgate(int group, int opt);
++
++/*!
++ * Implementing steps required to transition to low-power modes
++ *
++ * @param mode The desired low-power mode. Possible values are,
++ * WAIT_MODE, STOP_MODE or DSM_MODE
++ *
++ */
++void mxc_pm_lowpower(int mode);
++
++/*!
++ * Enables acknowledgement from module when entering stop or DSM mode.
++ *
++ * @param ack The desired module acknowledgement to enable.
++ *
++ */
++void mxc_pm_lp_ack_enable(int ack);
++
++/*!
++ * Disables acknowledgement from module when entering stop or DSM mode.
++ *
++ * @param ack The desired module acknowledgement to disable.
++ *
++ */
++void mxc_pm_lp_ack_disable(int ack);
++
++/*!
++ * Implementing steps required to set Integer Scaling
++ *
++ * @param armfreq The desired ARM frequency. AHB and IP
++ * frequency are changed depending on ARM
++ * frequency and the divider values.
++ * @param ahbfreq The desired AHB frequency
++ * @param ipfreq The desired IP frequency
++ *
++ * @return Returns 0 on success or
++ * Returns -PLL_LESS_ARM_ERR if pllfreq is less than
++ * desired core freq
++ */
++int mxc_pm_intscale(long armfreq, long ahbfreq, long ipfreq);
++
++/*!
++ * To calculate MFI, MFN, MFD values. Using this the output frequency
++ * whose value is calculated using,
++ * 2 * REF_FREQ * (MF / PDF), where
++ * REF_FREQ is 26 Mhz
++ * MF = MFI + (MFN + MFD)
++ * PDF is assumed to be 1
++ *
++ * @param armfreq The desired ARM frequency
++ * @param ahbfreq The desired AHB frequency
++ * @param ipfreq The desired IP frequency
++ *
++ * @return Returns 0 on success or
++ * Returns -1 on error
++ */
++int mxc_pm_pllscale(long armfreq, long ahbfreq, long ipfreq);
++
++/*!
++ * To change AP core frequency and/or voltage suitably
++ *
++ * @param armfreq The desired ARM frequency
++ * @param ahbfreq The desired AHB frequency
++ * @param ipfreq The desired IP frequency
++ *
++ * @return Returns -ERR_FREQ_INVALID on failure
++ * Returns 0 on success
++ */
++int mxc_pm_dvfs(unsigned long armfreq, long ahbfreq, long ipfreq);
++
++#endif
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/mach-mx2/include/mach/mxc_v4l2.h linux-2.6.28-karo/arch/arm/mach-mx2/include/mach/mxc_v4l2.h
+--- linux-2.6.28/arch/arm/mach-mx2/include/mach/mxc_v4l2.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/mach-mx2/include/mach/mxc_v4l2.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,56 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU Lesser General
++ * Public License. You may obtain a copy of the GNU Lesser General
++ * Public License Version 2.1 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/lgpl-license.html
++ * http://www.gnu.org/copyleft/lgpl.html
++ */
++
++/*!
++ * @file arch-mxc/mxc_v4l2.h
++ *
++ * @brief mxc V4L2 private structures
++ *
++ * @ingroup MXC_V4L2_CAPTURE
++ */
++
++#ifndef __ASM_ARCH_MXC_V4L2_H__
++#define __ASM_ARCH_MXC_V4L2_H__
++
++#define V4L2_CID_MXC_ROT (V4L2_CID_PRIVATE_BASE + 0)
++#define V4L2_CID_MXC_FLASH (V4L2_CID_PRIVATE_BASE + 1)
++#define V4L2_CID_MXC_FLICKER (V4L2_CID_PRIVATE_BASE + 2)
++#define V4L2_CID_MXC_TEAR_PROTECT (V4L2_CID_PRIVATE_BASE + 3)
++#define V4L2_CID_MXC_GAIN_LIMIT (V4L2_CID_PRIVATE_BASE + 4)
++
++/* V4L2_CID_MXC_ROT values */
++
++#define V4L2_MXC_ROTATE_NONE 0
++#define V4L2_MXC_ROTATE_VERT_FLIP 1
++#define V4L2_MXC_ROTATE_HORIZ_FLIP 2
++#define V4L2_MXC_ROTATE_180 3
++#define V4L2_MXC_ROTATE_90_RIGHT 4
++#define V4L2_MXC_ROTATE_90_RIGHT_VFLIP 5
++#define V4L2_MXC_ROTATE_90_RIGHT_HFLIP 6
++#define V4L2_MXC_ROTATE_90_LEFT 7
++
++/* V4L2_CID_MXC_FLICKER values */
++
++#define V4L2_MXC_FLICKER_DISABLE 0
++#define V4L2_MXC_FLICKER_60HZ 1
++#define V4L2_MXC_FLICKER_50HZ 2
++#define V4L2_MXC_FLICKER_AUTO 3
++
++struct v4l2_mxc_offset {
++ unsigned int y_offset;
++ unsigned int u_offset;
++ unsigned int v_offset;
++ unsigned int qp_offset;
++};
++
++#endif
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/mach-mx2/karo-tx27.c linux-2.6.28-karo/arch/arm/mach-mx2/karo-tx27.c
+--- linux-2.6.28/arch/arm/mach-mx2/karo-tx27.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/mach-mx2/karo-tx27.c 2009-03-11 13:31:40.000000000 +0100
+@@ -0,0 +1,1556 @@
++/*
++ * arch/arm/mach-mx27/karo-tx27.c
++ *
++ * Copyright (C) 2008 Lothar Wassmann <LW@KARO-electronics.de>
++ *
++ * based on: arch/arm/mach-mx27ads.c (C) Freescale Semiconductor, 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 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301
++ *
++ */
++
++#include <linux/types.h>
++#include <linux/sched.h>
++#include <linux/interrupt.h>
++#include <linux/init.h>
++#include <linux/ioport.h>
++#include <linux/platform_device.h>
++#include <linux/input.h>
++#include <linux/clk.h>
++#include <linux/delay.h>
++#include <linux/fb.h>
++#include <linux/i2c.h>
++#include <linux/i2c/at24.h>
++#include <linux/spi/spi.h>
++#include <linux/serial_8250.h>
++#include <linux/fec_enet.h>
++#include <mtd/mtd-abi.h>
++#include <linux/mtd/map.h>
++#include <linux/mtd/partitions.h>
++#include <asm/mach/flash.h>
++
++#include <linux/serial.h>
++#include <linux/fsl_devices.h>
++#include <linux/rtc/ds13xx.h>
++#include <linux/irq.h>
++#include <linux/mmc/host.h>
++#include <linux/gpio_keys.h>
++#include <linux/leds.h>
++
++#include <asm/setup.h>
++#include <asm/irq.h>
++#include <asm/mach-types.h>
++#include <asm/mach/arch.h>
++#include <asm/mach/time.h>
++#include <mach/common.h>
++#include <mach/hardware.h>
++#include <mach/gpio.h>
++#include <mach/iomux.h>
++#include <mach/irqs.h>
++#include <mach/clock.h>
++#include <mach/imxfb.h>
++#include <mach/imx_spi.h>
++#include <mach/imx_i2c.h>
++#include <mach/mmc.h>
++#include <mach/imx-uart.h>
++#include <mach/mxc_nand.h>
++#include <mach/ulpi.h>
++#include <mach/mxc_ehci.h>
++#include <mach/board-tx27.h>
++
++#include "crm_regs.h"
++#include "devices.h"
++
++#ifdef DEBUG
++int tx27_debug = 1;
++#define dbg_lvl(n) ((n) < tx27_debug)
++module_param(tx27_debug, int, S_IRUGO | S_IWUSR);
++
++#define DBG(lvl, fmt...) do { if (dbg_lvl(lvl)) printk(KERN_DEBUG fmt); } while (0)
++#else
++int tx27_debug;
++#define dbg_lvl(n) 0
++#define DBG(lvl, fmt...) do { } while (0)
++#endif
++
++#include "karo.h"
++
++#if defined(CONFIG_SERIAL_IMX) || defined(CONFIG_SERIAL_IMX_MODULE)
++static int tx27_uart_pins[][4] = {
++ {
++ PE12_PF_UART1_TXD,
++ PE13_PF_UART1_RXD,
++ PE14_PF_UART1_CTS,
++ PE15_PF_UART1_RTS,
++ },{
++ PE6_PF_UART2_TXD,
++ PE7_PF_UART2_RXD,
++ PE3_PF_UART2_CTS,
++ PE4_PF_UART2_RTS,
++ },{
++ PE8_PF_UART3_TXD,
++ PE9_PF_UART3_RXD,
++ PE10_PF_UART3_CTS,
++ PE11_PF_UART3_RTS,
++ },{
++ PB28_AF_UART4_TXD,
++ PB31_AF_UART4_RXD,
++ PB29_AF_UART4_CTS,
++ PB26_AF_UART4_RTS,
++ },{
++ PB18_AF_UART5_TXD,
++ PB19_AF_UART5_RXD,
++ PB20_AF_UART5_CTS,
++ PB21_AF_UART5_RTS,
++ },{
++ PB10_AF_UART6_TXD,
++ PB11_AF_UART6_RXD,
++ PB12_AF_UART6_CTS,
++ PB13_AF_UART6_RTS,
++ },
++};
++
++static int tx27_uart_init(struct platform_device *pdev)
++{
++ DBG(0, "%s: \n", __FUNCTION__);
++ return mxc_gpio_setup_multiple_pins(tx27_uart_pins[pdev->id],
++ ARRAY_SIZE(tx27_uart_pins[pdev->id]), "UART");
++}
++
++static int tx27_uart_exit(struct platform_device *pdev)
++{
++ DBG(0, "%s: \n", __FUNCTION__);
++ mxc_gpio_release_multiple_pins(tx27_uart_pins[pdev->id],
++ ARRAY_SIZE(tx27_uart_pins[pdev->id]));
++ return 0;
++}
++
++static struct imxuart_platform_data tx27_uart_ports[] = {
++ {
++ .init = tx27_uart_init,
++ .exit = tx27_uart_exit,
++ .flags = IMXUART_HAVE_RTSCTS,
++ },{
++ .init = tx27_uart_init,
++ .exit = tx27_uart_exit,
++ .flags = IMXUART_HAVE_RTSCTS,
++ },{
++ .init = tx27_uart_init,
++ .exit = tx27_uart_exit,
++ .flags = IMXUART_HAVE_RTSCTS,
++ },{
++ .init = tx27_uart_init,
++ .exit = tx27_uart_exit,
++ .flags = IMXUART_HAVE_RTSCTS,
++ },{
++ .init = tx27_uart_init,
++ .exit = tx27_uart_exit,
++ .flags = IMXUART_HAVE_RTSCTS,
++ },{
++ .init = tx27_uart_init,
++ .exit = tx27_uart_exit,
++ .flags = IMXUART_HAVE_RTSCTS,
++ },
++};
++
++static struct platform_device *tx27_uart_devices[] = {
++#if UART1_ENABLED
++ &mxc_uart_device0,
++#endif
++#if UART2_ENABLED
++ &mxc_uart_device1,
++#endif
++#if UART3_ENABLED
++ &mxc_uart_device2,
++#endif
++#if UART4_ENABLED
++ &mxc_uart_device3,
++#endif
++#if UART5_ENABLED
++ &mxc_uart_device4,
++#endif
++#if UART6_ENABLED
++ &mxc_uart_device5,
++#endif
++};
++#endif
++
++#ifdef CONFIG_USB_EHCI_MXC
++
++#define SMSC_VENDOR_ID 0x0424
++#define USB3317_PROD_ID 0x0006
++
++static int usb3317_set_vbus_power(void __iomem *view, int on)
++{
++ int vid, pid, ret = 0;
++
++ ret = ulpi_read(ISP1504_VID_HIGH, view);
++ if (ret < 0) {
++ goto err;
++ }
++ vid = ret << 8;
++
++ ret = ulpi_read(ISP1504_VID_LOW, view);
++ if (ret < 0) {
++ goto err;
++ }
++ vid |= ret;
++
++ ret = ulpi_read(ISP1504_PID_HIGH, view);
++ if (ret < 0) {
++ goto err;
++ }
++ pid = ret << 8;
++
++ ret = ulpi_read(ISP1504_PID_LOW, view);
++ if (ret < 0) {
++ goto err;
++ }
++ pid |= ret;
++
++ pr_info("ULPI Vendor ID 0x%x Product ID 0x%x\n", vid, pid);
++ if (vid != SMSC_VENDOR_ID || pid != USB3317_PROD_ID) {
++ pr_err("No USB3317 found\n");
++ return -1;
++ }
++
++ if (on) {
++ ret = ulpi_set(DRV_VBUS_EXT | /* enable external Vbus */
++ DRV_VBUS | /* enable internal Vbus */
++ USE_EXT_VBUS_IND | /* use external indicator */
++ CHRG_VBUS, /* charge Vbus */
++ ISP1504_OTGCTL, view);
++ } else {
++ ret = ulpi_clear(DRV_VBUS_EXT | /* disable external Vbus */
++ DRV_VBUS, /* disable internal Vbus */
++ ISP1504_OTGCTL, view);
++
++ ret |= ulpi_set(USE_EXT_VBUS_IND | /* use external indicator */
++ DISCHRG_VBUS, /* discharge Vbus */
++ ISP1504_OTGCTL, view);
++ }
++ return 0;
++
++ err:
++ printk(KERN_ERR "ULPI read failed with error %d\n", ret);
++ return ret;
++}
++
++static int tx27_usb_host_init(struct platform_device *pdev)
++{
++ int ret;
++ u32 temp;
++ unsigned long freq;
++ unsigned long flags;
++ struct clk *usb_clk = clk_get(NULL, "usb_clk");
++
++ if (IS_ERR(usb_clk)) {
++ ret = PTR_ERR(usb_clk);
++ printk(KERN_ERR "Failed to get usb_clk: %d\n", ret);
++ goto err;
++ }
++
++ ret = gpio_usbh2_active();
++ if (ret != 0) {
++ return ret;
++ }
++
++ freq = clk_get_rate(usb_clk);
++ if ((freq < 59999000) || (freq > 60001000)) {
++ printk(KERN_ERR "USB_CLK=%lu.%03luMHz, should be 60MHz\n",
++ freq / 1000000, freq / 1000 % 1000);
++ ret = clk_set_rate(usb_clk, 60000000);
++ if (ret != 0) {
++ printk(KERN_ERR "Failed to set usb_clk rate: %d\n", ret);
++ }
++ }
++ clk_put(usb_clk);
++ if (ret != 0) {
++ goto err;
++ }
++
++ local_irq_save(flags);
++ temp = readl(IO_ADDRESS(OTG_BASE_ADDR) + 0x600);
++ temp &= ~(3 << 21);
++ temp |= (1 << 5) | (1 << 16) | (1 << 19) | (1 << 20);
++ writel(temp, IO_ADDRESS(OTG_BASE_ADDR) + 0x600);
++ local_irq_restore(flags);
++
++ temp = readl(IO_ADDRESS(OTG_BASE_ADDR) + 0x584);
++ temp &= ~(3 << 30);
++ temp |= 2 << 30;
++ temp = 0x88001215;
++ /* select ULPI transceiver */
++ DBG(0, "%s: Changing USBH2_PORTSC1[%08lx] from %08x to %08x\n", __FUNCTION__,
++ OTG_BASE_ADDR + 0x584, readl(IO_ADDRESS(OTG_BASE_ADDR) + 0x584), temp);
++ writel(temp, IO_ADDRESS(OTG_BASE_ADDR) + 0x584);
++
++ mdelay(10);
++
++ ret = usb3317_set_vbus_power(IO_ADDRESS(OTG_BASE_ADDR + 0x570), 1);
++ if (ret != 0) {
++// goto err;
++ }
++ return 0;
++
++ err:
++ gpio_usbh2_inactive();
++ return ret;
++}
++
++static int tx27_usb_host_exit(struct platform_device *pdev)
++{
++ gpio_usbh2_inactive();
++ return 0;
++}
++
++static struct mxc_usbh_platform_data tx27_usbh2_data = {
++ .init = tx27_usb_host_init,
++ .exit = tx27_usb_host_exit,
++};
++
++int tx27_usbh2_init(void)
++{
++ int ret;
++
++ ret = mxc_register_device(&mxc_ehci2, &tx27_usbh2_data);
++ return ret;
++}
++device_initcall(tx27_usbh2_init);
++
++static int tx27_usb_otg_init(struct platform_device *pdev)
++{
++ int ret;
++ u32 temp;
++ unsigned long freq;
++ unsigned long flags;
++ struct clk *usb_clk = clk_get(NULL, "usb_clk");
++
++ if (IS_ERR(usb_clk)) {
++ ret = PTR_ERR(usb_clk);
++ printk(KERN_ERR "Failed to get usb_clk: %d\n", ret);
++ goto err;
++ }
++
++ ret = gpio_usbotg_hs_active();
++ if (ret != 0) {
++ return ret;
++ }
++
++ freq = clk_get_rate(usb_clk);
++ if ((freq < 59999000) || (freq > 60001000)) {
++ printk(KERN_ERR "USB_CLK=%lu.%03luMHz, should be 60MHz\n",
++ freq / 1000000, freq / 1000 % 1000);
++ ret = clk_set_rate(usb_clk, 60000000);
++ if (ret != 0) {
++ printk(KERN_ERR "Failed to set usb_clk rate: %d\n", ret);
++ }
++ }
++ clk_put(usb_clk);
++ if (ret != 0) {
++ goto err;
++ }
++
++ local_irq_save(flags);
++ temp = readl(IO_ADDRESS(OTG_BASE_ADDR) + 0x600);
++ temp &= ~(3 << 21);
++ temp |= (1 << 5) | (1 << 16) | (1 << 19) | (1 << 20);
++ writel(temp, IO_ADDRESS(OTG_BASE_ADDR) + 0x600);
++ local_irq_restore(flags);
++
++ temp = readl(IO_ADDRESS(OTG_BASE_ADDR) + 0x184);
++ temp &= ~(3 << 30);
++ temp |= 2 << 30;
++ temp = 0x88001215;
++ /* select ULPI transceiver */
++ DBG(0, "%s: Changing USBOTG_PORTSC1[%08lx] from %08x to %08x\n", __FUNCTION__,
++ OTG_BASE_ADDR + 0x184, readl(IO_ADDRESS(OTG_BASE_ADDR) + 0x184), temp);
++ writel(temp, IO_ADDRESS(OTG_BASE_ADDR) + 0x184);
++
++ mdelay(10);
++
++ ret = usb3317_set_vbus_power(IO_ADDRESS(OTG_BASE_ADDR + 0x170), 1);
++ if (ret != 0) {
++// goto err;
++ }
++ return 0;
++
++ err:
++ gpio_usbotg_hs_active();
++ return ret;
++}
++
++static int tx27_usb_otg_exit(struct platform_device *pdev)
++{
++ gpio_usbotg_hs_inactive();
++ return 0;
++}
++
++static struct mxc_usbh_platform_data tx27_otg_data = {
++ .init = tx27_usb_otg_init,
++ .exit = tx27_usb_otg_exit,
++};
++
++int tx27_otg_init(void)
++{
++ int ret;
++
++ ret = mxc_register_device(&mxc_otg, &tx27_otg_data);
++ return ret;
++}
++device_initcall(tx27_otg_init);
++#endif // CONFIG_USB_EHCI_MXC
++
++#if defined(CONFIG_FEC) || defined(CONFIG_FEC_MODULE)
++static struct resource fec_resources[] = {
++ {
++ .start = FEC_BASE_ADDR,
++ .end = FEC_BASE_ADDR + 0x18f,
++ .flags = IORESOURCE_MEM,
++ },
++ {
++ .start = FEC_BASE_ADDR + 0x200,
++ .end = FEC_BASE_ADDR + 0x2e3,
++ .flags = IORESOURCE_MEM,
++ },
++ {
++ .start = MXC_INT_FEC,
++ .end = MXC_INT_FEC,
++ .flags = IORESOURCE_IRQ,
++ },
++#ifdef FEC_MII_IRQ
++ {
++ .start = IRQ_GPIOD(16),
++ .end = IRQ_GPIOD(16),
++ .flags = IORESOURCE_IRQ,
++ },
++#endif
++};
++
++static struct clk *fec_clk;
++static int tx27_fec_suspend(struct platform_device *pdev)
++{
++ BUG_ON(fec_clk == NULL);
++ DBG(1, "%s: Switching FEC PHY off\n", __FUNCTION__);
++ gpio_fec_inactive();
++ clk_disable(fec_clk);
++ return 0;
++}
++
++static int tx27_fec_resume(struct platform_device *pdev)
++{
++ BUG_ON(fec_clk == NULL);
++ DBG(1, "%s: Switching FEC PHY on\n", __FUNCTION__);
++ clk_enable(fec_clk);
++ gpio_fec_active();
++ return 0;
++}
++
++static int fec_arch_init(struct platform_device *pdev)
++{
++ int ret;
++
++ DBG(0, "%s: Activating FEC GPIOs\n", __FUNCTION__);
++ dump_regs();
++ ret = gpio_fec_active();
++ if (ret) {
++ printk(KERN_ERR "%s: could not enable FEC gpios: %d\n", __FUNCTION__, ret);
++ return ret;
++ }
++ BUG_ON(fec_clk != NULL);
++ fec_clk = clk_get(&pdev->dev, "fec_clk");
++ if (unlikely(IS_ERR(fec_clk))) {
++ printk(KERN_ERR "Failed to get fec_clk\n");
++ return PTR_ERR(fec_clk);
++ }
++ DBG(0, "%s: Enabling FEC clock\n", __FUNCTION__);
++ clk_enable(fec_clk);
++ dump_regs();
++ return 0;
++}
++
++static void fec_arch_exit(struct platform_device *pdev)
++{
++ BUG_ON(fec_clk == NULL);
++ if (unlikely(IS_ERR(fec_clk))) {
++ printk(KERN_ERR "Failed to get fec_clk\n");
++ return;
++ }
++ DBG(0, "%s: Disabling FEC clock\n", __FUNCTION__);
++ clk_disable(fec_clk);
++ clk_put(fec_clk);
++ fec_clk = NULL;
++ DBG(0, "%s: Deactivating FEC GPIOs\n", __FUNCTION__);
++ gpio_fec_inactive();
++}
++
++static struct fec_enet_platform_data fec_data = {
++ .arch_init = fec_arch_init,
++ .arch_exit = fec_arch_exit,
++ .suspend = tx27_fec_suspend,
++ .resume = tx27_fec_resume,
++};
++
++static struct platform_device fec_device = {
++ .name = "fec_enet",
++ .id = -1,
++ .num_resources = ARRAY_SIZE(fec_resources),
++ .resource = fec_resources,
++ .dev = {
++ .platform_data = &fec_data,
++ .coherent_dma_mask = 0xFFFFFFFF,
++ },
++};
++#endif
++
++#if defined(CONFIG_KEYBOARD_GPIO) || defined(CONFIG_KEYBOARD_GPIO_MODULE)
++/* tx27 gpio keys driver */
++struct gpio_keys_button tx27_gpio_keys[] = {
++ {
++ .code = KEY_POWER,
++ .gpio = GPIO_PORTB + 24,
++ .active_low = 0,
++ .desc = "Power Button",
++ .type = EV_KEY, /* input event type (EV_KEY, EV_SW) */
++ .wakeup = 1, /* configure the button as a wake-up source */
++ .debounce_interval = 1, /* debounce ticks interval in msecs */
++ },
++};
++
++struct gpio_keys_platform_data tx27_gpio_keys_pdata = {
++ .buttons = tx27_gpio_keys,
++ .nbuttons = ARRAY_SIZE(tx27_gpio_keys),
++};
++
++static struct platform_device tx27_gpio_keys_device = {
++ .name = "gpio-keys",
++ .id = -1,
++ .dev = {
++ .platform_data = &tx27_gpio_keys_pdata,
++ },
++};
++#endif
++
++#if defined(CONFIG_LEDS_GPIO) || defined(CONFIG_LEDS_GPIO_MODULE)
++static struct gpio_led tx27_leds[] = {
++ {
++ .name = "Power-LED",
++ .default_trigger = "heartbeat",
++ .gpio = GPIO_PORTF + 13,
++ },
++};
++
++static struct gpio_led_platform_data tx27_led_data = {
++ .leds = tx27_leds,
++ .num_leds = ARRAY_SIZE(tx27_leds),
++};
++
++static struct platform_device tx27_led_device = {
++ .name = "leds-gpio",
++ .id = -1,
++ .dev = {
++ .platform_data = &tx27_led_data,
++ },
++};
++#endif
++
++#if defined(CONFIG_KEYBOARD_MXC) || defined(CONFIG_KEYBOARD_MXC_MODULE)
++/*!
++ * This array is used for mapping mx27 ADS keypad scancodes to input keyboard
++ * keycodes.
++ */
++static u16 tx27_kpd_keycodes[] = {
++ KEY_POWER,
++};
++
++static struct keypad_data tx27_keypad = {
++ .rowmax = 1,
++ .colmax = 1,
++ .irq = MXC_INT_KPP,
++ .learning = 0,
++ //.delay = 2, /* unused in the driver! */
++ .matrix = tx27_kpd_keycodes,
++};
++
++static struct resource tx27_kpp_resources[] = {
++ {
++ .start = MXC_INT_KPP,
++ .end = MXC_INT_KPP,
++ .flags = IORESOURCE_IRQ,
++ },
++};
++
++/* tx27 keypad driver */
++static struct platform_device tx27_keypad_device = {
++ .name = "mxc_keypad",
++ .id = 0,
++ .num_resources = ARRAY_SIZE(tx27_kpp_resources),
++ .resource = tx27_kpp_resources,
++ .dev = {
++ .platform_data = &tx27_keypad,
++ },
++};
++#endif
++
++/* MTD NAND flash */
++#if defined(CONFIG_MTD_NAND_MXC) || defined(CONFIG_MTD_NAND_MXC_MODULE)
++#if 0
++static struct mtd_partition tx27_nand_partitions[] = {
++ {
++ .name = "RedBoot",
++ .offset = 0,
++ .size = 0x00040000,
++ },
++ {
++ .name = "kernel",
++ .offset = MTDPART_OFS_APPEND,
++ .size = 0x001A0000,
++ },
++ {
++ .name = "rootfs",
++ .offset = MTDPART_OFS_APPEND,
++ .size = 0x07E000000,
++ },
++ {
++ .name = "FIS directory",
++ .offset = MTDPART_OFS_APPEND,
++ .size = 0x00003000,
++ .mask_flags = MTD_WRITEABLE,
++ },
++ {
++ .name = "RedBoot config",
++ .offset = MTDPART_OFS_APPEND,
++ .size = 0x00001000,
++ .mask_flags = MTD_WRITEABLE,
++ },
++};
++
++static struct flash_platform_data tx27_nand_data = {
++ .map_name = "nand_probe",
++ .name = "tx27-nand",
++ .parts = tx27_nand_partitions,
++ .nr_parts = ARRAY_SIZE(tx27_nand_partitions),
++ .width = 1,
++};
++#else
++static struct mxc_nand_platform_data tx27_nand_data = {
++ .hw_ecc = 1,
++ .width = 1,
++};
++#endif
++
++static struct resource tx27_nand_resources[] = {
++ {
++ .start = NFC_BASE_ADDR,
++ .end = NFC_BASE_ADDR + 0xfff,
++ .flags = IORESOURCE_MEM
++ }, {
++ .start = MXC_INT_NANDFC,
++ .end = MXC_INT_NANDFC,
++ .flags = IORESOURCE_IRQ
++ },
++};
++
++static struct platform_device tx27_nand_mtd_device = {
++ .name = "mxc_nand",
++ .id = 0,
++ .num_resources = ARRAY_SIZE(tx27_nand_resources),
++ .resource = tx27_nand_resources,
++ .dev = {
++ .platform_data = &tx27_nand_data,
++ },
++};
++#endif
++
++#if defined(CONFIG_FB_IMX) || defined(CONFIG_FB_IMX_MODULE)
++/*
++ * Setup GPIO for LCDC device to be active
++ *
++ */
++static unsigned int mx27_lcdc_gpios[] = {
++ MXC_PIN(A, 30, GPIO, GPIO_OUT | GPIO_DFLT_LOW), /* PA30 */
++ MXC_PIN(A, 25, GPIO, GPIO_OUT | GPIO_DFLT_LOW), /* PA25 */
++ MXC_PIN(A, 26, GPIO, GPIO_OUT | GPIO_DFLT_LOW), /* PA26 */
++ MXC_PIN(A, 24, GPIO, GPIO_OUT | GPIO_DFLT_LOW), /* PA24 */
++ MXC_PIN(A, 27, GPIO, GPIO_OUT | GPIO_DFLT_LOW), /* PA27 */
++ PA5_PF_LSCLK,
++ PA6_PF_LD0,
++ PA7_PF_LD1,
++ PA8_PF_LD2,
++ PA9_PF_LD3,
++ PA10_PF_LD4,
++ PA11_PF_LD5,
++ PA12_PF_LD6,
++ PA13_PF_LD7,
++ PA14_PF_LD8,
++ PA15_PF_LD9,
++ PA16_PF_LD10,
++ PA17_PF_LD11,
++ PA18_PF_LD12,
++ PA19_PF_LD13,
++ PA20_PF_LD14,
++ PA21_PF_LD15,
++ PA22_PF_LD16,
++ PA23_PF_LD17,
++ PA28_PF_HSYNC,
++ PA29_PF_VSYNC,
++ PA31_PF_OE_ACD,
++};
++
++static int tx27_gpio_lcdc_active(struct platform_device *dev)
++{
++ int ret;
++
++ DBG(0, "%s: Setting up GPIO pins for LCD\n", __FUNCTION__);
++ ret = mxc_gpio_setup_multiple_pins(mx27_lcdc_gpios,
++ ARRAY_SIZE(mx27_lcdc_gpios), "LCD");
++ if (ret) {
++ DBG(0, "%s: Failed to setup GPIO pins for LCD: %d\n",
++ __FUNCTION__, ret);
++ return ret;
++ }
++ return 0;
++}
++
++/*
++ * Setup GPIO for LCDC device to be inactive
++ *
++ */
++static int tx27_gpio_lcdc_inactive(struct platform_device *dev)
++{
++ mxc_gpio_release_multiple_pins(mx27_lcdc_gpios,
++ ARRAY_SIZE(mx27_lcdc_gpios));
++ return 0;
++}
++
++static struct imx_fb_platform_data tx27_fb_data[] __initdata = {
++ [0] = {
++ //.fb_mode = "Xenarc_700_Y-16",
++ .init = tx27_gpio_lcdc_active,
++ .exit = tx27_gpio_lcdc_inactive,
++ .lcd_power = NULL,
++ .backlight_power = NULL,
++
++ .pixclock = 34576,
++ .xres = 640,
++ .yres = 480,
++
++ .bpp = 16,
++
++ .hsync_len = 64,
++ .right_margin = 138 + 1,
++ .left_margin = 118 + 3,
++
++ .vsync_len = 7,
++ .upper_margin = 44,
++ .lower_margin = 44,
++#if 0
++ /* currently not used by driver! */
++ .sync = ((0*FB_SYNC_HOR_HIGH_ACT) |
++ (0*FB_SYNC_VERT_HIGH_ACT) |
++ (1*FB_SYNC_OE_ACT_HIGH)),
++#else
++ .pcr = PCR_TFT | PCR_COLOR | PCR_PBSIZ_8 |
++ PCR_BPIX_16 | PCR_FLMPOL | PCR_LPPOL | PCR_SCLK_SEL,
++ .dmacr = 0x80040060,
++#endif
++ .cmap_greyscale = 0,
++ .cmap_inverse = 0,
++ .cmap_static = 0,
++
++ .fixed_screen_cpu = NULL,
++ },
++ [1] = {
++ //.fb_mode = "SHARP LQ10D42-16",
++ .init = tx27_gpio_lcdc_active,
++ .exit = tx27_gpio_lcdc_inactive,
++ .lcd_power = NULL,
++ .backlight_power = NULL,
++
++ .pixclock = 34576,
++ .xres = 640,
++ .yres = 480,
++
++#ifdef USE_18BPP
++ .bpp = 32,
++#else
++ .bpp = 16,
++#endif
++ .hsync_len = 64,
++ .right_margin = 138 + 1,
++ .left_margin = 118 + 3,
++
++ .vsync_len = 7,
++ .upper_margin = 28,
++ .lower_margin = 60,
++#if 0
++ /* currently not used by driver! */
++ .sync = ((0*FB_SYNC_HOR_HIGH_ACT) |
++ (0*FB_SYNC_VERT_HIGH_ACT) |
++ (1*FB_SYNC_OE_ACT_HIGH)),
++#else
++ .pcr = PCR_TFT | PCR_COLOR | PCR_PBSIZ_8 |
++#ifdef USE_18BPP
++ PCR_BPIX_18 | PCR_END_SEL | PCR_FLMPOL | PCR_LPPOL | PCR_SCLK_SEL,
++#else
++ PCR_BPIX_16 | PCR_FLMPOL | PCR_LPPOL | PCR_SCLK_SEL,
++#endif
++ .dmacr = 0x80040060,
++#endif
++ .cmap_greyscale = 0,
++ .cmap_inverse = 0,
++ .cmap_static = 0,
++
++ .fixed_screen_cpu = NULL,
++ },
++ [2] = {
++ //.fb_mode = "SHARP LQ104V1DG61-16",
++ .init = tx27_gpio_lcdc_active,
++ .exit = tx27_gpio_lcdc_inactive,
++ .lcd_power = NULL,
++ .backlight_power = NULL,
++
++ .pixclock = 40000,
++ .xres = 640,
++ .yres = 480,
++
++#ifdef USE_18BPP
++ .bpp = 32,
++#else
++ .bpp = 16,
++#endif
++ .hsync_len = 32,
++ .right_margin = 32 + 1,
++ .left_margin = 0 + 3,
++
++ .vsync_len = 35,
++ .upper_margin = 0,
++ .lower_margin = 0,
++#if 0
++ /* currently not used by driver! */
++ .sync = ((0*FB_SYNC_HOR_HIGH_ACT) |
++ (0*FB_SYNC_VERT_HIGH_ACT) |
++ (1*FB_SYNC_OE_ACT_HIGH)),
++#else
++ .pcr = PCR_TFT | PCR_COLOR | PCR_PBSIZ_8 |
++#ifdef USE_18BPP
++ PCR_BPIX_18 | PCR_END_SEL | PCR_FLMPOL | PCR_LPPOL | PCR_SCLK_SEL,
++#else
++ PCR_BPIX_16 | PCR_FLMPOL | PCR_LPPOL | PCR_CLKPOL | PCR_SCLK_SEL,
++#endif
++ .dmacr = 0x80040060,
++#endif
++ .cmap_greyscale = 0,
++ .cmap_inverse = 0,
++ .cmap_static = 0,
++
++ .fixed_screen_cpu = NULL,
++ },
++};
++
++int __init karo_tx27_fb_init(void)
++{
++ int ret;
++
++ ret = mxc_register_device(&mxc_fb_device, &tx27_fb_data[0]);
++ if (ret != 0) {
++ DBG(0, "%s: Failed to register FB device: %d\n", __FUNCTION__, ret);
++ }
++ return ret;
++}
++device_initcall(karo_tx27_fb_init);
++#endif
++
++#if defined(CONFIG_MMC_MXC) || defined(CONFIG_MMC_MXC_MODULE)
++/*!
++ * Resource definition for the SDHC1
++ */
++static struct resource tx27_sdhc1_resources[] = {
++ {
++ .start = SDHC1_BASE_ADDR,
++ .end = SDHC1_BASE_ADDR + SZ_4K - 1,
++ .flags = IORESOURCE_MEM,
++ },
++ {
++ .start = MXC_INT_SDHC1,
++ .end = MXC_INT_SDHC1,
++ .flags = IORESOURCE_IRQ,
++ },
++ {
++ .start = IRQ_GPIOC(21),
++ .end = IRQ_GPIOC(21),
++ .flags = IORESOURCE_IRQ,
++ },
++ {
++ .name = "sdhc1",
++ .start = DMA_REQ_SDHC1,
++ .end = DMA_REQ_SDHC1,
++ .flags = IORESOURCE_DMA
++ },
++};
++
++/*!
++ * Resource definition for the SDHC2
++ */
++static struct resource tx27_sdhc2_resources[] = {
++ {
++ .start = SDHC2_BASE_ADDR,
++ .end = SDHC2_BASE_ADDR + SZ_4K - 1,
++ .flags = IORESOURCE_MEM,
++ },
++ {
++ .start = MXC_INT_SDHC2,
++ .end = MXC_INT_SDHC2,
++ .flags = IORESOURCE_IRQ,
++ },
++ {
++ .start = IRQ_GPIOC(22),
++ .end = IRQ_GPIOC(22),
++ .flags = IORESOURCE_IRQ,
++ },
++ {
++ .name = "sdhc2",
++ .start = DMA_REQ_SDHC2,
++ .end = DMA_REQ_SDHC2,
++ .flags = IORESOURCE_DMA
++ },
++};
++static inline int tx27_mmc_get_irq(int id)
++{
++ int irq;
++
++ switch (id) {
++ case 0:
++ irq = tx27_sdhc1_resources[2].start;
++ break;
++ case 1:
++ irq = tx27_sdhc2_resources[2].start;
++ break;
++ default:
++ BUG();
++ }
++ return irq;
++}
++
++static const char *tx27_mmc_irqdesc[] = {
++ "MMC card 0 detect",
++ "MMC card 1 detect",
++};
++
++static int tx27_mmc_init(struct device *dev, irqreturn_t (*mmc_detect_irq)(int, void *),
++ void *data)
++{
++ int err;
++ int id = to_platform_device(dev)->id;
++ struct mmc_host *host = data;
++ int irq = tx27_mmc_get_irq(id);
++
++ err = gpio_sdhc_active(id);
++ if (err) {
++ return err;
++ }
++
++ host->caps |= MMC_CAP_4_BIT_DATA;
++
++ err = request_irq(irq, mmc_detect_irq,
++ IRQF_TRIGGER_FALLING | IRQF_TRIGGER_RISING,
++ tx27_mmc_irqdesc[id], data);
++ if (err) {
++ printk(KERN_ERR "%s: MMC/SD: can't request MMC card detect IRQ %d\n",
++ __FUNCTION__, irq);
++ return err;
++ }
++ device_set_wakeup_capable(dev, 1);
++
++ return 0;
++}
++
++static void tx27_mmc_exit(struct device *dev, void *data)
++{
++ int id = to_platform_device(dev)->id;
++ int irq = tx27_mmc_get_irq(id);
++
++ free_irq(irq, data);
++ gpio_sdhc_inactive(id);
++}
++
++static int tx27_mmc_suspend(struct device *dev, pm_message_t state)
++{
++ int id = to_platform_device(dev)->id;
++ int irq = tx27_mmc_get_irq(id);
++
++ if (device_may_wakeup(dev)) {
++ DBG(0, "%s: Enabling IRQ %d wakeup\n", __FUNCTION__, irq);
++ return enable_irq_wake(irq);
++ }
++ return 0;
++}
++
++static int tx27_mmc_resume(struct device *dev)
++{
++ int id = to_platform_device(dev)->id;
++ int irq = tx27_mmc_get_irq(id);
++
++ if (device_may_wakeup(dev)) {
++ DBG(0, "%s: Disabling IRQ %d wakeup\n", __FUNCTION__, irq);
++ return disable_irq_wake(irq);
++ }
++ return 0;
++}
++
++static struct imxmmc_platform_data tx27_sdhc1_data = {
++ //.ocr_mask = MMC_VDD_32_33 | MMC_VDD_33_34,
++ //.min_clk = 150000,
++ //.max_clk = 25000000,
++ //.detect_delay = 20,
++ .init = tx27_mmc_init,
++ .exit = tx27_mmc_exit,
++ .suspend = tx27_mmc_suspend,
++ .resume = tx27_mmc_resume,
++};
++
++static struct imxmmc_platform_data tx27_sdhc2_data = {
++ //.ocr_mask = MMC_VDD_32_33 | MMC_VDD_33_34,
++ //.min_clk = 150000,
++ //.max_clk = 25000000,
++ //.detect_delay = 20,
++ .init = tx27_mmc_init,
++ .exit = tx27_mmc_exit,
++};
++
++static struct platform_device tx27_sdhc1_device = {
++ .name = "imx-mmc",
++ .id = 0,
++ .dev = {
++ .platform_data = &tx27_sdhc1_data,
++ },
++ .num_resources = ARRAY_SIZE(tx27_sdhc1_resources),
++ .resource = tx27_sdhc1_resources,
++};
++
++/*! Device Definition for MXC SDHC2 */
++static struct platform_device tx27_sdhc2_device = {
++ .name = "imx-mmc",
++ .id = 1,
++ .dev = {
++ .platform_data = &tx27_sdhc2_data,
++ },
++ .num_resources = ARRAY_SIZE(tx27_sdhc2_resources),
++ .resource = tx27_sdhc2_resources,
++};
++#endif
++
++#if defined(CONFIG_SPI_MXC) || defined(CONFIG_SPI_MXC_MODULE)
++static struct resource mxcspi1_resources[] = {
++ [0] = {
++ .start = CSPI1_BASE_ADDR,
++ .end = CSPI1_BASE_ADDR + SZ_4K - 1,
++ .flags = IORESOURCE_MEM,
++ },
++ [1] = {
++ .start = MXC_INT_CSPI1,
++ .end = MXC_INT_CSPI1,
++ .flags = IORESOURCE_IRQ,
++ },
++};
++
++static struct mxc_spi_master mxcspi1_data = {
++ .maxchipselect = 2,
++ .spi_version = 0,
++};
++
++static struct platform_device mxcspi1_device = {
++ .name = "mxc_spi",
++ .id = 0,
++ .dev = {
++ .platform_data = &mxcspi1_data,
++ },
++ .num_resources = ARRAY_SIZE(mxcspi1_resources),
++ .resource = mxcspi1_resources,
++};
++#endif // defined(CONFIG_SPI_MXC) || defined(CONFIG_SPI_MXC_MODULE)
++
++#if defined(CONFIG_AC97_BUS) || defined(CONFIG_AC97_BUS_MODULE)
++static u64 tx27_dma_mask = ~0UL;
++
++static void tx27_ac97_gpio_release(void)
++{
++ gpio_ac97_inactive();
++}
++
++static int tx27_ac97_init(struct platform_device *dev)
++{
++ int ret;
++
++ DBG(0, "%s: \n", __FUNCTION__);
++ ret = gpio_ac97_active();
++ if (ret) {
++
++ return ret;
++ }
++ return 0;
++}
++
++static void tx27_ac97_exit(struct platform_device *dev)
++{
++ DBG(0, "%s: Releasing AC97 GPIO pins\n", __FUNCTION__);
++ tx27_ac97_gpio_release();
++}
++
++static struct mxc_ac97_audio_ops tx27_ac97_ops = {
++ .init = tx27_ac97_init,
++ .exit = tx27_ac97_exit,
++ .startup = NULL,
++ .shutdown = NULL,
++ .suspend = NULL,
++ .resume = NULL,
++ .priv = NULL,
++};
++
++static struct platform_device ac97_device = {
++ .name = "mx27-ac97",
++ .id = -1,
++ .dev = {
++ .dma_mask = &tx27_dma_mask,
++ .coherent_dma_mask = ~0UL,
++ .platform_data = &tx27_ac97_ops,
++ },
++};
++#endif
++
++#if defined(CONFIG_RTC_DRV_DS13XX) || defined(CONFIG_RTC_DRV_DS13XX_MODULE)
++static struct ds13xx_platform_data tx27_ds1339_data = {
++ .type = ds_1339,
++ .ctrl = 1 << 2, /* set INTCN to disable SQW output */
++ .trc = DS1339_TRC_ENABLE | DS1339_DIODE_ENABLE | DS1339_TRC_250R,
++};
++
++static struct platform_device tx27_ds1339_device = {
++ .name = "rtc-ds13xx",
++ .dev = {
++ .platform_data = &tx27_ds1339_data,
++ },
++};
++#endif
++
++#if defined(CONFIG_VIDEO_MXC_EMMA_OUTPUT) || defined(CONFIG_VIDEO_MXC_EMMA_OUTPUT_MODULE)
++static u64 mxc_emma_dmamask = 0xffffffffUL;
++
++static struct platform_device tx27_v4l2out_device = {
++ .name = "MXC Video Output",
++ .id = 0,
++ .dev = {
++ .dma_mask = &mxc_emma_dmamask,
++ .coherent_dma_mask = ~0UL,
++ },
++};
++#endif
++
++#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
++static int mxc_i2c0_pins[] = {
++ /*
++ * it seems the data line misses a pullup, so we must enable
++ * the internal pullup as a local workaround
++ */
++ PD17_PF_I2C_DATA,
++ PD18_PF_I2C_CLK,
++};
++
++static int karo_tx27_i2c_0_init(struct platform_device *pdev)
++{
++ return mxc_gpio_setup_multiple_pins(mxc_i2c0_pins,
++ ARRAY_SIZE(mxc_i2c0_pins), "I2C0");
++}
++
++static int karo_tx27_i2c_0_exit(struct platform_device *pdev)
++{
++ mxc_gpio_release_multiple_pins(mxc_i2c0_pins,
++ ARRAY_SIZE(mxc_i2c0_pins));
++
++ return 0;
++}
++
++static struct imx_i2c_platform_data karo_tx27_i2c_0_data = {
++ .max_clk = 100000,
++ .init = karo_tx27_i2c_0_init,
++ .exit = karo_tx27_i2c_0_exit,
++};
++
++#if defined(CONFIG_RTC_DRV_DS13XX) || defined(CONFIG_RTC_DRV_DS13XX_MODULE)
++static struct ds13xx_platform_data karo_ds1339_data = {
++ .type = ds_1339,
++ .ctrl = 0,
++ .trc = DS1339_TRC_ENABLE | DS1339_DIODE_ENABLE | DS1339_TRC_250R,
++};
++#endif
++
++static struct at24_platform_data karo_tx27_eeprom = {
++ .byte_len = 2048,
++ .page_size = 32,
++ .flags = AT24_FLAG_ADDR16 | AT24_FLAG_TAKE8ADDR,
++};
++
++static struct i2c_board_info karo_i2c_0_boardinfo[] __initdata = {
++ {
++ I2C_BOARD_INFO("24c16", 0x50),
++ .platform_data = &karo_tx27_eeprom,
++ .type = "24c16",
++ },
++#if defined(CONFIG_RTC_DRV_DS13XX) || defined(CONFIG_RTC_DRV_DS13XX_MODULE)
++ {
++ I2C_BOARD_INFO("ds1339", 0x68/*DS1339_CHIP_ID*/),
++ .type = "ds1339",
++ .platform_data = &karo_ds1339_data,
++ },
++#endif
++#if defined(CONFIG_RTC_DRV_DS1307) || defined(CONFIG_RTC_DRV_DS1307_MODULE)
++ {
++ I2C_BOARD_INFO("ds1339", 0x68/*DS1339_CHIP_ID*/),
++ .type = "ds1339",
++ //.platform_data = &karo_ds1339_data,
++ },
++#endif
++};
++
++static int mxc_i2c_1_pins[] = {
++ /*
++ * it seems the data line misses a pullup, so we must enable
++ * the internal pullup as a local workaround
++ */
++ PC5_PF_I2C2_SDA,
++ PC6_PF_I2C2_SCL,
++};
++
++static int karo_tx27_i2c_1_init(struct platform_device *pdev)
++{
++ return mxc_gpio_setup_multiple_pins(mxc_i2c_1_pins,
++ ARRAY_SIZE(mxc_i2c_1_pins), "I2C_1");
++}
++
++static int karo_tx27_i2c_1_exit(struct platform_device *pdev)
++{
++ mxc_gpio_release_multiple_pins(mxc_i2c_1_pins,
++ ARRAY_SIZE(mxc_i2c_1_pins));
++
++ return 0;
++}
++
++static struct imx_i2c_platform_data karo_tx27_i2c_1_data = {
++ .max_clk = 100000,
++ .init = karo_tx27_i2c_1_init,
++ .exit = karo_tx27_i2c_1_exit,
++};
++
++static struct i2c_board_info karo_i2c_1_boardinfo[] __initdata = {
++ {
++ I2C_BOARD_INFO("lp3972", 0x34),
++ .type = "lp3972",
++ },
++};
++
++int __init karo_i2c_init(void)
++{
++ int ret;
++
++ DBG(0, "%s: Registering I2C bus 0\n", __FUNCTION__);
++ ret = mxc_register_device(&imx_i2c_device0, &karo_tx27_i2c_0_data);
++ if (ret != 0) {
++ printk(KERN_ERR "Failed to register I2C device: %d\n", ret);
++ return ret;
++ }
++ ret = i2c_register_board_info(0, karo_i2c_0_boardinfo,
++ ARRAY_SIZE(karo_i2c_0_boardinfo));
++ if (ret != 0) {
++ printk(KERN_ERR "Failed to register I2C board info: %d\n", ret);
++ platform_device_unregister(&imx_i2c_device0);
++ }
++
++ DBG(0, "%s: Registering I2C bus 1\n", __FUNCTION__);
++ ret = mxc_register_device(&imx_i2c_device1, &karo_tx27_i2c_1_data);
++ if (ret != 0) {
++ printk(KERN_ERR "Failed to register I2C device: %d\n", ret);
++ return ret;
++ }
++ ret = i2c_register_board_info(1, karo_i2c_1_boardinfo,
++ ARRAY_SIZE(karo_i2c_1_boardinfo));
++ if (ret != 0) {
++ printk(KERN_ERR "Failed to register I2C board info: %d\n", ret);
++ platform_device_unregister(&imx_i2c_device1);
++ }
++ return ret;
++}
++device_initcall(karo_i2c_init);
++#endif
++
++struct platform_dev_list {
++ struct platform_device *pdev;
++ int flag;
++} tx27_devices[] __initdata = {
++#if defined(CONFIG_KEYBOARD_GPIO) || defined(CONFIG_KEYBOARD_GPIO_MODULE)
++ { .pdev = &tx27_gpio_keys_device, .flag = -1, },
++#endif
++#if defined(CONFIG_LEDS_GPIO) || defined(CONFIG_LEDS_GPIO_MODULE)
++ { .pdev = &tx27_led_device, .flag = -1, },
++#endif
++#if defined(CONFIG_RTC_MXC) || defined(CONFIG_RTC_MXC_MODULE)
++ { .pdev = &mxc_rtc_device, .flag = -1, },
++#endif
++#if defined(CONFIG_MTD_NAND_MXC) || defined(CONFIG_MTD_NAND_MXC_MODULE)
++ { .pdev = &tx27_nand_mtd_device, .flag = 1, },
++#endif
++#if defined(CONFIG_KEYBOARD_MXC) || defined(CONFIG_KEYBOARD_MXC_MODULE)
++ { .pdev = &tx27_keypad_device, .flag = 1, },
++#endif
++#if defined(CONFIG_FEC) || defined(CONFIG_FEC_MODULE)
++ { .pdev = &fec_device, .flag = 1, },
++#endif
++#if defined(CONFIG_SPI_MXC) || defined(CONFIG_SPI_MXC_MODULE)
++ { .pdev = &mxcspi1_device, .flag = 1, },
++#endif
++#if defined(CONFIG_AC97_BUS) || defined(CONFIG_AC97_BUS_MODULE)
++ { .pdev = &ac97_device, .flag = 1, },
++#endif
++#if defined(CONFIG_MMC_MXC) || defined(CONFIG_MMC_MXC_MODULE)
++ { .pdev = &tx27_sdhc1_device, .flag = 1, },
++ { .pdev = &tx27_sdhc2_device, .flag = 1, },
++#endif
++#if defined(CONFIG_RTC_DRV_DS13XX) || defined(CONFIG_RTC_DRV_DS13XX_MODULE)
++ { .pdev = &tx27_ds1339_device, .flag = 1, },
++#endif
++#if defined(CONFIG_VIDEO_MXC_EMMA_OUTPUT) || defined(CONFIG_VIDEO_MXC_EMMA_OUTPUT_MODULE)
++ { .pdev = &tx27_v4l2out_device, .flag = 1, },
++#endif
++#if defined(CONFIG_MXC_VPU) || defined(CONFIG_MXC_VPU_MODULE)
++ { .pdev = &mxc_vpu_device, .flag = 1, },
++#endif
++};
++#define TX27_NUM_DEVICES ARRAY_SIZE(tx27_devices)
++
++#define OSC26M_ENABLE_PIN (GPIO_PORTB | 22)
++
++static int _clk_26m_enable(struct clk *clk)
++{
++ int ret;
++
++ DBG(0, "%s: Switching 26MHz oscillator on\n", __FUNCTION__);
++ ret = gpio_request(OSC26M_ENABLE_PIN, "OSC26m");
++ if (ret != 0) {
++ printk(KERN_ERR "Failed to request 26MHz oscillator enable GPIO: %d\n", ret);
++ return ret;
++ }
++ gpio_set_value(OSC26M_ENABLE_PIN, 1);
++ mxc_gpio_mode(OSC26M_ENABLE_PIN | GPIO_GPIO | GPIO_OUT);
++ return 0;
++}
++
++static void _clk_26m_disable(struct clk *clk)
++{
++ DBG(0, "%s: Switching 26MHz oscillator off\n", __FUNCTION__);
++ gpio_set_value(OSC26M_ENABLE_PIN, 0);
++ gpio_free(OSC26M_ENABLE_PIN);
++}
++
++static struct clk clk_26m = {
++ .name = "clk_26m",
++ .enable = _clk_26m_enable,
++ .disable = _clk_26m_disable,
++};
++
++#ifdef CONFIG_BASE_CLK_26MHz
++static __init void karo_tx27_clock_switch(struct clk *_26m_clk)
++{
++ int loops = 0;
++ u32 cscr = __raw_readl(CCM_CSCR);
++ u32 ccsr;
++
++ if (_26m_clk != NULL) {
++ DBG(0, "%s: Enabling 26MHz clock\n", __FUNCTION__);
++ clk_enable(_26m_clk);
++
++ __raw_writel(CCM_MPCTL0_PD_VAL(0) |
++ CCM_MPCTL0_MFD_VAL(51) |
++ CCM_MPCTL0_MFI_VAL(7) |
++ CCM_MPCTL0_MFN_VAL(35), CCM_MPCTL0);
++
++ __raw_writel(CCM_SPCTL0_PD_VAL(1) |
++ CCM_SPCTL0_MFD_VAL(12) |
++ CCM_SPCTL0_MFI_VAL(9) |
++ CCM_SPCTL0_MFN_VAL(3), CCM_SPCTL0);
++
++ cscr |= CCM_CSCR_MCU | CCM_CSCR_SP;
++ __raw_writel(cscr, CCM_CSCR);
++
++ cscr |= CCM_CSCR_MPLLRES | CCM_CSCR_SPLLRES;
++ __raw_writel(cscr, CCM_CSCR);
++ while (__raw_readl(CCM_CSCR) & (CCM_CSCR_MPLLRES | CCM_CSCR_SPLLRES)) {
++ udelay(1);
++ loops++;
++ }
++ printk("PLLs locked after %d loops: CSCR=%08x(%08x)\n", loops,
++ __raw_readl(CCM_CSCR), cscr);
++
++ cscr &= ~CCM_CSCR_FPM;
++ __raw_writel(cscr, CCM_CSCR);
++ DBG(9, "%s: Disabling FPM, DPLL and OSC26M\n", __FUNCTION__);
++ ccsr = __raw_readl(CCM_CCSR);
++ __raw_writel(ccsr & ~0x300, CCM_CCSR);
++ DBG(9, "changing CCSR from %08x to %08x(%08x)\n",
++ ccsr, ccsr & ~0x300, __raw_readl(CCM_CCSR));
++ } else {
++ printk(KERN_INFO "Changing SPCTL0 from %08x to %08x\n",
++ __raw_readl(CCM_SPCTL0), CCM_SPCTL0_PD_VAL(2) |
++ CCM_SPCTL0_MFD_VAL(755) |
++ CCM_SPCTL0_MFI_VAL(11) |
++ CCM_SPCTL0_MFN_VAL(-205));
++
++ __raw_writel(CCM_SPCTL0_PD_VAL(2) |
++ CCM_SPCTL0_MFD_VAL(755) |
++ CCM_SPCTL0_MFI_VAL(11) |
++ CCM_SPCTL0_MFN_VAL(-205), CCM_SPCTL0);
++ }
++}
++#else
++static inline void karo_tx27_clock_switch(struct clk *_26m_clk)
++{
++ printk(KERN_INFO "Changing SPCTL0 from %08x to %08x\n",
++ __raw_readl(CCM_SPCTL0), CCM_SPCTL0_PD_VAL(2) |
++ CCM_SPCTL0_MFD_VAL(755) |
++ CCM_SPCTL0_MFI_VAL(11) |
++ CCM_SPCTL0_MFN_VAL(-205));
++
++ __raw_writel(CCM_SPCTL0_PD_VAL(2) |
++ CCM_SPCTL0_MFD_VAL(755) |
++ CCM_SPCTL0_MFI_VAL(11) |
++ CCM_SPCTL0_MFN_VAL(-205), CCM_SPCTL0);
++}
++#endif
++
++static __init void karo_tx27_clock_init(void)
++{
++ struct clk *cpu_clk;
++ struct clk *_26m_clk = NULL;
++ int ret;
++
++ ret = clk_register(&clk_26m);
++ if (ret != 0) {
++ printk(KERN_ERR "Failed to register 26MHz clock: %d\n", ret);
++ goto no_26m;
++ }
++ _26m_clk = clk_get(NULL, "clk_26m");
++ if (IS_ERR(_26m_clk)) {
++ printk(KERN_ERR "Cannot request 26MHz clock: %ld\n", PTR_ERR(_26m_clk));
++ _26m_clk = NULL;
++ }
++ no_26m:
++ karo_tx27_clock_switch(_26m_clk);
++ mxc_clocks_init(26000000);
++ cpu_clk = clk_get(NULL, "cpu_clk");
++ if (!IS_ERR(cpu_clk)) {
++ printk(KERN_DEBUG "%s: Setting CPU clock to 400MHz\n", __FUNCTION__);
++ if (clk_set_rate(cpu_clk, 399000000) != 0) {
++ printk(KERN_ERR "Failed to set CPU clock rate\n");
++ }
++ } else {
++ printk(KERN_ERR "Failed to get CPU clock: %ld\n",
++ PTR_ERR(cpu_clk));
++ }
++ SHOW_REG(CCM_CSCR);
++}
++
++static __init void karo_tx27_board_init(void)
++{
++ int i;
++
++ DBG(0, "%s: \n", __FUNCTION__);
++ SHOW_REG(CCM_CSCR);
++
++ for (i = 0; i < ARRAY_SIZE(tx27_uart_devices); i++) {
++ int ret;
++ int port = tx27_uart_devices[i]->id;
++
++ DBG(0, "%s: Registering platform device[%d] @ %p dev %p: %s\n",
++ __FUNCTION__, i, tx27_uart_devices[i],
++ &tx27_uart_devices[i]->dev, tx27_uart_devices[i]->name);
++ ret = mxc_register_device(tx27_uart_devices[i],
++ &tx27_uart_ports[port]);
++ if (ret != 0) {
++ printk(KERN_WARNING "%s: Failed to register platform_device[%d]: %s: %d\n",
++ __FUNCTION__, i, tx27_uart_devices[i]->name, ret);
++ }
++ }
++ //mxc_cpu_common_init();
++ //karo_tx27_clock_init();
++ //early_console_setup(saved_command_line);
++
++ /* configure PF13 as output for Starterkit-5 LED */
++ //mxc_gpio_mode(MXC_PIN(F, 13, GPIO, GPIO_OUT | GPIO_DFLT_LOW));
++ dump_regs();
++
++ /* enable SSI3_INT (PC23) for IRQ probing */
++ set_irq_flags(gpio_to_irq(GPIO_PORTC | 23), IRQF_VALID | IRQF_PROBE);
++
++ for (i = 0; i < TX27_NUM_DEVICES; i++) {
++ int ret;
++
++ if (tx27_devices[i].pdev == NULL) continue;
++ if (!tx27_devices[i].flag) {
++ DBG(0, "%s: Skipping platform device[%d] @ %p dev %p: %s\n",
++ __FUNCTION__, i, tx27_devices[i].pdev, &tx27_devices[i].pdev->dev,
++ tx27_devices[i].pdev->name);
++ continue;
++ }
++ DBG(0, "%s: Registering platform device[%d] @ %p dev %p: %s\n",
++ __FUNCTION__, i, tx27_devices[i].pdev, &tx27_devices[i].pdev->dev,
++ tx27_devices[i].pdev->name);
++ ret = platform_device_register(tx27_devices[i].pdev);
++ if (ret) {
++ printk(KERN_WARNING "%s: Failed to register platform_device[%d]: %s: %d\n",
++ __FUNCTION__, i, tx27_devices[i].pdev->name, ret);
++ }
++ }
++ DBG(0, "%s: Done\n", __FUNCTION__);
++}
++
++static void __init karo_tx27_map_io(void)
++{
++ mxc_map_io();
++}
++
++static void __init karo_tx27_fixup(struct machine_desc *desc, struct tag *tags,
++ char **cmdline, struct meminfo *mi)
++{
++}
++
++static void __init karo_tx27_timer_init(void)
++{
++ DBG(0, "%s: \n", __FUNCTION__);
++ karo_tx27_clock_init();
++ mxc_timer_init("gpt_clk.0");
++ DBG(0, "%s: Done\n", __FUNCTION__);
++}
++
++struct sys_timer karo_tx27_timer = {
++ .init = karo_tx27_timer_init,
++};
++
++MACHINE_START(TX27, "Ka-Ro electronics TX27 module (Freescale i.MX27)")
++ /* Maintainer: <LW@KARO-electronics.de> */
++ .phys_io = AIPI_BASE_ADDR,
++ .io_pg_offst = ((unsigned long)AIPI_BASE_ADDR_VIRT >> 18) & 0xfffc,
++ .fixup = karo_tx27_fixup,
++ .map_io = karo_tx27_map_io,
++ .init_irq = mxc_init_irq,
++ .init_machine = karo_tx27_board_init,
++ .timer = &karo_tx27_timer,
++MACHINE_END
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/mach-mx2/karo.h linux-2.6.28-karo/arch/arm/mach-mx2/karo.h
+--- linux-2.6.28/arch/arm/mach-mx2/karo.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/mach-mx2/karo.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,45 @@
++#ifdef DEBUG
++#include <mach/iomux.h>
++
++#define SHOW_REG(reg) DBG(0, "%s[%08lx]=%08x\n", #reg, MXC_PHYS_ADDRESS(reg), __raw_readl(reg))
++#define SHOW_GPIO_REG(reg, port) \
++ DBG(0, "PT%c_%s[%08lx]=%08x\n", 'A' + port, #reg, \
++ GPIO_BASE_ADDR + MXC_##reg(port), __raw_readl(VA_GPIO_BASE + MXC_##reg(port)))
++
++static void dump_regs(void)
++{
++ int i;
++
++ SHOW_REG(CCM_CSCR);
++ SHOW_REG(CCM_MPCTL0);
++ SHOW_REG(CCM_MPCTL1);
++ SHOW_REG(CCM_SPCTL0);
++ SHOW_REG(CCM_SPCTL1);
++ SHOW_REG(CCM_OSC26MCTL);
++ SHOW_REG(CCM_PCDR0);
++ SHOW_REG(CCM_PCDR1);
++ SHOW_REG(CCM_PCCR0);
++ SHOW_REG(CCM_PCCR1);
++ SHOW_REG(CCM_CCSR);
++ SHOW_REG(CCM_PMCTL);
++ SHOW_REG(CCM_PMCOUNT);
++ SHOW_REG(CCM_WKGDCTL);
++ for (i = 0; i < 6; i++) {
++ SHOW_GPIO_REG(GIUS, i);
++ SHOW_GPIO_REG(DDIR, i);
++ SHOW_GPIO_REG(SSR, i);
++ SHOW_GPIO_REG(GPR, i);
++ SHOW_GPIO_REG(OCR1, i);
++ SHOW_GPIO_REG(OCR2, i);
++ SHOW_GPIO_REG(ICONFA1, i);
++ SHOW_GPIO_REG(ICONFA2, i);
++ SHOW_GPIO_REG(ICONFB1, i);
++ SHOW_GPIO_REG(ICONFB2, i);
++ }
++}
++#else
++static inline void dump_regs(void)
++{
++}
++#define SHOW_REG(reg) do {} while (0)
++#endif
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/mach-mx2/mxc_pm.c linux-2.6.28-karo/arch/arm/mach-mx2/mxc_pm.c
+--- linux-2.6.28/arch/arm/mach-mx2/mxc_pm.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/mach-mx2/mxc_pm.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,460 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @defgroup DPM_MX27 Power Management
++ * @ingroup MSL_MX27
++ */
++/*!
++ * @file mach-mx27/mxc_pm.c
++ *
++ * @brief This file contains the implementation of the Low-level power
++ * management driver. It modifies the registers of the PLL and clock module
++ * of the i.MX27.
++ *
++ * @ingroup DPM_MX27
++ */
++
++/*
++ * Include Files
++ */
++#include <linux/module.h>
++#include <linux/interrupt.h>
++#include <linux/delay.h>
++#include <asm/irq.h>
++#include <mach/mxc_pm.h>
++#include <mach/mxc.h>
++#include <mach/system.h>
++#include <asm/io.h>
++
++#include "crm_regs.h"
++
++/* Local defines */
++#define MAX_ARM_FREQ 400000000
++#define MAX_AHB_FREQ 133000000
++#define MAX_IPG_FREQ 66500000
++#define FREQ_COMP_TOLERANCE 100 /* tolerance percentage times 100 */
++#define MX27_LLPM_DEBUG 0
++
++/*
++ * Global variables
++ */
++#if 0
++/*!
++ * These variables hold the various clock values when the module is loaded.
++ * This is needed because these clocks are derived from MPLL and when MPLL
++ * output changes, these clocks need to be adjusted.
++ */
++static u32 perclk1, perclk2, perclk3, perclk4, nfcclk, cpuclk;
++
++/*!
++ * Compare two frequences using allowable tolerance
++ *
++ * The MX27 PLL can generate many frequencies. This function
++ * compares the generated frequency to the requested frequency
++ * and determines it they are within and acceptable tolerance.
++ *
++ * @param freq1 desired frequency
++ * @param freq2 generated frequency
++ *
++ * @return Returns 0 is frequencies are within talerance
++ * and non-zero is they are not.
++ */
++static s32 freq_equal(u32 freq1, u32 freq2)
++{
++ if (freq1 > freq2) {
++ return (freq1 - freq2) <= (freq1 / FREQ_COMP_TOLERANCE);
++ }
++ return (freq2 - freq1) <= (freq1 / FREQ_COMP_TOLERANCE);
++}
++
++/*!
++ * Select the PLL frequency based on the desired ARM frequency.
++ *
++ * The MPLL will be configured to output three frequencies, 400/333/266 MHz.
++ *
++ * @param armfreq Desired ARM frequency
++ *
++ * @return Returns one of the selected PLL frequency (400/333/266 MHz).
++ * Returns -1 on error.
++ *
++ */
++static s32 select_freq_pll(u32 armfreq)
++{
++ u32 div;
++
++ div = 266000000 / armfreq;
++ if ((div == 0) || (!freq_equal(armfreq, 266000000 / div))) {
++ div = 400000000 / armfreq;
++ if ((div == 0) || (!freq_equal(armfreq, 400000000 / div))) {
++ return -1;
++ }
++
++ return 400000000;
++ }
++
++ return 266000000;
++}
++
++/*!
++ * Check whether the desired ARM and AHB frequencies are valid.
++ *
++ * @param armfreq Desired ARM frequency
++ * @param ahbfreq Desired AHB frequency
++ *
++ * @return Returns 0 on success
++ * Return -1 on error
++ */
++static s32 mx27_pm_check_parameters(u32 armfreq, u32 ahbfreq)
++{
++ u32 ahbdiv;
++
++ /* No idea about minimum frequencies.. just a guess! */
++ if ((armfreq < 1000000) || (ahbfreq < 1000000)) {
++ printk("arm or ahb freq. too low\n");
++ return -1;
++ }
++
++ if ((armfreq > MAX_ARM_FREQ) || (ahbfreq > MAX_AHB_FREQ)) {
++ printk("arm or ahb freq. too high\n");
++ return -1;
++ }
++
++ /* AHB divider value is restricted to less than 8 */
++ ahbdiv = armfreq / ahbfreq;
++ if ((ahbdiv == 0) || (ahbdiv > 8)) {
++ printk("Invalid ahb frequency\n");
++ return -1;
++ }
++
++ return 0;
++}
++
++/*!
++ * Integer clock scaling
++ *
++ * Change the main ARM clock frequencies without changing the MPLL.
++ * The integer dividers (PRESC and BCLKDIV) are changed to obtain the
++ * desired frequency. Since NFC clock is derived from ARM frequency,
++ * NFCDIV is also adjusted.
++ *
++ * @param arm_freq Desired ARM frequency
++ * @param ahb_freq Desired AHB frequency
++ * @param pll_freq Current PLL frequency
++ *
++ * @return Returns 0
++ */
++static s32 mx27_pm_intscale(u32 arm_freq, u32 ahb_freq, s32 pll_freq)
++{
++ u32 pre_div, bclk_div, nfc_div;
++
++ /* Calculate ARM divider */
++ pre_div = pll_freq / arm_freq;
++ if (pre_div == 0)
++ pre_div = 1;
++
++ /* Calculate AHB divider */
++ bclk_div = arm_freq / ahb_freq;
++ if (bclk_div == 0)
++ bclk_div = 1;
++
++ if ((arm_freq / bclk_div) > ahb_freq)
++ bclk_div++;
++
++ /* NFC clock is dependent on ARM clock */
++ nfc_div = arm_freq / nfcclk;
++ if ((arm_freq / nfc_div) > nfcclk)
++ nfc_div++;
++
++ /* Adjust NFC divider */
++ mxc_set_clocks_div(NFC_CLK, nfc_div);
++
++#if MX27_LLPM_DEBUG
++ printk("DIVIDERS: PreDiv = %d BCLKDIV = %d \n", pre_div, bclk_div);
++ printk("Integer scaling\n");
++ printk("PLL = %d : ARM = %d: AHB = %d\n", pll_freq, arm_freq, ahb_freq);
++#endif
++
++ /*
++ * This part is tricky. What to adjust first (PRESC or BCLKDIV)?
++ * After trial and error, if current ARM frequency is greater than
++ * desired ARM frequency, then adjust PRESC first, else if current
++ * ARM frequency is less than desired ARM frequency, then adjust
++ * BCLKDIV first.
++ */
++ if (cpuclk > arm_freq) {
++ mxc_set_clocks_div(CPU_CLK, pre_div);
++ mxc_set_clocks_div(AHB_CLK, bclk_div);
++ } else {
++ mxc_set_clocks_div(AHB_CLK, bclk_div);
++ mxc_set_clocks_div(CPU_CLK, pre_div);
++ }
++
++ cpuclk = arm_freq;
++ mdelay(50);
++ return 0;
++}
++
++/*!
++ * Set dividers for various peripheral clocks.
++ *
++ * PERCLK1, PERCLK2, PERCLK3 and PERCLK4 are adjusted based on the MPLL
++ * output frequency.
++ *
++ * @param pll_freq Desired MPLL output frequency
++ */
++static void mx27_set_dividers(u32 pll_freq)
++{
++ s32 perdiv1, perdiv2, perdiv3, perdiv4;
++
++ perdiv1 = pll_freq / perclk1;
++ if ((pll_freq / perdiv1) > perclk1)
++ perdiv1++;
++
++ perdiv2 = pll_freq / perclk2;
++ if ((pll_freq / perdiv2) > perclk2)
++ perdiv2++;
++
++ perdiv3 = pll_freq / perclk3;
++ if ((pll_freq / perdiv3) > perclk3)
++ perdiv3++;
++
++ perdiv4 = pll_freq / perclk4;
++ if ((pll_freq / perdiv4) > perclk4)
++ perdiv4++;
++
++ mxc_set_clocks_div(PERCLK1, perdiv1);
++ mxc_set_clocks_div(PERCLK2, perdiv2);
++ mxc_set_clocks_div(PERCLK3, perdiv3);
++ mxc_set_clocks_div(PERCLK4, perdiv4);
++}
++
++/*!
++ * Change MPLL output frequency and adjust derived clocks to produce the
++ * desired frequencies.
++ *
++ * @param arm_freq Desired ARM frequency
++ * @param ahb_freq Desired AHB frequency
++ * @param org_pll Current PLL frequency
++ *
++ * @return Returns 0 on success
++ * Returns -1 on error
++ */
++static s32 mx27_pm_pllscale(u32 arm_freq, u32 ahb_freq, s32 org_pll)
++{
++ u32 mfi, mfn, mfd, pd = 1, cscr;
++ s32 pll_freq;
++
++ /* Obtain the PLL frequency for the desired ARM frequency */
++ pll_freq = select_freq_pll(arm_freq);
++ if (pll_freq == -1) {
++ return -1;
++ }
++
++ /* The MPCTL0 register values are programmed based on the oscillator */
++ cscr = __raw_readl(CCM_CSCR);
++ if ((cscr & CCM_CSCR_OSC26M) == 0) {
++ /* MPCTL0 register values are programmed for 400/266 MHz */
++ switch (pll_freq) {
++ case 400000000:
++ mfi = 7;
++ mfn = 9;
++ mfd = 12;
++ pd = 0;
++ break;
++
++ case 266000000:
++ mfi = 10;
++ mfn = 6;
++ mfd = 25;
++ break;
++
++ default:
++ return -1;
++ }
++ } else {
++ /* MPCTL0 register values are programmed for 400/266 MHz */
++ switch (pll_freq) {
++ case 400000000:
++ mfi = 12;
++ mfn = 2;
++ mfd = 3;
++ break;
++
++ case 266000000:
++ mfi = 8;
++ mfn = 10;
++ mfd = 31;
++ break;
++
++ default:
++ return -1;
++ }
++ }
++
++#if MX27_LLPM_DEBUG
++ printk("PLL scaling\n");
++ printk("PLL = %d : ARM = %d: AHB = %d\n", pll_freq, arm_freq, ahb_freq);
++#endif
++
++ /* Adjust the peripheral clock dividers for new PLL frequency */
++ mx27_set_dividers(pll_freq);
++
++ if (pll_freq > org_pll) {
++ /* Set the dividers first */
++ mx27_pm_intscale(arm_freq, ahb_freq, pll_freq);
++
++ /* Set the PLL */
++ mxc_pll_set(MCUPLL, mfi, pd, mfd, mfn);
++ mdelay(50);
++ } else {
++ /* Set the PLL first */
++ mxc_pll_set(MCUPLL, mfi, pd, mfd, mfn);
++ mdelay(50);
++
++ /* Set the dividers later */
++ mx27_pm_intscale(arm_freq, ahb_freq, pll_freq);
++ }
++
++ return 0;
++}
++#endif
++/*!
++ * Implement steps required to transition to low-power modes.
++ *
++ * @param mode The desired low-power mode. Possible values are,
++ * DOZE_MODE
++ * WAIT_MODE
++ * STOP_MODE
++ * DSM_MODE
++ */
++void mxc_pm_lowpower(s32 mode)
++{
++ u32 cscr;
++
++ local_irq_disable();
++
++ /* WAIT and DOZE execute WFI only */
++ switch (mode) {
++ case STOP_MODE:
++ case DSM_MODE:
++ /* Clear MPEN and SPEN to disable MPLL/SPLL */
++ cscr = __raw_readl(CCM_CSCR);
++ cscr &= 0xFFFFFFFC;
++ __raw_writel(cscr, CCM_CSCR);
++ break;
++ }
++
++ /* Executes WFI */
++ arch_idle();
++
++ local_irq_enable();
++}
++
++#if 0
++/*!
++ * Called to change the core frequency. This function internally decides
++ * whether to do integer scaling or pll scaling.
++ *
++ * @param arm_freq Desired ARM frequency
++ * @param ahb_freq Desired AHB frequency
++ * @param ipg_freq Desired IP frequency, constant AHB / 2 always.
++ *
++ * @return Returns 0 on success
++ * Returns -1 on error
++ */
++int mxc_pm_dvfs(unsigned long arm_freq, long ahb_freq, long ipg_freq)
++{
++ u32 divider;
++ s32 pll_freq, ret;
++ unsigned long flags;
++
++ if (mx27_pm_check_parameters(arm_freq, ahb_freq) != 0) {
++ return -1;
++ }
++
++ local_irq_save(flags);
++
++ /* Get the current PLL frequency */
++ pll_freq = mxc_pll_clock(MCUPLL);
++
++#if MX27_LLPM_DEBUG
++ printk("MCU PLL frequency is %d\n", pll_freq);
++#endif
++
++ /* Decide whether to do integer scaling or pll scaling */
++ if (arm_freq > pll_freq) {
++ /* Do PLL scaling */
++ ret = mx27_pm_pllscale(arm_freq, ahb_freq, pll_freq);
++ } else {
++ /* We need integer divider values */
++ divider = pll_freq / arm_freq;
++ if (!freq_equal(arm_freq, pll_freq / divider)) {
++ /* Do PLL scaling */
++ ret = mx27_pm_pllscale(arm_freq, ahb_freq, pll_freq);
++ } else {
++ /* Do integer scaling */
++ ret = mx27_pm_intscale(arm_freq, ahb_freq, pll_freq);
++ }
++ }
++
++ local_irq_restore(flags);
++ return ret;
++}
++#endif
++/*
++ * This API is not supported on i.MX27
++ */
++int mxc_pm_intscale(long armfreq, long ahbfreq, long ipfreq)
++{
++ return -MXC_PM_API_NOT_SUPPORTED;
++}
++
++/*
++ * This API is not supported on i.MX27
++ */
++int mxc_pm_pllscale(long armfreq, long ahbfreq, long ipfreq)
++{
++ return -MXC_PM_API_NOT_SUPPORTED;
++}
++
++/*!
++ * This function is used to load the module.
++ *
++ * @return Returns an Integer on success
++ */
++static int __init mxc_pm_init_module(void)
++{
++ printk(KERN_INFO "MX27: Power management module initialized\n");
++ return 0;
++}
++
++/*!
++ * This function is used to unload the module
++ */
++static void __exit mxc_pm_cleanup_module(void)
++{
++ printk(KERN_INFO "MX27: Power management module exit\n");
++}
++
++module_init(mxc_pm_init_module);
++module_exit(mxc_pm_cleanup_module);
++
++EXPORT_SYMBOL(mxc_pm_lowpower);
++//EXPORT_SYMBOL(mxc_pm_dvfs);
++EXPORT_SYMBOL(mxc_pm_pllscale);
++EXPORT_SYMBOL(mxc_pm_intscale);
++
++MODULE_AUTHOR("Freescale Semiconductor");
++MODULE_DESCRIPTION("i.MX27 low level PM driver");
++MODULE_LICENSE("GPL");
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/mach-mx2/pm.c linux-2.6.28-karo/arch/arm/mach-mx2/pm.c
+--- linux-2.6.28/arch/arm/mach-mx2/pm.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/mach-mx2/pm.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,88 @@
++/*
++ * linux/arch/arm/mach-mx27/pm.c
++ *
++ * MX27 Power Management Routines
++ *
++ * Original code for the SA11x0:
++ * Copyright (c) 2001 Cliff Brake <cbrake@accelent.com>
++ *
++ * Modified for the PXA250 by Nicolas Pitre:
++ * Copyright (c) 2002 Monta Vista Software, Inc.
++ *
++ * Modified for the OMAP1510 by David Singleton:
++ * Copyright (c) 2002 Monta Vista Software, Inc.
++ *
++ * Cleanup 2004 for OMAP1510/1610 by Dirk Behme <dirk.behme@de.bosch.com>
++ *
++ * Modified for the MX27
++ * Copyright 2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ *
++ * 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 SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED
++ * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
++ * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN
++ * NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
++ * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
++ * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
++ * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
++ * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
++ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
++ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
++ *
++ * 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.,
++ * 675 Mass Ave, Cambridge, MA 02139, USA.
++ */
++
++#include <linux/pm.h>
++#include <linux/sched.h>
++#include <linux/proc_fs.h>
++#include <linux/suspend.h>
++
++#include <asm/io.h>
++#include <mach/mxc_pm.h>
++
++/*
++ * TODO: whatta save?
++ */
++
++static int mx27_pm_enter(suspend_state_t state)
++{
++ pr_debug("%s: Entering state %d\n", __FUNCTION__, state);
++ switch (state) {
++ case PM_SUSPEND_MEM:
++ mxc_pm_lowpower(STOP_MODE);
++ break;
++
++ case PM_SUSPEND_STANDBY:
++ mxc_pm_lowpower(WAIT_MODE);
++ break;
++
++ case PM_SUSPEND_STOP:
++ mxc_pm_lowpower(DSM_MODE);
++ break;
++
++ default:
++ return -1;
++ }
++ return 0;
++}
++
++struct platform_suspend_ops mx27_pm_ops = {
++ .enter = mx27_pm_enter,
++ .valid = suspend_valid_only_mem,
++};
++
++static int __init mx27_pm_init(void)
++{
++ pr_debug("Power Management for Freescale MX27\n");
++ suspend_set_ops(&mx27_pm_ops);
++
++ return 0;
++}
++
++late_initcall(mx27_pm_init);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/mach-mx2/system.c linux-2.6.28-karo/arch/arm/mach-mx2/system.c
+--- linux-2.6.28/arch/arm/mach-mx2/system.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/mach-mx2/system.c 2009-03-11 13:31:09.000000000 +0100
+@@ -21,7 +21,9 @@
+
+ #include <linux/kernel.h>
+ #include <linux/clk.h>
++#include <linux/err.h>
+ #include <linux/io.h>
++#include <linux/init.h>
+
+ #include <mach/hardware.h>
+ #include <asm/proc-fns.h>
+@@ -43,21 +45,33 @@ void arch_idle(void)
+ #define WDOG_WCR_REG IO_ADDRESS(WDOG_BASE_ADDR)
+ #define WDOG_WCR_SRS (1 << 4)
+
++static struct clk *wdog_clk;
++
+ /*
+ * Reset the system. It is called by machine_restart().
+ */
+ void arch_reset(char mode)
+ {
+- struct clk *clk;
+-
+- clk = clk_get(NULL, "wdog_clk");
+- if (!clk) {
+- printk(KERN_ERR"Cannot activate the watchdog. Giving up\n");
+- return;
++ if (mode == 's') {
++ cpu_reset(0);
++ }
++ if (likely(wdog_clk != NULL)) {
++ clk_enable(wdog_clk);
+ }
+-
+- clk_enable(clk);
+
+ /* Assert SRS signal */
+ __raw_writew(__raw_readw(WDOG_WCR_REG) & ~WDOG_WCR_SRS, WDOG_WCR_REG);
+ }
++
++int __init mxc_wdog_init(void)
++{
++ wdog_clk = clk_get(NULL, "wdog_clk");
++ if (IS_ERR(wdog_clk)) {
++ int ret = PTR_ERR(wdog_clk);
++ printk(KERN_CRIT "Cannot get wdog clock; watchdog reset may not work\n");
++ wdog_clk = NULL;
++ return ret;
++ }
++ return 0;
++}
++postcore_initcall(mxc_wdog_init);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/mach-mx2/tx27_gpio.c linux-2.6.28-karo/arch/arm/mach-mx2/tx27_gpio.c
+--- linux-2.6.28/arch/arm/mach-mx2/tx27_gpio.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/mach-mx2/tx27_gpio.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,1019 @@
++/*
++ * arch/arm/mach-mx27/tx27_gpio.c
++ *
++ * Copyright (C) 2008 Lothar Wassmann <LW@KARO-electronics.de>
++ *
++ * 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 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301
++ */
++
++#include <linux/module.h>
++#include <linux/device.h>
++#include <linux/delay.h>
++#include <linux/err.h>
++#include <linux/clk.h>
++#include <mach/hardware.h>
++#include <mach/gpio.h>
++#include <mach/iomux.h>
++#include <mach/board-tx27.h>
++
++#include "crm_regs.h"
++
++#ifdef DEBUG
++extern int tx27_debug;
++#define dbg_lvl(n) ((n) < tx27_debug)
++#define DBG(lvl, fmt...) do { if (dbg_lvl(lvl)) printk(KERN_DEBUG fmt); } while (0)
++#else
++#define dbg_lvl(n) 0
++
++#define DBG(lvl, fmt...) do { } while (0)
++#endif
++
++#ifdef DEBUG
++#define try_request_mux(gmd) _try_request_mux(gmd, ARRAY_SIZE((gmd)), __FUNCTION__)
++static int _try_request_mux(const unsigned int *gmd, int num_gpios, const char *func)
++#else
++#define try_request_mux(gmd) _try_request_mux(gmd, ARRAY_SIZE((gmd)))
++static int _try_request_mux(const unsigned int *gmd, int num_gpios)
++#endif
++{
++ int i;
++ for (i = 0; i < num_gpios; i++) {
++ unsigned gpio = gmd[i] & (GPIO_PIN_MASK | GPIO_PORT_MASK);
++ int ret;
++
++ DBG(0, "Requesting GPIO: P%c%d (%s) for %s\n",
++ GPIO_PORT(gmd[i]) + 'A', GPIO_INDEX(gmd[i]),
++ MX27_PIN_NAME(gmd[i]), func);
++ ret = gpio_request(gpio, "");
++ if (ret != 0) {
++ while (--i >= 0) {
++ gpio_free(gmd[i] & (GPIO_PIN_MASK | GPIO_PORT_MASK));
++ }
++ return ret;
++ }
++ DBG(0, "%s: Configuring %s: %08x\n", __FUNCTION__,
++ MX27_PIN_NAME(gmd[i]), gmd[i]);
++ mxc_gpio_mode(gmd[i]);
++ }
++ return 0;
++}
++
++#ifdef DEBUG
++#define release_mux(gmd) _release_mux(gmd, ARRAY_SIZE((gmd)), __FUNCTION__)
++static void _release_mux(const unsigned int *gmd, int num_gpios,
++ const char *func)
++#else
++#define release_mux(gmd) _release_mux(gmd, ARRAY_SIZE((gmd)))
++static void _release_mux(const unsigned int *gmd, int num_gpios)
++#endif
++{
++ int i;
++
++ for (i = 0; i < num_gpios; i++) {
++ unsigned gpio = gmd[i] & (GPIO_PIN_MASK | GPIO_PORT_MASK);
++#ifdef DEBUG
++ DBG(0, "%s: Releasing GPIO port P%c%d (%s)\n", func,
++ GPIO_PORT(gpio) + 'A', GPIO_INDEX(gpio),
++ MX27_PIN_NAME(gmd[i]));
++#endif
++ gpio_free(gpio);
++ }
++}
++
++#define tx27_config_mux(__gmd, __puen) _tx27_config_mux(__gmd, ARRAY_SIZE((__gmd)), __puen)
++static void _tx27_config_mux(const unsigned int *gmd, int num_gpios, int puen)
++{
++ int i;
++
++ for (i = 0; i < num_gpios; i++) {
++ switch (puen) {
++ case 1:
++ DBG(0, "Enabling Pullup on %s\n",
++ MX27_PIN_NAME(gmd[i]));
++ mxc_gpio_mode(gmd[i] | GPIO_PUEN);
++ break;
++ case -1:
++ case 0:
++ DBG(0, "Disabling Pullup on %s\n",
++ MX27_PIN_NAME(gmd[i]));
++ mxc_gpio_mode(gmd[i] & ~GPIO_PUEN);
++ break;
++ }
++ }
++}
++
++#ifdef DEBUG
++#define dump_pins(__gmd) _dump_pins(__gmd, ARRAY_SIZE((__gmd)), __FUNCTION__)
++static void _dump_pins(const unsigned int *gmd, int num_gpios, const char *func)
++{
++ int i;
++
++ for (i = 0; i < num_gpios; i++) {
++ int gpio = IOMUX_TO_GPIO(gmd[i]);
++
++ DBG(0, "%s: %s (P%c%d) is: %d\n", func,
++ MX27_PIN_NAME(gmd[i]),
++ GPIO_PORT(gpio) + 'A', GPIO_INDEX(gpio),
++ gpio_get_value(gpio));
++ }
++}
++#else
++#define dump_pins(arg) do { } while (0)
++#endif
++
++/*
++ Setup GPIO for USB
++
++ PB22: 26MHz oscillator enable
++
++ PIN Configuration for USBOTG: High/Full speed OTG
++ PE2,PE1,PE0,PE24,PE25 -- PRIMARY
++ PC7 - PC13 -- PRIMARY
++ PB23,PB24 -- PRIMARY
++
++ PIN Configuration for USBH2: : High/Full/Low speed host
++ PA0 - PA4 -- PRIMARY
++ PD19, PD20,PD21,PD22,PD23,PD24,PD26 --Alternate (SECONDARY)
++
++ USBH1: Full/low speed host
++ not supported on TX27
++ */
++
++static unsigned int mx27_usbh2_gpios_active[] = {
++ PD19_AF_USBH2_DATA4,
++ PD20_AF_USBH2_DATA3,
++ PD21_AF_USBH2_DATA6,
++ PD22_AF_USBH2_DATA0,
++ PD23_AF_USBH2_DATA2,
++ PD24_AF_USBH2_DATA1,
++ PD26_AF_USBH2_DATA5,
++
++ PA0_PF_USBH2_CLK,
++ PA1_PF_USBH2_DIR,
++ PA2_PF_USBH2_DATA7,
++ PA3_PF_USBH2_NXT,
++ PA4_PF_USBH2_STP,
++
++ MXC_PIN(B, 31, AOUT, GPIO_IN), /* OC detect */
++};
++
++static unsigned int mx27_usbh2_gpios_inactive[] = {
++ MXC_PIN(D, 19, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(D, 20, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(D, 21, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(D, 22, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(D, 23, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(D, 24, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(D, 26, AOUT, GPIO_IN | GPIO_PUEN),
++
++ MXC_PIN(A, 0, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(A, 1, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(A, 2, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(A, 3, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(A, 4, AOUT, GPIO_IN | GPIO_PUEN),
++
++ MXC_PIN(B, 31, AOUT, GPIO_IN), /* OC detect */
++};
++
++/*
++ * conflicts with CSPI1 and CSPI2
++ */
++static struct clk *usbh2_clk;
++
++int gpio_usbh2_active(void)
++{
++ int ret;
++
++ DBG(0, "%s: \n", __FUNCTION__);
++ usbh2_clk = clk_get(NULL, "clk_26m");
++ if (IS_ERR(usbh2_clk)) {
++ ret = PTR_ERR(usbh2_clk);
++ printk(KERN_ERR "Cannot request 26MHz clock: %d\n", ret);
++ clk_put(usbh2_clk);
++ usbh2_clk = NULL;
++ return ret;
++ } else {
++ clk_enable(usbh2_clk);
++ }
++
++ ret = try_request_mux(mx27_usbh2_gpios_active);
++ return ret;
++}
++
++void gpio_usbh2_inactive(void)
++{
++ int i;
++
++ DBG(0, "%s: \n", __FUNCTION__);
++ if (usbh2_clk != NULL) {
++ clk_disable(usbh2_clk);
++ clk_put(usbh2_clk);
++ usbh2_clk = NULL;
++ }
++
++ for (i = ARRAY_SIZE(mx27_usbh2_gpios_inactive) - 1; i >= 0; i--) {
++ DBG(0, "Enabling Pullup on %s\n",
++ MX27_PIN_NAME(mx27_usbh2_gpios_inactive[i]));
++ mxc_gpio_mode(mx27_usbh2_gpios_inactive[i]);
++ }
++ release_mux(mx27_usbh2_gpios_active);
++}
++
++static unsigned int mx27_usbotg_hs_gpios_active[] = {
++ PC7_PF_USBOTG_DATA5,
++ PC8_PF_USBOTG_DATA6,
++ PC9_PF_USBOTG_DATA0,
++ PC10_PF_USBOTG_DATA2,
++ PC11_PF_USBOTG_DATA1,
++ PC13_PF_USBOTG_DATA3,
++ PC12_PF_USBOTG_DATA4,
++
++ PE24_PF_USBOTG_CLK,
++ PE2_PF_USBOTG_DIR,
++ PE25_PF_USBOTG_DATA7,
++ PE0_PF_USBOTG_NXT,
++ PE1_PF_USBOTG_STP,
++
++ MXC_PIN(B, 29, AOUT, GPIO_IN), /* OC detect */
++};
++
++static unsigned int mx27_usbotg_hs_gpios_inactive[] = {
++ MXC_PIN(C, 7, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(C, 8, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(C, 9, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(C, 10, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(C, 11, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(C, 13, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(C, 12, AOUT, GPIO_IN | GPIO_PUEN),
++
++ MXC_PIN(E, 24, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(E, 2, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(E, 25, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(E, 0, AOUT, GPIO_IN | GPIO_PUEN),
++ MXC_PIN(E, 1, AOUT, GPIO_IN | GPIO_PUEN),
++
++ MXC_PIN(B, 29, AOUT, GPIO_IN), /* OC detect */
++};
++
++static struct clk *usbotg_clk;
++
++int gpio_usbotg_hs_active(void)
++{
++ int ret;
++
++ usbotg_clk = clk_get(NULL, "clk_26m");
++ if (IS_ERR(usbotg_clk)) {
++ ret = PTR_ERR(usbotg_clk);
++ printk(KERN_ERR "Cannot request 26MHz clock: %d\n", ret);
++ clk_put(usbotg_clk);
++ usbotg_clk = NULL;
++ return ret;
++ } else {
++ clk_enable(usbotg_clk);
++ }
++
++ ret = try_request_mux(mx27_usbotg_hs_gpios_active);
++ return ret;
++}
++
++void gpio_usbotg_hs_inactive(void)
++{
++ int i;
++
++ if (usbotg_clk != NULL) {
++ clk_disable(usbotg_clk);
++ clk_put(usbotg_clk);
++ usbotg_clk = NULL;
++ }
++
++ for (i = 0; i < ARRAY_SIZE(mx27_usbotg_hs_gpios_inactive); i++) {
++ DBG(0, "Enabling Pullup on %s\n",
++ MX27_PIN_NAME(mx27_usbotg_hs_gpios_inactive[i]));
++ mxc_gpio_mode(mx27_usbotg_hs_gpios_inactive[i]);
++ }
++ release_mux(mx27_usbotg_hs_gpios_active);
++}
++
++int gpio_usbotg_fs_active(void)
++{
++ return gpio_usbotg_hs_active();
++}
++
++void gpio_usbotg_fs_inactive(void)
++{
++ gpio_usbotg_hs_inactive();
++}
++
++/************************************************************************/
++/* for i2c gpio */
++/* I2C1: PD17,PD18 -- Primary */
++/* I2C2: PC5,PC6 -- Primary */
++/************************************************************************/
++/*
++ * Setup GPIO for an I2C device to be active
++ */
++static unsigned int mx27_i2c_gpios[][2] = {
++ {
++ PD18_PF_I2C_CLK,
++ PD17_PF_I2C_DATA,
++ },{
++ PC6_PF_I2C2_SCL,
++ PC5_PF_I2C2_SDA,
++ }
++};
++
++int gpio_i2c_active(int i2c_num)
++{
++ if (i2c_num < 0 || i2c_num >= ARRAY_SIZE(mx27_i2c_gpios)) {
++ printk(KERN_ERR "%s: no compatible I2C adapter\n", __FUNCTION__);
++ return -EINVAL;
++ }
++ return try_request_mux(mx27_i2c_gpios[i2c_num]);
++}
++
++/*
++ * Setup GPIO for an I2C device to be inactive
++ */
++int gpio_i2c_inactive(int i2c_num)
++{
++ if (i2c_num < 0 || i2c_num >= ARRAY_SIZE(mx27_i2c_gpios)) {
++ return -EINVAL;
++ }
++ release_mux(mx27_i2c_gpios[i2c_num]);
++ return 0;
++}
++
++/*
++ * Setup GPIO for a CSPI device to be active
++ */
++static unsigned int mx27_cspi1_gpios[] = {
++ PD31_PF_CSPI1_MOSI,
++ PD30_PF_CSPI1_MISO,
++ PD29_PF_CSPI1_SCLK,
++ PD25_PF_CSPI1_RDY,
++ PD28_PF_CSPI1_SS0,
++ PD27_PF_CSPI1_SS1,
++ //PD26_PF_CSPI1_SS2, /* already in use by the USB controller */
++};
++
++static unsigned int mx27_cspi2_gpios[] = {
++ PD24_PF_CSPI2_MOSI,
++ PD23_PF_CSPI2_MISO,
++ PD22_PF_CSPI2_SCLK,
++ PD21_PF_CSPI2_SS0,
++ PD20_PF_CSPI2_SS1,
++ PD19_PF_CSPI2_SS2,
++};
++
++static unsigned int mx27_cspi3_gpios[] = {
++ PE18_AF_CSPI3_MISO,
++ PE22_AF_CSPI3_MOSI,
++ PE23_AF_CSPI3_SCLK,
++ PE21_AF_CSPI3_SS,
++};
++
++int gpio_spi_active(int cspi_mod)
++{
++
++ switch (cspi_mod) {
++ case 0:
++ /* SPI1 */
++ return try_request_mux(mx27_cspi1_gpios);
++ case 1:
++ /*SPI2 */
++ return try_request_mux(mx27_cspi2_gpios);
++ case 2:
++ /*SPI3 */
++ return try_request_mux(mx27_cspi3_gpios);
++ default:
++ return -EINVAL;
++ }
++}
++
++/*
++ * Setup GPIO for a CSPI device to be inactive
++ */
++int gpio_spi_inactive(int cspi_mod)
++{
++ switch (cspi_mod) {
++ case 0:
++ /* SPI1 */
++ release_mux(mx27_cspi1_gpios);
++ break;
++ case 1:
++ /*SPI2 */
++ release_mux(mx27_cspi2_gpios);
++ break;
++ case 2:
++ /*SPI3 */
++ release_mux(mx27_cspi3_gpios);
++ break;
++
++ default:
++ return -EINVAL;
++ }
++ return 0;
++}
++
++/*
++ * Setup GPIO for a nand flash device to be active
++ *
++ */
++static unsigned int mx27_nand_gpios[] = {
++ PF0_PF_NFRB,
++ PF3_PF_NFCE_B,
++ PF2_PF_NFWP_B,
++ PF1_PF_NFCLE,
++ PF4_PF_NFALE,
++ PF5_PF_NFRE_B,
++ PF6_PF_NFWE_B,
++};
++
++int gpio_nand_active(void)
++{
++ return try_request_mux(mx27_nand_gpios);
++}
++
++/*
++ * Setup GPIO for a nand flash device to be inactive
++ *
++ */
++void gpio_nand_inactive(void)
++{
++ release_mux(mx27_nand_gpios);
++}
++
++/*
++ * Setup GPIO for CSI device to be active
++ *
++ */
++static unsigned int mx27_csi_gpios[] = {
++ PB10_PF_CSI_D0,
++ PB11_PF_CSI_D1,
++ PB12_PF_CSI_D2,
++ PB13_PF_CSI_D3,
++ PB14_PF_CSI_D4,
++ PB15_PF_CSI_MCLK,
++ PB16_PF_CSI_PIXCLK,
++ PB17_PF_CSI_D5,
++ PB18_PF_CSI_D6,
++ PB19_PF_CSI_D7,
++ PB20_PF_CSI_VSYNC,
++ PB21_PF_CSI_HSYNC,
++};
++
++int gpio_sensor_active(void)
++{
++ return try_request_mux(mx27_csi_gpios);
++}
++
++void gpio_sensor_inactive(void)
++{
++ release_mux(mx27_csi_gpios);
++}
++
++#if 0
++/*
++ * Setup GPIO for LCDC device to be active
++ *
++ */
++static unsigned int mx27_lcdc_gpios[] = {
++ MXC_PIN(A, 30, GPIO, GPIO_OUT | GPIO_DFLT_LOW), /* PA30 */
++ MXC_PIN(A, 25, GPIO, GPIO_OUT | GPIO_DFLT_LOW), /* PA25 */
++ MXC_PIN(A, 26, GPIO, GPIO_OUT | GPIO_DFLT_LOW), /* PA26 */
++ MXC_PIN(A, 24, GPIO, GPIO_OUT | GPIO_DFLT_LOW), /* PA24 */
++ MXC_PIN(A, 27, GPIO, GPIO_OUT | GPIO_DFLT_LOW), /* PA27 */
++ PA5_PF_LSCLK,
++ PA6_PF_LD0,
++ PA7_PF_LD1,
++ PA8_PF_LD2,
++ PA9_PF_LD3,
++ PA10_PF_LD4,
++ PA11_PF_LD5,
++ PA12_PF_LD6,
++ PA13_PF_LD7,
++ PA14_PF_LD8,
++ PA15_PF_LD9,
++ PA16_PF_LD10,
++ PA17_PF_LD11,
++ PA18_PF_LD12,
++ PA19_PF_LD13,
++ PA20_PF_LD14,
++ PA21_PF_LD15,
++ PA22_PF_LD16,
++ PA23_PF_LD17,
++ PA28_PF_HSYNC,
++ PA29_PF_VSYNC,
++ PA31_PF_OE_ACD,
++};
++
++int tx27_gpio_lcdc_active(void)
++{
++ int ret;
++ ret = try_request_mux(mx27_lcdc_gpios);
++ if (ret) {
++ return ret;
++ }
++ return 0;
++}
++
++/*
++ * Setup GPIO for LCDC device to be inactive
++ *
++ */
++int tx27_gpio_lcdc_inactive(void)
++{
++ release_mux(mx27_lcdc_gpios);
++ return 0;
++}
++#endif
++
++/*
++ * GPIO settings not required for keypad
++ *
++ */
++int gpio_keypad_active(void)
++{
++ return 0;
++}
++
++/*
++ * GPIO settings not required for keypad
++ *
++ */
++void gpio_keypad_inactive(void)
++{
++}
++
++/*
++ * Setup GPIO for ATA device to be active
++ *
++ */
++static unsigned int mx27_ata_gpios[] = {
++ PD2_PF_ATA_DATA0,
++ PD3_PF_ATA_DATA1,
++ PD4_PF_ATA_DATA2,
++ PD5_PF_ATA_DATA3,
++ PD6_PF_ATA_DATA4,
++ PD7_PF_ATA_DATA5,
++ PD8_PF_ATA_DATA6,
++ PD9_PF_ATA_DATA7,
++ PD10_PF_ATA_DATA8,
++ PD11_PF_ATA_DATA9,
++ PD12_PF_ATA_DATA10,
++ PD13_PF_ATA_DATA11,
++ PD14_PF_ATA_DATA12,
++ PD15_PF_ATA_DATA13,
++ PD16_PF_ATA_DATA14,
++ PF23_PF_ATA_DATA15,
++
++ PF20_AF_PC_CD1_B,
++ PF19_AF_PC_CD2_B,
++ PF18_AF_PC_WAIT_B,
++ PF17_AF_PC_READY,
++ PF16_AF_PC_PWRON,
++ PF14_AF_PC_VS1,
++ PF13_AF_PC_VS2,
++ PF12_AF_PC_BVD1,
++ PF11_AF_PC_BVD2,
++ PF10_AF_PC_RST,
++ PF9_AF_IOIS16,
++ PF8_AF_PC_RW_B,
++ PF7_AF_PC_POE,
++};
++
++int gpio_ata_active(void)
++{
++ return try_request_mux(mx27_ata_gpios);
++}
++
++/*
++ * Setup GPIO for ATA device to be inactive
++ *
++ */
++void gpio_ata_inactive(void)
++{
++ release_mux(mx27_ata_gpios);
++}
++
++/*
++ * Setup GPIO for FEC device to be active
++ *
++ */
++static unsigned int mx27_fec_gpios_off[] = {
++ /* configure the PHY strap pins to the correct values */
++ MXC_PIN(D, 0, GPIO, GPIO_OUT | GPIO_DFLT_LOW),
++ MXC_PIN(D, 1, GPIO, GPIO_OUT | GPIO_DFLT_LOW),
++ MXC_PIN(D, 2, GPIO, GPIO_OUT | GPIO_DFLT_LOW),
++ MXC_PIN(D, 3, GPIO, GPIO_OUT | GPIO_DFLT_LOW),
++ MXC_PIN(D, 4, GPIO, GPIO_OUT | GPIO_DFLT_LOW),
++ MXC_PIN(D, 5, GPIO, GPIO_OUT | GPIO_DFLT_HIGH),
++ MXC_PIN(D, 6, GPIO, GPIO_OUT | GPIO_DFLT_HIGH),
++ MXC_PIN(D, 7, GPIO, GPIO_OUT | GPIO_DFLT_LOW),
++ MXC_PIN(D, 8, GPIO, GPIO_OUT | GPIO_DFLT_LOW),
++ MXC_PIN(D, 9, GPIO, GPIO_OUT | GPIO_DFLT_LOW),
++ MXC_PIN(D, 10, GPIO, GPIO_OUT | GPIO_DFLT_LOW),
++ MXC_PIN(D, 11, GPIO, GPIO_OUT | GPIO_DFLT_LOW),
++ MXC_PIN(D, 12, GPIO, GPIO_OUT | GPIO_DFLT_HIGH),
++ MXC_PIN(D, 13, GPIO, GPIO_OUT | GPIO_DFLT_LOW),
++ MXC_PIN(D, 14, GPIO, GPIO_OUT | GPIO_DFLT_LOW),
++ MXC_PIN(D, 15, GPIO, GPIO_OUT | GPIO_DFLT_LOW),
++};
++
++static unsigned int mx27_fec_pwr_gpios[] = {
++ MXC_PIN(B, 27, GPIO, GPIO_OUT), /* FEC PHY power on pin */
++ MXC_PIN(B, 30, GPIO, GPIO_OUT), /* keep FEC reset in original state */
++};
++
++static unsigned int mx27_fec_gpios_on[] = {
++ PD0_AIN_FEC_TXD0,
++ PD1_AIN_FEC_TXD1,
++ PD2_AIN_FEC_TXD2,
++ PD3_AIN_FEC_TXD3,
++ PD4_AOUT_FEC_RX_ER,
++ PD5_AOUT_FEC_RXD1,
++ PD6_AOUT_FEC_RXD2,
++ PD7_AOUT_FEC_RXD3,
++ PD8_AF_FEC_MDIO,
++ PD9_AIN_FEC_MDC,
++ PD10_AOUT_FEC_CRS,
++ PD11_AOUT_FEC_TX_CLK,
++ PD12_AOUT_FEC_RXD0,
++ PD13_AOUT_FEC_RX_DV,
++ PD14_AOUT_FEC_RX_CLK,
++ PD15_AOUT_FEC_COL,
++#ifndef FEC_MII_IRQ
++ PD16_AIN_FEC_TX_ER, /* TX_ER */
++#else
++ MXC_PIN(D, 16, AOUT, GPIO_IN), /* #INT */
++#endif
++ PF23_AIN_FEC_TX_EN,
++};
++
++#define TX27_FEC_PWR_GPIO (GPIO_PORTB | 27)
++#define TX27_FEC_RST_GPIO (GPIO_PORTB | 30)
++
++int gpio_fec_active(void)
++{
++ int ret;
++
++ ret = try_request_mux(mx27_fec_pwr_gpios);
++ if (ret) {
++ return ret;
++ }
++ /*
++ * If the PHY is already powered on, assume it has been
++ * correctly configured (by the boot loader)
++ */
++ if (0 && gpio_get_value(TX27_FEC_PWR_GPIO) &&
++ gpio_get_value(TX27_FEC_RST_GPIO)) {
++ ret = try_request_mux(mx27_fec_gpios_on);
++ if (ret) {
++ release_mux(mx27_fec_pwr_gpios);
++ return ret;
++ }
++ } else {
++ DBG(0, "%s: Switching FEC PHY power on\n", __FUNCTION__);
++ gpio_set_value(TX27_FEC_PWR_GPIO, 1);
++ DBG(0, "%s: Asserting FEC PHY reset\n", __FUNCTION__);
++ gpio_set_value(TX27_FEC_RST_GPIO, 0);
++ dump_pins(mx27_fec_pwr_gpios);
++ dump_pins(mx27_fec_gpios_on);
++ /* switch PHY strap pins into required state */
++ ret = try_request_mux(mx27_fec_gpios_off);
++ if (ret) {
++ release_mux(mx27_fec_pwr_gpios);
++ return ret;
++ }
++ DBG(0, "%s: Delaying for 22ms\n", __FUNCTION__);
++ mdelay(22);
++ dump_pins(mx27_fec_pwr_gpios);
++ dump_pins(mx27_fec_gpios_on);
++ DBG(0, "%s: Deasserting FEC PHY reset\n", __FUNCTION__);
++ gpio_set_value(TX27_FEC_RST_GPIO, 1);
++ tx27_config_mux(mx27_fec_gpios_on, -1);
++ }
++ return 0;
++}
++
++/*
++ * Setup GPIO for FEC device to be inactive
++ *
++ */
++void gpio_fec_inactive(void)
++{
++ tx27_config_mux(mx27_fec_gpios_off, -1);
++ DBG(0, "%s: Asserting FEC PHY reset\n", __FUNCTION__);
++ gpio_set_value(TX27_FEC_RST_GPIO, 0);
++ DBG(0, "%s: Switching FEC PHY power off\n", __FUNCTION__);
++ gpio_set_value(TX27_FEC_PWR_GPIO, 0);
++ release_mux(mx27_fec_gpios_off);
++ release_mux(mx27_fec_pwr_gpios);
++}
++
++/*
++ * Setup GPIO for SLCDC device to be active
++ *
++ */
++static unsigned int mx27_slcdc0_gpios[] = {
++ PC31_AIN_SLCDC2_CLK, /* CLK */
++ PC30_AIN_SLCDC2_CS, /* CS */
++ PC29_AIN_SLCDC2_RS, /* RS */
++ PC28_AIN_SLCDC2_D0, /* D0 */
++};
++
++static unsigned int mx27_slcdc1_gpios[] = {
++ PB5_AIN_SLCDC1_CLK, /* CLK */
++ PB6_AIN_SLCDC1_D0, /* D0 */
++ PB7_AIN_SLCDC1_RS, /* RS */
++ PB8_AIN_SLCDC1_CS, /* CS */
++};
++
++static unsigned int mx27_slcdc2_gpios[] = {
++ MXC_PIN(A, 6, GPIO, GPIO_OUT),
++ MXC_PIN(A, 7, GPIO, GPIO_OUT),
++ MXC_PIN(A, 8, GPIO, GPIO_OUT),
++ MXC_PIN(A, 9, GPIO, GPIO_OUT),
++ MXC_PIN(A, 10, GPIO, GPIO_OUT),
++ MXC_PIN(A, 11, GPIO, GPIO_OUT),
++ MXC_PIN(A, 12, GPIO, GPIO_OUT),
++ MXC_PIN(A, 13, GPIO, GPIO_OUT),
++ MXC_PIN(A, 14, GPIO, GPIO_OUT),
++ MXC_PIN(A, 15, GPIO, GPIO_OUT),
++ MXC_PIN(A, 16, GPIO, GPIO_OUT),
++ MXC_PIN(A, 17, GPIO, GPIO_OUT),
++ MXC_PIN(A, 18, GPIO, GPIO_OUT),
++ MXC_PIN(A, 19, GPIO, GPIO_OUT),
++ MXC_PIN(A, 20, GPIO, GPIO_OUT),
++ MXC_PIN(A, 21, GPIO, GPIO_OUT),
++};
++
++int gpio_slcdc_active(int type)
++{
++ switch (type) {
++ case 0:
++ return try_request_mux(mx27_slcdc0_gpios);
++ case 1:
++ return try_request_mux(mx27_slcdc1_gpios);
++ case 2:
++ return try_request_mux(mx27_slcdc2_gpios);
++ }
++ return -EINVAL;
++}
++
++/*
++ * Setup GPIO for SLCDC device to be inactive
++ *
++ */
++int gpio_slcdc_inactive(int type)
++{
++ switch (type) {
++ case 0:
++ release_mux(mx27_slcdc0_gpios);
++ break;
++ case 1:
++ release_mux(mx27_slcdc1_gpios);
++ break;
++ case 2:
++ release_mux(mx27_slcdc2_gpios);
++ break;
++ default:
++ return -EINVAL;
++ }
++ return 0;
++}
++
++static unsigned int mx27_ssi_gpios[][4] = {
++ {
++ PC28_PF_SSI3_FS,
++ PC29_PF_SSI3_RXD,
++ PC30_PF_SSI3_TXD,
++ PC31_PF_SSI3_CLK,
++ },
++ {
++ PC16_PF_SSI4_FS,
++ PC17_PF_SSI4_RXD,
++ PC18_PF_SSI4_TXD,
++ PC19_PF_SSI4_CLK,
++ },
++};
++
++int gpio_ssi_active(int ssi_num)
++{
++ int ret;
++ switch (ssi_num) {
++ case 0:
++ case 1:
++ ret = try_request_mux(mx27_ssi_gpios[ssi_num]);
++ break;
++ default:
++ return -EINVAL;
++ }
++ return ret;
++}
++
++/*
++ * Setup GPIO for a SSI port to be inactive
++ */
++int gpio_ssi_inactive(int ssi_num)
++{
++ if (ssi_num < 0 || ssi_num >= ARRAY_SIZE(mx27_ssi_gpios)) {
++ return -EINVAL;
++ }
++ release_mux(mx27_ssi_gpios[ssi_num]);
++ return 0;
++}
++
++static unsigned int mx27_ac97_gpios_off[] = {
++ MXC_PIN(E, 17, GPIO, GPIO_OUT | GPIO_DFLT_LOW), /* RESET_OUT_B */
++ PC31_PF_SSI3_CLK, /* AC97_BITCLK */
++ PC29_PF_SSI3_RXD, /* AC97_SDATAIN */
++ MXC_PIN(C, 23, AOUT, GPIO_IN), /* UCB1400 interrupt */
++ /* keep SDATA_OUT and SYNC deasserted to prevent UCB1400 from entering test mode */
++ MXC_PIN(C, 30, GPIO, GPIO_OUT | GPIO_DFLT_LOW), /* AC97_SDATAOUT */
++ MXC_PIN(C, 28, GPIO, GPIO_OUT | GPIO_DFLT_LOW), /* AC97_SYNC */
++};
++
++static unsigned int mx27_ac97_gpios_on[] = {
++ MXC_PIN(E, 17, GPIO, GPIO_OUT | GPIO_DFLT_HIGH), /* RESET_OUT_B */
++ PC30_PF_SSI3_TXD, /* AC97_SDATAOUT */
++ PC28_PF_SSI3_FS, /* AC97_SYNC */
++};
++
++int gpio_ac97_active(void)
++{
++ int ret;
++
++ ret = try_request_mux(mx27_ac97_gpios_off);
++ if (ret == 0) {
++ udelay(1);
++ dump_pins(mx27_ac97_gpios_off);
++ /* deassert UCB1400 reset and reconfigure SDATA_OUT and SYNC */
++ tx27_config_mux(mx27_ac97_gpios_on, -1);
++ }
++ return ret;
++}
++
++void gpio_ac97_inactive(void)
++{
++ DBG(0, "%s: Releasing AC97 GPIOS\n", __FUNCTION__);
++ release_mux(mx27_ac97_gpios_off);
++ DBG(0, "%s: Done\n", __FUNCTION__);
++}
++
++/*
++ * Setup GPIO for SDHC to be active
++ */
++static unsigned int mx27_sdhc_gpios[][6] = {
++ {
++ PE23_PF_SD1_CLK,
++ PE22_PF_SD1_CMD,
++ PE18_PF_SD1_D0,
++ PE19_PF_SD1_D1,
++ PE20_PF_SD1_D2,
++ PE21_PF_SD1_D3,
++ },
++ {
++ PB9_PF_SD2_CLK,
++ PB8_PF_SD2_CMD,
++ PB4_PF_SD2_D0,
++ PB5_PF_SD2_D1,
++ PB6_PF_SD2_D2,
++ PB7_PF_SD2_D3,
++ },
++ {
++ PD1_PF_SD3_CLK,
++ PD0_PF_SD3_CMD,
++ PD2_AF_SD3_D0,
++ PD3_AF_SD3_D1,
++ PD4_AF_SD3_D2,
++ PD5_AF_SD3_D3,
++ },
++};
++
++int gpio_sdhc_active(int module)
++{
++ int ret;
++ u16 data;
++ switch (module) {
++ case 0:
++ ret = try_request_mux(mx27_sdhc_gpios[module]);
++ if (ret == 0) {
++ /* 22k pull up for sd1 dat3 pins */
++ data = __raw_readw(IO_ADDRESS(SYSCTRL_BASE_ADDR + 0x54));
++ data |= 0x0c;
++ __raw_writew(data, IO_ADDRESS(SYSCTRL_BASE_ADDR + 0x54));
++ }
++ break;
++ case 1:
++ ret = try_request_mux(mx27_sdhc_gpios[module]);
++ if (ret == 0) {
++ /* 22k pull up for sd2 pins */
++ data = __raw_readw(IO_ADDRESS(SYSCTRL_BASE_ADDR + 0x54));
++ data |= 0xfff0;
++ __raw_writew(data, IO_ADDRESS(SYSCTRL_BASE_ADDR + 0x54));
++ }
++ break;
++ case 2:
++ ret = try_request_mux(mx27_sdhc_gpios[module]);
++ break;
++ default:
++ return -EINVAL;
++ }
++ return ret;
++}
++
++/*
++ * Setup GPIO for SDHC to be inactive
++ */
++int gpio_sdhc_inactive(int module)
++{
++ switch (module) {
++ case 0:
++ case 1:
++ case 2:
++ release_mux(mx27_sdhc_gpios[module]);
++ break;
++ default:
++ return -EINVAL;
++ }
++ return 0;
++}
++
++int gpio_owire_active(void)
++{
++ int ret;
++
++ ret = gpio_request(IOMUX_TO_GPIO(PE16_AF_OWIRE), "1Wire");
++ if (ret == 0) {
++ mxc_gpio_mode(PE16_AF_OWIRE);
++ }
++ return ret;
++}
++
++void gpio_owire_inactive(void)
++{
++ gpio_free(IOMUX_TO_GPIO(PE16_AF_OWIRE));
++}
++
++#define TX27_EXT_WAKEUP_GPIO (GPIO_PORTB | 24)
++
++#if 0
++static int tx27_gpio_init(void)
++{
++ int ret;
++
++ DBG(0, "%s: \n", __FUNCTION__);
++
++ /* USB_OC_B => EXT_WAKEUP */
++ ret = gpio_request(TX27_EXT_WAKEUP_GPIO, "EXT Wakeup");
++ if (ret) {
++ printk(KERN_WARNING "%s: Failed to request %s\n", __FUNCTION__,
++ MX27_PIN_NAME(TX27_EXT_WAKEUP_GPIO));
++ return ret;
++ }
++ mxc_gpio_mode(MXC_PIN(B, 24, AOUT, GPIO_IN));
++ return ret;
++}
++arch_initcall(tx27_gpio_init);
++#endif
++
++EXPORT_SYMBOL(gpio_usbh2_active);
++EXPORT_SYMBOL(gpio_usbh2_inactive);
++EXPORT_SYMBOL(gpio_usbotg_hs_active);
++EXPORT_SYMBOL(gpio_usbotg_hs_inactive);
++EXPORT_SYMBOL(gpio_usbotg_fs_active);
++EXPORT_SYMBOL(gpio_usbotg_fs_inactive);
++EXPORT_SYMBOL(gpio_i2c_active);
++EXPORT_SYMBOL(gpio_i2c_inactive);
++EXPORT_SYMBOL(gpio_spi_active);
++EXPORT_SYMBOL(gpio_spi_inactive);
++EXPORT_SYMBOL(gpio_nand_active);
++EXPORT_SYMBOL(gpio_nand_inactive);
++EXPORT_SYMBOL(gpio_sensor_active);
++EXPORT_SYMBOL(gpio_sensor_inactive);
++EXPORT_SYMBOL(gpio_keypad_active);
++EXPORT_SYMBOL(gpio_keypad_inactive);
++EXPORT_SYMBOL(gpio_ata_active);
++EXPORT_SYMBOL(gpio_ata_inactive);
++EXPORT_SYMBOL(gpio_fec_active);
++EXPORT_SYMBOL(gpio_fec_inactive);
++EXPORT_SYMBOL(gpio_slcdc_active);
++EXPORT_SYMBOL(gpio_slcdc_inactive);
++EXPORT_SYMBOL(gpio_ssi_active);
++EXPORT_SYMBOL(gpio_ssi_inactive);
++EXPORT_SYMBOL(gpio_sdhc_active);
++EXPORT_SYMBOL(gpio_sdhc_inactive);
++EXPORT_SYMBOL(gpio_owire_active);
++EXPORT_SYMBOL(gpio_owire_inactive);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/Kconfig linux-2.6.28-karo/arch/arm/plat-mxc/Kconfig
+--- linux-2.6.28/arch/arm/plat-mxc/Kconfig 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/Kconfig 2009-03-11 13:16:24.000000000 +0100
+@@ -8,6 +8,7 @@ choice
+
+ config ARCH_MX2
+ bool "MX2-based"
++ select MXC_EMMA
+ help
+ This enables support for systems based on the Freescale i.MX2 family
+
+@@ -23,6 +24,14 @@ source "arch/arm/mach-mx3/Kconfig"
+
+ endmenu
+
++config MXC_EMMA
++ bool
++ depends on ARCH_MXC
++
++config MXC_FB_IRAM
++ bool
++ depends on ARCH_MXC
++
+ config MXC_IRQ_PRIOR
+ bool "Use IRQ priority"
+ depends on ARCH_MXC
+@@ -34,4 +43,6 @@ config MXC_IRQ_PRIOR
+ requirements for timing.
+ Say N here, unless you have a specialized requirement.
+
++config MXC_ULPI
++ bool
+ endif
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/Makefile linux-2.6.28-karo/arch/arm/plat-mxc/Makefile
+--- linux-2.6.28/arch/arm/plat-mxc/Makefile 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/Makefile 2009-03-11 13:16:24.000000000 +0100
+@@ -5,4 +5,6 @@
+ # Common support
+ obj-y := irq.o clock.o gpio.o time.o devices.o
+
++obj-$(CONFIG_ARCH_MX1) += iomux-mx1-mx2.o dma-mx1-mx2.o
+ obj-$(CONFIG_ARCH_MX2) += iomux-mx1-mx2.o dma-mx1-mx2.o
++obj-$(CONFIG_MXC_ULPI) += ulpi.o
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/dma-mx1-mx2.c linux-2.6.28-karo/arch/arm/plat-mxc/dma-mx1-mx2.c
+--- linux-2.6.28/arch/arm/plat-mxc/dma-mx1-mx2.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/dma-mx1-mx2.c 2009-03-11 13:16:24.000000000 +0100
+@@ -151,7 +151,7 @@ static inline int imx_dma_sg_next(int ch
+ unsigned long now;
+
+ if (!imxdma->name) {
+- printk(KERN_CRIT "%s: called for not allocated channel %d\n",
++ printk(KERN_CRIT "%s: called for unallocated channel %d\n",
+ __func__, channel);
+ return 0;
+ }
+@@ -390,20 +390,21 @@ imx_dma_setup_handlers(int channel,
+ {
+ struct imx_dma_channel *imxdma = &imx_dma_channels[channel];
+ unsigned long flags;
++ int ret = 0;
+
+- if (!imxdma->name) {
+- printk(KERN_CRIT "%s: called for not allocated channel %d\n",
++ local_irq_save(flags);
++ if (imxdma->name) {
++ __raw_writel(1 << channel, DMA_BASE + DMA_DISR);
++ imxdma->irq_handler = irq_handler;
++ imxdma->err_handler = err_handler;
++ imxdma->data = data;
++ } else {
++ ret = -ENODEV;
++ printk(KERN_CRIT "%s: called for unallocated channel %d\n",
+ __func__, channel);
+- return -ENODEV;
+ }
+-
+- local_irq_save(flags);
+- __raw_writel(1 << channel, DMA_BASE + DMA_DISR);
+- imxdma->irq_handler = irq_handler;
+- imxdma->err_handler = err_handler;
+- imxdma->data = data;
+ local_irq_restore(flags);
+- return 0;
++ return ret;
+ }
+ EXPORT_SYMBOL(imx_dma_setup_handlers);
+
+@@ -419,17 +420,18 @@ imx_dma_setup_progression_handler(int ch
+ {
+ struct imx_dma_channel *imxdma = &imx_dma_channels[channel];
+ unsigned long flags;
++ int ret = 0;
+
+- if (!imxdma->name) {
+- printk(KERN_CRIT "%s: called for not allocated channel %d\n",
++ local_irq_save(flags);
++ if (imxdma->name) {
++ imxdma->prog_handler = prog_handler;
++ } else {
++ printk(KERN_CRIT "%s: called for unallocated channel %d\n",
+ __func__, channel);
+- return -ENODEV;
++ ret = -ENODEV;
+ }
+-
+- local_irq_save(flags);
+- imxdma->prog_handler = prog_handler;
+ local_irq_restore(flags);
+- return 0;
++ return ret;
+ }
+ EXPORT_SYMBOL(imx_dma_setup_progression_handler);
+
+@@ -451,16 +453,16 @@ void imx_dma_enable(int channel)
+
+ pr_debug("imxdma%d: imx_dma_enable\n", channel);
+
++ local_irq_save(flags);
+ if (!imxdma->name) {
+- printk(KERN_CRIT "%s: called for not allocated channel %d\n",
++ printk(KERN_CRIT "%s: called for unallocated channel %d\n",
+ __func__, channel);
+- return;
++ goto out;
+ }
+
+- if (imxdma->in_use)
+- return;
+-
+- local_irq_save(flags);
++ if (imxdma->in_use) {
++ goto out;
++ }
+
+ __raw_writel(1 << channel, DMA_BASE + DMA_DISR);
+ __raw_writel(__raw_readl(DMA_BASE + DMA_DIMR) & ~(1 << channel),
+@@ -482,7 +484,7 @@ void imx_dma_enable(int channel)
+ }
+ #endif
+ imxdma->in_use = 1;
+-
++ out:
+ local_irq_restore(flags);
+ }
+ EXPORT_SYMBOL(imx_dma_enable);
+@@ -512,6 +514,7 @@ void imx_dma_disable(int channel)
+ }
+ EXPORT_SYMBOL(imx_dma_disable);
+
++#ifdef CONFIG_ARCH_MX2
+ static void imx_dma_watchdog(unsigned long chno)
+ {
+ struct imx_dma_channel *imxdma = &imx_dma_channels[chno];
+@@ -523,6 +526,7 @@ static void imx_dma_watchdog(unsigned lo
+ if (imxdma->err_handler)
+ imxdma->err_handler(chno, imxdma->data, IMX_DMA_ERR_TIMEOUT);
+ }
++#endif
+
+ static irqreturn_t dma_err_handler(int irq, void *dev_id)
+ {
+@@ -675,16 +679,16 @@ int imx_dma_request(int channel, const c
+ {
+ struct imx_dma_channel *imxdma = &imx_dma_channels[channel];
+ unsigned long flags;
+- int ret;
++ int ret = -EINVAL;
+
+ /* basic sanity checks */
+ if (!name)
+- return -EINVAL;
++ return ret;
+
+ if (channel >= IMX_DMA_CHANNELS) {
+ printk(KERN_CRIT "%s: called for non-existed channel %d\n",
+ __func__, channel);
+- return -EINVAL;
++ return ret;
+ }
+
+ local_irq_save(flags);
+@@ -692,6 +696,8 @@ int imx_dma_request(int channel, const c
+ local_irq_restore(flags);
+ return -EBUSY;
+ }
++ imxdma->name = name;
++ local_irq_restore(flags);
+
+ #ifdef CONFIG_ARCH_MX2
+ ret = request_irq(MXC_INT_DMACH0 + channel, dma_irq_handler, 0, "DMA",
+@@ -706,13 +712,11 @@ int imx_dma_request(int channel, const c
+ imxdma->watchdog.data = channel;
+ #endif
+
+- imxdma->name = name;
+ imxdma->irq_handler = NULL;
+ imxdma->err_handler = NULL;
+ imxdma->data = NULL;
+ imxdma->sg = NULL;
+
+- local_irq_restore(flags);
+ return 0;
+ }
+ EXPORT_SYMBOL(imx_dma_request);
+@@ -726,14 +730,14 @@ void imx_dma_free(int channel)
+ unsigned long flags;
+ struct imx_dma_channel *imxdma = &imx_dma_channels[channel];
+
++ local_irq_save(flags);
+ if (!imxdma->name) {
+ printk(KERN_CRIT
+ "%s: trying to free free channel %d\n",
+ __func__, channel);
+- return;
++ goto out;
+ }
+
+- local_irq_save(flags);
+ /* Disable interrupts */
+ __raw_writel(__raw_readl(DMA_BASE + DMA_DIMR) | (1 << channel),
+ DMA_BASE + DMA_DIMR);
+@@ -744,7 +748,7 @@ void imx_dma_free(int channel)
+ #ifdef CONFIG_ARCH_MX2
+ free_irq(MXC_INT_DMACH0 + channel, NULL);
+ #endif
+-
++ out:
+ local_irq_restore(flags);
+ }
+ EXPORT_SYMBOL(imx_dma_free);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/gpio.c linux-2.6.28-karo/arch/arm/plat-mxc/gpio.c
+--- linux-2.6.28/arch/arm/plat-mxc/gpio.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/gpio.c 2009-03-11 18:50:00.000000000 +0100
+@@ -22,18 +22,42 @@
+ #include <linux/init.h>
+ #include <linux/io.h>
+ #include <linux/irq.h>
++#include <linux/interrupt.h>
+ #include <linux/gpio.h>
++#include <linux/sysdev.h>
++#include <linux/pm.h>
+ #include <mach/hardware.h>
+ #include <asm-generic/bug.h>
+
+ static struct mxc_gpio_port *mxc_gpio_ports;
+ static int gpio_table_size;
+
++#ifdef DEBUG
++static inline void __mxc_gpio_writel(unsigned int val,
++ volatile unsigned int __force *addr,
++ const char *fn)
++{
++ printk(KERN_DEBUG "%s: Writing %08x to %p\n", fn, val, addr);
++ __raw_writel(val addr);
++}
++
++static inline unsigned int __mxc_gpio_readl(volatile unsigned int __force *addr)
++{
++ printk(KERN_DEBUG "Writing %08x to %p\n", val, addr);
++ __raw_readl(val addr);
++}
++#define mxc_gpio_writel(v,a) __mxc_gpio_writel(v, a, __func__)
++#define mxc_gpio_readl(a) __mxc_gpio_readl(a, __func__)
++#else
++#define mxc_gpio_writel __raw_writel
++#define mxc_gpio_readl __raw_readl
++#endif
++
+ /* Note: This driver assumes 32 GPIOs are handled in one register */
+
+ static void _clear_gpio_irqstatus(struct mxc_gpio_port *port, u32 index)
+ {
+- __raw_writel(1 << index, port->base + GPIO_ISR);
++ mxc_gpio_writel(1 << index, port->base + GPIO_ISR);
+ }
+
+ static void _set_gpio_irqenable(struct mxc_gpio_port *port, u32 index,
+@@ -41,15 +65,21 @@ static void _set_gpio_irqenable(struct m
+ {
+ u32 l;
+
+- l = __raw_readl(port->base + GPIO_IMR);
++ l = mxc_gpio_readl(port->base + GPIO_IMR);
+ l = (l & (~(1 << index))) | (!!enable << index);
+- __raw_writel(l, port->base + GPIO_IMR);
++ mxc_gpio_writel(l, port->base + GPIO_IMR);
+ }
+
+-static void gpio_ack_irq(u32 irq)
++static void _set_gpio_edge_ctrl(struct mxc_gpio_port *port, u32 index,
++ int edge)
+ {
+- u32 gpio = irq_to_gpio(irq);
+- _clear_gpio_irqstatus(&mxc_gpio_ports[gpio / 32], gpio & 0x1f);
++ void __iomem *reg = port->base;
++ u32 bit, val;
++
++ reg += (index >> 4) ? GPIO_ICR2 : GPIO_ICR1;
++ bit = (index & 0xf) << 1;
++ val = mxc_gpio_readl(reg) & ~(0x3 << bit);
++ mxc_gpio_writel(val | (edge << bit), reg);
+ }
+
+ static void gpio_mask_irq(u32 irq)
+@@ -64,35 +94,73 @@ static void gpio_unmask_irq(u32 irq)
+ _set_gpio_irqenable(&mxc_gpio_ports[gpio / 32], gpio & 0x1f, 1);
+ }
+
++static int mxc_gpio_get(struct gpio_chip *chip, unsigned offset);
++
++static void gpio_ack_irq(u32 irq)
++{
++ u32 gpio = irq_to_gpio(irq);
++ struct mxc_gpio_port *port = &mxc_gpio_ports[gpio / 32];
++ int bit = 1 << (gpio & 0x1f);
++
++ if (port->level_irq & bit) {
++ gpio_mask_irq(irq);
++ } else if (port->both_edge_irq & bit) {
++ if (mxc_gpio_get(&port->chip, gpio & 0x1f)) {
++ _set_gpio_edge_ctrl(port, gpio, GPIO_INT_FALL_EDGE);
++ } else {
++ _set_gpio_edge_ctrl(port, gpio, GPIO_INT_RISE_EDGE);
++ }
++ }
++ _clear_gpio_irqstatus(port, gpio & 0x1f);
++}
++
+ static int gpio_set_irq_type(u32 irq, u32 type)
+ {
+ u32 gpio = irq_to_gpio(irq);
+ struct mxc_gpio_port *port = &mxc_gpio_ports[gpio / 32];
+- u32 bit, val;
++ u32 bitmask = 1 << (gpio & 0x1f);
+ int edge;
+- void __iomem *reg = port->base;
+
+ switch (type) {
+ case IRQ_TYPE_EDGE_RISING:
+ edge = GPIO_INT_RISE_EDGE;
++ port->level_irq &= ~bitmask;
++ port->both_edge_irq &= ~bitmask;
++ __set_irq_handler_unlocked(irq, handle_edge_irq);
+ break;
+ case IRQ_TYPE_EDGE_FALLING:
+ edge = GPIO_INT_FALL_EDGE;
++ port->level_irq &= ~bitmask;
++ port->both_edge_irq &= ~bitmask;
++ __set_irq_handler_unlocked(irq, handle_edge_irq);
+ break;
+ case IRQ_TYPE_LEVEL_LOW:
++ port->level_irq |= bitmask;
++ port->both_edge_irq &= ~bitmask;
+ edge = GPIO_INT_LOW_LEV;
++ __set_irq_handler_unlocked(irq, handle_level_irq);
+ break;
+ case IRQ_TYPE_LEVEL_HIGH:
++ port->level_irq |= bitmask;
++ port->both_edge_irq &= ~bitmask;
+ edge = GPIO_INT_HIGH_LEV;
++ __set_irq_handler_unlocked(irq, handle_level_irq);
++ break;
++ case IRQ_TYPE_EDGE_BOTH:
++ port->level_irq &= ~bitmask;
++ port->both_edge_irq |= bitmask;
++ if (mxc_gpio_get(&port->chip, gpio & 0x1f)) {
++ edge = GPIO_INT_FALL_EDGE;
++ } else {
++ edge = GPIO_INT_RISE_EDGE;
++ }
++ __set_irq_handler_unlocked(irq, handle_edge_irq);
+ break;
+- default: /* this includes IRQ_TYPE_EDGE_BOTH */
++ default:
+ return -EINVAL;
+ }
+
+- reg += GPIO_ICR1 + ((gpio & 0x10) >> 2); /* lower or upper register */
+- bit = gpio & 0xf;
+- val = __raw_readl(reg) & ~(0x3 << (bit << 1));
+- __raw_writel(val | (edge << (bit << 1)), reg);
++ _set_gpio_edge_ctrl(port, gpio, edge);
+ _clear_gpio_irqstatus(port, gpio & 0x1f);
+
+ return 0;
+@@ -115,15 +183,15 @@ static void mxc_gpio_irq_handler(struct
+ }
+ }
+
+-#ifdef CONFIG_ARCH_MX3
+-/* MX3 has one interrupt *per* gpio port */
++#if defined(CONFIG_ARCH_MX3) || defined(CONFIG_ARCH_MX1)
++/* MX1 and MX3 has one interrupt *per* gpio port */
+ static void mx3_gpio_irq_handler(u32 irq, struct irq_desc *desc)
+ {
+ u32 irq_stat;
+- struct mxc_gpio_port *port = (struct mxc_gpio_port *)get_irq_data(irq);
++ struct mxc_gpio_port *port = get_irq_data(irq);
+
+- irq_stat = __raw_readl(port->base + GPIO_ISR) &
+- __raw_readl(port->base + GPIO_IMR);
++ irq_stat = mxc_gpio_readl(port->base + GPIO_ISR) &
++ mxc_gpio_readl(port->base + GPIO_IMR);
+ BUG_ON(!irq_stat);
+ mxc_gpio_irq_handler(port, irq_stat);
+ }
+@@ -135,26 +203,52 @@ static void mx2_gpio_irq_handler(u32 irq
+ {
+ int i;
+ u32 irq_msk, irq_stat;
+- struct mxc_gpio_port *port = (struct mxc_gpio_port *)get_irq_data(irq);
++ struct mxc_gpio_port *port = get_irq_data(irq);
+
+ /* walk through all interrupt status registers */
+ for (i = 0; i < gpio_table_size; i++) {
+- irq_msk = __raw_readl(port[i].base + GPIO_IMR);
++ irq_msk = mxc_gpio_readl(port[i].base + GPIO_IMR);
+ if (!irq_msk)
+ continue;
+
+- irq_stat = __raw_readl(port[i].base + GPIO_ISR) & irq_msk;
++ irq_stat = mxc_gpio_readl(port[i].base + GPIO_ISR) & irq_msk;
+ if (irq_stat)
+ mxc_gpio_irq_handler(&port[i], irq_stat);
+ }
+ }
+ #endif
+
++/*
++ * Set interrupt number "irq" in the GPIO as a wakeup source.
++ * While system is running all registered GPIO interrupts need to have
++ * wakeup enabled. When system is suspended, only selected GPIO interrupts
++ * need to have wakeup enabled.
++ * @param irq interrupt source number
++ * @param enable enable as wakeup if equal to non-zero
++ * @return This function returns 0 on success.
++ */
++static int gpio_set_wake_irq(u32 irq, u32 enable)
++{
++ u32 gpio = irq_to_gpio(irq);
++ u32 gpio_idx = gpio & 0x1f;
++ struct mxc_gpio_port *port = &mxc_gpio_ports[gpio / 32];
++
++ if ((gpio / 32) >= gpio_table_size)
++ return -EINVAL;
++
++ if (enable)
++ port->suspend_wakeup |= (1 << gpio_idx);
++ else
++ port->suspend_wakeup &= ~(1 << gpio_idx);
++ return 0;
++}
++
+ static struct irq_chip gpio_irq_chip = {
+ .ack = gpio_ack_irq,
+ .mask = gpio_mask_irq,
+ .unmask = gpio_unmask_irq,
+ .set_type = gpio_set_irq_type,
++ .set_wake = gpio_set_wake_irq,
+ };
+
+ static void _set_gpio_direction(struct gpio_chip *chip, unsigned offset,
+@@ -164,12 +258,12 @@ static void _set_gpio_direction(struct g
+ container_of(chip, struct mxc_gpio_port, chip);
+ u32 l;
+
+- l = __raw_readl(port->base + GPIO_GDIR);
++ l = mxc_gpio_readl(port->base + GPIO_GDIR);
+ if (dir)
+ l |= 1 << offset;
+ else
+ l &= ~(1 << offset);
+- __raw_writel(l, port->base + GPIO_GDIR);
++ mxc_gpio_writel(l, port->base + GPIO_GDIR);
+ }
+
+ static void mxc_gpio_set(struct gpio_chip *chip, unsigned offset, int value)
+@@ -178,9 +272,12 @@ static void mxc_gpio_set(struct gpio_chi
+ container_of(chip, struct mxc_gpio_port, chip);
+ void __iomem *reg = port->base + GPIO_DR;
+ u32 l;
++ unsigned long flags;
+
+- l = (__raw_readl(reg) & (~(1 << offset))) | (value << offset);
+- __raw_writel(l, reg);
++ local_irq_save(flags);
++ l = (mxc_gpio_readl(reg) & (~(1 << offset))) | (value << offset);
++ mxc_gpio_writel(l, reg);
++ local_irq_restore(flags);
+ }
+
+ static int mxc_gpio_get(struct gpio_chip *chip, unsigned offset)
+@@ -188,20 +285,32 @@ static int mxc_gpio_get(struct gpio_chip
+ struct mxc_gpio_port *port =
+ container_of(chip, struct mxc_gpio_port, chip);
+
+- return (__raw_readl(port->base + GPIO_PSR) >> offset) & 1;
++ return (mxc_gpio_readl(port->base + GPIO_PSR) >> offset) & 1;
+ }
+
+ static int mxc_gpio_direction_input(struct gpio_chip *chip, unsigned offset)
+ {
++ unsigned long flags;
++
++ local_irq_save(flags);
+ _set_gpio_direction(chip, offset, 0);
++ local_irq_restore(flags);
+ return 0;
+ }
+
+ static int mxc_gpio_direction_output(struct gpio_chip *chip,
+ unsigned offset, int value)
+ {
+- _set_gpio_direction(chip, offset, 1);
++ unsigned long flags;
++
++ local_irq_save(flags);
++ /*
++ * set the output data register _before_ potentially changing
++ * the pin direction to prevent glitches on the output
++ */
+ mxc_gpio_set(chip, offset, value);
++ _set_gpio_direction(chip, offset, 1);
++ local_irq_restore(flags);
+ return 0;
+ }
+
+@@ -217,10 +326,10 @@ int __init mxc_gpio_init(struct mxc_gpio
+
+ for (i = 0; i < cnt; i++) {
+ /* disable the interrupt and clear the status */
+- __raw_writel(0, port[i].base + GPIO_IMR);
+- __raw_writel(~0, port[i].base + GPIO_ISR);
++ mxc_gpio_writel(0, port[i].base + GPIO_IMR);
++ mxc_gpio_writel(~0, port[i].base + GPIO_ISR);
+ for (j = port[i].virtual_irq_start;
+- j < port[i].virtual_irq_start + 32; j++) {
++ j < port[i].virtual_irq_start + 32; j++) {
+ set_irq_chip(j, &gpio_irq_chip);
+ set_irq_handler(j, handle_edge_irq);
+ set_irq_flags(j, IRQF_VALID);
+@@ -235,9 +344,9 @@ int __init mxc_gpio_init(struct mxc_gpio
+ port[i].chip.ngpio = 32;
+
+ /* its a serious configuration bug when it fails */
+- BUG_ON( gpiochip_add(&port[i].chip) < 0 );
++ BUG_ON(gpiochip_add(&port[i].chip) < 0);
+
+-#ifdef CONFIG_ARCH_MX3
++#if defined(CONFIG_ARCH_MX3) || defined(CONFIG_ARCH_MX1)
+ /* setup one handler for each entry */
+ set_irq_chained_handler(port[i].irq, mx3_gpio_irq_handler);
+ set_irq_data(port[i].irq, &port[i]);
+@@ -251,3 +360,126 @@ int __init mxc_gpio_init(struct mxc_gpio
+ #endif
+ return 0;
+ }
++
++#ifdef CONFIG_PM
++/*
++ * This function puts the GPIO in low-power mode/state.
++ * All the interrupts that are enabled are first saved.
++ * Only those interrupts which registers as a wake source by calling
++ * enable_irq_wake are enabled. All other interrupts are disabled.
++ *
++ * @param dev the system device structure used to give information
++ * on GPIO to suspend
++ * @param mesg the power state the device is entering
++ *
++ * @return The function always returns 0.
++ */
++static int mxc_gpio_suspend(struct sys_device *dev, pm_message_t mesg)
++{
++ int ret = 0;
++ int i;
++ int wakeup = 0;
++
++ for (i = 0; i < gpio_table_size; i++) {
++ struct mxc_gpio_port *port = &mxc_gpio_ports[i];
++
++ if (__raw_readl(port->base + GPIO_ISR) & port->suspend_wakeup)
++ return -EBUSY;
++
++ port->saved_wakeup = __raw_readl(port->base + GPIO_IMR);
++ __raw_writel(port->suspend_wakeup, port->base + GPIO_IMR);
++
++ if (port->suspend_wakeup) {
++ wakeup = 1;
++ }
++ if (ret != 0) {
++ while (--i >= 0) {
++ port = &mxc_gpio_ports[i];
++ __raw_writel(port->saved_wakeup,
++ port->base + GPIO_IMR);
++ }
++ return ret;
++ }
++ }
++ if (wakeup)
++ ret = enable_irq_wake(MXC_INT_GPIO);
++
++ return ret;
++}
++
++/*
++ * This function brings the GPIO back from low-power state.
++ * All the interrupts enabled before suspension are re-enabled from
++ * the saved information.
++ *
++ * @param dev the system device structure used to give information
++ * on GPIO to resume
++ *
++ * @return The function always returns 0.
++ */
++static int mxc_gpio_resume(struct sys_device *dev)
++{
++ int i;
++ int wakeup = 0;
++
++ for (i = 0; i < gpio_table_size; i++) {
++ struct mxc_gpio_port *port = &mxc_gpio_ports[i];
++
++ __raw_writel(port->saved_wakeup, port->base + GPIO_IMR);
++ if (port->suspend_wakeup) {
++ wakeup = 1;
++ }
++ }
++ if (wakeup)
++ disable_irq_wake(MXC_INT_GPIO);
++
++ return 0;
++}
++#else
++#define mxc_gpio_suspend NULL
++#define mxc_gpio_resume NULL
++#endif /* CONFIG_PM */
++
++/*
++ * This structure contains pointers to the power management callback functions.
++ */
++static struct sysdev_class mxc_gpio_sysclass = {
++ .name = "mxc_gpio",
++ .suspend = mxc_gpio_suspend,
++ .resume = mxc_gpio_resume,
++};
++
++/*
++ * This structure represents GPIO as a system device.
++ * System devices follow a slightly different driver model.
++ * They don't need to do dynammic driver binding, can't be probed,
++ * and don't reside on any type of peripheral bus.
++ * So, it is represented and treated a little differently.
++ */
++static struct sys_device mxc_gpio_device = {
++ .cls = &mxc_gpio_sysclass,
++};
++
++/*
++ * This function registers GPIO hardware as a system device and
++ * intializes all the GPIO ports if not already done.
++ * System devices will only be suspended with interrupts disabled, and
++ * after all other devices have been suspended. On resume, they will be
++ * resumed before any other devices, and also with interrupts disabled.
++ * This may get called early from board specific init
++ *
++ * @return This function returns 0 on success.
++ */
++int __init mxc_gpio_sys_init(void)
++{
++ int ret = 0;
++
++ ret = sysdev_class_register(&mxc_gpio_sysclass);
++ if (ret)
++ return ret;
++ ret = sysdev_register(&mxc_gpio_device);
++ if (ret != 0)
++ sysdev_class_unregister(&mxc_gpio_sysclass);
++
++ return ret;
++}
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/include/mach/debug-macro.S linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/debug-macro.S
+--- linux-2.6.28/arch/arm/plat-mxc/include/mach/debug-macro.S 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/debug-macro.S 2009-03-11 13:16:24.000000000 +0100
+@@ -28,6 +28,9 @@
+ #ifdef CONFIG_MACH_PCM038
+ #include <mach/board-pcm038.h>
+ #endif
++#ifdef CONFIG_MACH_TX27
++#include <mach/board-tx27.h>
++#endif
+ .macro addruart,rx
+ mrc p15, 0, \rx, c1, c0
+ tst \rx, #1 @ MMU enabled?
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/include/mach/gpio.h linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/gpio.h
+--- linux-2.6.28/arch/arm/plat-mxc/include/mach/gpio.h 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/gpio.h 2009-03-11 13:16:24.000000000 +0100
+@@ -35,6 +35,12 @@ struct mxc_gpio_port {
+ int irq;
+ int virtual_irq_start;
+ struct gpio_chip chip;
++ u32 level_irq;
++ u32 both_edge_irq;
++#ifdef CONFIG_PM
++ u32 suspend_wakeup;
++ u32 saved_wakeup;
++#endif
+ };
+
+ int mxc_gpio_init(struct mxc_gpio_port*, int);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/include/mach/imx_cam.h linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/imx_cam.h
+--- linux-2.6.28/arch/arm/plat-mxc/include/mach/imx_cam.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/imx_cam.h 2009-01-05 17:47:15.000000000 +0100
+@@ -0,0 +1,47 @@
++/*
++ imx-cam.h - i.MX27 camera driver header file
++
++ Copyright (C) 2003, Intel Corporation
++ Copyright (C) 2008, Sascha Hauer <s.hauer@pengutronix.de>
++
++ 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., 675 Mass Ave, Cambridge, MA 02139, USA.
++*/
++
++#ifndef __ASM_ARCH_CAMERA_H_
++#define __ASM_ARCH_CAMERA_H_
++
++#define MX27_CAMERA_SWAP16 (1 << 0)
++#define MX27_CAMERA_EXT_VSYNC (1 << 1)
++#define MX27_CAMERA_CCIR (1 << 2)
++#define MX27_CAMERA_CCIR_INTERLACE (1 << 3)
++#define MX27_CAMERA_HSYNC_HIGH (1 << 4)
++#define MX27_CAMERA_GATED_CLOCK (1 << 5)
++#define MX27_CAMERA_INV_DATA (1 << 6)
++#define MX27_CAMERA_PCLK_SAMPLE_RISING (1 << 7)
++#define MX27_CAMERA_PACK_DIR_MSB (1 << 8)
++
++struct mx27_camera_platform_data {
++ int (*init)(struct platform_device *);
++ int (*exit)(struct platform_device *);
++
++ unsigned long flags;
++
++ unsigned long clk;
++};
++
++extern int mx27_init_camera(struct mx27_camera_platform_data *);
++
++#endif /* __ASM_ARCH_CAMERA_H_ */
++
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/include/mach/imx_i2c.h linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/imx_i2c.h
+--- linux-2.6.28/arch/arm/plat-mxc/include/mach/imx_i2c.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/imx_i2c.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,81 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ * Copyright 2008 Juergen Beisert, kernel@pengutronix.de
++ *
++ * 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
++ */
++
++#ifndef __ASM_ARCH_MXC_I2C_H__
++#define __ASM_ARCH_MXC_I2C_H__
++
++struct imx_i2c_platform_data {
++ u32 max_clk;
++ int (*init)(struct platform_device *pdev);
++ int (*exit)(struct platform_device *pdev);
++};
++
++/*
++ * This file contains the I2C chip level configuration details.
++ *
++ * It also contains the API function that other drivers can use to read/write
++ * to the I2C device registers.
++ */
++
++/*
++ * This defines the string used to identify MXC I2C Bus drivers
++ */
++#define MXC_ADAPTER_NAME "MXC I2C Adapter"
++
++#define MXC_I2C_FLAG_READ 0x01 /* if set, is read; else is write */
++#define MXC_I2C_FLAG_POLLING 0x02 /* if set, is polling mode; else is interrupt mode */
++
++int mxc_i2c_read(int bus_id, unsigned int addr, char *reg, int reg_len,
++ char *buf, int num);
++
++int mxc_i2c_write(int bus_id, unsigned int addr, char *reg, int reg_len,
++ char *buf, int num);
++
++int mxc_i2c_polling_read(int bus_id, unsigned int addr, char *reg, int reg_len,
++ char *buf, int num);
++
++int mxc_i2c_polling_write(int bus_id, unsigned int addr, char *reg, int reg_len,
++ char *buf, int num);
++
++/* FIXME: This should be in a generic register file */
++
++/* Address offsets of the I2C registers */
++#define MXC_IADR 0x00 /* Address Register */
++#define MXC_IFDR 0x04 /* Freq div register */
++#define MXC_I2CR 0x08 /* Control regsiter */
++#define MXC_I2SR 0x0C /* Status register */
++#define MXC_I2DR 0x10 /* Data I/O register */
++
++/* Bit definitions of I2CR */
++#define MXC_I2CR_IEN 0x0080
++#define MXC_I2CR_IIEN 0x0040
++#define MXC_I2CR_MSTA 0x0020
++#define MXC_I2CR_MTX 0x0010
++#define MXC_I2CR_TXAK 0x0008
++#define MXC_I2CR_RSTA 0x0004
++
++/* Bit definitions of I2SR */
++#define MXC_I2SR_ICF 0x0080
++#define MXC_I2SR_IAAS 0x0040
++#define MXC_I2SR_IBB 0x0020
++#define MXC_I2SR_IAL 0x0010
++#define MXC_I2SR_SRW 0x0004
++#define MXC_I2SR_IIF 0x0002
++#define MXC_I2SR_RXAK 0x0001
++
++#endif /* __ASM_ARCH_MXC_I2C_H__ */
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/include/mach/imx_spi.h linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/imx_spi.h
+--- linux-2.6.28/arch/arm/plat-mxc/include/mach/imx_spi.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/imx_spi.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,220 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ * Copyright 2008 Juergen Beisert, kernel@pengutronix.de
++ * Copyright 2008 Luotao Fu, kernel@pengutronix.de
++ *
++ * 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
++ */
++
++#ifndef __MXC_SPI_MX27_H__
++#define __MXC_SPI_MX27_H__
++
++#include <mach/hardware.h>
++#include <asm/mach-types.h>
++
++/*
++ * This structure is used to define the SPI master controller's platform
++ * data. It includes the SPI bus number and the maximum number of
++ * slaves/chips it supports.
++ */
++struct mxc_spi_master {
++ unsigned int bus_num; /* bus number. */
++ unsigned int maxchipselect; /* number of chip selects. */
++ unsigned int spi_version; /* CSPI Hardware Version. */
++
++ int (*init)(struct platform_device *pdev);
++ int (*exit)(struct platform_device *pdev);
++};
++
++/* Register definitions.
++ * XXX: The ifdef CONFIG_ARCH_MX2 segments are merged from the
++ * freescale mxc_spi_mx27.h. This is actually highly strange, since according
++ * to the processor reference manuals the spi controllers on mx27 and mx31 are
++ * identical. We might want to clearify this after chatting with freescale on
++ * this issue */
++
++#define MXC_CSPIRXDATA 0x00
++#define MXC_CSPITXDATA 0x04
++#define MXC_CSPICTRL 0x08
++#define MXC_CSPIINT 0x0C
++
++#ifdef CONFIG_ARCH_MX2
++#define MXC_CSPIDMA 0x18
++#define MXC_CSPISTAT 0x0C
++#define MXC_CSPIPERIOD 0x14
++#define MXC_CSPITEST 0x10
++#define MXC_CSPIRESET 0x1C
++#define MXC_CSPICTRL_ENABLE (1 << 10)
++#else
++#define MXC_CSPIDMA 0x10
++#define MXC_CSPISTAT 0x14
++#define MXC_CSPIPERIOD 0x18
++#define MXC_CSPITEST 0x1C
++#define MXC_CSPIRESET 0x00
++#define MXC_CSPICTRL_ENABLE 0x1
++#endif
++
++#define MXC_CSPICTRL_DISABLE 0x0
++
++#ifdef CONFIG_ARCH_MX2
++#define MXC_CSPICTRL_MASTER (1 << 11)
++#else
++#define MXC_CSPICTRL_MASTER (1 << 1)
++#endif
++
++#define MXC_CSPICTRL_SLAVE 0x0
++
++#ifdef CONFIG_ARCH_MX2
++#define MXC_CSPICTRL_XCH (1 << 9)
++#define MXC_CSPICTRL_LOWPOL (1 << 5)
++#else
++#define MXC_CSPICTRL_XCH (1 << 2)
++#define MXC_CSPICTRL_SMC (1 << 3)
++#define MXC_CSPICTRL_LOWPOL (1 << 4)
++#endif
++
++#define MXC_CSPICTRL_HIGHPOL 0x0
++
++#ifdef CONFIG_ARCH_MX2
++#define MXC_CSPICTRL_PHA (1 << 6)
++#else
++#define MXC_CSPICTRL_PHA (1 << 5)
++#endif
++
++#define MXC_CSPICTRL_NOPHA 0x0
++
++#ifdef CONFIG_ARCH_MX2
++#define MXC_CSPICTRL_SSCTL (1 << 7)
++#define MXC_CSPICTRL_HIGHSSPOL (1 << 8)
++#else
++#define MXC_CSPICTRL_SSCTL (1 << 6)
++#define MXC_CSPICTRL_HIGHSSPOL (1 << 7)
++#endif
++
++#define MXC_CSPICTRL_LOWSSPOL 0x0
++#define MXC_CSPICTRL_CSMASK 0x3
++
++#ifdef CONFIG_ARCH_MX2
++#define MXC_CSPICTRL_MAXDATRATE 0x10
++#define MXC_CSPICTRL_DATAMASK 0x1F
++#define MXC_CSPICTRL_DATASHIFT 14
++/* This adjustment in the shift is valid only for even states only(i.e. divide
++ ratio of 2). SDHC_SPIEN is not set by default. If SDHC_SPIEN bit is set in
++ MXC_CSPICTRL, then divide ratio is 3, this shift adjustment is invalid. */
++#define MXC_CSPICTRL_ADJUST_SHIFT(x) ((x) = ((x) - 1) * 2)
++#else
++#define MXC_CSPICTRL_MAXDATRATE 0x7
++#define MXC_CSPICTRL_DATAMASK 0x7
++#define MXC_CSPICTRL_DATASHIFT 16
++#define MXC_CSPICTRL_ADJUST_SHIFT(x) ((x) -= 2)
++#endif
++
++#define MXC_CSPICTRL_CSSHIFT_0_7 12
++#define MXC_CSPICTRL_BCSHIFT_0_7 20
++#define MXC_CSPICTRL_BCMASK_0_7 0xFFF
++#define MXC_CSPICTRL_DRCTRLSHIFT_0_7 8
++
++#define MXC_CSPICTRL_CSSHIFT_0_5 12
++#define MXC_CSPICTRL_BCSHIFT_0_5 20
++#define MXC_CSPICTRL_BCMASK_0_5 0xFFF
++#define MXC_CSPICTRL_DRCTRLSHIFT_0_5 8
++
++#define MXC_CSPICTRL_CSSHIFT_0_4 24
++#define MXC_CSPICTRL_BCSHIFT_0_4 8
++#define MXC_CSPICTRL_BCMASK_0_4 0x1F
++#define MXC_CSPICTRL_DRCTRLSHIFT_0_4 20
++
++#define MXC_CSPICTRL_CSSHIFT_0_0 19
++#define MXC_CSPICTRL_BCSHIFT_0_0 0
++#define MXC_CSPICTRL_BCMASK_0_0 0x1F
++#define MXC_CSPICTRL_DRCTRLSHIFT_0_0 12
++
++#define MXC_CSPIINT_IRQSHIFT_0_7 8
++#define MXC_CSPIINT_IRQSHIFT_0_5 9
++#define MXC_CSPIINT_IRQSHIFT_0_4 9
++#define MXC_CSPIINT_IRQSHIFT_0_0 18
++
++#ifdef CONFIG_ARCH_MX2
++#define MXC_CSPIINT_TEEN (1 << 9)
++#define MXC_CSPIINT_THEN (1 << 10)
++#define MXC_CSPIINT_TFEN (1 << 11)
++#define MXC_CSPIINT_RREN (1 << 13)
++#define MXC_CSPIINT_RHEN (1 << 14)
++#define MXC_CSPIINT_RFEN (1 << 15)
++#define MXC_CSPIINT_ROEN (1 << 16)
++#else
++#define MXC_CSPIINT_TEEN (1 << 0)
++#define MXC_CSPIINT_THEN (1 << 1)
++#define MXC_CSPIINT_TFEN (1 << 2)
++#define MXC_CSPIINT_RREN (1 << 3)
++#define MXC_CSPIINT_RHEN (1 << 4)
++#define MXC_CSPIINT_RFEN (1 << 5)
++#define MXC_CSPIINT_ROEN (1 << 6)
++#endif
++
++#define MXC_CSPIINT_TCEN_0_7 (1 << 7)
++#define MXC_CSPIINT_TCEN_0_5 (1 << 8)
++#define MXC_CSPIINT_TCEN_0_4 (1 << 8)
++
++#ifdef CONFIG_ARCH_MX2
++#define MXC_CSPIINT_TCEN_0_0 (1 << 12)
++#else
++#define MXC_CSPIINT_TCEN_0_0 (1 << 3)
++#endif
++
++#define MXC_CSPIINT_BOEN_0_7 0
++#define MXC_CSPIINT_BOEN_0_5 (1 << 7)
++#define MXC_CSPIINT_BOEN_0_4 (1 << 7)
++
++#ifdef CONFIG_ARCH_MX2
++#define MXC_CSPIINT_BOEN_0_0 (1 << 17)
++#else
++#define MXC_CSPIINT_BOEN_0_0 (1 << 8)
++#endif
++
++#define MXC_CSPISTAT_TE (1 << 0)
++#define MXC_CSPISTAT_TH (1 << 1)
++#define MXC_CSPISTAT_TF (1 << 2)
++#define MXC_CSPISTAT_RR (1 << 3)
++#define MXC_CSPISTAT_RH (1 << 4)
++#define MXC_CSPISTAT_RF (1 << 5)
++#define MXC_CSPISTAT_RO (1 << 6)
++#define MXC_CSPISTAT_TC_0_7 (1 << 7)
++#define MXC_CSPISTAT_TC_0_5 (1 << 8)
++#define MXC_CSPISTAT_TC_0_4 (1 << 8)
++#define MXC_CSPISTAT_TC_0_0 (1 << 3)
++#define MXC_CSPISTAT_BO_0_7 0
++#define MXC_CSPISTAT_BO_0_5 (1 << 7)
++#define MXC_CSPISTAT_BO_0_4 (1 << 7)
++#define MXC_CSPISTAT_BO_0_0 (1 << 8)
++
++#define MXC_CSPIPERIOD_32KHZ (1 << 15)
++
++#define MXC_CSPITEST_LBC (1 << 14)
++
++/*
++ * This structure contains information that differs with
++ * SPI master controller hardware version
++ */
++struct mxc_spi_unique_def {
++ unsigned int intr_bit_shift; /* Width of valid bits in MXC_CSPIINT. */
++ unsigned int cs_shift; /* Chip Select shift. */
++ unsigned int bc_shift; /* Bit count shift. */
++ unsigned int bc_mask; /* Bit count mask. */
++ unsigned int drctrl_shift; /* Data Control shift. */
++ unsigned int xfer_complete; /* Transfer Complete shift. */
++ unsigned int bc_overflow; /* Bit counnter overflow shift. */
++};
++
++#endif /*__MXC_SPI_MX27_H__ */
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/include/mach/imx_ssi.h linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/imx_ssi.h
+--- linux-2.6.28/arch/arm/plat-mxc/include/mach/imx_ssi.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/imx_ssi.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,191 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ * Copyright 2008 Juergen Beisert, kernel@pengutronix.de
++ *
++ * 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
++ */
++
++#ifndef __ASM_ARCH_MXC_SSI_H
++#define __ASM_ARCH_MXC_SSI_H
++
++#include <linux/platform_device.h>
++
++struct imx_ssi_platform_data {
++ int (*init)(struct platform_device *pdev);
++ int (*exit)(struct platform_device *pdev);
++};
++
++#define SSI_STX0 0x00
++#define SSI_STX1 0x04
++#define SSI_SRX0 0x08
++#define SSI_SRX1 0x0c
++
++#define SSI_SCR 0x10
++# define SSI_SCR_CLK_IST (1 << 9)
++# define SSI_SCR_CLK_IST_SHIFT 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_I2S_MODE_SHIFT 5
++
++#define SSI_SISR 0x14
++# define SSI_SISR_MASK ((1 << 19) - 1)
++# 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 0x18
++# 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 0x1c
++# define SSI_STCR_TXBIT0 (1 << 9)
++# define SSI_STCR_TFEN1 (1 << 8)
++# define SSI_STCR_TFEN0 (1 << 7)
++# define SSI_FIFO_ENABLE_0_SHIFT 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 0x20
++# define SSI_SRCR_RXBIT0 (1 << 9)
++# define SSI_SRCR_RFEN1 (1 << 8)
++# define SSI_SRCR_RFEN0 (1 << 7)
++# define SSI_FIFO_ENABLE_0_SHIFT 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_SRCCR 0x28
++# define SSI_SRCCR_DIV2 (1 << 18)
++# define SSI_SRCCR_PSR (1 << 17)
++# 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_STCCR 0x24
++# define SSI_STCCR_DIV2 (1 << 18)
++# define SSI_STCCR_PSR (1 << 17)
++# 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_SFCSR 0x2c
++# define SSI_SFCSR_RFCNT1(x) (((x) & 0xf) << 28)
++# define SSI_RX_FIFO_1_COUNT_SHIFT 28
++# define SSI_SFCSR_TFCNT1(x) (((x) & 0xf) << 24)
++# define SSI_TX_FIFO_1_COUNT_SHIFT 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_RX_FIFO_0_COUNT_SHIFT 12
++# define SSI_SFCSR_TFCNT0(x) (((x) & 0xf) << 8)
++# define SSI_TX_FIFO_0_COUNT_SHIFT 8
++# define SSI_SFCSR_RFWM0(x) (((x) & 0xf) << 4)
++# define SSI_SFCSR_TFWM0(x) (((x) & 0xf) << 0)
++# define SSI_SFCSR_RFWM0_MASK (0xf << 4)
++# define SSI_SFCSR_TFWM0_MASK (0xf << 0)
++
++#define SSI_STR 0x30
++# 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 0x34
++# 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_WAIT_MASK (0x3 << 1)
++# define SSI_SOR_SYNRST (1 << 0)
++
++#define SSI_SACNT 0x38
++# 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)
++
++#define SSI_SACADD 0x3c
++#define SSI_SRMSK 0x4c
++#define SSI_SACDAT 0x40
++#define SSI_SATAG 0x44
++#define SSI_STMSK 0x48
++
++#endif /* __ASM_ARCH_MXC_SSI_H */
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/include/mach/imxfb.h linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/imxfb.h
+--- linux-2.6.28/arch/arm/plat-mxc/include/mach/imxfb.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/imxfb.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,88 @@
++/*
++ * This structure describes the machine which we are running on.
++ */
++
++#define PCR_TFT (1 << 31)
++#define PCR_COLOR (1 << 30)
++#define PCR_PBSIZ_1 (0 << 28)
++#define PCR_PBSIZ_2 (1 << 28)
++#define PCR_PBSIZ_4 (2 << 28)
++#define PCR_PBSIZ_8 (3 << 28)
++#define PCR_BPIX_1 (0 << 25)
++#define PCR_BPIX_2 (1 << 25)
++#define PCR_BPIX_4 (2 << 25)
++#define PCR_BPIX_8 (3 << 25)
++#define PCR_BPIX_12 (4 << 25)
++#define PCR_BPIX_16 (5 << 25)
++#define PCR_BPIX_18 (6 << 25)
++#define PCR_PIXPOL (1 << 24)
++#define PCR_FLMPOL (1 << 23)
++#define PCR_LPPOL (1 << 22)
++#define PCR_CLKPOL (1 << 21)
++#define PCR_OEPOL (1 << 20)
++#define PCR_SCLKIDLE (1 << 19)
++#define PCR_END_SEL (1 << 18)
++#define PCR_END_BYTE_SWAP (1 << 17)
++#define PCR_REV_VS (1 << 16)
++#define PCR_ACD_SEL (1 << 15)
++#define PCR_ACD(x) (((x) & 0x7f) << 8)
++#define PCR_SCLK_SEL (1 << 7)
++#define PCR_SHARP (1 << 6)
++#define PCR_PCD(x) ((x) & 0x3f)
++
++#define PWMR_CLS(x) (((x) & 0x1ff) << 16)
++#define PWMR_LDMSK (1 << 15)
++#define PWMR_SCR1 (1 << 10)
++#define PWMR_SCR0 (1 << 9)
++#define PWMR_CC_EN (1 << 8)
++#define PWMR_PW(x) ((x) & 0xff)
++
++#define LSCR1_PS_RISE_DELAY(x) (((x) & 0x7f) << 26)
++#define LSCR1_CLS_RISE_DELAY(x) (((x) & 0x3f) << 16)
++#define LSCR1_REV_TOGGLE_DELAY(x) (((x) & 0xf) << 8)
++#define LSCR1_GRAY2(x) (((x) & 0xf) << 4)
++#define LSCR1_GRAY1(x) (((x) & 0xf))
++
++#define DMACR_BURST (1 << 31)
++#define DMACR_HM(x) (((x) & 0xf) << 16)
++#define DMACR_TM(x) ((x) & 0xf)
++
++struct imx_fb_platform_data {
++ u_long pixclock;
++
++ u_short xres;
++ u_short yres;
++
++ u_int nonstd;
++ u_char bpp;
++ u_char hsync_len;
++ u_char left_margin;
++ u_char right_margin;
++
++ u_char vsync_len;
++ u_char upper_margin;
++ u_char lower_margin;
++ u_char sync;
++
++ u_int cmap_greyscale:1,
++ cmap_inverse:1,
++ cmap_static:1,
++ unused:29;
++
++ u_int pcr;
++ u_int pwmr;
++ u_int lscr1;
++ u_int dmacr;
++
++ u_char * fixed_screen_cpu;
++ dma_addr_t fixed_screen_dma;
++
++ int (*init)(struct platform_device*);
++ int (*exit)(struct platform_device*);
++ void (*lcd_power)(int);
++ void (*backlight_power)(int);
++};
++
++#ifdef ARCH_IMX
++void set_imx_fb_info(struct imx_fb_platform_data *);
++#endif
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/include/mach/iomux-mx1-mx2.h linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/iomux-mx1-mx2.h
+--- linux-2.6.28/arch/arm/plat-mxc/include/mach/iomux-mx1-mx2.h 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/iomux-mx1-mx2.h 2009-03-11 13:16:24.000000000 +0100
+@@ -19,14 +19,12 @@
+ #ifndef _MXC_GPIO_MX1_MX2_H
+ #define _MXC_GPIO_MX1_MX2_H
+
++#if !defined(_MXC_GPIO_MX1_H) && !defined(_MXC_GPIO_MX2_H)
++#error "Please don't include mach/iomux-mx1-mx2.h directly, include <mach/iomux.h> instead"
++#endif
++
+ #include <linux/io.h>
+
+-#define MXC_GPIO_ALLOC_MODE_NORMAL 0
+-#define MXC_GPIO_ALLOC_MODE_NO_ALLOC 1
+-#define MXC_GPIO_ALLOC_MODE_TRY_ALLOC 2
+-#define MXC_GPIO_ALLOC_MODE_ALLOC_ONLY 4
+-#define MXC_GPIO_ALLOC_MODE_RELEASE 8
+-
+ /*
+ * GPIO Module and I/O Multiplexer
+ * x = 0..3 for reg_A, reg_B, reg_C, reg_D
+@@ -50,13 +48,6 @@
+ #define MXC_SWR(x) (0x3c + ((x) << 8))
+ #define MXC_PUEN(x) (0x40 + ((x) << 8))
+
+-#ifdef CONFIG_ARCH_MX1
+-# define GPIO_PORT_MAX 3
+-#endif
+-#ifdef CONFIG_ARCH_MX2
+-# define GPIO_PORT_MAX 5
+-#endif
+-
+ #ifndef GPIO_PORT_MAX
+ # error "GPIO config port count unknown!"
+ #endif
+@@ -66,331 +57,67 @@
+ #define GPIO_PORT_SHIFT 5
+ #define GPIO_PORT_MASK (0x7 << GPIO_PORT_SHIFT)
+
+-#define GPIO_PORTA (0 << GPIO_PORT_SHIFT)
+-#define GPIO_PORTB (1 << GPIO_PORT_SHIFT)
+-#define GPIO_PORTC (2 << GPIO_PORT_SHIFT)
+-#define GPIO_PORTD (3 << GPIO_PORT_SHIFT)
+-#define GPIO_PORTE (4 << GPIO_PORT_SHIFT)
+-#define GPIO_PORTF (5 << GPIO_PORT_SHIFT)
+-
+-#define GPIO_OUT (1 << 8)
+-#define GPIO_IN (0 << 8)
+-#define GPIO_PUEN (1 << 9)
+-
+-#define GPIO_PF (1 << 10)
+-#define GPIO_AF (1 << 11)
+-
+-#define GPIO_OCR_SHIFT 12
+-#define GPIO_OCR_MASK (3 << GPIO_OCR_SHIFT)
+-#define GPIO_AIN (0 << GPIO_OCR_SHIFT)
+-#define GPIO_BIN (1 << GPIO_OCR_SHIFT)
+-#define GPIO_CIN (2 << GPIO_OCR_SHIFT)
+-#define GPIO_GPIO (3 << GPIO_OCR_SHIFT)
+-
+-#define GPIO_AOUT_SHIFT 14
+-#define GPIO_AOUT_MASK (3 << GPIO_AOUT_SHIFT)
+-#define GPIO_AOUT (0 << GPIO_AOUT_SHIFT)
+-#define GPIO_AOUT_ISR (1 << GPIO_AOUT_SHIFT)
+-#define GPIO_AOUT_0 (2 << GPIO_AOUT_SHIFT)
+-#define GPIO_AOUT_1 (3 << GPIO_AOUT_SHIFT)
+-
+-#define GPIO_BOUT_SHIFT 16
+-#define GPIO_BOUT_MASK (3 << GPIO_BOUT_SHIFT)
+-#define GPIO_BOUT (0 << GPIO_BOUT_SHIFT)
+-#define GPIO_BOUT_ISR (1 << GPIO_BOUT_SHIFT)
+-#define GPIO_BOUT_0 (2 << GPIO_BOUT_SHIFT)
+-#define GPIO_BOUT_1 (3 << GPIO_BOUT_SHIFT)
++#define GPIO_INDEX(gpio_cookie) ((gpio_cookie) & GPIO_PIN_MASK)
++#define GPIO_PORT(gpio_cookie) (((gpio_cookie) & GPIO_PORT_MASK) >> GPIO_PORT_SHIFT)
++#define IOMUX_TO_GPIO(gpio_cookie) ((gpio_cookie) & (GPIO_PORT_MASK | GPIO_PIN_MASK))
++
++#define GPIO_PORTA (0 << GPIO_PORT_SHIFT)
++#define GPIO_PORTB (1 << GPIO_PORT_SHIFT)
++#define GPIO_PORTC (2 << GPIO_PORT_SHIFT)
++#define GPIO_PORTD (3 << GPIO_PORT_SHIFT)
++#define GPIO_PORTE (4 << GPIO_PORT_SHIFT)
++#define GPIO_PORTF (5 << GPIO_PORT_SHIFT)
++
++#define GPIO_OUT (1 << 8)
++#define GPIO_IN (0 << 8)
++#define GPIO_PUEN (1 << 9)
++
++#define GPIO_PF (1 << 10)
++#define GPIO_AF (1 << 11)
++
++#define GPIO_OCR_SHIFT 12
++#define GPIO_OCR_MASK (3 << GPIO_OCR_SHIFT)
++#define GPIO_AIN (0 << GPIO_OCR_SHIFT)
++#define GPIO_BIN (1 << GPIO_OCR_SHIFT)
++#define GPIO_CIN (2 << GPIO_OCR_SHIFT)
++#define GPIO_GPIO (3 << GPIO_OCR_SHIFT)
++
++#define GPIO_AOUT_SHIFT 14
++#define GPIO_AOUT_MASK (3 << GPIO_AOUT_SHIFT)
++#define GPIO_AOUT (0 << GPIO_AOUT_SHIFT)
++#define GPIO_AOUT_ISR (1 << GPIO_AOUT_SHIFT)
++#define GPIO_AOUT_0 (2 << GPIO_AOUT_SHIFT)
++#define GPIO_AOUT_1 (3 << GPIO_AOUT_SHIFT)
++
++#define GPIO_BOUT_SHIFT 16
++#define GPIO_BOUT_MASK (3 << GPIO_BOUT_SHIFT)
++#define GPIO_BOUT (0 << GPIO_BOUT_SHIFT)
++#define GPIO_BOUT_ISR (1 << GPIO_BOUT_SHIFT)
++#define GPIO_BOUT_0 (2 << GPIO_BOUT_SHIFT)
++#define GPIO_BOUT_1 (3 << GPIO_BOUT_SHIFT)
++
++#define GPIO_DFLT_LOW (1 << 18)
++#define GPIO_DFLT_HIGH (1 << 19)
+
+ extern void mxc_gpio_mode(int gpio_mode);
+ extern int mxc_gpio_setup_multiple_pins(const int *pin_list, unsigned count,
+- int alloc_mode, const char *label);
++ const char *label);
++extern void mxc_gpio_release_multiple_pins(const int *pin_list, int count);
+
+ /*-------------------------------------------------------------------------*/
+
+-/* assignements for GPIO alternate/primary functions */
+-
+-/* FIXME: This list is not completed. The correct directions are
+- * missing on some (many) pins
+- */
+-#ifdef CONFIG_ARCH_MX1
+-#define PA0_AIN_SPI2_CLK (GPIO_GIUS | GPIO_PORTA | GPIO_OUT | 0)
+-#define PA0_AF_ETMTRACESYNC (GPIO_PORTA | GPIO_AF | 0)
+-#define PA1_AOUT_SPI2_RXD (GPIO_GIUS | GPIO_PORTA | GPIO_IN | 1)
+-#define PA1_PF_TIN (GPIO_PORTA | GPIO_PF | 1)
+-#define PA2_PF_PWM0 (GPIO_PORTA | GPIO_OUT | GPIO_PF | 2)
+-#define PA3_PF_CSI_MCLK (GPIO_PORTA | GPIO_PF | 3)
+-#define PA4_PF_CSI_D0 (GPIO_PORTA | GPIO_PF | 4)
+-#define PA5_PF_CSI_D1 (GPIO_PORTA | GPIO_PF | 5)
+-#define PA6_PF_CSI_D2 (GPIO_PORTA | GPIO_PF | 6)
+-#define PA7_PF_CSI_D3 (GPIO_PORTA | GPIO_PF | 7)
+-#define PA8_PF_CSI_D4 (GPIO_PORTA | GPIO_PF | 8)
+-#define PA9_PF_CSI_D5 (GPIO_PORTA | GPIO_PF | 9)
+-#define PA10_PF_CSI_D6 (GPIO_PORTA | GPIO_PF | 10)
+-#define PA11_PF_CSI_D7 (GPIO_PORTA | GPIO_PF | 11)
+-#define PA12_PF_CSI_VSYNC (GPIO_PORTA | GPIO_PF | 12)
+-#define PA13_PF_CSI_HSYNC (GPIO_PORTA | GPIO_PF | 13)
+-#define PA14_PF_CSI_PIXCLK (GPIO_PORTA | GPIO_PF | 14)
+-#define PA15_PF_I2C_SDA (GPIO_PORTA | GPIO_OUT | GPIO_PF | 15)
+-#define PA16_PF_I2C_SCL (GPIO_PORTA | GPIO_OUT | GPIO_PF | 16)
+-#define PA17_AF_ETMTRACEPKT4 (GPIO_PORTA | GPIO_AF | 17)
+-#define PA17_AIN_SPI2_SS (GPIO_GIUS | GPIO_PORTA | GPIO_OUT | 17)
+-#define PA18_AF_ETMTRACEPKT5 (GPIO_PORTA | GPIO_AF | 18)
+-#define PA19_AF_ETMTRACEPKT6 (GPIO_PORTA | GPIO_AF | 19)
+-#define PA20_AF_ETMTRACEPKT7 (GPIO_PORTA | GPIO_AF | 20)
+-#define PA21_PF_A0 (GPIO_PORTA | GPIO_PF | 21)
+-#define PA22_PF_CS4 (GPIO_PORTA | GPIO_PF | 22)
+-#define PA23_PF_CS5 (GPIO_PORTA | GPIO_PF | 23)
+-#define PA24_PF_A16 (GPIO_PORTA | GPIO_PF | 24)
+-#define PA24_AF_ETMTRACEPKT0 (GPIO_PORTA | GPIO_AF | 24)
+-#define PA25_PF_A17 (GPIO_PORTA | GPIO_PF | 25)
+-#define PA25_AF_ETMTRACEPKT1 (GPIO_PORTA | GPIO_AF | 25)
+-#define PA26_PF_A18 (GPIO_PORTA | GPIO_PF | 26)
+-#define PA26_AF_ETMTRACEPKT2 (GPIO_PORTA | GPIO_AF | 26)
+-#define PA27_PF_A19 (GPIO_PORTA | GPIO_PF | 27)
+-#define PA27_AF_ETMTRACEPKT3 (GPIO_PORTA | GPIO_AF | 27)
+-#define PA28_PF_A20 (GPIO_PORTA | GPIO_PF | 28)
+-#define PA28_AF_ETMPIPESTAT0 (GPIO_PORTA | GPIO_AF | 28)
+-#define PA29_PF_A21 (GPIO_PORTA | GPIO_PF | 29)
+-#define PA29_AF_ETMPIPESTAT1 (GPIO_PORTA | GPIO_AF | 29)
+-#define PA30_PF_A22 (GPIO_PORTA | GPIO_PF | 30)
+-#define PA30_AF_ETMPIPESTAT2 (GPIO_PORTA | GPIO_AF | 30)
+-#define PA31_PF_A23 (GPIO_PORTA | GPIO_PF | 31)
+-#define PA31_AF_ETMTRACECLK (GPIO_PORTA | GPIO_AF | 31)
+-#define PB8_PF_SD_DAT0 (GPIO_PORTB | GPIO_PF | GPIO_PUEN | 8)
+-#define PB8_AF_MS_PIO (GPIO_PORTB | GPIO_AF | 8)
+-#define PB9_PF_SD_DAT1 (GPIO_PORTB | GPIO_PF | GPIO_PUEN | 9)
+-#define PB9_AF_MS_PI1 (GPIO_PORTB | GPIO_AF | 9)
+-#define PB10_PF_SD_DAT2 (GPIO_PORTB | GPIO_PF | GPIO_PUEN | 10)
+-#define PB10_AF_MS_SCLKI (GPIO_PORTB | GPIO_AF | 10)
+-#define PB11_PF_SD_DAT3 (GPIO_PORTB | GPIO_PF | 11)
+-#define PB11_AF_MS_SDIO (GPIO_PORTB | GPIO_AF | 11)
+-#define PB12_PF_SD_CLK (GPIO_PORTB | GPIO_PF | 12)
+-#define PB12_AF_MS_SCLK0 (GPIO_PORTB | GPIO_AF | 12)
+-#define PB13_PF_SD_CMD (GPIO_PORTB | GPIO_PF | GPIO_PUEN | 13)
+-#define PB13_AF_MS_BS (GPIO_PORTB | GPIO_AF | 13)
+-#define PB14_AF_SSI_RXFS (GPIO_PORTB | GPIO_AF | 14)
+-#define PB15_AF_SSI_RXCLK (GPIO_PORTB | GPIO_AF | 15)
+-#define PB16_AF_SSI_RXDAT (GPIO_PORTB | GPIO_IN | GPIO_AF | 16)
+-#define PB17_AF_SSI_TXDAT (GPIO_PORTB | GPIO_OUT | GPIO_AF | 17)
+-#define PB18_AF_SSI_TXFS (GPIO_PORTB | GPIO_AF | 18)
+-#define PB19_AF_SSI_TXCLK (GPIO_PORTB | GPIO_AF | 19)
+-#define PB20_PF_USBD_AFE (GPIO_PORTB | GPIO_PF | 20)
+-#define PB21_PF_USBD_OE (GPIO_PORTB | GPIO_PF | 21)
+-#define PB22_PFUSBD_RCV (GPIO_PORTB | GPIO_PF | 22)
+-#define PB23_PF_USBD_SUSPND (GPIO_PORTB | GPIO_PF | 23)
+-#define PB24_PF_USBD_VP (GPIO_PORTB | GPIO_PF | 24)
+-#define PB25_PF_USBD_VM (GPIO_PORTB | GPIO_PF | 25)
+-#define PB26_PF_USBD_VPO (GPIO_PORTB | GPIO_PF | 26)
+-#define PB27_PF_USBD_VMO (GPIO_PORTB | GPIO_PF | 27)
+-#define PB28_PF_UART2_CTS (GPIO_PORTB | GPIO_OUT | GPIO_PF | 28)
+-#define PB29_PF_UART2_RTS (GPIO_PORTB | GPIO_IN | GPIO_PF | 29)
+-#define PB30_PF_UART2_TXD (GPIO_PORTB | GPIO_OUT | GPIO_PF | 30)
+-#define PB31_PF_UART2_RXD (GPIO_PORTB | GPIO_IN | GPIO_PF | 31)
+-#define PC3_PF_SSI_RXFS (GPIO_PORTC | GPIO_PF | 3)
+-#define PC4_PF_SSI_RXCLK (GPIO_PORTC | GPIO_PF | 4)
+-#define PC5_PF_SSI_RXDAT (GPIO_PORTC | GPIO_IN | GPIO_PF | 5)
+-#define PC6_PF_SSI_TXDAT (GPIO_PORTC | GPIO_OUT | GPIO_PF | 6)
+-#define PC7_PF_SSI_TXFS (GPIO_PORTC | GPIO_PF | 7)
+-#define PC8_PF_SSI_TXCLK (GPIO_PORTC | GPIO_PF | 8)
+-#define PC9_PF_UART1_CTS (GPIO_PORTC | GPIO_OUT | GPIO_PF | 9)
+-#define PC10_PF_UART1_RTS (GPIO_PORTC | GPIO_IN | GPIO_PF | 10)
+-#define PC11_PF_UART1_TXD (GPIO_PORTC | GPIO_OUT | GPIO_PF | 11)
+-#define PC12_PF_UART1_RXD (GPIO_PORTC | GPIO_IN | GPIO_PF | 12)
+-#define PC13_PF_SPI1_SPI_RDY (GPIO_PORTC | GPIO_PF | 13)
+-#define PC14_PF_SPI1_SCLK (GPIO_PORTC | GPIO_PF | 14)
+-#define PC15_PF_SPI1_SS (GPIO_PORTC | GPIO_PF | 15)
+-#define PC16_PF_SPI1_MISO (GPIO_PORTC | GPIO_PF | 16)
+-#define PC17_PF_SPI1_MOSI (GPIO_PORTC | GPIO_PF | 17)
+-#define PC24_BIN_UART3_RI (GPIO_GIUS | GPIO_PORTC | GPIO_OUT | GPIO_BIN | 24)
+-#define PC25_BIN_UART3_DSR (GPIO_GIUS | GPIO_PORTC | GPIO_OUT | GPIO_BIN | 25)
+-#define PC26_AOUT_UART3_DTR (GPIO_GIUS | GPIO_PORTC | GPIO_IN | 26)
+-#define PC27_BIN_UART3_DCD (GPIO_GIUS | GPIO_PORTC | GPIO_OUT | GPIO_BIN | 27)
+-#define PC28_BIN_UART3_CTS (GPIO_GIUS | GPIO_PORTC | GPIO_OUT | GPIO_BIN | 28)
+-#define PC29_AOUT_UART3_RTS (GPIO_GIUS | GPIO_PORTC | GPIO_IN | 29)
+-#define PC30_BIN_UART3_TX (GPIO_GIUS | GPIO_PORTC | GPIO_BIN | 30)
+-#define PC31_AOUT_UART3_RX (GPIO_GIUS | GPIO_PORTC | GPIO_IN | 31)
+-#define PD6_PF_LSCLK (GPIO_PORTD | GPIO_OUT | GPIO_PF | 6)
+-#define PD7_PF_REV (GPIO_PORTD | GPIO_PF | 7)
+-#define PD7_AF_UART2_DTR (GPIO_GIUS | GPIO_PORTD | GPIO_IN | GPIO_AF | 7)
+-#define PD7_AIN_SPI2_SCLK (GPIO_GIUS | GPIO_PORTD | GPIO_AIN | 7)
+-#define PD8_PF_CLS (GPIO_PORTD | GPIO_PF | 8)
+-#define PD8_AF_UART2_DCD (GPIO_PORTD | GPIO_OUT | GPIO_AF | 8)
+-#define PD8_AIN_SPI2_SS (GPIO_GIUS | GPIO_PORTD | GPIO_AIN | 8)
+-#define PD9_PF_PS (GPIO_PORTD | GPIO_PF | 9)
+-#define PD9_AF_UART2_RI (GPIO_PORTD | GPIO_OUT | GPIO_AF | 9)
+-#define PD9_AOUT_SPI2_RXD (GPIO_GIUS | GPIO_PORTD | GPIO_IN | 9)
+-#define PD10_PF_SPL_SPR (GPIO_PORTD | GPIO_OUT | GPIO_PF | 10)
+-#define PD10_AF_UART2_DSR (GPIO_PORTD | GPIO_OUT | GPIO_AF | 10)
+-#define PD10_AIN_SPI2_TXD (GPIO_GIUS | GPIO_PORTD | GPIO_OUT | 10)
+-#define PD11_PF_CONTRAST (GPIO_PORTD | GPIO_OUT | GPIO_PF | 11)
+-#define PD12_PF_ACD_OE (GPIO_PORTD | GPIO_OUT | GPIO_PF | 12)
+-#define PD13_PF_LP_HSYNC (GPIO_PORTD | GPIO_OUT | GPIO_PF | 13)
+-#define PD14_PF_FLM_VSYNC (GPIO_PORTD | GPIO_OUT | GPIO_PF | 14)
+-#define PD15_PF_LD0 (GPIO_PORTD | GPIO_OUT | GPIO_PF | 15)
+-#define PD16_PF_LD1 (GPIO_PORTD | GPIO_OUT | GPIO_PF | 16)
+-#define PD17_PF_LD2 (GPIO_PORTD | GPIO_OUT | GPIO_PF | 17)
+-#define PD18_PF_LD3 (GPIO_PORTD | GPIO_OUT | GPIO_PF | 18)
+-#define PD19_PF_LD4 (GPIO_PORTD | GPIO_OUT | GPIO_PF | 19)
+-#define PD20_PF_LD5 (GPIO_PORTD | GPIO_OUT | GPIO_PF | 20)
+-#define PD21_PF_LD6 (GPIO_PORTD | GPIO_OUT | GPIO_PF | 21)
+-#define PD22_PF_LD7 (GPIO_PORTD | GPIO_OUT | GPIO_PF | 22)
+-#define PD23_PF_LD8 (GPIO_PORTD | GPIO_OUT | GPIO_PF | 23)
+-#define PD24_PF_LD9 (GPIO_PORTD | GPIO_OUT | GPIO_PF | 24)
+-#define PD25_PF_LD10 (GPIO_PORTD | GPIO_OUT | GPIO_PF | 25)
+-#define PD26_PF_LD11 (GPIO_PORTD | GPIO_OUT | GPIO_PF | 26)
+-#define PD27_PF_LD12 (GPIO_PORTD | GPIO_OUT | GPIO_PF | 27)
+-#define PD28_PF_LD13 (GPIO_PORTD | GPIO_OUT | GPIO_PF | 28)
+-#define PD29_PF_LD14 (GPIO_PORTD | GPIO_OUT | GPIO_PF | 29)
+-#define PD30_PF_LD15 (GPIO_PORTD | GPIO_OUT | GPIO_PF | 30)
+-#define PD31_PF_TMR2OUT (GPIO_PORTD | GPIO_PF | 31)
+-#define PD31_BIN_SPI2_TXD (GPIO_GIUS | GPIO_PORTD | GPIO_BIN | 31)
+-#endif
+-
+-#ifdef CONFIG_ARCH_MX2
+-#define PA0_PF_USBH2_CLK (GPIO_PORTA | GPIO_PF | 0)
+-#define PA1_PF_USBH2_DIR (GPIO_PORTA | GPIO_PF | 1)
+-#define PA2_PF_USBH2_DATA7 (GPIO_PORTA | GPIO_PF | 2)
+-#define PA3_PF_USBH2_NXT (GPIO_PORTA | GPIO_PF | 3)
+-#define PA4_PF_USBH2_STP (GPIO_PORTA | GPIO_PF | 4)
+-#define PA5_PF_LSCLK (GPIO_PORTA | GPIO_OUT | GPIO_PF | 5)
+-#define PA6_PF_LD0 (GPIO_PORTA | GPIO_OUT | GPIO_PF | 6)
+-#define PA7_PF_LD1 (GPIO_PORTA | GPIO_OUT | GPIO_PF | 7)
+-#define PA8_PF_LD2 (GPIO_PORTA | GPIO_OUT | GPIO_PF | 8)
+-#define PA9_PF_LD3 (GPIO_PORTA | GPIO_OUT | GPIO_PF | 9)
+-#define PA10_PF_LD4 (GPIO_PORTA | GPIO_OUT | GPIO_PF | 10)
+-#define PA11_PF_LD5 (GPIO_PORTA | GPIO_OUT | GPIO_PF | 11)
+-#define PA12_PF_LD6 (GPIO_PORTA | GPIO_OUT | GPIO_PF | 12)
+-#define PA13_PF_LD7 (GPIO_PORTA | GPIO_OUT | GPIO_PF | 13)
+-#define PA14_PF_LD8 (GPIO_PORTA | GPIO_OUT | GPIO_PF | 14)
+-#define PA15_PF_LD9 (GPIO_PORTA | GPIO_OUT | GPIO_PF | 15)
+-#define PA16_PF_LD10 (GPIO_PORTA | GPIO_OUT | GPIO_PF | 16)
+-#define PA17_PF_LD11 (GPIO_PORTA | GPIO_OUT | GPIO_PF | 17)
+-#define PA18_PF_LD12 (GPIO_PORTA | GPIO_OUT | GPIO_PF | 18)
+-#define PA19_PF_LD13 (GPIO_PORTA | GPIO_OUT | GPIO_PF | 19)
+-#define PA20_PF_LD14 (GPIO_PORTA | GPIO_OUT | GPIO_PF | 20)
+-#define PA21_PF_LD15 (GPIO_PORTA | GPIO_OUT | GPIO_PF | 21)
+-#define PA22_PF_LD16 (GPIO_PORTA | GPIO_OUT | GPIO_PF | 22)
+-#define PA23_PF_LD17 (GPIO_PORTA | GPIO_OUT | GPIO_PF | 23)
+-#define PA24_PF_REV (GPIO_PORTA | GPIO_OUT | GPIO_PF | 24)
+-#define PA25_PF_CLS (GPIO_PORTA | GPIO_OUT | GPIO_PF | 25)
+-#define PA26_PF_PS (GPIO_PORTA | GPIO_OUT | GPIO_PF | 26)
+-#define PA27_PF_SPL_SPR (GPIO_PORTA | GPIO_OUT | GPIO_PF | 27)
+-#define PA28_PF_HSYNC (GPIO_PORTA | GPIO_OUT | GPIO_PF | 28)
+-#define PA29_PF_VSYNC (GPIO_PORTA | GPIO_OUT | GPIO_PF | 29)
+-#define PA30_PF_CONTRAST (GPIO_PORTA | GPIO_OUT | GPIO_PF | 30)
+-#define PA31_PF_OE_ACD (GPIO_PORTA | GPIO_OUT | GPIO_PF | 31)
+-#define PB10_PF_CSI_D0 (GPIO_PORTB | GPIO_OUT | GPIO_PF | 10)
+-#define PB10_AF_UART6_TXD (GPIO_PORTB | GPIO_OUT | GPIO_AF | 10)
+-#define PB11_PF_CSI_D1 (GPIO_PORTB | GPIO_OUT | GPIO_PF | 11)
+-#define PB11_AF_UART6_RXD (GPIO_PORTB | GPIO_IN | GPIO_AF | 11)
+-#define PB12_PF_CSI_D2 (GPIO_PORTB | GPIO_OUT | GPIO_PF | 12)
+-#define PB12_AF_UART6_CTS (GPIO_PORTB | GPIO_OUT | GPIO_AF | 12)
+-#define PB13_PF_CSI_D3 (GPIO_PORTB | GPIO_OUT | GPIO_PF | 13)
+-#define PB13_AF_UART6_RTS (GPIO_PORTB | GPIO_IN | GPIO_AF | 13)
+-#define PB14_PF_CSI_D4 (GPIO_PORTB | GPIO_OUT | GPIO_PF | 14)
+-#define PB15_PF_CSI_MCLK (GPIO_PORTB | GPIO_OUT | GPIO_PF | 15)
+-#define PB16_PF_CSI_PIXCLK (GPIO_PORTB | GPIO_OUT | GPIO_PF | 16)
+-#define PB17_PF_CSI_D5 (GPIO_PORTB | GPIO_OUT | GPIO_PF | 17)
+-#define PB18_PF_CSI_D6 (GPIO_PORTB | GPIO_OUT | GPIO_PF | 18)
+-#define PB18_AF_UART5_TXD (GPIO_PORTB | GPIO_OUT | GPIO_AF | 18)
+-#define PB19_PF_CSI_D7 (GPIO_PORTB | GPIO_OUT | GPIO_PF | 19)
+-#define PB19_AF_UART5_RXD (GPIO_PORTB | GPIO_IN | GPIO_AF | 19)
+-#define PB20_PF_CSI_VSYNC (GPIO_PORTB | GPIO_OUT | GPIO_PF | 20)
+-#define PB20_AF_UART5_CTS (GPIO_PORTB | GPIO_OUT | GPIO_AF | 20)
+-#define PB21_PF_CSI_HSYNC (GPIO_PORTB | GPIO_OUT | GPIO_PF | 21)
+-#define PB21_AF_UART5_RTS (GPIO_PORTB | GPIO_IN | GPIO_AF | 21)
+-#define PB22_PF_USBH1_SUSP (GPIO_PORTB | GPIO_PF | 22)
+-#define PB23_PF_USB_PWR (GPIO_PORTB | GPIO_PF | 23)
+-#define PB24_PF_USB_OC_B (GPIO_PORTB | GPIO_PF | 24)
+-#define PB25_PF_USBH1_RCV (GPIO_PORTB | GPIO_PF | 25)
+-#define PB26_PF_USBH1_FS (GPIO_PORTB | GPIO_PF | 26)
+-#define PB27_PF_USBH1_OE_B (GPIO_PORTB | GPIO_PF | 27)
+-#define PB28_PF_USBH1_TXDM (GPIO_PORTB | GPIO_PF | 28)
+-#define PB29_PF_USBH1_TXDP (GPIO_PORTB | GPIO_PF | 29)
+-#define PB30_PF_USBH1_RXDM (GPIO_PORTB | GPIO_PF | 30)
+-#define PB31_PF_USBH1_RXDP (GPIO_PORTB | GPIO_PF | 31)
+-#define PB26_AF_UART4_RTS (GPIO_PORTB | GPIO_IN | GPIO_PF | 26)
+-#define PB28_AF_UART4_TXD (GPIO_PORTB | GPIO_OUT | GPIO_AF | 28)
+-#define PB29_AF_UART4_CTS (GPIO_PORTB | GPIO_OUT | GPIO_AF | 29)
+-#define PB31_AF_UART4_RXD (GPIO_PORTB | GPIO_IN | GPIO_AF | 31)
+-#define PC5_PF_I2C2_SDA (GPIO_PORTC | GPIO_IN | GPIO_PF | 5)
+-#define PC6_PF_I2C2_SCL (GPIO_PORTC | GPIO_IN | GPIO_PF | 6)
+-#define PC16_PF_SSI4_FS (GPIO_PORTC | GPIO_IN | GPIO_PF | 16)
+-#define PC17_PF_SSI4_RXD (GPIO_PORTC | GPIO_IN | GPIO_PF | 17)
+-#define PC18_PF_SSI4_TXD (GPIO_PORTC | GPIO_IN | GPIO_PF | 18)
+-#define PC19_PF_SSI4_CLK (GPIO_PORTC | GPIO_IN | GPIO_PF | 19)
+-#define PC20_PF_SSI1_FS (GPIO_PORTC | GPIO_IN | GPIO_PF | 20)
+-#define PC21_PF_SSI1_RXD (GPIO_PORTC | GPIO_IN | GPIO_PF | 21)
+-#define PC22_PF_SSI1_TXD (GPIO_PORTC | GPIO_IN | GPIO_PF | 22)
+-#define PC23_PF_SSI1_CLK (GPIO_PORTC | GPIO_IN | GPIO_PF | 23)
+-#define PC24_PF_SSI2_FS (GPIO_PORTC | GPIO_IN | GPIO_PF | 24)
+-#define PC25_PF_SSI2_RXD (GPIO_PORTC | GPIO_IN | GPIO_PF | 25)
+-#define PC26_PF_SSI2_TXD (GPIO_PORTC | GPIO_IN | GPIO_PF | 26)
+-#define PC27_PF_SSI2_CLK (GPIO_PORTC | GPIO_IN | GPIO_PF | 27)
+-#define PC28_PF_SSI3_FS (GPIO_PORTC | GPIO_IN | GPIO_PF | 28)
+-#define PC29_PF_SSI3_RXD (GPIO_PORTC | GPIO_IN | GPIO_PF | 29)
+-#define PC30_PF_SSI3_TXD (GPIO_PORTC | GPIO_IN | GPIO_PF | 30)
+-#define PC31_PF_SSI3_CLK (GPIO_PORTC | GPIO_IN | GPIO_PF | 31)
+-#define PD0_AIN_FEC_TXD0 (GPIO_PORTD | GPIO_OUT | GPIO_AIN | 0)
+-#define PD1_AIN_FEC_TXD1 (GPIO_PORTD | GPIO_OUT | GPIO_AIN | 1)
+-#define PD2_AIN_FEC_TXD2 (GPIO_PORTD | GPIO_OUT | GPIO_AIN | 2)
+-#define PD3_AIN_FEC_TXD3 (GPIO_PORTD | GPIO_OUT | GPIO_AIN | 3)
+-#define PD4_AOUT_FEC_RX_ER (GPIO_PORTD | GPIO_IN | GPIO_AOUT | 4)
+-#define PD5_AOUT_FEC_RXD1 (GPIO_PORTD | GPIO_IN | GPIO_AOUT | 5)
+-#define PD6_AOUT_FEC_RXD2 (GPIO_PORTD | GPIO_IN | GPIO_AOUT | 6)
+-#define PD7_AOUT_FEC_RXD3 (GPIO_PORTD | GPIO_IN | GPIO_AOUT | 7)
+-#define PD8_AF_FEC_MDIO (GPIO_PORTD | GPIO_IN | GPIO_AF | 8)
+-#define PD9_AIN_FEC_MDC (GPIO_PORTD | GPIO_OUT | GPIO_AIN | 9)
+-#define PD10_AOUT_FEC_CRS (GPIO_PORTD | GPIO_IN | GPIO_AOUT | 10)
+-#define PD11_AOUT_FEC_TX_CLK (GPIO_PORTD | GPIO_IN | GPIO_AOUT | 11)
+-#define PD12_AOUT_FEC_RXD0 (GPIO_PORTD | GPIO_IN | GPIO_AOUT | 12)
+-#define PD13_AOUT_FEC_RX_DV (GPIO_PORTD | GPIO_IN | GPIO_AOUT | 13)
+-#define PD14_AOUT_FEC_CLR (GPIO_PORTD | GPIO_IN | GPIO_AOUT | 14)
+-#define PD15_AOUT_FEC_COL (GPIO_PORTD | GPIO_IN | GPIO_AOUT | 15)
+-#define PD16_AIN_FEC_TX_ER (GPIO_PORTD | GPIO_OUT | GPIO_AIN | 16)
+-#define PD17_PF_I2C_DATA (GPIO_PORTD | GPIO_OUT | GPIO_PF | 17)
+-#define PD18_PF_I2C_CLK (GPIO_PORTD | GPIO_OUT | GPIO_PF | 18)
+-#define PD19_AF_USBH2_DATA4 (GPIO_PORTD | GPIO_AF | 19)
+-#define PD20_AF_USBH2_DATA3 (GPIO_PORTD | GPIO_AF | 20)
+-#define PD21_AF_USBH2_DATA6 (GPIO_PORTD | GPIO_AF | 21)
+-#define PD22_AF_USBH2_DATA0 (GPIO_PORTD | GPIO_AF | 22)
+-#define PD23_AF_USBH2_DATA2 (GPIO_PORTD | GPIO_AF | 23)
+-#define PD24_AF_USBH2_DATA1 (GPIO_PORTD | GPIO_AF | 24)
+-#define PD25_PF_CSPI1_RDY (GPIO_PORTD | GPIO_OUT | GPIO_PF | 25)
+-#define PD26_PF_CSPI1_SS2 (GPIO_PORTD | GPIO_OUT | GPIO_PF | 26)
+-#define PD26_AF_USBH2_DATA5 (GPIO_PORTD | GPIO_AF | 26)
+-#define PD27_PF_CSPI1_SS1 (GPIO_PORTD | GPIO_OUT | GPIO_PF | 27)
+-#define PD28_PF_CSPI1_SS0 (GPIO_PORTD | GPIO_OUT | GPIO_PF | 28)
+-#define PD29_PF_CSPI1_SCLK (GPIO_PORTD | GPIO_OUT | GPIO_PF | 29)
+-#define PD30_PF_CSPI1_MISO (GPIO_PORTD | GPIO_IN | GPIO_PF | 30)
+-#define PD31_PF_CSPI1_MOSI (GPIO_PORTD | GPIO_OUT | GPIO_PF | 31)
+-#define PF23_AIN_FEC_TX_EN (GPIO_PORTF | GPIO_OUT | GPIO_AIN | 23)
+-#define PE3_PF_UART2_CTS (GPIO_PORTE | GPIO_OUT | GPIO_PF | 3)
+-#define PE4_PF_UART2_RTS (GPIO_PORTE | GPIO_IN | GPIO_PF | 4)
+-#define PE6_PF_UART2_TXD (GPIO_PORTE | GPIO_OUT | GPIO_PF | 6)
+-#define PE7_PF_UART2_RXD (GPIO_PORTE | GPIO_IN | GPIO_PF | 7)
+-#define PE8_PF_UART3_TXD (GPIO_PORTE | GPIO_OUT | GPIO_PF | 8)
+-#define PE9_PF_UART3_RXD (GPIO_PORTE | GPIO_IN | GPIO_PF | 9)
+-#define PE10_PF_UART3_CTS (GPIO_PORTE | GPIO_OUT | GPIO_PF | 10)
+-#define PE11_PF_UART3_RTS (GPIO_PORTE | GPIO_IN | GPIO_PF | 11)
+-#define PE12_PF_UART1_TXD (GPIO_PORTE | GPIO_OUT | GPIO_PF | 12)
+-#define PE13_PF_UART1_RXD (GPIO_PORTE | GPIO_IN | GPIO_PF | 13)
+-#define PE14_PF_UART1_CTS (GPIO_PORTE | GPIO_OUT | GPIO_PF | 14)
+-#define PE15_PF_UART1_RTS (GPIO_PORTE | GPIO_IN | GPIO_PF | 15)
+-#define PE16_AF_RTCK (GPIO_PORTE | GPIO_OUT | GPIO_AF | 16)
+-#define PE16_PF_RTCK (GPIO_PORTE | GPIO_OUT | GPIO_PF | 16)
+-#define PE18_AF_CSPI3_MISO (GPIO_PORTE | GPIO_IN | GPIO_AF | 18)
+-#define PE21_AF_CSPI3_SS (GPIO_PORTE | GPIO_OUT | GPIO_AF | 21)
+-#define PE22_AF_CSPI3_MOSI (GPIO_PORTE | GPIO_OUT | GPIO_AF | 22)
+-#define PE23_AF_CSPI3_SCLK (GPIO_PORTE | GPIO_OUT | GPIO_AF | 23)
+-#endif
++#define MXC_PIN(port,gpio,fn,flags) \
++ (GPIO_PORT##port | GPIO_##fn | (flags) | (gpio))
++#define MXC_DEFINE_PIN(port,gpio,fn,name,flags) \
++ P##port##gpio##_##fn##_##name = MXC_PIN(port,gpio,fn,flags)
+
+ /* decode irq number to use with IMR(x), ISR(x) and friends */
+-#define IRQ_TO_REG(irq) ((irq - MXC_MAX_INT_LINES) >> 5)
++#define IRQ_TO_REG(irq) ((irq - MXC_MAX_INT_LINES) >> 5)
+
+-#define IRQ_GPIOA(x) (MXC_MAX_INT_LINES + x)
+-#define IRQ_GPIOB(x) (IRQ_GPIOA(32) + x)
+-#define IRQ_GPIOC(x) (IRQ_GPIOB(32) + x)
+-#define IRQ_GPIOD(x) (IRQ_GPIOC(32) + x)
++#define IRQ_GPIOA(x) (MXC_MAX_INT_LINES + x)
++#define IRQ_GPIOB(x) (IRQ_GPIOA(32) + x)
++#define IRQ_GPIOC(x) (IRQ_GPIOB(32) + x)
++#define IRQ_GPIOD(x) (IRQ_GPIOC(32) + x)
++#define IRQ_GPIOE(x) (IRQ_GPIOD(32) + x)
+
+ #endif /* _MXC_GPIO_MX1_MX2_H */
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/include/mach/ipu.h linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/ipu.h
+--- linux-2.6.28/arch/arm/plat-mxc/include/mach/ipu.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/ipu.h 2009-01-05 17:47:15.000000000 +0100
+@@ -0,0 +1,193 @@
++/*
++ * Copyright (C) 2008
++ * Guennadi Liakhovetski, DENX Software Engineering, <lg@denx.de>
++ *
++ * 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 _IPU_H_
++#define _IPU_H_
++
++#include <linux/types.h>
++#include <linux/dmaengine.h>
++
++/* IPU DMA Controller channel definitions. */
++enum ipu_channel {
++ IDMAC_IC_0 = 0, /* IC (encoding task) to memory */
++ IDMAC_IC_1 = 1, /* IC (viewfinder task) to memory */
++ IDMAC_ADC_0 = 1,
++ IDMAC_IC_2 = 2,
++ IDMAC_ADC_1 = 2,
++ IDMAC_IC_3 = 3,
++ IDMAC_IC_4 = 4,
++ IDMAC_IC_5 = 5,
++ IDMAC_IC_6 = 6,
++ IDMAC_IC_7 = 7, /* IC (sensor data) to memory */
++ IDMAC_IC_8 = 8,
++ IDMAC_IC_9 = 9,
++ IDMAC_IC_10 = 10,
++ IDMAC_IC_11 = 11,
++ IDMAC_IC_12 = 12,
++ IDMAC_IC_13 = 13,
++ IDMAC_SDC_0 = 14, /* Background synchronous display data */
++ IDMAC_SDC_1 = 15, /* Foreground data (overlay) */
++ IDMAC_SDC_2 = 16,
++ IDMAC_SDC_3 = 17,
++ IDMAC_ADC_2 = 18,
++ IDMAC_ADC_3 = 19,
++ IDMAC_ADC_4 = 20,
++ IDMAC_ADC_5 = 21,
++ IDMAC_ADC_6 = 22,
++ IDMAC_ADC_7 = 23,
++ IDMAC_PF_0 = 24,
++ IDMAC_PF_1 = 25,
++ IDMAC_PF_2 = 26,
++ IDMAC_PF_3 = 27,
++ IDMAC_PF_4 = 28,
++ IDMAC_PF_5 = 29,
++ IDMAC_PF_6 = 30,
++ IDMAC_PF_7 = 31,
++};
++
++/* Order significant! */
++enum ipu_channel_status {
++ IPU_CHANNEL_FREE,
++ IPU_CHANNEL_GRANTED,
++ IPU_CHANNEL_INITIALIZED,
++ IPU_CHANNEL_READY,
++ IPU_CHANNEL_ENABLED,
++};
++
++#define IPU_CHANNELS_NUM 32
++
++enum pixel_fmt {
++ /* 1 byte */
++ IPU_PIX_FMT_GENERIC,
++ IPU_PIX_FMT_RGB332,
++ IPU_PIX_FMT_YUV420P,
++ IPU_PIX_FMT_YUV422P,
++ IPU_PIX_FMT_YUV420P2,
++ IPU_PIX_FMT_YVU422P,
++ /* 2 bytes */
++ IPU_PIX_FMT_RGB565,
++ IPU_PIX_FMT_RGB666,
++ IPU_PIX_FMT_BGR666,
++ IPU_PIX_FMT_YUYV,
++ IPU_PIX_FMT_UYVY,
++ /* 3 bytes */
++ IPU_PIX_FMT_RGB24,
++ IPU_PIX_FMT_BGR24,
++ /* 4 bytes */
++ IPU_PIX_FMT_GENERIC_32,
++ IPU_PIX_FMT_RGB32,
++ IPU_PIX_FMT_BGR32,
++ IPU_PIX_FMT_ABGR32,
++ IPU_PIX_FMT_BGRA32,
++ IPU_PIX_FMT_RGBA32,
++};
++
++enum ipu_color_space {
++ IPU_COLORSPACE_RGB,
++ IPU_COLORSPACE_YCBCR,
++ IPU_COLORSPACE_YUV
++};
++
++/**
++ * Enumeration of IPU rotation modes
++ */
++enum ipu_rotate_mode {
++ /* Note the enum values correspond to BAM value */
++ IPU_ROTATE_NONE = 0,
++ IPU_ROTATE_VERT_FLIP = 1,
++ IPU_ROTATE_HORIZ_FLIP = 2,
++ IPU_ROTATE_180 = 3,
++ IPU_ROTATE_90_RIGHT = 4,
++ IPU_ROTATE_90_RIGHT_VFLIP = 5,
++ IPU_ROTATE_90_RIGHT_HFLIP = 6,
++ IPU_ROTATE_90_LEFT = 7,
++};
++
++struct ipu_platform_data {
++ unsigned int irq_base;
++};
++
++/**
++ * Enumeration of DI ports for ADC.
++ */
++typedef enum {
++ DISP0,
++ DISP1,
++ DISP2,
++ DISP3
++} display_port_t;
++
++struct idmac_video_param {
++ unsigned short in_width;
++ unsigned short in_height;
++ uint32_t in_pixel_fmt;
++ unsigned short out_width;
++ unsigned short out_height;
++ uint32_t out_pixel_fmt;
++ unsigned short out_stride;
++ bool graphics_combine_en;
++ bool global_alpha_en;
++ bool key_color_en;
++ display_port_t disp;
++ unsigned short out_left;
++ unsigned short out_top;
++};
++
++/*
++ * Union of initialization parameters for a logical channel. So far only video
++ * parameters are used.
++ */
++union ipu_channel_param {
++ struct idmac_video_param video;
++};
++
++struct idmac_tx_desc {
++ struct dma_async_tx_descriptor txd;
++ struct scatterlist *sg; /* scatterlist for this */
++ unsigned int sg_len; /* tx-descriptor. */
++ struct list_head list;
++};
++
++struct idmac_channel {
++ struct dma_chan dma_chan;
++ dma_cookie_t completed; /* last completed cookie */
++ union ipu_channel_param params;
++ enum ipu_channel link; /* input channel, linked to the output */
++ enum ipu_channel_status status;
++ struct idmac_client *iclient; /* Only one client per channel */
++ unsigned int n_tx_desc;
++ struct idmac_tx_desc *desc; /* allocated tx-descriptors */
++ struct scatterlist *sg[2]; /* scatterlist elements in buffer-0 and -1 */
++ struct list_head free_list; /* free tx-descriptors */
++ struct list_head queue; /* queued tx-descriptors */
++ spinlock_t lock; /* protects sg[0,1], queue */
++ struct mutex chan_mutex; /* protects status, cookie, free_list */
++ unsigned int eof_irq;
++ bool sec_chan_en;
++ int active_buffer;
++};
++
++struct idmac_channel_rq {
++ enum ipu_channel channel;
++ struct idmac_channel *ichannel;
++};
++
++struct idmac_client {
++ int n_channels;
++ struct idmac_channel_rq *channels;
++ struct dma_client dma_client;
++};
++
++extern unsigned long ipu_clk_get_rate(void);
++
++#define to_tx_desc(tx) container_of(tx, struct idmac_tx_desc, txd)
++#define to_idmac_chan(c) container_of(c, struct idmac_channel, dma_chan)
++#define to_idmac_client(i) container_of(i, struct idmac_client, dma_client)
++
++#endif
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/include/mach/irqs.h linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/irqs.h
+--- linux-2.6.28/arch/arm/plat-mxc/include/mach/irqs.h 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/irqs.h 2009-03-11 13:16:24.000000000 +0100
+@@ -14,4 +14,9 @@
+ #include <mach/hardware.h>
+ extern void imx_irq_set_priority(unsigned char irq, unsigned char prio);
+
++/* all normal IRQs can be FIQs */
++#define FIQ_START 0
++/* switch between IRQ and FIQ */
++extern int mxc_set_irq_fiq(unsigned int irq, unsigned int type);
++
+ #endif /* __ASM_ARCH_MXC_IRQS_H__ */
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/include/mach/memory.h linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/memory.h
+--- linux-2.6.28/arch/arm/plat-mxc/include/mach/memory.h 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/memory.h 2009-03-11 13:16:24.000000000 +0100
+@@ -26,4 +26,5 @@
+ */
+ #define __bus_to_virt(a) __phys_to_virt(a)
+
++#define CONSISTENT_DMA_SIZE (14 * SZ_1M)
+ #endif /* __ASM_ARCH_MXC_MEMORY_H__ */
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/include/mach/mmc.h linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/mmc.h
+--- linux-2.6.28/arch/arm/plat-mxc/include/mach/mmc.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/mmc.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,19 @@
++#ifndef ASMARM_ARCH_MMC_H
++#define ASMARM_ARCH_MMC_H
++
++#include <linux/mmc/host.h>
++
++struct device;
++
++struct imxmmc_platform_data {
++ int (*get_ro)(struct device *);
++ int (*init)(struct device *, irq_handler_t, void *);
++ void (*exit)(struct device *, void *);
++ void (*setpower)(struct device *, unsigned int vdd);
++ int (*suspend)(struct device *, pm_message_t state);
++ int (*resume)(struct device *);
++};
++
++extern void imx_set_mmc_info(struct imxmmc_platform_data *info);
++
++#endif
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/include/mach/mx27.h linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/mx27.h
+--- linux-2.6.28/arch/arm/plat-mxc/include/mach/mx27.h 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/mx27.h 2009-03-11 13:16:24.000000000 +0100
+@@ -24,12 +24,20 @@
+ #error "Do not include directly."
+ #endif
+
++/* the DMA code supports SG list chaining */
++#define ARCH_HAS_SG_CHAIN
++
+ /* IRAM */
+ #define IRAM_BASE_ADDR 0xFFFF4C00 /* internal ram */
+
+ /* Register offests */
++#ifndef __ASSEMBLY__
++#define AIPI_BASE_ADDR 0x10000000UL
++#define AIPI_BASE_ADDR_VIRT ((void __iomem *)0xF4000000)
++#else
+ #define AIPI_BASE_ADDR 0x10000000
+ #define AIPI_BASE_ADDR_VIRT 0xF4000000
++#endif
+ #define AIPI_SIZE SZ_1M
+
+ #define DMA_BASE_ADDR (AIPI_BASE_ADDR + 0x01000)
+@@ -72,7 +80,8 @@
+ /* for mx27*/
+ #define OTG_BASE_ADDR USBOTG_BASE_ADDR
+ #define SAHARA_BASE_ADDR (AIPI_BASE_ADDR + 0x25000)
+-#define EMMA_BASE_ADDR (AIPI_BASE_ADDR + 0x26400)
++#define EMMA_PP_BASE_ADDR (AIPI_BASE_ADDR + 0x26000)
++#define EMMA_PRP_BASE_ADDR (AIPI_BASE_ADDR + 0x26400)
+ #define CCM_BASE_ADDR (AIPI_BASE_ADDR + 0x27000)
+ #define SYSCTRL_BASE_ADDR (AIPI_BASE_ADDR + 0x27800)
+ #define IIM_BASE_ADDR (AIPI_BASE_ADDR + 0x28000)
+@@ -91,16 +100,26 @@
+
+ #define AVIC_BASE_ADDR 0x10040000
+
++#ifndef __ASSEMBLY__
++#define SAHB1_BASE_ADDR 0x80000000UL
++#define SAHB1_BASE_ADDR_VIRT ((void __iomem *)0xF4100000UL)
++#else
+ #define SAHB1_BASE_ADDR 0x80000000
+ #define SAHB1_BASE_ADDR_VIRT 0xF4100000
++#endif
+ #define SAHB1_SIZE SZ_1M
+
+ #define CSI_BASE_ADDR (SAHB1_BASE_ADDR + 0x0000)
+ #define ATA_BASE_ADDR (SAHB1_BASE_ADDR + 0x1000)
+
+ /* NAND, SDRAM, WEIM, M3IF, EMI controllers */
++#ifndef __ASSEMBLY__
++#define X_MEMC_BASE_ADDR 0xD8000000UL
++#define X_MEMC_BASE_ADDR_VIRT ((void __iomem *)0xF4200000UL)
++#else
+ #define X_MEMC_BASE_ADDR 0xD8000000
+ #define X_MEMC_BASE_ADDR_VIRT 0xF4200000
++#endif
+ #define X_MEMC_SIZE SZ_1M
+
+ #define NFC_BASE_ADDR (X_MEMC_BASE_ADDR)
+@@ -127,14 +146,24 @@
+ * and returning the virtual address. If the physical address is not mapped,
+ * it returns 0xDEADBEEF
+ */
+-#define IO_ADDRESS(x) \
+- (void __iomem *) \
+- (((x >= AIPI_BASE_ADDR) && (x < (AIPI_BASE_ADDR + AIPI_SIZE))) ? \
+- AIPI_IO_ADDRESS(x) : \
+- ((x >= SAHB1_BASE_ADDR) && (x < (SAHB1_BASE_ADDR + SAHB1_SIZE))) ? \
+- SAHB1_IO_ADDRESS(x) : \
+- ((x >= X_MEMC_BASE_ADDR) && (x < (X_MEMC_BASE_ADDR + X_MEMC_SIZE))) ? \
+- X_MEMC_IO_ADDRESS(x) : 0xDEADBEEF)
++#define IO_ADDRESS(x) \
++ ((((x) >= AIPI_BASE_ADDR) && ((x) < (AIPI_BASE_ADDR + AIPI_SIZE))) ? \
++ AIPI_IO_ADDRESS(x) : \
++ (((x) >= SAHB1_BASE_ADDR) && ((x) < (SAHB1_BASE_ADDR + SAHB1_SIZE))) ? \
++ SAHB1_IO_ADDRESS(x) : \
++ (((x) >= X_MEMC_BASE_ADDR) && ((x) < (X_MEMC_BASE_ADDR + X_MEMC_SIZE))) ? \
++ X_MEMC_IO_ADDRESS(x) : NULL)
++
++#define MXC_VADDR_RANGE(v,n) \
++ (((v)) >= n##_BASE_ADDR_VIRT) && \
++ (((v)) < n##_BASE_ADDR_VIRT + n##_SIZE) ? \
++ ((v)-n##_BASE_ADDR_VIRT + n##_BASE_ADDR) :
++
++#define MXC_PHYS_ADDRESS(v) \
++ (unsigned long)(MXC_VADDR_RANGE(v,AIPI) \
++ MXC_VADDR_RANGE(v,SAHB1) \
++ MXC_VADDR_RANGE(v,X_MEMC) \
++ 0UL)
+
+ /* define the address mapping macros: in physical address order */
+ #define AIPI_IO_ADDRESS(x) \
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/include/mach/mxc_ehci.h linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/mxc_ehci.h
+--- linux-2.6.28/arch/arm/plat-mxc/include/mach/mxc_ehci.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/mxc_ehci.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,9 @@
++#ifndef __INCLUDE_ASM_ARCH_MXC_EHCI_H
++#define __INCLUDE_ASM_ARCH_MXC_EHCI_H
++
++struct mxc_usbh_platform_data {
++ int (*init)(struct platform_device *pdev);
++ int (*exit)(struct platform_device *pdev);
++};
++#endif /* __INCLUDE_ASM_ARCH_MXC_EHCI_H */
++
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/include/mach/mxc_timer.h linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/mxc_timer.h
+--- linux-2.6.28/arch/arm/plat-mxc/include/mach/mxc_timer.h 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/mxc_timer.h 2009-03-11 13:16:24.000000000 +0100
+@@ -26,7 +26,7 @@
+ #include <linux/clk.h>
+ #include <mach/hardware.h>
+
+-#ifdef CONFIG_ARCH_IMX
++#ifdef CONFIG_ARCH_MX1
+ #define TIMER_BASE IO_ADDRESS(TIM1_BASE_ADDR)
+ #define TIMER_INTERRUPT TIM1_INT
+
+@@ -65,7 +65,7 @@ static void gpt_irq_acknowledge(void)
+ {
+ __raw_writel(0, TIMER_BASE + MXC_TSTAT);
+ }
+-#endif /* CONFIG_ARCH_IMX */
++#endif /* CONFIG_ARCH_MX1 */
+
+ #ifdef CONFIG_ARCH_MX2
+ #define TIMER_BASE IO_ADDRESS(GPT1_BASE_ADDR)
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/include/mach/system.h linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/system.h
+--- linux-2.6.28/arch/arm/plat-mxc/include/mach/system.h 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/system.h 2009-03-11 13:16:24.000000000 +0100
+@@ -21,14 +21,7 @@
+ #ifndef __ASM_ARCH_MXC_SYSTEM_H__
+ #define __ASM_ARCH_MXC_SYSTEM_H__
+
+-static inline void arch_idle(void)
+-{
+- cpu_do_idle();
+-}
+-
+-static inline void arch_reset(char mode)
+-{
+- cpu_reset(0);
+-}
++extern void arch_idle(void);
++extern void arch_reset(char mode);
+
+ #endif /* __ASM_ARCH_MXC_SYSTEM_H__ */
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/include/mach/ulpi.h linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/ulpi.h
+--- linux-2.6.28/arch/arm/plat-mxc/include/mach/ulpi.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/include/mach/ulpi.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,31 @@
++#ifndef __MACH_ULPI_H
++#define __MACH_ULPI_H
++
++int ulpi_set(u8 bits, int reg, void __iomem *view);
++int ulpi_clear(u8 bits, int reg, void __iomem *view);
++int ulpi_read(int reg, void __iomem *view);
++
++/* ISP 1504 register addresses */
++#define ISP1504_VID_LOW 0x00 /* Vendor ID low */
++#define ISP1504_VID_HIGH 0x01 /* Vendor ID high */
++#define ISP1504_PID_LOW 0x02 /* Product ID low */
++#define ISP1504_PID_HIGH 0x03 /* Product ID high */
++#define ISP1504_ITFCTL 0x07 /* Interface Control */
++#define ISP1504_OTGCTL 0x0A /* OTG Control */
++
++/* add to above register address to access Set/Clear functions */
++#define ISP1504_REG_SET 0x01
++#define ISP1504_REG_CLEAR 0x02
++
++/* 1504 OTG Control Register bits */
++#define USE_EXT_VBUS_IND (1 << 7) /* Use ext. Vbus indicator */
++#define DRV_VBUS_EXT (1 << 6) /* Drive Vbus external */
++#define DRV_VBUS (1 << 5) /* Drive Vbus */
++#define CHRG_VBUS (1 << 4) /* Charge Vbus */
++#define DISCHRG_VBUS (1 << 3) /* Discharge Vbus */
++#define DM_PULL_DOWN (1 << 2) /* enable DM Pull Down */
++#define DP_PULL_DOWN (1 << 1) /* enable DP Pull Down */
++#define ID_PULL_UP (1 << 0) /* enable ID Pull Up */
++
++#endif /* __MACH_ULPI_H */
++
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/iomux-mx1-mx2.c linux-2.6.28-karo/arch/arm/plat-mxc/iomux-mx1-mx2.c
+--- linux-2.6.28/arch/arm/plat-mxc/iomux-mx1-mx2.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/iomux-mx1-mx2.c 2009-03-11 13:16:24.000000000 +0100
+@@ -1,9 +1,10 @@
+ /*
+- * arch/arm/mach-mxc/generic.c
++ * arch/arm/plat-mxc/iomux-mx1-mx2.c
+ *
+- * author: Sascha Hauer
+- * Created: april 20th, 2004
++ * Author: Sascha Hauer
++ * Created: April 20th, 2004
+ * Copyright: Synertronixx GmbH
++// FIXME: This is most likely as incorrect as the filename comment above
+ *
+ * Common code for i.MX machines
+ *
+@@ -32,7 +33,7 @@
+
+ #include <mach/hardware.h>
+ #include <asm/mach/map.h>
+-#include <mach/iomux-mx1-mx2.h>
++#include <mach/iomux.h>
+
+ void mxc_gpio_mode(int gpio_mode)
+ {
+@@ -40,6 +41,31 @@ void mxc_gpio_mode(int gpio_mode)
+ unsigned int port = (gpio_mode & GPIO_PORT_MASK) >> GPIO_PORT_SHIFT;
+ unsigned int ocr = (gpio_mode & GPIO_OCR_MASK) >> GPIO_OCR_SHIFT;
+ unsigned int tmp;
++ unsigned long flags;
++ char buf[64];
++
++ local_irq_save(flags);
++
++ if (gpio_mode & GPIO_OUT) {
++ switch (gpio_mode & (GPIO_DFLT_LOW | GPIO_DFLT_HIGH)) {
++ case 0:
++ break;
++ case GPIO_DFLT_LOW:
++ tmp = __raw_readl(VA_GPIO_BASE + MXC_DR(port));
++ tmp &= ~(1 << pin);
++ __raw_writel(tmp, VA_GPIO_BASE + MXC_DR(port));
++ break;
++ case GPIO_DFLT_HIGH:
++ tmp = __raw_readl(VA_GPIO_BASE + MXC_DR(port));
++ tmp |= (1 << pin);
++ __raw_writel(tmp, VA_GPIO_BASE + MXC_DR(port));
++ break;
++ default:
++ printk(KERN_ERR
++ "GPIO_DFLT_LOW and GPIO_DFLT_HIGH both set for P%c%d\n",
++ port + 'A', pin);
++ }
++ }
+
+ /* Pullup enable */
+ tmp = __raw_readl(VA_GPIO_BASE + MXC_PUEN(port));
+@@ -106,51 +132,50 @@ void mxc_gpio_mode(int gpio_mode)
+ tmp |= ((gpio_mode >> GPIO_BOUT_SHIFT) & 3) << (pin * 2);
+ __raw_writel(tmp, VA_GPIO_BASE + MXC_ICONFB2(port));
+ }
++ local_irq_restore(flags);
+ }
+ EXPORT_SYMBOL(mxc_gpio_mode);
+
+ int mxc_gpio_setup_multiple_pins(const int *pin_list, unsigned count,
+- int alloc_mode, const char *label)
++ const char *label)
+ {
+- const int *p = pin_list;
++ const int *p;
+ int i;
+- unsigned gpio;
+- unsigned mode;
++ int ret = -EINVAL;
+
+- for (i = 0; i < count; i++) {
+- gpio = *p & (GPIO_PIN_MASK | GPIO_PORT_MASK);
+- mode = *p & ~(GPIO_PIN_MASK | GPIO_PORT_MASK);
++ /* Try to obtain all requested GPIOs */
++ for (i = 0, p = pin_list; i < count; i++, p++) {
++ unsigned gpio = *p & (GPIO_PIN_MASK | GPIO_PORT_MASK);
+
+ if (gpio >= (GPIO_PORT_MAX + 1) * 32)
+ goto setup_error;
+
+- if (alloc_mode & MXC_GPIO_ALLOC_MODE_RELEASE)
+- gpio_free(gpio);
+- else if (!(alloc_mode & MXC_GPIO_ALLOC_MODE_NO_ALLOC))
+- if (gpio_request(gpio, label)
+- && !(alloc_mode & MXC_GPIO_ALLOC_MODE_TRY_ALLOC))
+- goto setup_error;
+-
+- if (!(alloc_mode & (MXC_GPIO_ALLOC_MODE_ALLOC_ONLY |
+- MXC_GPIO_ALLOC_MODE_RELEASE)))
+- mxc_gpio_mode(gpio | mode);
+-
+- p++;
++ ret = gpio_request(gpio, label);
++ if (ret)
++ goto setup_error;
++ }
++ /* Reconfigure all requested pins */
++ for (i = 0, p = pin_list; i < count; i++, p++) {
++ mxc_gpio_mode(*p);
+ }
+ return 0;
+
+ setup_error:
+- if (alloc_mode & (MXC_GPIO_ALLOC_MODE_NO_ALLOC |
+- MXC_GPIO_ALLOC_MODE_TRY_ALLOC))
+- return -EINVAL;
+-
+- while (p != pin_list) {
+- p--;
+- gpio = *p & (GPIO_PIN_MASK | GPIO_PORT_MASK);
++ mxc_gpio_release_multiple_pins(pin_list, i);
++ return ret;
++}
++EXPORT_SYMBOL(mxc_gpio_setup_multiple_pins);
++
++void mxc_gpio_release_multiple_pins(const int *pin_list, int count)
++{
++ const int *p;
++ int i;
++
++ for (i = 0, p = pin_list; i < count; i++, p++) {
++ unsigned gpio = *p & (GPIO_PIN_MASK | GPIO_PORT_MASK);
+ gpio_free(gpio);
+ }
+
+- return -EINVAL;
+ }
+-EXPORT_SYMBOL(mxc_gpio_setup_multiple_pins);
++EXPORT_SYMBOL(mxc_gpio_release_multiple_pins);
+
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/irq.c linux-2.6.28-karo/arch/arm/plat-mxc/irq.c
+--- linux-2.6.28/arch/arm/plat-mxc/irq.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/irq.c 2009-03-11 18:48:21.000000000 +0100
+@@ -17,9 +17,13 @@
+ * MA 02110-1301, USA.
+ */
+
++#include <linux/module.h>
++#include <linux/sysdev.h>
+ #include <linux/irq.h>
+ #include <linux/io.h>
++#include <linux/pm.h>
+ #include <mach/common.h>
++#include <asm/mach/irq.h>
+
+ #define AVIC_BASE IO_ADDRESS(AVIC_BASE_ADDR)
+ #define AVIC_INTCNTL (AVIC_BASE + 0x00) /* int control reg */
+@@ -65,6 +69,28 @@ void imx_irq_set_priority(unsigned char
+ EXPORT_SYMBOL(imx_irq_set_priority);
+ #endif
+
++#ifdef CONFIG_FIQ
++int mxc_set_irq_fiq(unsigned int irq, unsigned int type)
++{
++ unsigned int irqt;
++
++ if (irq >= MXC_MAX_INT_LINES)
++ return -EINVAL;
++
++ if (irq < MXC_MAX_INT_LINES / 2) {
++ irqt = __raw_readl(AVIC_INTTYPEL) & ~(1 << irq);
++ __raw_writel(irqt | (!!type << irq), AVIC_INTTYPEL);
++ } else {
++ irq -= MXC_MAX_INT_LINES / 2;
++ irqt = __raw_readl(AVIC_INTTYPEH) & ~(1 << irq);
++ __raw_writel(irqt | (!!type << irq), AVIC_INTTYPEH);
++ }
++
++ return 0;
++}
++EXPORT_SYMBOL(mxc_set_irq_fiq);
++#endif /* CONFIG_FIQ */
++
+ /* Disable interrupt number "irq" in the AVIC */
+ static void mxc_mask_irq(unsigned int irq)
+ {
+@@ -77,12 +103,94 @@ static void mxc_unmask_irq(unsigned int
+ __raw_writel(irq, AVIC_INTENNUM);
+ }
+
++static u32 saved_wakeup_low, saved_wakeup_high;
++static u32 suspend_wakeup_low, suspend_wakeup_high;
++
++/* Set interrupt number "irq" in the AVIC as a wake-up source */
++static int mxc_set_wake_irq(unsigned int irq, unsigned int enable)
++{
++ uint32_t *wakeup_intr;
++ uint32_t irq_bit;
++
++ if (irq < 32) {
++ wakeup_intr = &suspend_wakeup_low;
++ irq_bit = 1 << irq;
++ } else {
++ wakeup_intr = &suspend_wakeup_high;
++ irq_bit = 1 << (irq - 32);
++ }
++
++ if (enable) {
++ *wakeup_intr |= irq_bit;
++ } else {
++ *wakeup_intr &= ~irq_bit;
++ }
++
++ return 0;
++}
++
+ static struct irq_chip mxc_avic_chip = {
+ .ack = mxc_mask_irq,
+ .mask = mxc_mask_irq,
+ .unmask = mxc_unmask_irq,
++ .set_wake = mxc_set_wake_irq,
++};
++
++#ifdef CONFIG_PM
++/* This function puts the AVIC in low-power mode/state.
++ * All interrupts that are enabled are first saved.
++ * Only those interrupts which are registered as a wake source by calling
++ * enable_irq_wake are enabled. All other interrupts are disabled.
++ */
++static int mxc_avic_suspend(struct sys_device *dev, pm_message_t mesg)
++{
++ saved_wakeup_high = __raw_readl(AVIC_INTENABLEH);
++ saved_wakeup_low = __raw_readl(AVIC_INTENABLEL);
++
++ __raw_writel(suspend_wakeup_high, AVIC_INTENABLEH);
++ __raw_writel(suspend_wakeup_low, AVIC_INTENABLEL);
++
++ return 0;
++}
++
++/* This function brings the AVIC back from low-power state.
++ * All interrupts that were enabled prior to suspend are re-enabled.
++ */
++static int mxc_avic_resume(struct sys_device *dev)
++{
++ __raw_writel(saved_wakeup_high, AVIC_INTENABLEH);
++ __raw_writel(saved_wakeup_low, AVIC_INTENABLEL);
++
++ return 0;
++}
++
++#else
++#define mxc_avic_suspend NULL
++#define mxc_avic_resume NULL
++#endif /* CONFIG_PM */
++
++/*!
++ * This structure contains pointers to the power management callback functions.
++ */
++static struct sysdev_class mxc_avic_sysclass = {
++ .name = "mxc_irq",
++ .suspend = mxc_avic_suspend,
++ .resume = mxc_avic_resume,
++};
++
++/*!
++ * This structure represents AVIC as a system device.
++ * System devices follow a slightly different driver model.
++ * They don't need to do dynammic driver binding, can't be probed,
++ * and don't reside on any type of peripheral bus.
++ * So, it is represented and treated a little differently.
++ */
++struct sys_device mxc_avic_device = {
++ .cls = &mxc_avic_sysclass,
+ };
+
++static int __init mxc_avic_sysinit(void);
++
+ /*
+ * This function initializes the AVIC hardware and disables all the
+ * interrupts. It registers the interrupt enable and disable functions
+@@ -91,7 +199,6 @@ static struct irq_chip mxc_avic_chip = {
+ void __init mxc_init_irq(void)
+ {
+ int i;
+- u32 reg;
+
+ /* put the AVIC into the reset value with
+ * all interrupts disabled
+@@ -119,5 +226,30 @@ void __init mxc_init_irq(void)
+ /* init architectures chained interrupt handler */
+ mxc_register_gpios();
+
++#ifdef CONFIG_FIQ
++ /* Initialize FIQ */
++ init_FIQ();
++#endif
++
+ printk(KERN_INFO "MXC IRQ initialized\n");
+ }
++
++extern int __init mxc_gpio_sys_init(void);
++
++/* This function registers AVIC hardware as a system device */
++static int __init mxc_avic_sysinit(void)
++{
++ int ret;
++
++ ret = sysdev_class_register(&mxc_avic_sysclass);
++ if (ret)
++ return ret;
++ ret = sysdev_register(&mxc_avic_device);
++ if (ret)
++ return ret;
++
++ ret = mxc_gpio_sys_init();
++
++ return ret;
++}
++arch_initcall(mxc_avic_sysinit);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/arch/arm/plat-mxc/ulpi.c linux-2.6.28-karo/arch/arm/plat-mxc/ulpi.c
+--- linux-2.6.28/arch/arm/plat-mxc/ulpi.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/arch/arm/plat-mxc/ulpi.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,120 @@
++/*
++ * Copyright 2008 Sascha Hauer, Pengutronix <s.hauer@pengutronix.de>
++ *
++ * 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., 51 Franklin Street, Fifth Floor, Boston,
++ * MA 02110-1301, USA.
++ */
++
++#include <linux/module.h>
++#include <linux/kernel.h>
++#include <linux/io.h>
++#include <linux/delay.h>
++
++#include <mach/ulpi.h>
++
++/* ULPIVIEW register bits */
++#define ULPIVW_WU (1 << 31) /* Wakeup */
++#define ULPIVW_RUN (1 << 30) /* read/write run */
++#define ULPIVW_WRITE (1 << 29) /* 0 = read 1 = write */
++#define ULPIVW_SS (1 << 27) /* SyncState */
++#define ULPIVW_PORT_MASK 0x07 /* Port field */
++#define ULPIVW_PORT_SHIFT 24
++#define ULPIVW_ADDR_MASK 0xFF /* data address field */
++#define ULPIVW_ADDR_SHIFT 16
++#define ULPIVW_RDATA_MASK 0xFF /* read data field */
++#define ULPIVW_RDATA_SHIFT 8
++#define ULPIVW_WDATA_MASK 0xFF /* write data field */
++#define ULPIVW_WDATA_SHIFT 0
++
++static int ulpi_poll(void __iomem *view, uint32_t bit)
++{
++ uint32_t data;
++ int timeout = 10000;
++
++ data = __raw_readl(view);
++ while (data & bit) {
++ if (!timeout--)
++ return -ETIMEDOUT;
++
++ udelay(1);
++ data = __raw_readl(view);
++ }
++ return (data >> ULPIVW_RDATA_SHIFT) & ULPIVW_RDATA_MASK;
++}
++
++int ulpi_read(int reg, void __iomem *view)
++{
++ int ret;
++
++ /* make sure interface is running */
++ if (!(__raw_readl(view) && ULPIVW_SS)) {
++ __raw_writel(ULPIVW_WU, view);
++
++ /* wait for wakeup */
++ ret = ulpi_poll(view, ULPIVW_WU);
++ if (ret < 0)
++ return ret;
++ }
++
++ /* read the register */
++ __raw_writel((ULPIVW_RUN | (reg << ULPIVW_ADDR_SHIFT)), view);
++
++ /* wait for completion */
++ return ulpi_poll(view, ULPIVW_RUN);
++}
++EXPORT_SYMBOL(ulpi_read);
++
++int ulpi_set(u8 bits, int reg, void __iomem *view)
++{
++ int ret;
++
++ /* make sure the interface is running */
++ if (!(__raw_readl(view) && ULPIVW_SS)) {
++ __raw_writel(ULPIVW_WU, view);
++ /* wait for wakeup */
++ ret = ulpi_poll(view, ULPIVW_WU);
++ if (ret < 0)
++ return ret;
++ }
++
++ __raw_writel((ULPIVW_RUN | ULPIVW_WRITE |
++ ((reg + ISP1504_REG_SET) << ULPIVW_ADDR_SHIFT) |
++ ((bits & ULPIVW_WDATA_MASK) << ULPIVW_WDATA_SHIFT)),
++ view);
++
++ /* wait for completion */
++ ret = ulpi_poll(view, ULPIVW_RUN);
++ if (ret < 0)
++ return ret;
++ return 0;
++}
++EXPORT_SYMBOL(ulpi_set);
++
++int ulpi_clear(u8 bits, int reg, void __iomem *view)
++{
++ int ret;
++
++ __raw_writel((ULPIVW_RUN | ULPIVW_WRITE |
++ ((reg + ISP1504_REG_CLEAR) << ULPIVW_ADDR_SHIFT) |
++ ((bits & ULPIVW_WDATA_MASK) << ULPIVW_WDATA_SHIFT)),
++ view);
++
++ /* wait for completion */
++ ret = ulpi_poll(view, ULPIVW_RUN);
++ if (ret < 0)
++ return ret;
++ return 0;
++}
++EXPORT_SYMBOL(ulpi_clear);
++
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/Makefile linux-2.6.28-karo/drivers/Makefile
+--- linux-2.6.28/drivers/Makefile 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/Makefile 2009-03-11 13:16:24.000000000 +0100
+@@ -102,3 +102,4 @@ obj-$(CONFIG_SSB) += ssb/
+ obj-$(CONFIG_VIRTIO) += virtio/
+ obj-$(CONFIG_REGULATOR) += regulator/
+ obj-$(CONFIG_STAGING) += staging/
++obj-$(CONFIG_DRIVERS_MXC) += mxc/
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/dma/dmaengine.c linux-2.6.28-karo/drivers/dma/dmaengine.c
+--- linux-2.6.28/drivers/dma/dmaengine.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/dma/dmaengine.c 2009-03-11 13:16:24.000000000 +0100
+@@ -322,7 +322,12 @@ void dma_async_client_unregister(struct
+
+ mutex_lock(&dma_list_mutex);
+ /* free all channels the client is holding */
+- list_for_each_entry(device, &dma_device_list, global_node)
++ list_for_each_entry(device, &dma_device_list, global_node) {
++ /* Client only can have channels from one DMA controller */
++ if (client->slave && client->slave->dma_dev &&
++ client->slave->dma_dev != device->dev)
++ continue;
++
+ list_for_each_entry(chan, &device->channels, device_node) {
+ ack = client->event_callback(client, chan,
+ DMA_RESOURCE_REMOVED);
+@@ -332,6 +337,7 @@ void dma_async_client_unregister(struct
+ chan->client_count--;
+ }
+ }
++ }
+
+ list_del(&client->global_node);
+ mutex_unlock(&dma_list_mutex);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/gpio/gpiolib.c linux-2.6.28-karo/drivers/gpio/gpiolib.c
+--- linux-2.6.28/drivers/gpio/gpiolib.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/gpio/gpiolib.c 2009-03-11 13:16:24.000000000 +0100
+@@ -789,6 +789,7 @@ int gpio_request(unsigned gpio, const ch
+ } else {
+ status = -EBUSY;
+ module_put(chip->owner);
++ goto done;
+ }
+
+ if (chip->request) {
+@@ -842,9 +843,14 @@ void gpio_free(unsigned gpio)
+ desc_set_label(desc, NULL);
+ module_put(desc->chip->owner);
+ clear_bit(FLAG_REQUESTED, &desc->flags);
+- } else
++ } else {
++ if (!chip) {
++ printk(KERN_ERR "%s: No chip for GPIO%d\n", __FUNCTION__, gpio);
++ } else {
++ printk(KERN_ERR "%s: GPIO%d not requested\n", __FUNCTION__, gpio);
++ }
+ WARN_ON(extra_checks);
+-
++ }
+ spin_unlock_irqrestore(&gpio_lock, flags);
+ }
+ EXPORT_SYMBOL_GPL(gpio_free);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/i2c/busses/Kconfig linux-2.6.28-karo/drivers/i2c/busses/Kconfig
+--- linux-2.6.28/drivers/i2c/busses/Kconfig 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/i2c/busses/Kconfig 2009-03-11 13:16:24.000000000 +0100
+@@ -355,6 +355,16 @@ config I2C_IBM_IIC
+ This driver can also be built as a module. If so, the module
+ will be called i2c-ibm_iic.
+
++config I2C_MXC
++ tristate "MXC I2C support"
++ depends on I2C && ARCH_MXC
++ help
++ Say Y here if you want to use the IIC bus controller on Freescale
++ i.MX2 family of processors (like i.MX21 and i.MX27).
++
++ This driver can also be built as a module. If so, the module
++ will be called i2c-mxc.
++
+ config I2C_IOP3XX
+ tristate "Intel IOPx3xx and IXP4xx on-chip I2C interface"
+ depends on ARCH_IOP32X || ARCH_IOP33X || ARCH_IXP4XX || ARCH_IOP13XX
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/i2c/busses/Makefile linux-2.6.28-karo/drivers/i2c/busses/Makefile
+--- linux-2.6.28/drivers/i2c/busses/Makefile 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/i2c/busses/Makefile 2009-03-11 13:16:24.000000000 +0100
+@@ -11,6 +11,7 @@ obj-$(CONFIG_I2C_AMD756_S4882) += i2c-am
+ obj-$(CONFIG_I2C_AMD8111) += i2c-amd8111.o
+ obj-$(CONFIG_I2C_I801) += i2c-i801.o
+ obj-$(CONFIG_I2C_ISCH) += i2c-isch.o
++obj-$(CONFIG_I2C_MXC) += i2c-mxc.o
+ obj-$(CONFIG_I2C_NFORCE2) += i2c-nforce2.o
+ obj-$(CONFIG_I2C_NFORCE2_S4985) += i2c-nforce2-s4985.o
+ obj-$(CONFIG_I2C_PIIX4) += i2c-piix4.o
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/i2c/busses/i2c-mxc.c linux-2.6.28-karo/drivers/i2c/busses/i2c-mxc.c
+--- linux-2.6.28/drivers/i2c/busses/i2c-mxc.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/i2c/busses/i2c-mxc.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,667 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ * Copyright 2008 Juergen Beisert, kernel@pengutronix.de
++ *
++ * 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
++ */
++
++/*
++ * Driver for the Freescale Semiconductor MXC I2C buses.
++ * Based on i2c driver algorithm for PCF8584 adapters
++ */
++
++#include <linux/module.h>
++#include <linux/kernel.h>
++#include <linux/delay.h>
++#include <linux/interrupt.h>
++#include <linux/platform_device.h>
++#include <linux/err.h>
++#include <linux/i2c.h>
++#include <linux/clk.h>
++#include <asm/irq.h>
++#include <asm/io.h>
++#include <mach/clock.h>
++#include <mach/imx_i2c.h>
++
++#define DRV_NAME "imx_i2c"
++
++typedef struct {
++ struct i2c_adapter adap;
++ wait_queue_head_t wq;
++ void __iomem *membase; /* FIXME iomem? */
++ int irq;
++ unsigned int clkdiv;
++ struct clk *clk;
++ bool low_power;
++ struct imx_i2c_platform_data *pdata;
++ bool transfer_done;
++ bool tx_success;
++} imx_i2c_device;
++
++struct clk_div_table {
++ int reg_value;
++ int div;
++};
++
++static const struct clk_div_table i2c_clk_table[] = {
++ {0x20, 22}, {0x21, 24}, {0x22, 26}, {0x23, 28},
++ {0, 30}, {1, 32}, {0x24, 32}, {2, 36},
++ {0x25, 36}, {0x26, 40}, {3, 42}, {0x27, 44},
++ {4, 48}, {0x28, 48}, {5, 52}, {0x29, 56},
++ {6, 60}, {0x2A, 64}, {7, 72}, {0x2B, 72},
++ {8, 80}, {0x2C, 80}, {9, 88}, {0x2D, 96},
++ {0xA, 104}, {0x2E, 112}, {0xB, 128}, {0x2F, 128},
++ {0xC, 144}, {0xD, 160}, {0x30, 160}, {0xE, 192},
++ {0x31, 192}, {0x32, 224}, {0xF, 240}, {0x33, 256},
++ {0x10, 288}, {0x11, 320}, {0x34, 320}, {0x12, 384},
++ {0x35, 384}, {0x36, 448}, {0x13, 480}, {0x37, 512},
++ {0x14, 576}, {0x15, 640}, {0x38, 640}, {0x16, 768},
++ {0x39, 768}, {0x3A, 896}, {0x17, 960}, {0x3B, 1024},
++ {0x18, 1152}, {0x19, 1280}, {0x3C, 1280}, {0x1A, 1536},
++ {0x3D, 1536}, {0x3E, 1792}, {0x1B, 1920}, {0x3F, 2048},
++ {0x1C, 2304}, {0x1D, 2560}, {0x1E, 3072}, {0x1F, 3840},
++ {0, 0}
++};
++
++/* Transmit a STOP signal to the slave device */
++static void imx_i2c_stop(imx_i2c_device * dev)
++{
++ unsigned int cr;
++ int retry = 16;
++
++ cr = __raw_readw(dev->membase + MXC_I2CR);
++ cr &= ~(MXC_I2CR_MSTA | MXC_I2CR_MTX);
++ __raw_writew(cr, dev->membase + MXC_I2CR);
++
++ /*
++ * Make sure STOP meets setup requirement.
++ */
++ for (;;) {
++ unsigned int sr = __raw_readw(dev->membase + MXC_I2SR);
++ if ((sr & MXC_I2SR_IBB) == 0) break;
++ if (retry-- <= 0) {
++ printk(KERN_DEBUG "Bus busy\n");
++ break;
++ }
++ udelay(3);
++ }
++}
++
++/*
++ * Wait for the transmission of the data byte to complete. This function waits
++ * till we get a signal from the interrupt service routine indicating completion
++ * of the address cycle or we time out.
++ * The function returns 0 on success or -1 if an ack was not received
++ */
++static int imx_i2c_wait_for_tc(imx_i2c_device * dev, int trans_flag)
++{
++ int retry = 16;
++
++ while (retry-- && !dev->transfer_done) {
++ wait_event_interruptible_timeout(dev->wq,
++ dev->transfer_done,
++ dev->adap.timeout);
++ }
++ dev->transfer_done = false;
++
++ if (retry <= 0) {
++ /* Unable to send data */
++ printk(KERN_DEBUG "Data not transmitted\n");
++ return -1;
++ } else if (!(trans_flag & I2C_M_RD)) {
++ if (!dev->tx_success) {
++ /*
++ * An ACK was not received for a transmitted byte. Don't
++ * print a message here as slow devices might cause this
++ * condition quite often. This is not an error as
++ * routines may try to look if a device is ready.
++ */
++ return -1;
++ }
++ }
++
++ return 0;
++}
++
++/* Transmit a START signal to the slave device */
++static void imx_i2c_start(imx_i2c_device * dev, struct i2c_msg *msg)
++{
++ unsigned int cr, sr;
++ unsigned int addr_trans;
++ int retry = 16;
++
++ /*
++ * Set the slave address and the requested transfer mode
++ * in the data register
++ */
++ addr_trans = msg->addr << 1;
++ if (msg->flags & I2C_M_RD) {
++ addr_trans |= 0x01;
++ }
++
++ /* Set the Master bit */
++ cr = __raw_readw(dev->membase + MXC_I2CR);
++ cr |= MXC_I2CR_MSTA;
++ __raw_writew(cr, dev->membase + MXC_I2CR);
++
++ /* Wait till the Bus Busy bit is set */
++ sr = __raw_readw(dev->membase + MXC_I2SR);
++ while (retry-- && (!(sr & MXC_I2SR_IBB))) {
++ udelay(3);
++ sr = __raw_readw(dev->membase + MXC_I2SR);
++ }
++ if (retry <= 0) {
++ printk(KERN_DEBUG "Could not grab Bus ownership\n");
++ }
++
++ /* Set the Transmit bit */
++ cr = __raw_readw(dev->membase + MXC_I2CR);
++ cr |= MXC_I2CR_MTX;
++ __raw_writew(cr, dev->membase + MXC_I2CR);
++
++ __raw_writew(addr_trans, dev->membase + MXC_I2DR);
++}
++
++/* Transmit a REPEAT START to the slave device */
++static void imx_i2c_repstart(imx_i2c_device * dev, struct i2c_msg *msg)
++{
++ unsigned int cr;
++ unsigned int addr_trans;
++
++ /*
++ * Set the slave address and the requested transfer mode
++ * in the data register
++ */
++ addr_trans = msg->addr << 1;
++ if (msg->flags & I2C_M_RD) {
++ addr_trans |= 0x01;
++ }
++ cr = __raw_readw(dev->membase + MXC_I2CR);
++ cr |= MXC_I2CR_RSTA;
++ __raw_writew(cr, dev->membase + MXC_I2CR);
++ udelay(3);
++ __raw_writew(addr_trans, dev->membase + MXC_I2DR);
++}
++
++/*
++ * Read the received data. The function waits till data is available or times
++ * out. Generates a stop signal if this is the last message to be received.
++ * Sends an ack for all the bytes received except the last byte.
++ * The function returns the number of bytes read or -1 on time out.
++ */
++static int imx_i2c_readbytes(imx_i2c_device * dev, struct i2c_msg *msg,
++ int last, int addr_comp)
++{
++ int i;
++ char *buf = msg->buf;
++ int len = msg->len;
++ unsigned int cr;
++
++ cr = __raw_readw(dev->membase + MXC_I2CR);
++
++ /* switch to receive mode */
++ cr &= ~MXC_I2CR_MTX;
++ /*
++ * Clear the TXAK bit to gen an ack when receiving only one byte.
++ */
++ if (len == 1)
++ cr |= MXC_I2CR_TXAK;
++ else
++ cr &= ~MXC_I2CR_TXAK;
++
++ __raw_writew(cr, dev->membase + MXC_I2CR);
++ /*
++ * Dummy read only at the end of an address cycle
++ */
++ if (addr_comp > 0)
++ __raw_readw(dev->membase + MXC_I2DR);
++
++ for (i = 0; i < len; i++) {
++ /* Wait for data transmission to complete */
++ if (imx_i2c_wait_for_tc(dev, msg->flags)) {
++ imx_i2c_stop(dev);
++ return -1;
++ }
++ /* Do not generate an ACK for the last byte */
++ if (i == (len - 2)) {
++ cr = __raw_readw(dev->membase + MXC_I2CR);
++ cr |= MXC_I2CR_TXAK;
++ __raw_writew(cr, dev->membase + MXC_I2CR);
++ } else if (i == (len - 1)) {
++ if (last)
++ imx_i2c_stop(dev);
++ }
++ /* Read the data */
++ *buf++ = __raw_readw(dev->membase + MXC_I2DR);
++ }
++
++ return i;
++}
++
++/*
++ * Write the data to the data register. Generates a stop signal if this is
++ * the last message to be sent or if no ack was received for the data sent.
++ * The function returns the number of bytes written or -1 on time out
++ * or if no ack was received for the data that was sent.
++ */
++static int imx_i2c_writebytes(imx_i2c_device * dev, struct i2c_msg *msg,
++ int last)
++{
++ int i;
++ char *buf = msg->buf;
++ int len = msg->len;
++ unsigned int cr;
++
++ cr = __raw_readw(dev->membase + MXC_I2CR);
++ /* switch to transmit mode */
++ cr |= MXC_I2CR_MTX;
++ __raw_writew(cr, dev->membase + MXC_I2CR);
++
++ for (i = 0; i < len; i++) {
++ /* Write the data */
++ __raw_writew(*buf++, dev->membase + MXC_I2DR);
++ if (imx_i2c_wait_for_tc(dev, msg->flags)) {
++ imx_i2c_stop(dev);
++ return -1;
++ }
++ }
++ if (last > 0) {
++ imx_i2c_stop(dev);
++ }
++
++ return i;
++}
++
++/* Function enables the I2C module and initializes the registers */
++static void imx_i2c_module_en(imx_i2c_device * dev, int trans_flag)
++{
++ clk_enable(dev->clk);
++ /* Set the frequency divider */
++ __raw_writew(dev->clkdiv, dev->membase + MXC_IFDR);
++ /* Clear the status register */
++ __raw_writew(0x0, dev->membase + MXC_I2SR);
++ /* Enable I2C and its interrupts */
++ __raw_writew(MXC_I2CR_IEN, dev->membase + MXC_I2CR);
++ __raw_writew(MXC_I2CR_IEN | MXC_I2CR_IIEN, dev->membase + MXC_I2CR);
++}
++
++/* Disables the I2C module */
++static void imx_i2c_module_dis(imx_i2c_device * dev)
++{
++ __raw_writew(0x0, dev->membase + MXC_I2CR);
++ clk_disable(dev->clk);
++}
++
++/*
++ * The function is registered in the adapter structure. It is called when an MXC
++ * driver wishes to transfer data to a device connected to the I2C device.
++ * The function returns the number of messages transferred, -EREMOTEIO on I2C
++ * failure and a 0 if the num argument is less than 0.
++ */
++static int imx_i2c_xfer(struct i2c_adapter *adap, struct i2c_msg msgs[],
++ int num)
++{
++ imx_i2c_device *dev = (imx_i2c_device *) (i2c_get_adapdata(adap));
++ int i, ret = 0, addr_comp = 0;
++ unsigned int sr;
++
++ if (dev->low_power) {
++ printk(KERN_ERR "I2C Device in low power mode\n");
++ return -EREMOTEIO;
++ }
++
++ if (num < 1)
++ return 0;
++
++ imx_i2c_module_en(dev, msgs[0].flags);
++ sr = __raw_readw(dev->membase + MXC_I2SR);
++
++ /* Check bus state */
++ if (sr & MXC_I2SR_IBB) {
++ imx_i2c_module_dis(dev);
++ printk(KERN_DEBUG "Bus busy\n");
++ return -EREMOTEIO;
++ }
++
++ dev->transfer_done = false;
++ dev->tx_success = false;
++ for (i = 0; i < num && ret >= 0; i++) {
++ addr_comp = 0;
++ /*
++ * Send the slave address and transfer direction in the
++ * address cycle
++ */
++ if (i == 0) {
++ /* Send a start or repeat start signal */
++ imx_i2c_start(dev, &msgs[0]);
++ /* Wait for the address cycle to complete */
++ if (imx_i2c_wait_for_tc(dev, msgs[0].flags)) {
++ imx_i2c_stop(dev);
++ imx_i2c_module_dis(dev);
++ return -EREMOTEIO;
++ }
++ addr_comp = 1;
++ } else {
++ /*
++ * Generate repeat start only if required i.e the address
++ * changed or the transfer direction changed
++ */
++ if ((msgs[i].addr != msgs[i - 1].addr) ||
++ ((msgs[i].flags & I2C_M_RD) !=
++ (msgs[i - 1].flags & I2C_M_RD))) {
++ imx_i2c_repstart(dev, &msgs[i]);
++ /* Wait for the address cycle to complete */
++ if (imx_i2c_wait_for_tc(dev, msgs[i].flags)) {
++ imx_i2c_stop(dev);
++ imx_i2c_module_dis(dev);
++ return -EREMOTEIO;
++ }
++ addr_comp = 1;
++ }
++ }
++
++ /* Transfer the data */
++ if (msgs[i].flags & I2C_M_RD) {
++ /* Read the data */
++ ret = imx_i2c_readbytes(dev, &msgs[i], (i + 1 == num),
++ addr_comp);
++ if (ret < 0) {
++ printk(KERN_ERR "imx_i2c_readbytes: fail.\n");
++ break;
++ }
++ } else {
++ /* Write the data */
++ ret = imx_i2c_writebytes(dev, &msgs[i], (i + 1 == num));
++ if (ret < 0) {
++ printk(KERN_ERR "imx_i2c_writebytes: fail.\n");
++ break;
++ }
++ }
++ }
++
++ imx_i2c_module_dis(dev);
++ return i;
++}
++
++/*
++ * Returns the i2c functionality supported by this driver.
++ * Returns the functionality that is supported.
++ */
++static u32 imx_i2c_func(struct i2c_adapter *adap)
++{
++ return I2C_FUNC_I2C | I2C_FUNC_SMBUS_EMUL;
++}
++
++/*
++ * Stores the pointers for the i2c algorithm functions. The algorithm functions
++ * is used by the i2c bus driver to talk to the i2c bus
++ */
++static struct i2c_algorithm imx_i2c_algorithm = {
++ .master_xfer = imx_i2c_xfer,
++ .functionality = imx_i2c_func
++};
++
++/*
++ * Interrupt Service Routine. It signals to the process about the data transfer
++ * completion. Also sets a flag if bus arbitration is lost.
++ * The function returns IRQ_HANDLED.
++ */
++static irqreturn_t imx_i2c_handler(int irq, void *dev_id)
++{
++ imx_i2c_device *dev = dev_id;
++ unsigned int sr, cr;
++
++ sr = __raw_readw(dev->membase + MXC_I2SR);
++ cr = __raw_readw(dev->membase + MXC_I2CR);
++
++ /*
++ * Clear the interrupt bit
++ */
++ __raw_writew(0x0, dev->membase + MXC_I2SR);
++
++ if (sr & MXC_I2SR_IAL) {
++ printk(KERN_DEBUG "Bus Arbitration lost\n");
++ } else {
++ /* Interrupt due byte transfer completion */
++ dev->tx_success = false;
++ /* Check if RXAK is received in Transmit mode */
++ if ((cr & MXC_I2CR_MTX) && (!(sr & MXC_I2SR_RXAK))) {
++ dev->tx_success = true;
++ }
++ dev->transfer_done = true;
++ wake_up_interruptible(&dev->wq);
++ }
++
++ return IRQ_HANDLED;
++}
++
++/*
++ * This function is called to put the I2C adapter in a low power state. Refer to the
++ * document driver-model/driver.txt in the kernel source tree for more
++ * information.
++ * The function returns 0 on success and -1 on failure.
++ */
++static int mxci2c_suspend(struct platform_device *pdev, pm_message_t state)
++{
++ imx_i2c_device *mxcdev = platform_get_drvdata(pdev);
++ unsigned int sr;
++
++ if (mxcdev == NULL)
++ return -1;
++
++ /* Prevent further calls to be processed */
++ mxcdev->low_power = true;
++ /* Wait till we finish the current transfer */
++ sr = __raw_readw(mxcdev->membase + MXC_I2SR);
++ while (sr & MXC_I2SR_IBB) {
++ msleep(10);
++ sr = __raw_readw(mxcdev->membase + MXC_I2SR);
++ }
++
++ mxcdev->pdata->exit(pdev);
++
++ return 0;
++}
++
++/*
++ * This function is called to bring the I2C adapter back from a low power state. Refer
++ * to the document driver-model/driver.txt in the kernel source tree for more
++ * information.
++ * The function returns 0 on success and -1 on failure
++ */
++static int mxci2c_resume(struct platform_device *pdev)
++{
++ imx_i2c_device *mxcdev = platform_get_drvdata(pdev);
++
++ if (mxcdev == NULL)
++ return -1;
++
++ mxcdev->low_power = false;
++ mxcdev->pdata->init(pdev);
++
++ return 0;
++}
++
++/*
++ * This function is called during the driver binding process.
++ * The function always returns 0.
++ */
++static int mxci2c_probe(struct platform_device *pdev)
++{
++ imx_i2c_device *imx_i2c;
++ struct imx_i2c_platform_data *i2c_plat_data = pdev->dev.platform_data;
++ struct resource *res;
++ int id = pdev->id;
++ u32 clk_freq;
++ int ret;
++ int i;
++
++ imx_i2c = kzalloc(sizeof(imx_i2c_device), GFP_KERNEL);
++ if (!imx_i2c) {
++ return -ENOMEM;
++ }
++
++ if (i2c_plat_data == NULL) {
++ dev_err(&pdev->dev, "No platform data for device!\n");
++ return -ENODEV;
++ }
++ imx_i2c->pdata = i2c_plat_data;
++ /* claim the region we will work on */
++ res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
++ if (res == NULL) {
++ ret = -ENODEV;
++ goto err1;
++ }
++ if (!request_mem_region(res->start, resource_size(res), DRV_NAME)) {
++ dev_err(&pdev->dev, "request_mem_region failed for IMX I2C %d\n", id);
++ ret = -EBUSY;
++ goto err1;
++ }
++
++ imx_i2c->membase = ioremap(res->start, resource_size(res));
++ if (imx_i2c->membase == NULL) {
++ ret = -ENOMEM;
++ goto err2;
++ }
++ /* Claim the I2C irq line */
++ imx_i2c->irq = platform_get_irq(pdev, 0);
++ if (imx_i2c->irq < 0) {
++ dev_err(&pdev->dev, "No interrupt defined for IMX I2C %d\n", id);
++ ret = imx_i2c->irq;
++ goto err3;
++ }
++ ret = request_irq(imx_i2c->irq, imx_i2c_handler,
++ 0, DRV_NAME, imx_i2c);
++ if (ret < 0) {
++ dev_err(&pdev->dev, "Cannot claim interrupt %d for IMX I2C %d\n",
++ imx_i2c->irq, id);
++ goto err3;
++ }
++
++ init_waitqueue_head(&imx_i2c->wq);
++
++ imx_i2c->low_power = false;
++ imx_i2c->tx_success = false;
++ imx_i2c->transfer_done = false;
++
++ imx_i2c->pdata->init(pdev);
++
++ imx_i2c->clk = clk_get(&pdev->dev, "i2c_clk");
++ if (IS_ERR(imx_i2c->clk)) {
++ ret = PTR_ERR(imx_i2c->clk);
++ dev_err(&pdev->dev, "Cannot get clock for for IMX I2C %d: %d\n", id, ret);
++ goto err4;
++ }
++
++ clk_freq = clk_get_rate(imx_i2c->clk);
++ imx_i2c->clkdiv = -1;
++ if (i2c_plat_data->max_clk) {
++ /* Calculate divider and round up any fractional part */
++ int div = (clk_freq + i2c_plat_data->max_clk - 1) /
++ i2c_plat_data->max_clk;
++ for (i = 0; i2c_clk_table[i].div != 0; i++) {
++ if (i2c_clk_table[i].div >= div) {
++ imx_i2c->clkdiv = i2c_clk_table[i].reg_value;
++ break;
++ }
++ }
++ }
++ if (imx_i2c->clkdiv == -1) {
++ i = ARRAY_SIZE(i2c_clk_table) - 2;
++ imx_i2c->clkdiv = i2c_clk_table[i].reg_value;
++ }
++ dev_dbg(&pdev->dev, "i2c speed is %d/%d = %d bps, reg val = 0x%02X\n",
++ clk_freq, i2c_clk_table[i].div,
++ clk_freq / i2c_clk_table[i].div, imx_i2c->clkdiv);
++
++ /*
++ * Set the adapter information
++ */
++ strcpy(imx_i2c->adap.name, MXC_ADAPTER_NAME);
++ imx_i2c->adap.id = id;
++ imx_i2c->adap.nr = id;
++ imx_i2c->adap.algo = &imx_i2c_algorithm;
++ imx_i2c->adap.timeout = 1;
++ platform_set_drvdata(pdev, imx_i2c);
++ i2c_set_adapdata(&imx_i2c->adap, imx_i2c);
++ if ((ret = i2c_add_numbered_adapter(&imx_i2c->adap)) < 0) {
++ dev_err(&pdev->dev, "Cannot register the IMX I2C %d master\n", id);
++ goto err5;
++ }
++
++ return 0;
++err5:
++ clk_put(imx_i2c->clk);
++err4:
++ imx_i2c->pdata->exit(pdev);
++ free_irq(imx_i2c->irq, imx_i2c);
++err3:
++ iounmap(imx_i2c->membase);
++err2:
++ release_mem_region(res->start, resource_size(res));
++err1:
++ dev_err(&pdev->dev, "failed to probe i2c adapter\n");
++ kfree(imx_i2c);
++ return ret;
++}
++
++/*
++ * Dissociates the driver from the I2C device.
++ */
++static int mxci2c_remove(struct platform_device *pdev)
++{
++ imx_i2c_device *imx_i2c = platform_get_drvdata(pdev);
++ struct resource *res;
++
++ free_irq(imx_i2c->irq, imx_i2c);
++ i2c_del_adapter(&imx_i2c->adap);
++ imx_i2c->pdata->exit(pdev);
++ clk_put(imx_i2c->clk);
++ iounmap(imx_i2c->membase);
++ res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
++ release_mem_region(res->start, resource_size(res));
++ platform_set_drvdata(pdev, NULL);
++ return 0;
++}
++
++/*
++ * This structure contains pointers to the power management callback functions.
++ */
++static struct platform_driver mxci2c_driver = {
++ .driver = {
++ .name = DRV_NAME,
++ .owner = THIS_MODULE,
++ },
++ .probe = mxci2c_probe,
++ .remove = mxci2c_remove,
++ .suspend = mxci2c_suspend,
++ .resume = mxci2c_resume,
++};
++
++static int __init imx_i2c_init(void)
++{
++ return platform_driver_register(&mxci2c_driver);
++}
++
++static void __exit imx_i2c_exit(void)
++{
++ platform_driver_unregister(&mxci2c_driver);
++}
++
++subsys_initcall(imx_i2c_init);
++module_exit(imx_i2c_exit);
++
++MODULE_AUTHOR("Freescale Semiconductor, Inc.");
++MODULE_DESCRIPTION("MXC I2C driver");
++MODULE_LICENSE("GPL");
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/Kconfig linux-2.6.28-karo/drivers/media/video/Kconfig
+--- linux-2.6.28/drivers/media/video/Kconfig 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/Kconfig 2009-03-11 13:16:24.000000000 +0100
+@@ -531,6 +531,33 @@ config VIDEO_W9966
+ Check out <file:Documentation/video4linux/w9966.txt> for more
+ information.
+
++config VIDEO_MXC_CAMERA
++ tristate "MXC Video For Linux Camera"
++ depends on VIDEO_DEV && ARCH_MXC
++ default y
++ ---help---
++ This is the video4linux2 capture driver based on MXC IPU/eMMA module.
++
++source "drivers/media/video/mxc/capture/Kconfig"
++
++config VIDEO_MXC_OUTPUT
++ tristate "MXC Video For Linux Video Output"
++ depends on VIDEO_DEV && ARCH_MXC
++ default y
++ ---help---
++ This is the video4linux2 output driver based on MXC IPU/eMMA module.
++
++source "drivers/media/video/mxc/output/Kconfig"
++
++config VIDEO_MXC_OPL
++ tristate
++ depends on VIDEO_DEV && ARCH_MXC
++ default n
++ ---help---
++ This is the ARM9-optimized OPL (Open Primitives Library) software
++ rotation/mirroring implementation. It may be used by eMMA video
++ capture or output device.
++
+ config VIDEO_CPIA
+ tristate "CPiA Video For Linux"
+ depends on VIDEO_V4L1
+@@ -893,4 +920,12 @@ config USB_S2255
+
+ endif # V4L_USB_DRIVERS
+
++config VIDEO_MX27
++ tristate "i.MX27 Quick Capture Interface driver"
++ depends on VIDEO_DEV && MACH_MX27
++ select SOC_CAMERA
++ select VIDEOBUF_DMA_CONTIG
++ ---help---
++ This is a v4l2 driver for the i.MX27 Capture Interface
++
+ endif # VIDEO_CAPTURE_DRIVERS
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/Makefile linux-2.6.28-karo/drivers/media/video/Makefile
+--- linux-2.6.28/drivers/media/video/Makefile 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/Makefile 2009-03-11 13:16:24.000000000 +0100
+@@ -57,6 +57,11 @@ obj-$(CONFIG_VIDEO_ZORAN) += zoran/
+ obj-$(CONFIG_VIDEO_PMS) += pms.o
+ obj-$(CONFIG_VIDEO_VINO) += vino.o indycam.o
+ obj-$(CONFIG_VIDEO_STRADIS) += stradis.o
++obj-$(CONFIG_VIDEO_MXC_IPU_CAMERA) += mxc/capture/
++obj-$(CONFIG_VIDEO_MXC_EMMA_CAMERA) += mxc/capture/
++obj-$(CONFIG_VIDEO_MXC_IPU_OUTPUT) += mxc/output/
++obj-$(CONFIG_VIDEO_MXC_EMMA_OUTPUT) += mxc/output/
++obj-$(CONFIG_VIDEO_MXC_OPL) += mxc/opl/
+ obj-$(CONFIG_VIDEO_CPIA) += cpia.o
+ obj-$(CONFIG_VIDEO_CPIA_PP) += cpia_pp.o
+ obj-$(CONFIG_VIDEO_CPIA_USB) += cpia_usb.o
+@@ -128,6 +133,7 @@ obj-$(CONFIG_VIDEO_VIVI) += vivi.o
+ obj-$(CONFIG_VIDEO_CX23885) += cx23885/
+
+ obj-$(CONFIG_VIDEO_PXA27x) += pxa_camera.o
++obj-$(CONFIG_VIDEO_MX27) += mx27_camera.o
+ obj-$(CONFIG_VIDEO_SH_MOBILE_CEU) += sh_mobile_ceu_camera.o
+ obj-$(CONFIG_SOC_CAMERA) += soc_camera.o
+ obj-$(CONFIG_SOC_CAMERA_MT9M001) += mt9m001.o
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mx27_camera.c linux-2.6.28-karo/drivers/media/video/mx27_camera.c
+--- linux-2.6.28/drivers/media/video/mx27_camera.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mx27_camera.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,1185 @@
++/*
++ * V4L2 Driver for MX27 camera host
++ *
++ * Copyright (C) 2008, Sascha Hauer, Pengutronix
++ *
++ * 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/init.h>
++#include <linux/module.h>
++#include <linux/io.h>
++#include <linux/delay.h>
++#include <linux/dma-mapping.h>
++#include <linux/errno.h>
++#include <linux/fs.h>
++#include <linux/interrupt.h>
++#include <linux/kernel.h>
++#include <linux/mm.h>
++#include <linux/moduleparam.h>
++#include <linux/time.h>
++#include <linux/version.h>
++#include <linux/device.h>
++#include <linux/platform_device.h>
++#include <linux/mutex.h>
++#include <linux/clk.h>
++
++#include <media/v4l2-common.h>
++#include <media/v4l2-dev.h>
++#include <media/videobuf-dma-contig.h>
++#include <media/soc_camera.h>
++
++#include <linux/videodev2.h>
++
++#include <mach/imx_cam.h>
++#include <asm/dma.h>
++#include <mach/dma-mx1-mx2.h>
++
++#include <asm/dma.h>
++
++#define MX27_CAM_DRV_NAME "mx27-camera"
++#define MX27_CAM_VERSION_CODE KERNEL_VERSION(0, 0, 5) /* FIXME: Whats this? */
++
++static const char *mx27_cam_driver_description = "i.MX27_Camera";
++
++/* reset values */
++#define CSICR1_RESET_VAL 0x40000800
++#define CSICR2_RESET_VAL 0x0
++#define CSICR3_RESET_VAL 0x0
++
++/* csi control reg 1 */
++#define CSICR1_SWAP16_EN (1 << 31)
++#define CSICR1_EXT_VSYNC (1 << 30)
++#define CSICR1_EOF_INTEN (1 << 29)
++#define CSICR1_PRP_IF_EN (1 << 28)
++#define CSICR1_CCIR_MODE (1 << 27)
++#define CSICR1_COF_INTEN (1 << 26)
++#define CSICR1_SF_OR_INTEN (1 << 25)
++#define CSICR1_RF_OR_INTEN (1 << 24)
++#define CSICR1_STATFF_LEVEL (3 << 22)
++#define CSICR1_STATFF_INTEN (1 << 21)
++#define CSICR1_RXFF_LEVEL(l) (((l) & 3) << 19)
++#define CSICR1_RXFF_INTEN (1 << 18)
++#define CSICR1_SOF_POL (1 << 17)
++#define CSICR1_SOF_INTEN (1 << 16)
++#define CSICR1_MCLKDIV(d) (((d) & 0xF) << 12)
++#define CSICR1_HSYNC_POL (1 << 11)
++#define CSICR1_CCIR_EN (1 << 10)
++#define CSICR1_MCLKEN (1 << 9)
++#define CSICR1_FCC (1 << 8)
++#define CSICR1_PACK_DIR (1 << 7)
++#define CSICR1_CLR_STATFIFO (1 << 6)
++#define CSICR1_CLR_RXFIFO (1 << 5)
++#define CSICR1_GCLK_MODE (1 << 4)
++#define CSICR1_INV_DATA (1 << 3)
++#define CSICR1_INV_PCLK (1 << 2)
++#define CSICR1_REDGE (1 << 1)
++
++#define SHIFT_STATFF_LEVEL 22
++#define SHIFT_RXFF_LEVEL 19
++#define SHIFT_MCLKDIV 12
++
++/* control reg 3 */
++#define CSICR3_FRMCNT (0xFFFF << 16)
++#define CSICR3_FRMCNT_RST (1 << 15)
++#define CSICR3_CSI_SUP (1 << 3)
++#define CSICR3_ZERO_PACK_EN (1 << 2)
++#define CSICR3_ECC_INT_EN (1 << 1)
++#define CSICR3_ECC_AUTO_EN (1 << 0)
++
++#define SHIFT_FRMCNT 16
++
++/* csi status reg */
++#define CSISR_SFF_OR_INT (1 << 25)
++#define CSISR_RFF_OR_INT (1 << 24)
++#define CSISR_STATFF_INT (1 << 21)
++#define CSISR_RXFF_INT (1 << 18)
++#define CSISR_EOF_INT (1 << 17)
++#define CSISR_SOF_INT (1 << 16)
++#define CSISR_F2_INT (1 << 15)
++#define CSISR_F1_INT (1 << 14)
++#define CSISR_COF_INT (1 << 13)
++#define CSISR_ECC_INT (1 << 1)
++#define CSISR_DRDY (1 << 0)
++
++#define CSICR1 0x00
++#define CSICR2 0x04
++#define CSISR 0x08
++#define CSISTATFIFO 0x0c
++#define CSIRFIFO 0x10
++#define CSIRXCNT 0x14
++#define CSICR3 0x1C
++
++/* EMMA PrP */
++#define PRP_CNTL 0x00
++#define PRP_INTR_CNTL 0x04
++#define PRP_INTRSTATUS 0x08
++#define PRP_SOURCE_Y_PTR 0x0c
++#define PRP_SOURCE_CB_PTR 0x10
++#define PRP_SOURCE_CR_PTR 0x14
++#define PRP_DEST_RGB1_PTR 0x18
++#define PRP_DEST_RGB2_PTR 0x1c
++#define PRP_DEST_Y_PTR 0x20
++#define PRP_DEST_CB_PTR 0x24
++#define PRP_DEST_CR_PTR 0x28
++#define PRP_SRC_FRAME_SIZE 0x2c
++#define PRP_DEST_CH1_LINE_STRIDE 0x30
++#define PRP_SRC_PIXEL_FORMAT_CNTL 0x34
++#define PRP_CH1_PIXEL_FORMAT_CNTL 0x38
++#define PRP_CH1_OUT_IMAGE_SIZE 0x3c
++#define PRP_CH2_OUT_IMAGE_SIZE 0x40
++#define PRP_SRC_LINE_STRIDE 0x44
++#define PRP_CSC_COEF_012 0x48
++#define PRP_CSC_COEF_345 0x4c
++#define PRP_CSC_COEF_678 0x50
++#define PRP_CH1_RZ_HORI_COEF1 0x54
++#define PRP_CH1_RZ_HORI_COEF2 0x58
++#define PRP_CH1_RZ_HORI_VALID 0x5c
++#define PRP_CH1_RZ_VERT_COEF1 0x60
++#define PRP_CH1_RZ_VERT_COEF2 0x64
++#define PRP_CH1_RZ_VERT_VALID 0x68
++#define PRP_CH2_RZ_HORI_COEF1 0x6c
++#define PRP_CH2_RZ_HORI_COEF2 0x70
++#define PRP_CH2_RZ_HORI_VALID 0x74
++#define PRP_CH2_RZ_VERT_COEF1 0x78
++#define PRP_CH2_RZ_VERT_COEF2 0x7c
++#define PRP_CH2_RZ_VERT_VALID 0x80
++
++#define PRP_CNTL_CH1EN (1 << 0)
++#define PRP_CNTL_CH2EN (1 << 1)
++#define PRP_CNTL_CSIEN (1 << 2)
++#define PRP_CNTL_DATA_IN_YUV420 (0 << 3)
++#define PRP_CNTL_DATA_IN_YUV422 (1 << 3)
++#define PRP_CNTL_DATA_IN_RGB16 (2 << 3)
++#define PRP_CNTL_DATA_IN_RGB32 (3 << 3)
++#define PRP_CNTL_CH1_OUT_RGB8 (0 << 5)
++#define PRP_CNTL_CH1_OUT_RGB16 (1 << 5)
++#define PRP_CNTL_CH1_OUT_RGB32 (2 << 5)
++#define PRP_CNTL_CH1_OUT_YUV422 (3 << 5)
++#define PRP_CNTL_CH2_OUT_YUV420 (0 << 7)
++#define PRP_CNTL_CH2_OUT_YUV422 (1 << 7)
++#define PRP_CNTL_CH2_OUT_YUV444 (2 << 7)
++#define PRP_CNTL_CH1_LEN (1 << 9)
++#define PRP_CNTL_CH2_LEN (1 << 10)
++#define PRP_CNTL_SKIP_FRAME (1 << 11)
++#define PRP_CNTL_SWRST (1 << 12)
++#define PRP_CNTL_CLKEN (1 << 13)
++#define PRP_CNTL_WEN (1 << 14)
++#define PRP_CNTL_CH1BYP (1 << 15)
++#define PRP_CNTL_IN_TSKIP(x) ((x) << 16)
++#define PRP_CNTL_CH1_TSKIP(x) ((x) << 19)
++#define PRP_CNTL_CH2_TSKIP(x) ((x) << 22)
++#define PRP_CNTL_INPUT_FIFO_LEVEL(x) ((x) << 25)
++#define PRP_CNTL_RZ_FIFO_LEVEL(x) ((x) << 27)
++#define PRP_CNTL_CH2B1EN (1 << 29)
++#define PRP_CNTL_CH2B2EN (1 << 30)
++#define PRP_CNTL_CH2FEN (1 << 31)
++
++/* IRQ Enable and status register */
++#define PRP_INTR_RDERR (1 << 0)
++#define PRP_INTR_CH1WERR (1 << 1)
++#define PRP_INTR_CH2WERR (1 << 2)
++#define PRP_INTR_CH1FC (1 << 3)
++#define PRP_INTR_CH2FC (1 << 5)
++#define PRP_INTR_LBOVF (1 << 7)
++#define PRP_INTR_CH2OVF (1 << 8)
++
++#define mx27_camera_emma(pcdev) (pcdev->use_emma)
++
++/* Currently we do not need irqs. All we need is DMA callback
++ * Leave it here for reference for some time.
++ */
++#undef MX27_CAMERA_USE_IRQ
++
++struct mx27_camera_dev {
++ struct device *dev;
++ struct soc_camera_device *icd;
++ struct clk *clk_csi, *clk_emma;
++
++ unsigned int irq_csi, irq_emma;
++ void __iomem *base_csi, *base_emma;
++
++ struct mx27_camera_platform_data *pdata;
++ struct resource *res_csi, *res_emma;
++ unsigned long platform_flags;
++
++ struct list_head capture;
++ struct list_head active_bufs;
++
++ spinlock_t lock;
++
++ int dma;
++ struct mx27_buffer *active;
++
++ int use_emma;
++
++ unsigned int csicr1;
++
++ void __iomem *discard_buffer;
++ dma_addr_t discard_buffer_dma;
++ size_t discard_size;
++};
++
++/* buffer for one video frame */
++struct mx27_buffer {
++ /* common v4l buffer stuff -- must be first */
++ struct videobuf_buffer vb;
++
++ const struct soc_camera_data_format *fmt;
++
++ int bufnum;
++};
++
++static DEFINE_MUTEX(camera_lock);
++
++static int mclk_get_divisor(struct mx27_camera_dev *pcdev)
++{
++ dev_info(pcdev->dev, "%s not implemented. Running at max speed\n",
++ __func__);
++
++#if 0
++ unsigned int mclk = pcdev->pdata->clk_csi;
++ unsigned int pclk = clk_get_rate(pcdev->clk_csi);
++ int i;
++
++ dev_dbg(pcdev->dev, "%s: %ld %ld\n", __func__, mclk, pclk);
++
++ for (i = 0; i < 0xf; i++)
++ if ((i + 1) * 2 * mclk <= pclk)
++ break;
++ return i;
++#endif
++ return 0;
++}
++
++static void mx27_camera_deactivate(struct mx27_camera_dev *pcdev)
++{
++ clk_disable(pcdev->clk_csi);
++ writel(0, pcdev->base_csi + CSICR1);
++ if (mx27_camera_emma(pcdev))
++ writel(0, pcdev->base_emma + PRP_CNTL);
++}
++
++/* The following two functions absolutely depend on the fact, that
++ * there can be only one camera on mx27 quick capture interface */
++static int mx27_camera_add_device(struct soc_camera_device *icd)
++{
++ struct soc_camera_host *ici = to_soc_camera_host(icd->dev.parent);
++ struct mx27_camera_dev *pcdev = ici->priv;
++ int ret;
++ u32 csicr1;
++
++ mutex_lock(&camera_lock);
++
++ if (pcdev->icd) {
++ ret = -EBUSY;
++ goto ebusy;
++ }
++
++ dev_info(&icd->dev, "Camera driver attached to camera %d\n",
++ icd->devnum);
++
++ clk_enable(pcdev->clk_csi);
++
++ csicr1 = CSICR1_MCLKDIV(mclk_get_divisor(pcdev)) |
++ CSICR1_MCLKEN;
++ if (mx27_camera_emma(pcdev)) {
++ csicr1 |= CSICR1_PRP_IF_EN | CSICR1_FCC |
++ CSICR1_RXFF_LEVEL(0);
++ } else
++ csicr1 |= CSICR1_SOF_INTEN | CSICR1_RXFF_LEVEL(2);
++
++ pcdev->csicr1 = csicr1;
++ writel(pcdev->csicr1, pcdev->base_csi + CSICR1);
++
++ ret = icd->ops->init(icd);
++
++ if (!ret)
++ pcdev->icd = icd;
++
++ebusy:
++ mutex_unlock(&camera_lock);
++
++ return ret;
++}
++
++static void mx27_camera_remove_device(struct soc_camera_device *icd)
++{
++ struct soc_camera_host *ici = to_soc_camera_host(icd->dev.parent);
++ struct mx27_camera_dev *pcdev = ici->priv;
++
++ BUG_ON(icd != pcdev->icd);
++
++ dev_info(&icd->dev, "Camera driver detached from camera %d\n",
++ icd->devnum);
++
++ icd->ops->release(icd);
++
++ mx27_camera_deactivate(pcdev);
++
++ if (pcdev->discard_buffer) {
++ dma_free_coherent(NULL, pcdev->discard_size,
++ pcdev->discard_buffer,
++ pcdev->discard_buffer_dma);
++ }
++ pcdev->discard_buffer = 0;
++
++ pcdev->icd = NULL;
++}
++
++static void mx27_camera_dma_enable(struct mx27_camera_dev *pcdev)
++{
++ u32 tmp;
++
++ imx_dma_enable(pcdev->dma);
++
++ tmp = readl(pcdev->base_csi + CSICR1);
++ tmp |= CSICR1_RF_OR_INTEN;
++ writel(tmp, pcdev->base_csi + CSICR1);
++}
++
++static irqreturn_t mx27_camera_irq(int irq_csi, void *data)
++{
++ struct mx27_camera_dev *pcdev = data;
++ u32 status = readl(pcdev->base_csi + CSISR);
++
++ if (status & CSISR_SOF_INT && pcdev->active) {
++ u32 tmp;
++
++ tmp = readl(pcdev->base_csi + CSICR1);
++ tmp |= CSICR1_CLR_RXFIFO;
++ writel(tmp, pcdev->base_csi + CSICR1);
++ mx27_camera_dma_enable(pcdev);
++ writel(CSISR_SOF_INT | CSISR_RFF_OR_INT,
++ pcdev->base_csi + CSISR);
++ status &= ~CSISR_RFF_OR_INT;
++ }
++
++ writel(CSISR_SOF_INT | CSISR_RFF_OR_INT, pcdev->base_csi + CSISR);
++
++ return IRQ_HANDLED;
++}
++
++static unsigned int vid_limit = 16; /* Video memory limit, in Mb */
++
++/*
++ * Videobuf operations
++ */
++static int mx27_videobuf_setup(struct videobuf_queue *vq, unsigned int *count,
++ unsigned int *size)
++{
++ struct soc_camera_device *icd = vq->priv_data;
++
++ dev_dbg(&icd->dev, "count=%d, size=%d\n", *count, *size);
++
++ *size = icd->width * icd->height *
++ ((icd->current_fmt->depth + 7) >> 3);
++
++ if (0 == *count)
++ *count = 32;
++ while (*size * *count > vid_limit * 1024 * 1024)
++ (*count)--;
++
++ return 0;
++}
++
++static void free_buffer(struct videobuf_queue *vq, struct mx27_buffer *buf)
++{
++ struct soc_camera_device *icd = vq->priv_data;
++
++ BUG_ON(in_interrupt());
++
++ dev_dbg(&icd->dev, "%s (vb=0x%p) 0x%08lx %d\n", __func__,
++ &buf->vb, buf->vb.baddr, buf->vb.bsize);
++
++ /* This waits until this buffer is out of danger, i.e., until it is no
++ * longer in STATE_QUEUED or STATE_ACTIVE */
++ videobuf_waiton(&buf->vb, 0, 0);
++
++ videobuf_dma_contig_free(vq, &buf->vb);
++ dev_dbg(&icd->dev, "%s freed\n", __func__);
++
++ buf->vb.state = VIDEOBUF_NEEDS_INIT;
++}
++
++static int mx27_videobuf_prepare(struct videobuf_queue *vq,
++ struct videobuf_buffer *vb, enum v4l2_field field)
++{
++ struct soc_camera_device *icd = vq->priv_data;
++ struct mx27_buffer *buf = container_of(vb, struct mx27_buffer, vb);
++ int ret = 0;
++
++#ifdef DEBUG
++ /* This can be useful if you want to see if we actually fill
++ * the buffer with something */
++ memset((void *)vb->baddr, 0xaa, vb->bsize);
++#endif
++
++ dev_dbg(&icd->dev, "%s (vb=0x%p) 0x%08lx %d\n", __func__,
++ vb, vb->baddr, vb->bsize);
++
++ if (buf->fmt != icd->current_fmt ||
++ vb->width != icd->width ||
++ vb->height != icd->height ||
++ vb->field != field) {
++ buf->fmt = icd->current_fmt;
++ vb->width = icd->width;
++ vb->height = icd->height;
++ vb->field = field;
++ vb->state = VIDEOBUF_NEEDS_INIT;
++ }
++
++ vb->size = vb->width * vb->height * ((buf->fmt->depth + 7) >> 3);
++ if (vb->baddr && vb->bsize < vb->size) {
++ ret = -EINVAL;
++ goto out;
++ }
++
++ if (vb->state == VIDEOBUF_NEEDS_INIT) {
++ ret = videobuf_iolock(vq, vb, NULL);
++ if (ret)
++ goto fail;
++
++ vb->state = VIDEOBUF_PREPARED;
++ }
++
++ return 0;
++
++fail:
++ free_buffer(vq, buf);
++out:
++ return ret;
++}
++
++static void mx27_videobuf_queue(struct videobuf_queue *vq,
++ struct videobuf_buffer *vb)
++{
++ struct soc_camera_device *icd = vq->priv_data;
++ struct soc_camera_host *ici =
++ to_soc_camera_host(icd->dev.parent);
++ struct mx27_camera_dev *pcdev = ici->priv;
++ struct mx27_buffer *buf = container_of(vb, struct mx27_buffer, vb);
++ unsigned long flags;
++ int ret;
++
++ dev_dbg(&icd->dev, "%s (vb=0x%p) 0x%08lx %d\n", __func__,
++ vb, vb->baddr, vb->bsize);
++
++ spin_lock_irqsave(&pcdev->lock, flags);
++
++ if (mx27_camera_emma(pcdev)) {
++ list_add_tail(&vb->queue, &pcdev->capture);
++ vb->state = VIDEOBUF_QUEUED;
++ } else {
++ list_add_tail(&vb->queue, &pcdev->capture);
++ vb->state = VIDEOBUF_ACTIVE;
++
++ if (!pcdev->active) {
++ ret = imx_dma_setup_single(pcdev->dma,
++ videobuf_to_dma_contig(vb), vb->size,
++ CSI_BASE_ADDR + 0x10, DMA_MODE_READ);
++ if (ret) {
++ vb->state = VIDEOBUF_ERROR;
++ wake_up(&vb->done);
++ goto out;
++ }
++
++ pcdev->active = buf;
++ }
++ }
++
++out:
++ spin_unlock_irqrestore(&pcdev->lock, flags);
++}
++
++static void mx27_videobuf_release(struct videobuf_queue *vq,
++ struct videobuf_buffer *vb)
++{
++ struct mx27_buffer *buf = container_of(vb, struct mx27_buffer, vb);
++
++#ifdef DEBUG
++ struct soc_camera_device *icd = vq->priv_data;
++
++ dev_dbg(&icd->dev, "%s (vb=0x%p) 0x%08lx %d\n", __func__,
++ vb, vb->baddr, vb->bsize);
++
++ switch (vb->state) {
++ case VIDEOBUF_ACTIVE:
++ dev_info(&icd->dev, "%s (active)\n", __func__);
++ break;
++ case VIDEOBUF_QUEUED:
++ dev_info(&icd->dev, "%s (queued)\n", __func__);
++ break;
++ case VIDEOBUF_PREPARED:
++ dev_info(&icd->dev, "%s (prepared)\n", __func__);
++ break;
++ default:
++ dev_info(&icd->dev, "%s (unknown) %d\n", __func__,
++ vb->state);
++ break;
++ }
++#endif
++
++ free_buffer(vq, buf);
++}
++
++static struct videobuf_queue_ops mx27_videobuf_ops = {
++ .buf_setup = mx27_videobuf_setup,
++ .buf_prepare = mx27_videobuf_prepare,
++ .buf_queue = mx27_videobuf_queue,
++ .buf_release = mx27_videobuf_release,
++};
++
++static void mx27_camera_init_videobuf(struct videobuf_queue *q,
++ struct soc_camera_device *icd)
++{
++ struct soc_camera_host *ici = to_soc_camera_host(icd->dev.parent);
++ struct mx27_camera_dev *pcdev = ici->priv;
++
++ /* We must pass NULL as dev pointer, then all pci_* dma operations
++ * transform to normal dma_* ones. */
++ videobuf_queue_dma_contig_init(q, &mx27_videobuf_ops, NULL,
++ &pcdev->lock, V4L2_BUF_TYPE_VIDEO_CAPTURE,
++ V4L2_FIELD_NONE, sizeof(struct mx27_buffer), icd);
++}
++
++#define MX27_BUS_FLAGS (SOCAM_DATAWIDTH_8 | \
++ SOCAM_MASTER | \
++ SOCAM_VSYNC_ACTIVE_HIGH | \
++ SOCAM_HSYNC_ACTIVE_HIGH | \
++ SOCAM_HSYNC_ACTIVE_LOW | \
++ SOCAM_PCLK_SAMPLE_RISING | \
++ SOCAM_PCLK_SAMPLE_FALLING)
++
++static int mx27_camera_emma_prp_reset(struct mx27_camera_dev *pcdev)
++{
++ unsigned int cntl;
++
++ cntl = readl(pcdev->base_emma + PRP_CNTL);
++ writel(PRP_CNTL_SWRST, pcdev->base_emma + PRP_CNTL);
++ while (readl(pcdev->base_emma + PRP_CNTL) & PRP_CNTL_SWRST)
++ barrier();
++
++ return 0;
++}
++
++static int mx27_camera_set_bus_param(struct soc_camera_device *icd,
++ __u32 pixfmt)
++{
++ struct soc_camera_host *ici =
++ to_soc_camera_host(icd->dev.parent);
++ struct mx27_camera_dev *pcdev = ici->priv;
++ unsigned long camera_flags, common_flags;
++ int ret = 0;
++ u32 csicr1 = pcdev->csicr1;
++
++ camera_flags = icd->ops->query_bus_param(icd);
++
++ common_flags = soc_camera_bus_param_compatible(camera_flags,
++ MX27_BUS_FLAGS);
++ if (!common_flags)
++ return -EINVAL;
++
++ if ((common_flags & SOCAM_HSYNC_ACTIVE_HIGH) &&
++ (common_flags & SOCAM_HSYNC_ACTIVE_LOW)) {
++ if (pcdev->platform_flags & MX27_CAMERA_HSYNC_HIGH)
++ common_flags &= ~SOCAM_HSYNC_ACTIVE_LOW;
++ else
++ common_flags &= ~SOCAM_HSYNC_ACTIVE_HIGH;
++ }
++
++ if ((common_flags & SOCAM_PCLK_SAMPLE_RISING) &&
++ (common_flags & SOCAM_PCLK_SAMPLE_FALLING)) {
++ if (pcdev->platform_flags & MX27_CAMERA_PCLK_SAMPLE_RISING)
++ common_flags &= ~SOCAM_PCLK_SAMPLE_FALLING;
++ else
++ common_flags &= ~SOCAM_PCLK_SAMPLE_RISING;
++ }
++
++ ret = icd->ops->set_bus_param(icd, common_flags);
++ if (ret < 0)
++ return ret;
++
++ if (common_flags & SOCAM_PCLK_SAMPLE_FALLING)
++ csicr1 |= CSICR1_INV_PCLK;
++ if (common_flags & SOCAM_HSYNC_ACTIVE_HIGH)
++ csicr1 |= CSICR1_HSYNC_POL;
++ if (pcdev->platform_flags & MX27_CAMERA_SWAP16)
++ csicr1 |= CSICR1_SWAP16_EN;
++ if (pcdev->platform_flags & MX27_CAMERA_EXT_VSYNC)
++ csicr1 |= CSICR1_EXT_VSYNC;
++ if (pcdev->platform_flags & MX27_CAMERA_CCIR)
++ csicr1 |= CSICR1_CCIR_EN;
++ if (pcdev->platform_flags & MX27_CAMERA_CCIR_INTERLACE)
++ csicr1 |= CSICR1_CCIR_MODE;
++ if (pcdev->platform_flags & MX27_CAMERA_GATED_CLOCK)
++ csicr1 |= CSICR1_GCLK_MODE;
++ if (pcdev->platform_flags & MX27_CAMERA_INV_DATA)
++ csicr1 |= CSICR1_INV_DATA;
++ if (pcdev->platform_flags & MX27_CAMERA_PACK_DIR_MSB)
++ csicr1 |= CSICR1_PACK_DIR;
++
++ if (mx27_camera_emma(pcdev)) {
++ int bytesperline = (icd->width * icd->current_fmt->depth) >> 3;
++
++ if (mx27_camera_emma_prp_reset(pcdev))
++ return -ENODEV;
++
++ if (pcdev->discard_buffer)
++ dma_free_coherent(NULL, pcdev->discard_size,
++ pcdev->discard_buffer,
++ pcdev->discard_buffer_dma);
++
++ /* I didn't manage to properly enable/disable the prp
++ * on a per frame basis during running transfers,
++ * thus we allocate a buffer here and use it to
++ * discard frames when no buffer is available.
++ * Feel free to work on this ;)
++ */
++ pcdev->discard_size = icd->height * bytesperline;
++ pcdev->discard_buffer = dma_alloc_coherent(NULL,
++ pcdev->discard_size, &pcdev->discard_buffer_dma,
++ GFP_KERNEL);
++ if (!pcdev->discard_buffer)
++ return -ENOMEM;
++
++ writel(pcdev->discard_buffer_dma,
++ pcdev->base_emma + PRP_DEST_RGB1_PTR);
++ writel(pcdev->discard_buffer_dma,
++ pcdev->base_emma + PRP_DEST_RGB2_PTR);
++
++ /* We only use the EMMA engine to get rid of the f**king
++ * DMA Engine. No color space consversion at the moment.
++ * We adjust incoming and outgoing pixelformat to rgb16
++ * and adjust the bytesperline accordingly.
++ */
++ writel(PRP_CNTL_CH1EN |
++ PRP_CNTL_CSIEN |
++ PRP_CNTL_DATA_IN_RGB16 |
++ PRP_CNTL_CH1_OUT_RGB16 |
++ PRP_CNTL_CH1_LEN |
++ PRP_CNTL_CH1BYP |
++ PRP_CNTL_CH1_TSKIP(0) |
++ PRP_CNTL_IN_TSKIP(0),
++ pcdev->base_emma + PRP_CNTL);
++
++ writel(((bytesperline >> 1) << 16) | icd->height,
++ pcdev->base_emma + PRP_SRC_FRAME_SIZE);
++ writel(((bytesperline >> 1) << 16) | icd->height,
++ pcdev->base_emma + PRP_CH1_OUT_IMAGE_SIZE);
++ writel(bytesperline,
++ pcdev->base_emma + PRP_DEST_CH1_LINE_STRIDE);
++ writel(0x2ca00565,
++ pcdev->base_emma + PRP_SRC_PIXEL_FORMAT_CNTL);
++ writel(0x2ca00565,
++ pcdev->base_emma + PRP_CH1_PIXEL_FORMAT_CNTL);
++
++ /* Enable interrupts */
++ writel(PRP_INTR_RDERR |
++ PRP_INTR_CH1WERR |
++ PRP_INTR_CH2WERR |
++ PRP_INTR_CH1FC |
++ PRP_INTR_CH2FC |
++ PRP_INTR_LBOVF |
++ PRP_INTR_CH2OVF
++ , pcdev->base_emma + PRP_INTR_CNTL);
++ }
++
++ pcdev->csicr1 = csicr1;
++
++ writel(csicr1, pcdev->base_csi + CSICR1);
++
++ return 0;
++}
++
++static int mx27_camera_try_bus_param(struct soc_camera_device *icd,
++ __u32 pixfmt)
++{
++ unsigned long bus_flags, camera_flags;
++
++ bus_flags = MX27_BUS_FLAGS;
++
++ camera_flags = icd->ops->query_bus_param(icd);
++
++ return soc_camera_bus_param_compatible(camera_flags, bus_flags) ?
++ 0 : -EINVAL;
++}
++
++static int mx27_camera_set_fmt_cap(struct soc_camera_device *icd,
++ __u32 pixfmt, struct v4l2_rect *rect)
++{
++ return icd->ops->set_fmt_cap(icd, pixfmt, rect);
++}
++
++static int mx27_camera_try_fmt_cap(struct soc_camera_device *icd,
++ struct v4l2_format *f)
++{
++ return 0;
++}
++
++static int mx27_camera_querycap(struct soc_camera_host *ici,
++ struct v4l2_capability *cap)
++{
++ /* cap->name is set by the friendly caller:-> */
++ strlcpy(cap->card, mx27_cam_driver_description, sizeof(cap->card));
++ cap->version = MX27_CAM_VERSION_CODE;
++ cap->capabilities = V4L2_CAP_VIDEO_CAPTURE | V4L2_CAP_STREAMING;
++
++ return 0;
++}
++
++static int mx27_camera_reqbufs(struct soc_camera_file *icf,
++ struct v4l2_requestbuffers *p)
++{
++ int i;
++
++ for (i = 0; i < p->count; i++) {
++ struct mx27_buffer *buf = container_of(icf->vb_vidq.bufs[i],
++ struct mx27_buffer, vb);
++ INIT_LIST_HEAD(&buf->vb.queue);
++ }
++
++ return 0;
++}
++
++static void mx27_camera_frame_done(struct mx27_camera_dev *pcdev, int state)
++{
++ struct videobuf_buffer *vb;
++ struct mx27_buffer *buf;
++ int ret;
++
++ if (!pcdev->active) {
++ dev_err(pcdev->dev, "%s called with no active buffer!\n",
++ __func__);
++ return;
++ }
++
++ vb = &pcdev->active->vb;
++ buf = container_of(vb, struct mx27_buffer, vb);
++ WARN_ON(list_empty(&vb->queue));
++ dev_dbg(pcdev->dev, "%s (vb=0x%p) 0x%08lx %d\n", __func__,
++ vb, vb->baddr, vb->bsize);
++
++ /* _init is used to debug races, see comment in pxa_camera_reqbufs() */
++ list_del_init(&vb->queue);
++ vb->state = state;
++ do_gettimeofday(&vb->ts);
++ vb->field_count++;
++
++ wake_up(&vb->done);
++
++ if (list_empty(&pcdev->capture)) {
++ pcdev->active = NULL;
++ return;
++ }
++
++ pcdev->active = list_entry(pcdev->capture.next,
++ struct mx27_buffer, vb.queue);
++
++ vb = &pcdev->active->vb;
++ vb->state = VIDEOBUF_ACTIVE;
++
++ ret = imx_dma_setup_single(pcdev->dma, videobuf_to_dma_contig(vb),
++ vb->size, CSI_BASE_ADDR + 0x10, DMA_MODE_READ);
++ if (ret) {
++ vb->state = VIDEOBUF_ERROR;
++ wake_up(&vb->done);
++ return;
++ }
++}
++
++static void mx27_camera_dma_err_callback(int channel, void *data, int err)
++{
++ struct mx27_camera_dev *pcdev = data;
++ unsigned long flags;
++
++ spin_lock_irqsave(&pcdev->lock, flags);
++
++ mx27_camera_frame_done(pcdev, VIDEOBUF_ERROR);
++
++ spin_unlock_irqrestore(&pcdev->lock, flags);
++}
++
++static void mx27_camera_dma_callback(int channel, void *data)
++{
++ struct mx27_camera_dev *pcdev = data;
++ unsigned long flags;
++
++ spin_lock_irqsave(&pcdev->lock, flags);
++
++ mx27_camera_frame_done(pcdev, VIDEOBUF_DONE);
++
++ spin_unlock_irqrestore(&pcdev->lock, flags);
++}
++
++static unsigned int mx27_camera_poll(struct file *file, poll_table *pt)
++{
++ struct soc_camera_file *icf = file->private_data;
++ struct mx27_buffer *buf;
++
++ buf = list_entry(icf->vb_vidq.stream.next, struct mx27_buffer,
++ vb.stream);
++
++ poll_wait(file, &buf->vb.done, pt);
++
++ if (buf->vb.state == VIDEOBUF_DONE ||
++ buf->vb.state == VIDEOBUF_ERROR)
++ return POLLIN | POLLRDNORM;
++
++ return 0;
++}
++
++/* Should beallocated dynamically too, but we have only one. */
++static struct soc_camera_host_ops mx27_soc_camera_host_ops = {
++ .owner = THIS_MODULE,
++ .add = mx27_camera_add_device,
++ .remove = mx27_camera_remove_device,
++ .set_fmt_cap = mx27_camera_set_fmt_cap,
++ .try_fmt_cap = mx27_camera_try_fmt_cap,
++ .init_videobuf = mx27_camera_init_videobuf,
++ .reqbufs = mx27_camera_reqbufs,
++ .poll = mx27_camera_poll,
++ .querycap = mx27_camera_querycap,
++ .try_bus_param = mx27_camera_try_bus_param,
++ .set_bus_param = mx27_camera_set_bus_param,
++};
++
++static void mx27_camera_frame_done_emma(struct mx27_camera_dev *pcdev,
++ int bufnum, int state)
++{
++ struct mx27_buffer *buf;
++ struct videobuf_buffer *vb;
++ unsigned long phys;
++
++ if (!list_empty(&pcdev->active_bufs)) {
++ buf = list_entry(pcdev->active_bufs.next,
++ struct mx27_buffer, vb.queue);
++
++ if (buf->bufnum == bufnum) {
++ vb = &buf->vb;
++#ifdef DEBUG
++ phys = videobuf_to_dma_contig(vb);
++ if (readl(pcdev->base_emma + PRP_DEST_RGB1_PTR +
++ 4 * bufnum) != phys) {
++ dev_err(pcdev->dev, "%p != %p\n", phys,
++ readl(pcdev->base_emma +
++ PRP_DEST_RGB1_PTR +
++ 4 * bufnum));
++ }
++#endif
++ dev_dbg(pcdev->dev, "%s (vb=0x%p) 0x%08lx %d\n",
++ __func__, vb, vb->baddr, vb->bsize);
++
++ list_del(&vb->queue);
++ vb->state = state;
++ do_gettimeofday(&vb->ts);
++ vb->field_count++;
++
++ wake_up(&vb->done);
++ }
++ }
++
++ if (list_empty(&pcdev->capture)) {
++ writel(pcdev->discard_buffer_dma, pcdev->base_emma +
++ PRP_DEST_RGB1_PTR + 4 * bufnum);
++ return;
++ }
++
++ buf = list_entry(pcdev->capture.next,
++ struct mx27_buffer, vb.queue);
++
++ buf->bufnum = bufnum;
++
++ list_move_tail(pcdev->capture.next, &pcdev->active_bufs);
++
++ vb = &buf->vb;
++ vb->state = VIDEOBUF_ACTIVE;
++
++ phys = videobuf_to_dma_contig(vb);
++ writel(phys, pcdev->base_emma + PRP_DEST_RGB1_PTR + 4 * bufnum);
++}
++
++static irqreturn_t mx27_camera_emma_irq(int irq_emma, void *data)
++{
++ struct mx27_camera_dev *pcdev = data;
++ unsigned int status = readl(pcdev->base_emma + PRP_INTRSTATUS);
++
++ if (status & (1 << 6))
++ mx27_camera_frame_done_emma(pcdev, 0, VIDEOBUF_DONE);
++ if (status & (1 << 5))
++ mx27_camera_frame_done_emma(pcdev, 1, VIDEOBUF_DONE);
++ if (status & (1 << 7)) {
++ uint32_t cntl;
++ cntl = readl(pcdev->base_emma + PRP_CNTL);
++ writel(cntl & ~PRP_CNTL_CH1EN, pcdev->base_emma + PRP_CNTL);
++ writel(cntl, pcdev->base_emma + PRP_CNTL);
++ }
++
++ writel(status, pcdev->base_emma + PRP_INTRSTATUS);
++
++ return IRQ_HANDLED;
++}
++
++/* Should be allocated dynamically too, but we have only one. */
++static struct soc_camera_host mx27_soc_camera_host = {
++ .drv_name = MX27_CAM_DRV_NAME,
++ .ops = &mx27_soc_camera_host_ops,
++};
++
++int mx27_camera_emma_init(struct mx27_camera_dev *pcdev)
++{
++ struct resource *res_emma = pcdev->res_emma;
++ int err = 0;
++
++ if (!request_mem_region(res_emma->start, resource_size(res_emma),
++ MX27_CAM_DRV_NAME)) {
++ err = -EBUSY;
++ goto out;
++ }
++
++ pcdev->base_emma = ioremap(res_emma->start, resource_size(res_emma));
++ if (!pcdev->base_emma) {
++ err = -ENOMEM;
++ goto exit_release;
++ }
++
++ err = request_irq(pcdev->irq_emma, mx27_camera_emma_irq, 0,
++ MX27_CAM_DRV_NAME, pcdev);
++ if (err) {
++ dev_err(pcdev->dev, "Camera EMMA interrupt register failed \n");
++ goto exit_iounmap;
++ }
++
++ pcdev->clk_emma = clk_get(pcdev->dev, "emma_clk");
++ if (IS_ERR(pcdev->clk_emma)) {
++ err = PTR_ERR(pcdev->clk_emma);
++ goto exit_free_irq;
++ }
++
++ clk_enable(pcdev->clk_emma);
++
++ err = mx27_camera_emma_prp_reset(pcdev);
++ if (err)
++ goto exit_clk_emma_put;
++
++ return err;
++
++exit_clk_emma_put:
++ clk_disable(pcdev->clk_emma);
++ clk_put(pcdev->clk_emma);
++exit_free_irq:
++ free_irq(pcdev->irq_emma, pcdev);
++exit_iounmap:
++ iounmap(pcdev->base_emma);
++exit_release:
++ release_mem_region(res_emma->start, resource_size(res_emma));
++out:
++ return err;
++}
++
++static int mx27_camera_probe(struct platform_device *pdev)
++{
++ struct mx27_camera_dev *pcdev;
++ struct resource *res_csi, *res_emma;
++ void __iomem *base_csi;
++ unsigned int irq_csi, irq_emma;
++ int err = 0;
++
++ dev_info(&pdev->dev, "initialising\n");
++
++ res_csi = platform_get_resource(pdev, IORESOURCE_MEM, 0);
++ irq_csi = platform_get_irq(pdev, 0);
++ if (!res_csi || !irq_csi) {
++ dev_err(&pdev->dev, "No platform irq\n");
++ err = -ENODEV;
++ goto exit;
++ }
++
++ pcdev = kzalloc(sizeof(*pcdev), GFP_KERNEL);
++ if (!pcdev) {
++ dev_err(&pdev->dev, "Could not allocate pcdev\n");
++ err = -ENOMEM;
++ goto exit;
++ }
++
++ pcdev->clk_csi = clk_get(&pdev->dev, "csi_perclk");
++ if (IS_ERR(pcdev->clk_csi)) {
++ err = PTR_ERR(pcdev->clk_csi);
++ goto exit_kfree;
++ }
++
++ dev_info(&pdev->dev, "Camera clock frequency: %ld\n",
++ clk_get_rate(pcdev->clk_csi));
++
++ /* Initialize DMA */
++ pcdev->dma = imx_dma_request_by_prio("CSI RX DMA", DMA_PRIO_HIGH);
++ if (pcdev->dma < 0) {
++ dev_err(&pdev->dev,
++ "mxc_v4l_read failed to request DMA channel\n");
++ err = -EIO;
++ goto exit_clk_put;
++ }
++
++ err = imx_dma_setup_handlers(pcdev->dma, mx27_camera_dma_callback,
++ mx27_camera_dma_err_callback, pcdev);
++ if (err != 0) {
++ dev_err(&pdev->dev,
++ "mxc_v4l_read failed to set DMA callback\n");
++ err = -EIO;
++ goto exit_dma_free;
++ }
++
++ imx_dma_config_channel(pcdev->dma,
++ IMX_DMA_MEMSIZE_32 | IMX_DMA_TYPE_FIFO,
++ IMX_DMA_MEMSIZE_32 | IMX_DMA_TYPE_LINEAR,
++ DMA_REQ_CSI_RX, 1);
++
++ imx_dma_config_burstlen(pcdev->dma, 64);
++
++ dev_set_drvdata(&pdev->dev, pcdev);
++ pcdev->res_csi = res_csi;
++
++ pcdev->pdata = pdev->dev.platform_data;
++ pcdev->platform_flags = pcdev->pdata->flags;
++
++ clk_set_rate(pcdev->clk_csi, pcdev->pdata->clk * 2);
++
++ INIT_LIST_HEAD(&pcdev->capture);
++ INIT_LIST_HEAD(&pcdev->active_bufs);
++ spin_lock_init(&pcdev->lock);
++
++ /*
++ * Request the regions.
++ */
++ if (!request_mem_region(res_csi->start, resource_size(res_csi),
++ MX27_CAM_DRV_NAME)) {
++ err = -EBUSY;
++ goto exit_dma_free;
++ }
++
++ base_csi = ioremap(res_csi->start, resource_size(res_csi));
++ if (!base_csi) {
++ err = -ENOMEM;
++ goto exit_release;
++ }
++ pcdev->irq_csi = irq_csi;
++ pcdev->base_csi = base_csi;
++ pcdev->dev = &pdev->dev;
++
++ pcdev->pdata->init(pdev);
++
++ err = request_irq(pcdev->irq_csi, mx27_camera_irq, 0, MX27_CAM_DRV_NAME,
++ pcdev);
++ if (err) {
++ dev_err(pcdev->dev, "Camera interrupt register failed \n");
++ goto exit_iounmap;
++ }
++
++ mx27_soc_camera_host.priv = pcdev;
++ mx27_soc_camera_host.dev.parent = &pdev->dev;
++ mx27_soc_camera_host.nr = pdev->id;
++
++ /* EMMA support */
++ res_emma = platform_get_resource(pdev, IORESOURCE_MEM, 1);
++ irq_emma = platform_get_irq(pdev, 1);
++
++ if (res_emma && irq_emma) {
++ dev_info(&pdev->dev, "Using EMMA\n");
++ pcdev->use_emma = 1;
++ pcdev->res_emma = res_emma;
++ pcdev->irq_emma = irq_emma;
++ if (mx27_camera_emma_init(pcdev))
++ goto exit_free_irq;
++ }
++
++ err = soc_camera_host_register(&mx27_soc_camera_host);
++ if (err)
++ goto exit_free_emma;
++
++ return 0;
++
++exit_free_emma:
++ free_irq(pcdev->irq_emma, pcdev);
++ clk_disable(pcdev->clk_emma);
++ clk_put(pcdev->clk_emma);
++ iounmap(pcdev->base_emma);
++ release_mem_region(res_emma->start, resource_size(res_emma));
++exit_free_irq:
++ free_irq(pcdev->irq_csi, pcdev);
++exit_iounmap:
++ iounmap(base_csi);
++exit_release:
++ release_mem_region(res_csi->start, resource_size(res_csi));
++exit_dma_free:
++ imx_dma_free(pcdev->dma);
++exit_clk_put:
++ clk_put(pcdev->clk_csi);
++exit_kfree:
++ kfree(pcdev);
++exit:
++ return err;
++}
++
++static int __devexit mx27_camera_remove(struct platform_device *pdev)
++{
++ struct mx27_camera_dev *pcdev = platform_get_drvdata(pdev);
++ struct resource *res;
++
++ clk_put(pcdev->clk_csi);
++ imx_dma_free(pcdev->dma);
++ free_irq(pcdev->irq_csi, pcdev);
++ if (mx27_camera_emma(pcdev))
++ free_irq(pcdev->irq_emma, pcdev);
++
++ soc_camera_host_unregister(&mx27_soc_camera_host);
++
++ iounmap(pcdev->base_csi);
++
++ if (mx27_camera_emma(pcdev)) {
++ clk_disable(pcdev->clk_emma);
++ clk_put(pcdev->clk_emma);
++ iounmap(pcdev->base_emma);
++ res = pcdev->res_emma;
++ release_mem_region(res->start, resource_size(res));
++ }
++
++ pcdev->pdata->exit(pdev);
++
++ res = pcdev->res_csi;
++ release_mem_region(res->start, resource_size(res));
++
++ kfree(pcdev);
++
++ dev_info(&pdev->dev, "MX27 Camera driver unloaded\n");
++
++ return 0;
++}
++
++static struct platform_driver mx27_camera_driver = {
++ .driver = {
++ .name = MX27_CAM_DRV_NAME,
++ },
++ .probe = mx27_camera_probe,
++ .remove = __exit_p(mx27_camera_remove),
++};
++
++
++static int __devinit mx27_camera_init(void)
++{
++ return platform_driver_register(&mx27_camera_driver);
++}
++
++static void __exit mx27_camera_exit(void)
++{
++ return platform_driver_unregister(&mx27_camera_driver);
++}
++
++module_init(mx27_camera_init);
++module_exit(mx27_camera_exit);
++
++MODULE_DESCRIPTION("i.MX27 SoC Camera Host driver");
++MODULE_AUTHOR("Sascha Hauer <sha@pengutronix.de>");
++MODULE_LICENSE("GPL");
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/capture/Kconfig linux-2.6.28-karo/drivers/media/video/mxc/capture/Kconfig
+--- linux-2.6.28/drivers/media/video/mxc/capture/Kconfig 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/capture/Kconfig 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,80 @@
++if VIDEO_MXC_CAMERA
++
++menu "MXC Camera/V4L2 PRP Features support"
++config VIDEO_MXC_IPU_CAMERA
++ bool
++ depends on VIDEO_MXC_CAMERA && MXC_IPU
++ default y
++
++config VIDEO_MXC_EMMA_CAMERA
++ bool "MX27 eMMA support"
++ depends on VIDEO_MXC_CAMERA && MXC_EMMA && FB_MXC_SYNC_PANEL && (MXC_CAMERA_MICRON111 || MXC_CAMERA_MC521DA || MXC_CAMERA_OV2640)
++ select VIDEO_MXC_OPL
++ default y
++
++config VIDEO_MXC_CSI_DMA
++ bool "CSI-DMA Still Image Capture support"
++ depends on VIDEO_MXC_EMMA_CAMERA
++ default n
++ ---help---
++ Use CSI-DMA method instead of CSI-PrP link to capture still image. This allows
++ to use less physical contiguous memory to capture big resolution still image. But
++ with this method the CSC (Color Space Conversion) and resize are not supported.
++ If unsure, say N.
++
++choice
++ prompt "Select Camera"
++ default MXC_CAMERA_MICRON111
++ depends on (VIDEO_MXC_CAMERA && I2C_MXC)
++
++config MXC_CAMERA_MICRON111
++ tristate "Micron mt9v111 camera support"
++ ---help---
++ If you plan to use the mt9v111 Camera with your MXC system, say Y here.
++
++config MXC_CAMERA_MC521DA
++ tristate "Magnachip mc521da camera support"
++ ---help---
++ If you plan to use the mc521da Camera with your MXC system, say Y here.
++
++config MXC_CAMERA_OV2640
++ tristate "OmniVision ov2640 camera support"
++ ---help---
++ If you plan to use the ov2640 Camera with your MXC system, say Y here.
++endchoice
++
++config MXC_IPU_PRP_VF_SDC
++ tristate "Pre-Processor VF SDC library"
++ depends on (VIDEO_MXC_IPU_CAMERA && FB_MXC_SYNC_PANEL && (MXC_CAMERA_MICRON111 || MXC_CAMERA_MC521DA || MXC_CAMERA_OV2640))
++ default y
++ ---help---
++ Use case PRP_VF_SDC:
++ Preprocessing image from smart sensor for viewfinder and
++ displaying it on synchronous display with SDC use case.
++ If SDC BG is selected, Rotation will not be supported.
++ CSI -> IC (PRP VF) -> MEM
++ MEM -> IC (ROT) -> MEM
++ MEM -> SDC (FG/BG)
++
++config MXC_IPU_PRP_VF_ADC
++ tristate "Pre-Processor VF ADC library"
++ depends on (VIDEO_MXC_IPU_CAMERA && FB_MXC_ASYNC_PANEL && (MXC_CAMERA_MICRON111 || MXC_CAMERA_MC521DA || MXC_CAMERA_OV2640))
++ default y
++ ---help---
++ Use case PRP_VF_ADC:
++ Preprocessing image from smart sensor for viewfinder and
++ displaying it on asynchronous display.
++ CSI -> IC (PRP VF) -> ADC2
++
++config MXC_IPU_PRP_ENC
++ tristate "Pre-processor Encoder library"
++ depends on (VIDEO_MXC_IPU_CAMERA && (MXC_CAMERA_MICRON111 || MXC_CAMERA_MC521DA || MXC_CAMERA_OV2640))
++ default y
++ ---help---
++ Use case PRP_ENC:
++ Preprocessing image from smart sensor for encoder.
++ CSI -> IC (PRP ENC) -> MEM
++
++endmenu
++
++endif
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/capture/Makefile linux-2.6.28-karo/drivers/media/video/mxc/capture/Makefile
+--- linux-2.6.28/drivers/media/video/mxc/capture/Makefile 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/capture/Makefile 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,18 @@
++ifeq ($(CONFIG_VIDEO_MXC_IPU_CAMERA),y)
++ obj-$(CONFIG_VIDEO_MXC_CAMERA) += mxc_v4l2_capture.o
++ obj-$(CONFIG_MXC_IPU_PRP_VF_ADC) += ipu_prp_vf_adc.o
++ obj-$(CONFIG_MXC_IPU_PRP_VF_SDC) += ipu_prp_vf_sdc.o ipu_prp_vf_sdc_bg.o
++ obj-$(CONFIG_MXC_IPU_PRP_ENC) += ipu_prp_enc.o ipu_still.o
++endif
++
++mx27_capture-objs := mx27_prphw.o mx27_prpsw.o mx27_v4l2_capture.o
++obj-$(CONFIG_VIDEO_MXC_EMMA_CAMERA) += mx27_csi.o mx27_capture.o
++
++mc521da_camera-objs := mc521da.o sensor_clock.o
++obj-$(CONFIG_MXC_CAMERA_MC521DA) += mc521da_camera.o
++
++mt9v111_camera-objs := mt9v111.o sensor_clock.o
++obj-$(CONFIG_MXC_CAMERA_MICRON111) += mt9v111_camera.o
++
++ov2640_camera-objs := ov2640.o sensor_clock.o
++obj-$(CONFIG_MXC_CAMERA_OV2640) += ov2640_camera.o
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/capture/mc521da.c linux-2.6.28-karo/drivers/media/video/mxc/capture/mc521da.c
+--- linux-2.6.28/drivers/media/video/mxc/capture/mc521da.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/capture/mc521da.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,702 @@
++/*
++ * Copyright 2006-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @file mc521da.c
++ *
++ * @brief MC521DA camera driver functions
++ *
++ * @ingroup Camera
++ */
++
++#include <linux/module.h>
++#include <linux/init.h>
++#include <linux/slab.h>
++#include <linux/ctype.h>
++#include <linux/types.h>
++#include <linux/delay.h>
++#include <linux/device.h>
++#include <linux/i2c.h>
++#include <linux/clk.h>
++#include <asm/arch/mxc_i2c.h>
++#include "mxc_v4l2_capture.h"
++
++#define MC521DA_I2C_ADDRESS 0x22
++#define MC521DA_TERM 0xFF
++
++typedef struct {
++ u16 width;
++ u16 height;
++} mc521da_image_format;
++
++struct mc521da_reg {
++ u8 reg;
++ u8 val;
++};
++
++static sensor_interface *interface_param = NULL;
++
++static mc521da_image_format format[2] = {
++ {
++ .width = 1600,
++ .height = 1200,
++ },
++ {
++ .width = 640,
++ .height = 480,
++ },
++};
++
++const static struct mc521da_reg mc521da_initial[] = {
++ /*----------------------------------------------------------
++ * Sensor Setting Start
++ *----------------------------------------------------------
++ */
++ {0xff, 0x01}, /* Sensor setting start */
++ {0x01, 0x10}, /* Wavetable script, generated by waveman */
++ {0x10, 0x64},
++ {0x03, 0x00}, {0x04, 0x06}, {0x05, 0x30}, {0x06, 0x02}, {0x08, 0x00},
++ {0x03, 0x01}, {0x04, 0x41}, {0x05, 0x70}, {0x06, 0x03}, {0x08, 0x00},
++ {0x03, 0x02}, {0x04, 0x55}, {0x05, 0x30}, {0x06, 0x03}, {0x08, 0x00},
++ {0x03, 0x03}, {0x04, 0x5A}, {0x05, 0x30}, {0x06, 0x02}, {0x08, 0x00},
++ {0x03, 0x04}, {0x04, 0x7A}, {0x05, 0x30}, {0x06, 0x06}, {0x08, 0x00},
++ {0x03, 0x05}, {0x04, 0x9C}, {0x05, 0x30}, {0x06, 0x0F}, {0x08, 0x00},
++ {0x03, 0x06}, {0x04, 0x73}, {0x05, 0x31}, {0x06, 0x06}, {0x08, 0x00},
++ {0x03, 0x07}, {0x04, 0x2D}, {0x05, 0x3B}, {0x06, 0x06}, {0x08, 0x00},
++ {0x03, 0x08}, {0x04, 0x32}, {0x05, 0x33}, {0x06, 0x06}, {0x08, 0x00},
++ {0x03, 0x09}, {0x04, 0x67}, {0x05, 0x63}, {0x06, 0x06}, {0x08, 0x00},
++ {0x03, 0x0a}, {0x04, 0x6C}, {0x05, 0x23}, {0x06, 0x0E}, {0x08, 0x00},
++ {0x03, 0x0b}, {0x04, 0x71}, {0x05, 0x23}, {0x06, 0x06}, {0x08, 0x00},
++ {0x03, 0x0c}, {0x04, 0x30}, {0x05, 0x2F}, {0x06, 0x06}, {0x08, 0x00},
++ {0x03, 0x0d}, {0x04, 0x00}, {0x05, 0x00}, {0x06, 0x06}, {0x08, 0x00},
++ {0x07, 0x0e},
++
++ /* Start Address */
++ {0x10, 0x64}, {0x14, 0x10}, {0x15, 0x00},
++
++ /* SYNC */
++ {0x18, 0x40}, {0x19, 0x00}, {0x1A, 0x03}, {0x1B, 0x00},
++
++ /* X-Y Mirror */
++ {0x11, 0x00}, {0xda, 0x00}, /* X mirror OFF, Y Mirror OFF */
++
++ /* Frame height */
++ {0x1c, 0x13}, {0x1d, 0x04}, {0x0e, 0x4b}, {0x0f, 0x05},
++ {0x9e, 0x04}, {0x9d, 0xc6}, {0xcc, 0x14}, {0xcd, 0x05},
++
++ /* Frame width */
++ {0x0c, 0x35}, {0x0d, 0x07}, {0x9b, 0x10}, {0x9c, 0x07},
++ {0x93, 0x21},
++
++ {0x01, 0x01}, {0x40, 0x00}, {0x41, 0x00}, {0x42, 0xf0},
++ {0x43, 0x03}, {0x44, 0x0a}, {0x45, 0x00}, {0x3b, 0x40},
++ {0x38, 0x18}, {0x3c, 0x00}, {0x20, 0x00}, {0x21, 0x01},
++ {0x22, 0x00}, {0x23, 0x01}, {0x24, 0x00}, {0x25, 0x01},
++ {0x26, 0x00}, {0x27, 0x01}, {0xb9, 0x04}, {0xb8, 0xc3},
++ {0xbb, 0x04}, {0xba, 0xc3}, {0xbf, 0x04}, {0xbe, 0xc3},
++
++ /* Ramp */
++ {0x57, 0x07}, {0x56, 0xd6}, {0x55, 0x03}, {0x54, 0x74},
++ {0x9f, 0x99}, {0x94, 0x80}, {0x91, 0x78}, {0x92, 0x8b},
++
++ /* Output Mode */
++ {0x52, 0x10}, {0x51, 0x00},
++
++ /* Analog Gain and Output driver */
++ {0x28, 0x00}, {0xdd, 0x82}, {0xdb, 0x00}, {0xdc, 0x00},
++
++ /* Update */
++ {0x00, 0x84},
++
++ /* PLL ADC clock = 75 MHz */
++ {0xb5, 0x60}, {0xb4, 0x02}, {0xb5, 0x20},
++
++ /*----------------------------------------------*/
++ /* ISP Setting Start */
++ /*----------------------------------------------*/
++ {0xff, 0x02},
++ {0x01, 0xbd}, {0x02, 0xf8}, {0x03, 0x3a}, {0x04, 0x00}, {0x0e, 0x00},
++
++ /* Output mode */
++ {0x88, 0x00}, {0x87, 0x11},
++
++ /* Threshold */
++ {0xb6, 0x1b}, {0x0d, 0xc0}, {0x24, 0x00}, {0x25, 0x00}, {0x26, 0x00},
++
++ /* Image Effect */
++ {0x3f, 0x80}, {0x40, 0x00}, {0x41, 0x00}, {0x42, 0x80}, {0x43, 0x00},
++ {0x44, 0x00}, {0x45, 0x00}, {0x46, 0x00}, {0x56, 0x80}, {0x57, 0x20},
++ {0x58, 0x20}, {0x59, 0x02}, {0x5a, 0x00}, {0x5b, 0x78}, {0x5c, 0x7c},
++ {0x5d, 0x84}, {0x5e, 0x85}, {0x5f, 0x78}, {0x60, 0x7e}, {0x61, 0x82},
++ {0x62, 0x85}, {0x63, 0x00}, {0x64, 0x80}, {0x65, 0x00}, {0x66, 0x80},
++ {0x67, 0x80}, {0x68, 0x80},
++
++ /* Auto Focus */
++ {0x6e, 0x02}, {0x6f, 0xe5}, {0x70, 0x08}, {0x71, 0x01}, {0x72, 0x00},
++
++ /* Decimator */
++ {0x78, 0xff}, {0x79, 0xff}, {0x7a, 0x70}, {0x7b, 0x00}, {0x7c, 0x00},
++ {0x7d, 0x00}, {0x7e, 0xc8}, {0x7f, 0xc8}, {0x80, 0x96}, {0x81, 0x96},
++ {0x82, 0x00}, {0x83, 0x00}, {0x84, 0x00}, {0x85, 0x00}, {0x86, 0x00},
++
++ /* Luminance Info */
++ {0xf9, 0x20}, {0xb7, 0x7f}, {0xb8, 0x28}, {0xb9, 0x08},
++ {0xf9, 0xa0}, {0xb7, 0x10}, {0xb9, 0x00},
++ {0xf9, 0x40}, {0xb7, 0x7f}, {0xb8, 0x28}, {0xb9, 0x08},
++ {0xf9, 0xc0}, {0xb7, 0x08}, {0xb9, 0x00},
++ {0xf9, 0x60}, {0xb7, 0x7f}, {0xb8, 0x28}, {0xb9, 0x08},
++ {0xf9, 0xe0}, {0xb7, 0x05}, {0xb9, 0x00},
++ {0xf9, 0x00}, {0xb7, 0x03}, {0xb8, 0x2d}, {0xb9, 0xcd},
++ {0xf9, 0x80}, {0xb7, 0x02}, {0xb9, 0x00},
++
++ /* AE */
++ {0x8a, 0x00}, {0x89, 0xc0}, {0x8c, 0x32}, {0x8d, 0x96}, {0x8e, 0x25},
++ {0x8f, 0x70}, {0x90, 0x12}, {0x91, 0x41}, {0x9e, 0x2e}, {0x9f, 0x2e},
++ {0xa0, 0x0b}, {0xa1, 0x71}, {0xa2, 0xb0}, {0xa3, 0x09}, {0xa4, 0x89},
++ {0xa5, 0x68}, {0xa6, 0x1a}, {0xa7, 0xb3}, {0xa8, 0xf0}, {0xa9, 0x19},
++ {0xaa, 0x6a}, {0xab, 0x6b}, {0xac, 0x01}, {0xad, 0xe8}, {0xae, 0x48},
++ {0xaf, 0x01}, {0xb0, 0x96}, {0xb1, 0xe6}, {0xb2, 0x03}, {0xb3, 0x00},
++ {0xb4, 0x10}, {0xb5, 0x00}, {0xb6, 0x04}, {0xba, 0x44}, {0xbb, 0x3a},
++ {0xbc, 0x01}, {0xbd, 0x08}, {0xbe, 0xa0}, {0xbf, 0x01}, {0xc0, 0x82},
++ {0x8a, 0xe1}, {0x8b, 0x8c},
++
++ /* AWB */
++ {0xc8, 0x00}, {0xc9, 0x00}, {0xca, 0x40}, {0xcb, 0xB0}, {0xcc, 0x40},
++ {0xcd, 0xff}, {0xce, 0x19}, {0xcf, 0x40}, {0xd0, 0x01}, {0xd1, 0x43},
++ {0xd2, 0x80}, {0xd3, 0x80}, {0xd4, 0xf1}, {0xdf, 0x00}, {0xe0, 0x8f},
++ {0xe1, 0x8f}, {0xe2, 0x53}, {0xe3, 0x97}, {0xe4, 0x1f}, {0xe5, 0x3b},
++ {0xe6, 0x9c}, {0xe7, 0x2e}, {0xe8, 0x03}, {0xe9, 0x02},
++
++ /* Neutral CCM */
++ {0xfa, 0x00}, {0xd5, 0x3f}, {0xd6, 0x8c}, {0xd7, 0x43}, {0xd8, 0x08},
++ {0xd9, 0x27}, {0xda, 0x7e}, {0xdb, 0x17}, {0xdc, 0x1a}, {0xdd, 0x47},
++ {0xde, 0xa1},
++
++ /* Blue CCM */
++ {0xfa, 0x01}, {0xd5, 0x3f}, {0xd6, 0x77}, {0xd7, 0x34}, {0xd8, 0x03},
++ {0xd9, 0x18}, {0xda, 0x6e}, {0xdb, 0x16}, {0xdc, 0x0f}, {0xdd, 0x29},
++ {0xde, 0x77},
++
++ /* Red CCM */
++ {0xfa, 0x02}, {0xd5, 0x3f}, {0xd6, 0x7d}, {0xd7, 0x2f}, {0xd8, 0x0e},
++ {0xd9, 0x1e}, {0xda, 0x76}, {0xdb, 0x18}, {0xdc, 0x29}, {0xdd, 0x51},
++ {0xde, 0xba},
++
++ /* AWB */
++ {0xea, 0x00}, {0xeb, 0x1a}, {0xc8, 0x33}, {0xc9, 0xc2},
++
++ {0xed, 0x02}, {0xee, 0x02},
++
++ /* AFD */
++ {0xf0, 0x11}, {0xf1, 0x03}, {0xf2, 0x05}, {0xf5, 0x05}, {0xf6, 0x32},
++ {0xf7, 0x32},
++
++ /* Lens Shading */
++ {0xf9, 0x00}, {0x05, 0x04}, {0x06, 0xff}, {0x07, 0xf2}, {0x08, 0x00},
++ {0x09, 0x00}, {0x0a, 0xf2}, {0x0b, 0xff}, {0x0c, 0xff},
++ {0xf9, 0x01}, {0x05, 0x04}, {0x06, 0xff}, {0x07, 0x8b}, {0x08, 0x16},
++ {0x09, 0x16}, {0x0a, 0x8b}, {0x0b, 0xff}, {0x0c, 0xe0},
++ {0xf9, 0x02}, {0x05, 0x04}, {0x06, 0xff}, {0x07, 0x8b}, {0x08, 0x16},
++ {0x09, 0x16}, {0x0a, 0x8b}, {0x0b, 0xff}, {0x0c, 0xe0},
++ {0xf9, 0x03}, {0x05, 0x04}, {0x06, 0xff}, {0x07, 0x7c}, {0x08, 0x26},
++ {0x09, 0x26}, {0x0a, 0x7c}, {0x0b, 0xd0}, {0x0c, 0xe0},
++ {0xf9, 0x04}, {0x05, 0x0d}, {0x06, 0x40}, {0x07, 0xa0}, {0x08, 0x00},
++ {0x09, 0x00}, {0x0a, 0xa0}, {0x0b, 0x40}, {0x0c, 0xe0},
++ {0xf9, 0x05}, {0x05, 0x0d}, {0x06, 0x40}, {0x07, 0xa0}, {0x08, 0x00},
++ {0x09, 0x00}, {0x0a, 0xa0}, {0x0b, 0x40}, {0x0c, 0xa0},
++ {0xf9, 0x06}, {0x05, 0x0d}, {0x06, 0x40}, {0x07, 0xa0}, {0x08, 0x00},
++ {0x09, 0x00}, {0x0a, 0xa0}, {0x0b, 0x40}, {0x0c, 0xa0},
++ {0xf9, 0x07}, {0x05, 0x0d}, {0x06, 0x40}, {0x07, 0xa0}, {0x08, 0x00},
++ {0x09, 0x00}, {0x0a, 0xa0}, {0x0b, 0x40}, {0x0c, 0xa0},
++
++ /* Edge setting */
++ {0x73, 0x68}, {0x74, 0x40}, {0x75, 0x00}, {0x76, 0xff}, {0x77, 0x80},
++ {0x4f, 0x80}, {0x50, 0x82}, {0x51, 0x82}, {0x52, 0x08},
++
++ /* Interpolation Setting */
++ {0x23, 0x7f}, {0x22, 0x08}, {0x18, 0xff}, {0x19, 0x00},
++ {0x40, 0x00}, {0x53, 0xff}, {0x54, 0x0a}, {0x55, 0xc2},
++ {0x1b, 0x18},
++
++ {0xfa, 0x00}, {0x15, 0x0c}, {0x22, 0x00}, {0x0e, 0xef}, {0x1f, 0x1d},
++ {0x20, 0x2d}, {0x1c, 0x01}, {0x1d, 0x02}, {0x1e, 0x03}, {0x0e, 0xee},
++ {0x12, 0x10}, {0x16, 0x10}, {0x17, 0x02}, {0x1a, 0x01},
++ {0xfa, 0x04}, {0x0e, 0xef}, {0x1c, 0x01}, {0x1d, 0x02}, {0x1e, 0x03},
++ {0x1f, 0x11}, {0x20, 0x11}, {0x0e, 0xee}, {0x12, 0x03}, {0x16, 0x10},
++ {0x17, 0x02}, {0x1a, 0xee},
++ {0xfa, 0x08}, {0x0e, 0xef}, {0x1c, 0x01}, {0x1d, 0x02}, {0x1e, 0x03},
++ {0x1f, 0x00}, {0x20, 0x00}, {0x0e, 0xee}, {0x12, 0x03}, {0x16, 0x10},
++ {0x17, 0x02}, {0x1a, 0x22},
++
++ /* Gamma Correction */
++ {0x27, 0x62}, {0x28, 0x00}, {0x27, 0x62}, {0x28, 0x00}, {0x29, 0x00},
++ {0x2a, 0x00}, {0x2f, 0x03}, {0x30, 0x10}, {0x31, 0x2b}, {0x32, 0x50},
++ {0x33, 0x70}, {0x34, 0x90}, {0x35, 0xB0}, {0x36, 0xD0}, {0x37, 0x00},
++ {0x38, 0x18}, {0x39, 0x57}, {0x3a, 0x89}, {0x3b, 0xac}, {0x3c, 0xc9},
++ {0x3d, 0xde}, {0x3e, 0xef}, {0x2b, 0x00}, {0x2c, 0x00}, {0x2d, 0x40},
++ {0x2e, 0xab},
++
++ /* Contrast */
++ {0x47, 0x10}, {0x48, 0x1f}, {0x49, 0xe3}, {0x4a, 0xf0}, {0x4b, 0x08},
++ {0x4c, 0x14}, {0x4d, 0xe9}, {0x4e, 0xf5}, {0x98, 0x8a},
++
++ {0xfa, 0x00},
++ {MC521DA_TERM, MC521DA_TERM}
++};
++
++static int mc521da_attach(struct i2c_adapter *adapter);
++static int mc521da_detach(struct i2c_client *client);
++
++static struct i2c_driver mc521da_i2c_driver = {
++ .driver = {
++ .owner = THIS_MODULE,
++ .name = "MC521DA Client",
++ },
++ .attach_adapter = mc521da_attach,
++ .detach_client = mc521da_detach,
++};
++
++static struct i2c_client mc521da_i2c_client = {
++ .name = "MC521DA I2C dev",
++ .addr = MC521DA_I2C_ADDRESS,
++ .driver = &mc521da_i2c_driver,
++};
++
++/*
++ * Function definitions
++ */
++static int mc521da_i2c_client_xfer(unsigned int addr, char *reg,
++ int reg_len, char *buf, int num,
++ int tran_flag)
++{
++ struct i2c_msg msg[2];
++ int ret;
++
++ msg[0].addr = addr;
++ msg[0].len = reg_len;
++ msg[0].buf = reg;
++ msg[0].flags = tran_flag;
++ msg[0].flags &= ~I2C_M_RD;
++
++ msg[1].addr = addr;
++ msg[1].len = num;
++ msg[1].buf = buf;
++ msg[1].flags = tran_flag;
++
++ if (tran_flag & MXC_I2C_FLAG_READ) {
++ msg[1].flags |= I2C_M_RD;
++ } else {
++ msg[1].flags &= ~I2C_M_RD;
++ }
++
++ ret = i2c_transfer(mc521da_i2c_client.adapter, msg, 2);
++ if (ret >= 0)
++ return 0;
++
++ return ret;
++}
++
++static int mc521da_read_reg(u8 * reg, u8 * val)
++{
++ return mc521da_i2c_client_xfer(MC521DA_I2C_ADDRESS, reg, 1, val, 1,
++ MXC_I2C_FLAG_READ);
++}
++
++static int mc521da_write_reg(u8 reg, u8 val)
++{
++ u8 temp1, temp2;
++ temp1 = reg;
++ temp2 = val;
++ return mc521da_i2c_client_xfer(MC521DA_I2C_ADDRESS, &temp1, 1, &temp2,
++ 1, 0);
++}
++
++static int mc521da_write_regs(const struct mc521da_reg reglist[])
++{
++ int err;
++ const struct mc521da_reg *next = reglist;
++
++ while (!((next->reg == MC521DA_TERM) && (next->val == MC521DA_TERM))) {
++ err = mc521da_write_reg(next->reg, next->val);
++ if (err) {
++ return err;
++ }
++ next++;
++ }
++ return 0;
++}
++
++/*!
++ * mc521da sensor downscale function
++ * @param downscale bool
++ * @return Error code indicating success or failure
++ */
++static u8 mc521da_sensor_downscale(bool downscale)
++{
++ u8 reg[1], data;
++ u32 i = 0;
++
++ if (downscale == true) {
++ // VGA
++ mc521da_write_reg(0xff, 0x01);
++
++ mc521da_write_reg(0x52, 0x30);
++ mc521da_write_reg(0x51, 0x00);
++
++ mc521da_write_reg(0xda, 0x01);
++ mc521da_write_reg(0x00, 0x8C);
++
++ /* Wait for changes to take effect */
++ reg[0] = 0x00;
++ while (i < 256) {
++ i++;
++ mc521da_read_reg(reg, &data);
++ if ((data & 0x80) == 0)
++ break;
++ msleep(5);
++ }
++
++ /* ISP */
++ mc521da_write_reg(0xff, 0x02);
++
++ mc521da_write_reg(0x03, 0x3b); /* Enable Decimator */
++
++ mc521da_write_reg(0x7a, 0x74);
++ mc521da_write_reg(0x7b, 0x01);
++ mc521da_write_reg(0x7e, 0x50);
++ mc521da_write_reg(0x7f, 0x50);
++ mc521da_write_reg(0x80, 0x3c);
++ mc521da_write_reg(0x81, 0x3c);
++ } else {
++ //UXGA
++ mc521da_write_reg(0xff, 0x01);
++ mc521da_write_reg(0x52, 0x10);
++ mc521da_write_reg(0x51, 0x00);
++ mc521da_write_reg(0xda, 0x00);
++
++ /* update */
++ mc521da_write_reg(0x00, 0x84);
++
++ /* Wait for changes to take effect */
++ reg[0] = 0x00;
++ while (i < 256) {
++ i++;
++ mc521da_read_reg(reg, &data);
++ if ((data & 0x80) == 0)
++ break;
++ msleep(5);
++ }
++
++ /* ISP */
++ mc521da_write_reg(0xff, 0x02);
++
++ mc521da_write_reg(0x03, 0x3a);
++
++ mc521da_write_reg(0x7a, 0x70);
++ mc521da_write_reg(0x7b, 0x00);
++ mc521da_write_reg(0x7e, 0xc8);
++ mc521da_write_reg(0x7f, 0xc8);
++ mc521da_write_reg(0x80, 0x96);
++ mc521da_write_reg(0x81, 0x96);
++ }
++
++ return 0;
++}
++
++/*!
++ * mc521da sensor interface Initialization
++ * @param param sensor_interface *
++ * @param width u32
++ * @param height u32
++ * @return None
++ */
++static void mc521da_interface(sensor_interface * param, u32 width, u32 height)
++{
++ param->clk_mode = 0x0; //gated
++ param->pixclk_pol = 0x0;
++ param->data_width = 0x1;
++ param->data_pol = 0x0;
++ param->ext_vsync = 0x0;
++ param->Vsync_pol = 0x0;
++ param->Hsync_pol = 0x0;
++ param->width = width - 1;
++ param->height = height - 1;
++ param->pixel_fmt = IPU_PIX_FMT_UYVY;
++}
++
++extern void gpio_sensor_reset(bool flag);
++
++/*!
++ * mc521da Reset function
++ *
++ * @return None
++ */
++static sensor_interface *mc521da_reset(void)
++{
++ if (interface_param == NULL)
++ return NULL;
++
++ mc521da_interface(interface_param, format[1].width, format[1].height);
++ set_mclk_rate(&interface_param->mclk);
++
++ gpio_sensor_reset(true);
++ msleep(10);
++ gpio_sensor_reset(false);
++ msleep(50);
++
++ return interface_param;
++}
++
++/*!
++ * mc521da sensor configuration
++ *
++ * @param frame_rate int *
++ * @param high_quality int
++ * @return sensor_interface *
++ */
++static sensor_interface *mc521da_config(int *frame_rate, int high_quality)
++{
++ int num_clock_per_row, err;
++ int max_rate = 0;
++ int index = 1;
++ u16 frame_height;
++
++ if (high_quality == 1)
++ index = 0;
++
++ err = mc521da_write_regs(mc521da_initial);
++ if (err) {
++ /* Reduce the MCLK */
++ interface_param->mclk = 20000000;
++ mc521da_reset();
++
++ printk(KERN_INFO "mc521da: mclk reduced\n");
++ mc521da_write_regs(mc521da_initial);
++ }
++
++ mc521da_interface(interface_param, format[index].width,
++ format[index].height);
++
++ if (index == 0) {
++ mc521da_sensor_downscale(false);
++ } else {
++ mc521da_sensor_downscale(true);
++ }
++
++ num_clock_per_row = 1845;
++ max_rate = interface_param->mclk * 3 * (index + 1)
++ / (2 * num_clock_per_row * 1300);
++
++ if ((*frame_rate > max_rate) || (*frame_rate == 0)) {
++ *frame_rate = max_rate;
++ }
++
++ frame_height = 1300 * max_rate / (*frame_rate);
++
++ *frame_rate = interface_param->mclk * 3 * (index + 1)
++ / (2 * num_clock_per_row * frame_height);
++
++ mc521da_write_reg(0xff, 0x01);
++ mc521da_write_reg(0xE, frame_height & 0xFF);
++ mc521da_write_reg(0xF, (frame_height & 0xFF00) >> 8);
++ mc521da_write_reg(0xCC, frame_height & 0xFF);
++ mc521da_write_reg(0xCD, (frame_height & 0xFF00) >> 8);
++
++ return interface_param;
++}
++
++/*!
++ * mc521da sensor set color configuration
++ *
++ * @param bright int
++ * @param saturation int
++ * @param red int
++ * @param green int
++ * @param blue int
++ * @return None
++ */
++static void
++mc521da_set_color(int bright, int saturation, int red, int green, int blue)
++{
++ /* Select ISP */
++ mc521da_write_reg(0xff, 0x02);
++
++ mc521da_write_reg(0x41, bright);
++ mc521da_write_reg(0xca, red);
++ mc521da_write_reg(0xcb, green);
++ mc521da_write_reg(0xcc, blue);
++}
++
++/*!
++ * mc521da sensor get color configuration
++ *
++ * @param bright int *
++ * @param saturation int *
++ * @param red int *
++ * @param green int *
++ * @param blue int *
++ * @return None
++ */
++static void
++mc521da_get_color(int *bright, int *saturation, int *red, int *green, int *blue)
++{
++ u8 reg[1];
++ u8 *pdata;
++
++ *saturation = 0;
++
++ /* Select ISP */
++ mc521da_write_reg(0xff, 0x02);
++
++ reg[0] = 0x41;
++ pdata = (u8 *) bright;
++ mc521da_read_reg(reg, pdata);
++
++ reg[0] = 0xCA;
++ pdata = (u8 *) red;
++ mc521da_read_reg(reg, pdata);
++
++ reg[0] = 0xCB;
++ pdata = (u8 *) green;
++ mc521da_read_reg(reg, pdata);
++
++ reg[0] = 0xCC;
++ pdata = (u8 *) blue;
++ mc521da_read_reg(reg, pdata);
++}
++
++struct camera_sensor camera_sensor_if = {
++ set_color:mc521da_set_color,
++ get_color:mc521da_get_color,
++ config:mc521da_config,
++ reset:mc521da_reset,
++};
++
++/*!
++ * mc521da I2C detect_client function
++ *
++ * @param adapter struct i2c_adapter *
++ * @param address int
++ * @param kind int
++ *
++ * @return Error code indicating success or failure
++ */
++static int mc521da_detect_client(struct i2c_adapter *adapter, int address,
++ int kind)
++{
++ mc521da_i2c_client.adapter = adapter;
++ if (i2c_attach_client(&mc521da_i2c_client)) {
++ mc521da_i2c_client.adapter = NULL;
++ printk(KERN_ERR "mc521da_attach: i2c_attach_client failed\n");
++ return -1;
++ }
++
++ interface_param = (sensor_interface *)
++ kmalloc(sizeof(sensor_interface), GFP_KERNEL);
++ if (!interface_param) {
++ printk(KERN_ERR "mc521da_attach: kmalloc failed \n");
++ return -1;
++ }
++
++ interface_param->mclk = 25000000;
++
++ printk(KERN_INFO "mc521da Detected\n");
++
++ return 0;
++}
++
++static unsigned short normal_i2c[] = { MC521DA_I2C_ADDRESS, I2C_CLIENT_END };
++
++/* Magic definition of all other variables and things */
++I2C_CLIENT_INSMOD;
++
++static int mc521da_attach(struct i2c_adapter *adap)
++{
++ uint32_t mclk = 25000000;
++ struct clk *clk;
++ int err;
++
++ clk = clk_get(NULL, "csi_clk");
++ clk_enable(clk);
++ set_mclk_rate(&mclk);
++
++ gpio_sensor_reset(true);
++ msleep(10);
++ gpio_sensor_reset(false);
++ msleep(100);
++
++ err = i2c_probe(adap, &addr_data, &mc521da_detect_client);
++
++ clk_disable(clk);
++ clk_put(clk);
++
++ return err;
++}
++
++/*!
++ * mc521da I2C detach function
++ *
++ * @param client struct i2c_client *
++ * @return Error code indicating success or failure
++ */
++static int mc521da_detach(struct i2c_client *client)
++{
++ int err;
++
++ if (!mc521da_i2c_client.adapter)
++ return -1;
++
++ err = i2c_detach_client(&mc521da_i2c_client);
++ mc521da_i2c_client.adapter = NULL;
++
++ if (interface_param)
++ kfree(interface_param);
++ interface_param = NULL;
++
++ return err;
++}
++
++extern void gpio_sensor_active(void);
++extern void gpio_sensor_inactive(void);
++
++/*!
++ * mc521da init function
++ *
++ * @return Error code indicating success or failure
++ */
++static __init int mc521da_init(void)
++{
++ gpio_sensor_active();
++ return i2c_add_driver(&mc521da_i2c_driver);
++}
++
++/*!
++ * mc521da cleanup function
++ *
++ * @return Error code indicating success or failure
++ */
++static void __exit mc521da_clean(void)
++{
++ i2c_del_driver(&mc521da_i2c_driver);
++ gpio_sensor_inactive();
++}
++
++module_init(mc521da_init);
++module_exit(mc521da_clean);
++
++/* Exported symbols for modules. */
++EXPORT_SYMBOL(camera_sensor_if);
++
++MODULE_AUTHOR("Freescale Semiconductor, Inc.");
++MODULE_DESCRIPTION("MC521DA Camera Driver");
++MODULE_LICENSE("GPL");
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/capture/mt9v111.c linux-2.6.28-karo/drivers/media/video/mxc/capture/mt9v111.c
+--- linux-2.6.28/drivers/media/video/mxc/capture/mt9v111.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/capture/mt9v111.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,932 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @file mt9v111.c
++ *
++ * @brief mt9v111 camera driver functions
++ *
++ * @ingroup Camera
++ */
++
++#include <linux/module.h>
++#include <linux/init.h>
++#include <linux/slab.h>
++#include <linux/ctype.h>
++#include <linux/types.h>
++#include <linux/delay.h>
++#include <linux/device.h>
++#include <linux/i2c.h>
++#include <linux/clk.h>
++#include <asm/arch/mxc_i2c.h>
++#include "mxc_v4l2_capture.h"
++#include "mt9v111.h"
++
++#ifdef MT9V111_DEBUG
++static u16 testpattern = 0;
++#endif
++
++static sensor_interface *interface_param = NULL;
++static mt9v111_conf mt9v111_device;
++static int reset_frame_rate = 30;
++
++#define MT9V111_FRAME_RATE_NUM 20
++
++static mt9v111_image_format format[2] = {
++ {
++ .index = 0,
++ .width = 640,
++ .height = 480,
++ },
++ {
++ .index = 1,
++ .width = 352,
++ .height = 288,
++ },
++};
++
++static int mt9v111_attach(struct i2c_adapter *adapter);
++static int mt9v111_detach(struct i2c_client *client);
++
++static struct i2c_driver mt9v111_i2c_driver = {
++ .driver = {
++ .owner = THIS_MODULE,
++ .name = "MT9V111 Client",
++ },
++ .attach_adapter = mt9v111_attach,
++ .detach_client = mt9v111_detach,
++};
++
++static struct i2c_client mt9v111_i2c_client = {
++ .name = "mt9v111 I2C dev",
++ .addr = MT9V111_I2C_ADDRESS,
++ .driver = &mt9v111_i2c_driver,
++};
++
++/*
++ * Function definitions
++ */
++
++static u16 mt9v111_endian_swap16(u16 data)
++{
++ u16 temp;
++
++ temp = data;
++ temp = ((data >> 8) & 0xff) | ((data << 8) & 0xff00);
++
++ return temp;
++}
++
++static int mt9v111_i2c_client_xfer(unsigned int addr, char *reg, int reg_len,
++ char *buf, int num, int tran_flag)
++{
++ struct i2c_msg msg[2];
++ int ret;
++
++ msg[0].addr = addr;
++ msg[0].len = reg_len;
++ msg[0].buf = reg;
++ msg[0].flags = tran_flag;
++ msg[0].flags &= ~I2C_M_RD;
++
++ msg[1].addr = addr;
++ msg[1].len = num;
++ msg[1].buf = buf;
++ msg[1].flags = tran_flag;
++
++ if (tran_flag & MXC_I2C_FLAG_READ) {
++ msg[1].flags |= I2C_M_RD;
++ } else {
++ msg[1].flags &= ~I2C_M_RD;
++ }
++
++ ret = i2c_transfer(mt9v111_i2c_client.adapter, msg, 2);
++ if (ret >= 0)
++ return 0;
++
++ return ret;
++}
++
++static int mt9v111_read_reg(u8 * reg, u16 * val)
++{
++ return mt9v111_i2c_client_xfer(MT9V111_I2C_ADDRESS, reg, 1,
++ (u8 *) val, 2, MXC_I2C_FLAG_READ);
++}
++
++static int mt9v111_write_reg(u8 reg, u16 val)
++{
++ u8 temp1;
++ u16 temp2;
++ temp1 = reg;
++ temp2 = mt9v111_endian_swap16(val);
++ pr_debug("write reg %x val %x.\n", reg, val);
++ return mt9v111_i2c_client_xfer(MT9V111_I2C_ADDRESS, &temp1, 1,
++ (u8 *) & temp2, 2, 0);
++}
++
++/*!
++ * Initialize mt9v111_sensor_lib
++ * Libarary for Sensor configuration through I2C
++ *
++ * @param coreReg Core Registers
++ * @param ifpReg IFP Register
++ *
++ * @return status
++ */
++static u8 mt9v111_sensor_lib(mt9v111_coreReg * coreReg, mt9v111_IFPReg * ifpReg)
++{
++ u8 reg;
++ u16 data;
++ u8 error = 0;
++
++ /*
++ * setup to IFP registers
++ */
++ reg = MT9V111I_ADDR_SPACE_SEL;
++ data = ifpReg->addrSpaceSel;
++ mt9v111_write_reg(reg, data);
++
++ // Operation Mode Control
++ reg = MT9V111I_MODE_CONTROL;
++ data = ifpReg->modeControl;
++ mt9v111_write_reg(reg, data);
++
++ // Output format
++ reg = MT9V111I_FORMAT_CONTROL;
++ data = ifpReg->formatControl; // Set bit 12
++ mt9v111_write_reg(reg, data);
++
++ // Flicker Control
++ reg = MT9V111I_FLICKER_CONTROL;
++ data = ifpReg->flickerCtrl;
++ mt9v111_write_reg(reg, data);
++
++ // AE limit 4
++ reg = MT9V111I_SHUTTER_WIDTH_LIMIT_AE;
++ data = ifpReg->gainLimitAE;
++ mt9v111_write_reg(reg, data);
++
++ reg = MT9V111I_OUTPUT_FORMAT_CTRL2;
++ data = ifpReg->outputFormatCtrl2;
++ mt9v111_write_reg(reg, data);
++
++ reg = MT9V111I_AE_SPEED;
++ data = ifpReg->AESpeed;
++ mt9v111_write_reg(reg, data);
++
++ /* output image size */
++ reg = MT9V111i_H_PAN;
++ data = 0x8000 | ifpReg->HPan;
++ mt9v111_write_reg(reg, data);
++
++ reg = MT9V111i_H_ZOOM;
++ data = 0x8000 | ifpReg->HZoom;
++ mt9v111_write_reg(reg, data);
++
++ reg = MT9V111i_H_SIZE;
++ data = 0x8000 | ifpReg->HSize;
++ mt9v111_write_reg(reg, data);
++
++ reg = MT9V111i_V_PAN;
++ data = 0x8000 | ifpReg->VPan;
++ mt9v111_write_reg(reg, data);
++
++ reg = MT9V111i_V_ZOOM;
++ data = 0x8000 | ifpReg->VZoom;
++ mt9v111_write_reg(reg, data);
++
++ reg = MT9V111i_V_SIZE;
++ data = 0x8000 | ifpReg->VSize;
++ mt9v111_write_reg(reg, data);
++
++ reg = MT9V111i_H_PAN;
++ data = ~0x8000 & ifpReg->HPan;
++ mt9v111_write_reg(reg, data);
++#if 0
++ reg = MT9V111I_UPPER_SHUTTER_DELAY_LIM;
++ data = ifpReg->upperShutterDelayLi;
++ mt9v111_write_reg(reg, data);
++
++ reg = MT9V111I_SHUTTER_60;
++ data = ifpReg->shutter_width_60;
++ mt9v111_write_reg(reg, data);
++
++ reg = MT9V111I_SEARCH_FLICK_60;
++ data = ifpReg->search_flicker_60;
++ mt9v111_write_reg(reg, data);
++#endif
++
++ /*
++ * setup to sensor core registers
++ */
++ reg = MT9V111I_ADDR_SPACE_SEL;
++ data = coreReg->addressSelect;
++ mt9v111_write_reg(reg, data);
++
++ // enable changes and put the Sync bit on
++ reg = MT9V111S_OUTPUT_CTRL;
++ data = MT9V111S_OUTCTRL_SYNC | MT9V111S_OUTCTRL_CHIP_ENABLE | 0x3000;
++ mt9v111_write_reg(reg, data);
++
++ // min PIXCLK - Default
++ reg = MT9V111S_PIXEL_CLOCK_SPEED;
++ data = coreReg->pixelClockSpeed;
++ mt9v111_write_reg(reg, data);
++
++ //Setup image flipping / Dark rows / row/column skip
++ reg = MT9V111S_READ_MODE;
++ data = coreReg->readMode;
++ mt9v111_write_reg(reg, data);
++
++ //zoom 0
++ reg = MT9V111S_DIGITAL_ZOOM;
++ data = coreReg->digitalZoom;
++ mt9v111_write_reg(reg, data);
++
++ // min H-blank
++ reg = MT9V111S_HOR_BLANKING;
++ data = coreReg->horizontalBlanking;
++ mt9v111_write_reg(reg, data);
++
++ // min V-blank
++ reg = MT9V111S_VER_BLANKING;
++ data = coreReg->verticalBlanking;
++ mt9v111_write_reg(reg, data);
++
++ reg = MT9V111S_SHUTTER_WIDTH;
++ data = coreReg->shutterWidth;
++ mt9v111_write_reg(reg, data);
++
++ reg = MT9V111S_SHUTTER_DELAY;
++ data = ifpReg->upperShutterDelayLi;
++ mt9v111_write_reg(reg, data);
++
++ // changes become effective
++ reg = MT9V111S_OUTPUT_CTRL;
++ data = MT9V111S_OUTCTRL_CHIP_ENABLE | 0x3000;
++ mt9v111_write_reg(reg, data);
++
++ return error;
++}
++
++/*!
++ * mt9v111 sensor interface Initialization
++ * @param param sensor_interface *
++ * @param width u32
++ * @param height u32
++ * @return None
++ */
++static void mt9v111_interface(sensor_interface * param, u32 width, u32 height)
++{
++ param->Vsync_pol = 0x0;
++ param->clk_mode = 0x0; //gated
++ param->pixclk_pol = 0x0;
++ param->data_width = 0x1;
++ param->data_pol = 0x0;
++ param->ext_vsync = 0x0;
++ param->Vsync_pol = 0x0;
++ param->Hsync_pol = 0x0;
++ param->width = width - 1;
++ param->height = height - 1;
++ param->pixel_fmt = IPU_PIX_FMT_UYVY;
++ param->mclk = 27000000;
++}
++
++/*!
++ * MT9V111 frame rate calculate
++ *
++ * @param frame_rate int *
++ * @param mclk int
++ * @return None
++ */
++static void mt9v111_rate_cal(int *frame_rate, int mclk)
++{
++ int num_clock_per_row;
++ int max_rate = 0;
++
++ mt9v111_device.coreReg->horizontalBlanking = MT9V111_HORZBLANK_MIN;
++
++ num_clock_per_row = (format[0].width + 114 + MT9V111_HORZBLANK_MIN) * 2;
++ max_rate = mclk / (num_clock_per_row *
++ (format[0].height + MT9V111_VERTBLANK_DEFAULT));
++
++ if ((*frame_rate > max_rate) || (*frame_rate == 0)) {
++ *frame_rate = max_rate;
++ }
++
++ mt9v111_device.coreReg->verticalBlanking
++ = mclk / (*frame_rate * num_clock_per_row) - format[0].height;
++
++ reset_frame_rate = *frame_rate;
++}
++
++/*!
++ * MT9V111 sensor configuration
++ *
++ * @param frame_rate int *
++ * @param high_quality int
++ * @return sensor_interface *
++ */
++sensor_interface *mt9v111_config(int *frame_rate, int high_quality)
++{
++ u32 out_width, out_height;
++
++ if (interface_param == NULL)
++ return NULL;
++
++ mt9v111_device.coreReg->addressSelect = MT9V111I_SEL_SCA;
++ mt9v111_device.ifpReg->addrSpaceSel = MT9V111I_SEL_IFP;
++
++ mt9v111_device.coreReg->windowHeight = MT9V111_WINHEIGHT;
++ mt9v111_device.coreReg->windowWidth = MT9V111_WINWIDTH;
++ mt9v111_device.coreReg->zoomColStart = 0;
++ mt9v111_device.coreReg->zomRowStart = 0;
++ mt9v111_device.coreReg->digitalZoom = 0x0;
++
++ mt9v111_device.coreReg->verticalBlanking = MT9V111_VERTBLANK_DEFAULT;
++ mt9v111_device.coreReg->horizontalBlanking = MT9V111_HORZBLANK_MIN;
++ mt9v111_device.coreReg->pixelClockSpeed = 0;
++ mt9v111_device.coreReg->readMode = 0xd0a1;
++
++ mt9v111_device.ifpReg->outputFormatCtrl2 = 0;
++ mt9v111_device.ifpReg->gainLimitAE = 0x300;
++ mt9v111_device.ifpReg->AESpeed = 0x80;
++
++ // here is the default value
++ mt9v111_device.ifpReg->formatControl = 0xc800;
++ mt9v111_device.ifpReg->modeControl = 0x708e;
++ mt9v111_device.ifpReg->awbSpeed = 0x4514;
++ mt9v111_device.ifpReg->flickerCtrl = 0x02;
++ mt9v111_device.coreReg->shutterWidth = 0xf8;
++
++ out_width = 640;
++ out_height = 480;
++
++ /*output size */
++ mt9v111_device.ifpReg->HPan = 0;
++ mt9v111_device.ifpReg->HZoom = 640;
++ mt9v111_device.ifpReg->HSize = out_width;
++ mt9v111_device.ifpReg->VPan = 0;
++ mt9v111_device.ifpReg->VZoom = 480;
++ mt9v111_device.ifpReg->VSize = out_height;
++
++ mt9v111_interface(interface_param, out_width, out_height);
++ set_mclk_rate(&interface_param->mclk);
++ mt9v111_rate_cal(frame_rate, interface_param->mclk);
++ mt9v111_sensor_lib(mt9v111_device.coreReg, mt9v111_device.ifpReg);
++
++ return interface_param;
++}
++
++/*!
++ * mt9v111 sensor set color configuration
++ *
++ * @param bright int
++ * @param saturation int
++ * @param red int
++ * @param green int
++ * @param blue int
++ * @return None
++ */
++static void
++mt9v111_set_color(int bright, int saturation, int red, int green, int blue)
++{
++ u8 reg;
++ u16 data;
++
++ switch (saturation) {
++ case 100:
++ mt9v111_device.ifpReg->awbSpeed = 0x4514;
++ break;
++ case 150:
++ mt9v111_device.ifpReg->awbSpeed = 0x6D14;
++ break;
++ case 75:
++ mt9v111_device.ifpReg->awbSpeed = 0x4D14;
++ break;
++ case 50:
++ mt9v111_device.ifpReg->awbSpeed = 0x5514;
++ break;
++ case 37:
++ mt9v111_device.ifpReg->awbSpeed = 0x5D14;
++ break;
++ case 25:
++ mt9v111_device.ifpReg->awbSpeed = 0x6514;
++ break;
++ default:
++ mt9v111_device.ifpReg->awbSpeed = 0x4514;
++ break;
++ }
++
++ reg = MT9V111I_ADDR_SPACE_SEL;
++ data = mt9v111_device.ifpReg->addrSpaceSel;
++ mt9v111_write_reg(reg, data);
++
++ // Operation Mode Control
++ reg = MT9V111I_AWB_SPEED;
++ data = mt9v111_device.ifpReg->awbSpeed;
++ mt9v111_write_reg(reg, data);
++}
++
++/*!
++ * mt9v111 sensor get color configuration
++ *
++ * @param bright int *
++ * @param saturation int *
++ * @param red int *
++ * @param green int *
++ * @param blue int *
++ * @return None
++ */
++static void
++mt9v111_get_color(int *bright, int *saturation, int *red, int *green, int *blue)
++{
++ *saturation = (mt9v111_device.ifpReg->awbSpeed & 0x3800) >> 11;
++ switch (*saturation) {
++ case 0:
++ *saturation = 100;
++ break;
++ case 1:
++ *saturation = 75;
++ break;
++ case 2:
++ *saturation = 50;
++ break;
++ case 3:
++ *saturation = 37;
++ break;
++ case 4:
++ *saturation = 25;
++ break;
++ case 5:
++ *saturation = 150;
++ break;
++ case 6:
++ *saturation = 0;
++ break;
++ default:
++ *saturation = 0;
++ break;
++ }
++}
++
++/*!
++ * mt9v111 sensor set AE measurement window mode configuration
++ *
++ * @param ae_mode int
++ * @return None
++ */
++static void mt9v111_set_ae_mode(int ae_mode)
++{
++ u8 reg;
++ u16 data;
++
++ mt9v111_device.ifpReg->modeControl &= 0xfff3;
++ mt9v111_device.ifpReg->modeControl |= (ae_mode & 0x03) << 2;
++
++ reg = MT9V111I_ADDR_SPACE_SEL;
++ data = mt9v111_device.ifpReg->addrSpaceSel;
++ mt9v111_write_reg(reg, data);
++
++ reg = MT9V111I_MODE_CONTROL;
++ data = mt9v111_device.ifpReg->modeControl;
++ mt9v111_write_reg(reg, data);
++}
++
++/*!
++ * mt9v111 sensor get AE measurement window mode configuration
++ *
++ * @param ae_mode int *
++ * @return None
++ */
++static void mt9v111_get_ae_mode(int *ae_mode)
++{
++ if (ae_mode != NULL) {
++ *ae_mode = (mt9v111_device.ifpReg->modeControl & 0xc) >> 2;
++ }
++}
++
++/*!
++ * mt9v111 sensor enable/disable AE
++ *
++ * @param active int
++ * @return None
++ */
++static void mt9v111_set_ae(int active)
++{
++ u8 reg;
++ u16 data;
++
++ mt9v111_device.ifpReg->modeControl &= 0xBFFF;
++ mt9v111_device.ifpReg->modeControl |= (active & 0x01) << 14;
++
++ reg = MT9V111I_ADDR_SPACE_SEL;
++ data = mt9v111_device.ifpReg->addrSpaceSel;
++ mt9v111_write_reg(reg, data);
++
++ reg = MT9V111I_MODE_CONTROL;
++ data = mt9v111_device.ifpReg->modeControl;
++ mt9v111_write_reg(reg, data);
++}
++
++ /*!
++ * mt9v111 sensor set AE gain
++ *
++ * @param active int
++ * @return None
++ */
++static void mt9v111_set_ae_limit(int limit)
++{
++ u8 reg;
++ u16 data;
++
++ mt9v111_device.ifpReg->gainLimitAE = (limit & 0x1F) << 5;
++
++ reg = MT9V111I_ADDR_SPACE_SEL;
++ data = mt9v111_device.ifpReg->addrSpaceSel;
++ mt9v111_write_reg(reg, data);
++
++ reg = MT9V111I_SHUTTER_WIDTH_LIMIT_AE;
++ data = mt9v111_device.ifpReg->gainLimitAE;
++ mt9v111_write_reg(reg, data);
++}
++
++
++/*!
++ * mt9v111 sensor enable/disable auto white balance
++ *
++ * @param active int
++ * @return None
++ */
++static void mt9v111_set_awb(int active)
++{
++ u8 reg;
++ u16 data;
++
++ mt9v111_device.ifpReg->modeControl &= 0xFFFD;
++ mt9v111_device.ifpReg->modeControl |= (active & 0x01) << 1;
++
++ reg = MT9V111I_ADDR_SPACE_SEL;
++ data = mt9v111_device.ifpReg->addrSpaceSel;
++ mt9v111_write_reg(reg, data);
++
++ reg = MT9V111I_MODE_CONTROL;
++ data = mt9v111_device.ifpReg->modeControl;
++ mt9v111_write_reg(reg, data);
++}
++
++/*!
++ * mt9v111 sensor set the flicker control
++ *
++ * @param control int
++ * @return None
++ */
++static void mt9v111_flicker_control(int control)
++{
++ u8 reg;
++ u16 data;
++
++ reg = MT9V111I_ADDR_SPACE_SEL;
++ data = mt9v111_device.ifpReg->addrSpaceSel;
++ mt9v111_write_reg(reg, data);
++
++ switch (control) {
++ case V4L2_MXC_FLICKER_DISABLE:
++ mt9v111_device.ifpReg->formatControl &= ~(0x01 << 11);
++
++ reg = MT9V111I_FORMAT_CONTROL;
++ data = mt9v111_device.ifpReg->formatControl;
++ mt9v111_write_reg(reg, data);
++ break;
++
++ case V4L2_MXC_FLICKER_50HZ:
++ if (!(mt9v111_device.ifpReg->formatControl & (0x01 << 11))) {
++ mt9v111_device.ifpReg->formatControl |= (0x01 << 11);
++ reg = MT9V111I_FORMAT_CONTROL;
++ data = mt9v111_device.ifpReg->formatControl;
++ mt9v111_write_reg(reg, data);
++ }
++ mt9v111_device.ifpReg->flickerCtrl = 0x01;
++ reg = MT9V111I_FLICKER_CONTROL;
++ data = mt9v111_device.ifpReg->flickerCtrl;
++ mt9v111_write_reg(reg, data);
++ break;
++
++ case V4L2_MXC_FLICKER_60HZ:
++ if (!(mt9v111_device.ifpReg->formatControl & (0x01 << 11))) {
++ mt9v111_device.ifpReg->formatControl |= (0x01 << 11);
++ reg = MT9V111I_FORMAT_CONTROL;
++ data = mt9v111_device.ifpReg->formatControl;
++ mt9v111_write_reg(reg, data);
++ }
++ mt9v111_device.ifpReg->flickerCtrl = 0x03;
++ reg = MT9V111I_FLICKER_CONTROL;
++ data = mt9v111_device.ifpReg->flickerCtrl;
++ mt9v111_write_reg(reg, data);
++ break;
++
++ case V4L2_MXC_FLICKER_AUTO:
++ if (!(mt9v111_device.ifpReg->formatControl & (0x01 << 11))) {
++ mt9v111_device.ifpReg->formatControl |= (0x01 << 11);
++ reg = MT9V111I_FORMAT_CONTROL;
++ data = mt9v111_device.ifpReg->formatControl;
++ mt9v111_write_reg(reg, data);
++ }
++ mt9v111_device.ifpReg->flickerCtrl = 0x10;
++ reg = MT9V111I_FLICKER_CONTROL;
++ data = mt9v111_device.ifpReg->flickerCtrl;
++ mt9v111_write_reg(reg, data);
++ break;
++ }
++ return;
++
++}
++
++/*!
++ * mt9v111 Get mode&flicker control parameters
++ *
++ * @return None
++ */
++static void mt9v111_get_control_params(int *ae, int *awb, int *flicker)
++{
++ if ((ae != NULL) && (awb != NULL) && (flicker != NULL)) {
++ *ae = (mt9v111_device.ifpReg->modeControl & 0x4000) >> 14;
++ *awb = (mt9v111_device.ifpReg->modeControl & 0x02) >> 1;
++ *flicker = (mt9v111_device.ifpReg->formatControl & 0x800) >> 9;
++ if (*flicker) {
++ *flicker = (mt9v111_device.ifpReg->flickerCtrl & 0x03);
++ switch (*flicker) {
++ case 1:
++ *flicker = V4L2_MXC_FLICKER_50HZ;
++ break;
++ case 3:
++ *flicker = V4L2_MXC_FLICKER_60HZ;
++ break;
++ default:
++ *flicker = V4L2_MXC_FLICKER_AUTO;
++ break;
++ }
++ } else
++ *flicker = V4L2_MXC_FLICKER_DISABLE;
++ }
++ return;
++}
++
++/*!
++ * mt9v111 Reset function
++ *
++ * @return None
++ */
++static sensor_interface *mt9v111_reset(void)
++{
++ return mt9v111_config(&reset_frame_rate, 0);
++}
++
++/*!
++ * mt9v111 get_status function
++ *
++ * @return int
++ */
++static int mt9v111_get_status(void)
++{
++ int retval = 0;
++ u8 reg;
++ u16 data = 0;
++
++ if (!interface_param)
++ return -ENODEV;
++
++ reg = MT9V111I_ADDR_SPACE_SEL;
++ data = MT9V111I_SEL_SCA;
++ retval = mt9v111_write_reg(reg, data);
++
++ reg = MT9V111S_SENSOR_CORE_VERSION;
++ retval = mt9v111_read_reg(&reg, &data);
++ data = mt9v111_endian_swap16(data);
++ if (MT9V111_CHIP_VERSION != data)
++ retval = -ENODEV;
++
++ return retval;
++}
++
++struct camera_sensor camera_sensor_if = {
++ .set_color = mt9v111_set_color,
++ .get_color = mt9v111_get_color,
++ .set_ae_mode = mt9v111_set_ae_mode,
++ .get_ae_mode = mt9v111_get_ae_mode,
++ .set_ae = mt9v111_set_ae,
++ .set_ae_limit = mt9v111_set_ae_limit,
++ .set_awb = mt9v111_set_awb,
++ .flicker_control = mt9v111_flicker_control,
++ .get_control_params = mt9v111_get_control_params,
++ .config = mt9v111_config,
++ .reset = mt9v111_reset,
++ .get_status = mt9v111_get_status,
++};
++
++#ifdef MT9V111_DEBUG
++/*!
++ * Set sensor to test mode, which will generate test pattern.
++ *
++ * @return none
++ */
++static void mt9v111_test_pattern(bool flag)
++{
++ u8 reg;
++ u16 data;
++
++ // switch to sensor registers
++ reg = MT9V111I_ADDR_SPACE_SEL;
++ data = MT9V111I_SEL_SCA;
++ mt9v111_write_reg(reg, data);
++
++ if (flag == true) {
++ testpattern = MT9V111S_OUTCTRL_TEST_MODE;
++
++ reg = MT9V111S_ROW_NOISE_CTRL;
++ data = mt9v111_read_reg(&reg, &data) & 0xBF;
++ data = mt9v111_endian_swap16(data);
++ mt9v111_write_reg(reg, data);
++
++ reg = MT9V111S_TEST_DATA;
++ data = 0;
++ mt9v111_write_reg(reg, data);
++
++ reg = MT9V111S_OUTPUT_CTRL;
++ // changes take effect
++ data = MT9V111S_OUTCTRL_CHIP_ENABLE | testpattern | 0x3000;
++ mt9v111_write_reg(reg, data);
++ } else {
++ testpattern = 0;
++
++ reg = MT9V111S_ROW_NOISE_CTRL;
++ data = mt9v111_read_reg(&reg, &data) | 0x40;
++ data = mt9v111_endian_swap16(data);
++ mt9v111_write_reg(reg, data);
++
++ reg = MT9V111S_OUTPUT_CTRL;
++ // changes take effect
++ data = MT9V111S_OUTCTRL_CHIP_ENABLE | testpattern | 0x3000;
++ mt9v111_write_reg(reg, data);
++ }
++}
++#endif
++
++/*!
++ * mt9v111 I2C detect_client function
++ *
++ * @param adapter struct i2c_adapter *
++ * @param address int
++ * @param kind int
++ *
++ * @return Error code indicating success or failure
++ */
++static int mt9v111_detect_client(struct i2c_adapter *adapter, int address,
++ int kind)
++{
++ mt9v111_i2c_client.adapter = adapter;
++ if (i2c_attach_client(&mt9v111_i2c_client)) {
++ mt9v111_i2c_client.adapter = NULL;
++ printk(KERN_ERR "mt9v111_attach: i2c_attach_client failed\n");
++ return -1;
++ }
++
++ interface_param = (sensor_interface *)
++ kmalloc(sizeof(sensor_interface), GFP_KERNEL);
++ if (!interface_param) {
++ printk(KERN_ERR "mt9v111_attach: kmalloc failed \n");
++ return -1;
++ }
++
++ printk(KERN_INFO "MT9V111 Detected\n");
++
++ return 0;
++}
++
++static unsigned short normal_i2c[] = { MT9V111_I2C_ADDRESS, I2C_CLIENT_END };
++
++/* Magic definition of all other variables and things */
++I2C_CLIENT_INSMOD;
++
++/*!
++ * mt9v111 I2C attach function
++ *
++ * @param adapter struct i2c_adapter *
++ * @return Error code indicating success or failure
++ */
++static int mt9v111_attach(struct i2c_adapter *adap)
++{
++ uint32_t mclk = 27000000;
++ struct clk *clk;
++ int err;
++
++ clk = clk_get(NULL, "csi_clk");
++ clk_enable(clk);
++ set_mclk_rate(&mclk);
++
++ err = i2c_probe(adap, &addr_data, &mt9v111_detect_client);
++ clk_disable(clk);
++ clk_put(clk);
++
++ return err;
++}
++
++/*!
++ * mt9v111 I2C detach function
++ *
++ * @param client struct i2c_client *
++ * @return Error code indicating success or failure
++ */
++static int mt9v111_detach(struct i2c_client *client)
++{
++ int err;
++
++ if (!mt9v111_i2c_client.adapter)
++ return -1;
++
++ err = i2c_detach_client(&mt9v111_i2c_client);
++ mt9v111_i2c_client.adapter = NULL;
++
++ if (interface_param)
++ kfree(interface_param);
++ interface_param = NULL;
++
++ return err;
++}
++
++extern void gpio_sensor_active(void);
++
++/*!
++ * MT9V111 init function
++ *
++ * @return Error code indicating success or failure
++ */
++static __init int mt9v111_init(void)
++{
++ u8 err;
++
++ gpio_sensor_active();
++
++ mt9v111_device.coreReg = (mt9v111_coreReg *)
++ kmalloc(sizeof(mt9v111_coreReg), GFP_KERNEL);
++ if (!mt9v111_device.coreReg)
++ return -1;
++
++ memset(mt9v111_device.coreReg, 0, sizeof(mt9v111_coreReg));
++
++ mt9v111_device.ifpReg = (mt9v111_IFPReg *)
++ kmalloc(sizeof(mt9v111_IFPReg), GFP_KERNEL);
++ if (!mt9v111_device.ifpReg) {
++ kfree(mt9v111_device.coreReg);
++ mt9v111_device.coreReg = NULL;
++ return -1;
++ }
++
++ memset(mt9v111_device.ifpReg, 0, sizeof(mt9v111_IFPReg));
++
++ err = i2c_add_driver(&mt9v111_i2c_driver);
++ return err;
++}
++
++extern void gpio_sensor_inactive(void);
++/*!
++ * MT9V111 cleanup function
++ *
++ * @return Error code indicating success or failure
++ */
++static void __exit mt9v111_clean(void)
++{
++ if (mt9v111_device.coreReg) {
++ kfree(mt9v111_device.coreReg);
++ mt9v111_device.coreReg = NULL;
++ }
++
++ if (mt9v111_device.ifpReg) {
++ kfree(mt9v111_device.ifpReg);
++ mt9v111_device.ifpReg = NULL;
++ }
++
++ i2c_del_driver(&mt9v111_i2c_driver);
++
++ gpio_sensor_inactive();
++}
++
++module_init(mt9v111_init);
++module_exit(mt9v111_clean);
++
++/* Exported symbols for modules. */
++EXPORT_SYMBOL(camera_sensor_if);
++
++MODULE_AUTHOR("Freescale Semiconductor, Inc.");
++MODULE_DESCRIPTION("Mt9v111 Camera Driver");
++MODULE_LICENSE("GPL");
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/capture/mt9v111.h linux-2.6.28-karo/drivers/media/video/mxc/capture/mt9v111.h
+--- linux-2.6.28/drivers/media/video/mxc/capture/mt9v111.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/capture/mt9v111.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,435 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @defgroup Camera Sensor Drivers
++ */
++
++/*!
++ * @file mt9v111.h
++ *
++ * @brief MT9V111 Camera Header file
++ *
++ * It include all the defines for bitmaps operations, also two main structure
++ * one for IFP interface structure, other for sensor core registers.
++ *
++ * @ingroup Camera
++ */
++
++#ifndef MT9V111_H_
++#define MT9V111_H_
++
++/*!
++ * mt9v111 CHIP VERSION
++ */
++#define MT9V111_CHIP_VERSION 0x823A
++
++/*!
++ * mt9v111 IFP REGISTER BANK MAP
++ */
++#define MT9V111I_ADDR_SPACE_SEL 0x1
++#define MT9V111I_BASE_MAXTRIX_SIGN 0x2
++#define MT9V111I_BASE_MAXTRIX_SCALE15 0x3
++#define MT9V111I_BASE_MAXTRIX_SCALE69 0x4
++#define MT9V111I_APERTURE_GAIN 0x5
++#define MT9V111I_MODE_CONTROL 0x6
++#define MT9V111I_SOFT_RESET 0x7
++#define MT9V111I_FORMAT_CONTROL 0x8
++#define MT9V111I_BASE_MATRIX_CFK1 0x9
++#define MT9V111I_BASE_MATRIX_CFK2 0xa
++#define MT9V111I_BASE_MATRIX_CFK3 0xb
++#define MT9V111I_BASE_MATRIX_CFK4 0xc
++#define MT9V111I_BASE_MATRIX_CFK5 0xd
++#define MT9V111I_BASE_MATRIX_CFK6 0xe
++#define MT9V111I_BASE_MATRIX_CFK7 0xf
++#define MT9V111I_BASE_MATRIX_CFK8 0x10
++#define MT9V111I_BASE_MATRIX_CFK9 0x11
++#define MT9V111I_AWB_POSITION 0x12
++#define MT9V111I_AWB_RED_GAIN 0x13
++#define MT9V111I_AWB_BLUE_GAIN 0x14
++#define MT9V111I_DELTA_MATRIX_CF_SIGN 0x15
++#define MT9V111I_DELTA_MATRIX_CF_D1 0x16
++#define MT9V111I_DELTA_MATRIX_CF_D2 0x17
++#define MT9V111I_DELTA_MATRIX_CF_D3 0x18
++#define MT9V111I_DELTA_MATRIX_CF_D4 0x19
++#define MT9V111I_DELTA_MATRIX_CF_D5 0x1a
++#define MT9V111I_DELTA_MATRIX_CF_D6 0x1b
++#define MT9V111I_DELTA_MATRIX_CF_D7 0x1c
++#define MT9V111I_DELTA_MATRIX_CF_D8 0x1d
++#define MT9V111I_DELTA_MATRIX_CF_D9 0x1e
++#define MT9V111I_LUMINANCE_LIMIT_WB 0x20
++#define MT9V111I_RBG_MANUUAL_WB 0x21
++#define MT9V111I_AWB_RED_LIMIT 0x22
++#define MT9V111I_AWB_BLUE_LIMIT 0x23
++#define MT9V111I_MATRIX_ADJUST_LIMIT 0x24
++#define MT9V111I_AWB_SPEED 0x25
++#define MT9V111I_H_BOUND_AE 0x26
++#define MT9V111I_V_BOUND_AE 0x27
++#define MT9V111I_H_BOUND_AE_CEN_WIN 0x2b
++#define MT9V111I_V_BOUND_AE_CEN_WIN 0x2c
++#define MT9V111I_BOUND_AWB_WIN 0x2d
++#define MT9V111I_AE_PRECISION_TARGET 0x2e
++#define MT9V111I_AE_SPEED 0x2f
++#define MT9V111I_RED_AWB_MEASURE 0x30
++#define MT9V111I_LUMA_AWB_MEASURE 0x31
++#define MT9V111I_BLUE_AWB_MEASURE 0x32
++#define MT9V111I_LIMIT_SHARP_SATU_CTRL 0x33
++#define MT9V111I_LUMA_OFFSET 0x34
++#define MT9V111I_CLIP_LIMIT_OUTPUT_LUMI 0x35
++#define MT9V111I_GAIN_LIMIT_AE 0x36
++#define MT9V111I_SHUTTER_WIDTH_LIMIT_AE 0x37
++#define MT9V111I_UPPER_SHUTTER_DELAY_LIM 0x39
++#define MT9V111I_OUTPUT_FORMAT_CTRL2 0x3a
++#define MT9V111I_IPF_BLACK_LEVEL_SUB 0x3b
++#define MT9V111I_IPF_BLACK_LEVEL_ADD 0x3c
++#define MT9V111I_ADC_LIMIT_AE_ADJ 0x3d
++#define MT9V111I_GAIN_THRE_CCAM_ADJ 0x3e
++#define MT9V111I_LINEAR_AE 0x3f
++#define MT9V111I_THRESHOLD_EDGE_DEFECT 0x47
++#define MT9V111I_LUMA_SUM_MEASURE 0x4c
++#define MT9V111I_TIME_ADV_SUM_LUMA 0x4d
++#define MT9V111I_MOTION 0x52
++#define MT9V111I_GAMMA_KNEE_Y12 0x53
++#define MT9V111I_GAMMA_KNEE_Y34 0x54
++#define MT9V111I_GAMMA_KNEE_Y56 0x55
++#define MT9V111I_GAMMA_KNEE_Y78 0x56
++#define MT9V111I_GAMMA_KNEE_Y90 0x57
++#define MT9V111I_GAMMA_VALUE_Y0 0x58
++#define MT9V111I_SHUTTER_60 0x59
++#define MT9V111I_FLICKER_CONTROL 0x5B
++#define MT9V111I_SEARCH_FLICK_60 0x5c
++#define MT9V111I_RATIO_IMAGE_GAIN_BASE 0x5e
++#define MT9V111I_RATIO_IMAGE_GAIN_DELTA 0x5f
++#define MT9V111I_SIGN_VALUE_REG5F 0x60
++#define MT9V111I_AE_GAIN 0x62
++#define MT9V111I_MAX_GAIN_AE 0x67
++#define MT9V111I_LENS_CORRECT_CTRL 0x80
++#define MT9V111I_SHADING_PARAMETER1 0x81
++#define MT9V111I_SHADING_PARAMETER2 0x82
++#define MT9V111I_SHADING_PARAMETER3 0x83
++#define MT9V111I_SHADING_PARAMETER4 0x84
++#define MT9V111I_SHADING_PARAMETER5 0x85
++#define MT9V111I_SHADING_PARAMETER6 0x86
++#define MT9V111I_SHADING_PARAMETER7 0x87
++#define MT9V111I_SHADING_PARAMETER8 0x88
++#define MT9V111I_SHADING_PARAMETER9 0x89
++#define MT9V111I_SHADING_PARAMETER10 0x8A
++#define MT9V111I_SHADING_PARAMETER11 0x8B
++#define MT9V111I_SHADING_PARAMETER12 0x8C
++#define MT9V111I_SHADING_PARAMETER13 0x8D
++#define MT9V111I_SHADING_PARAMETER14 0x8E
++#define MT9V111I_SHADING_PARAMETER15 0x8F
++#define MT9V111I_SHADING_PARAMETER16 0x90
++#define MT9V111I_SHADING_PARAMETER17 0x91
++#define MT9V111I_SHADING_PARAMETER18 0x92
++#define MT9V111I_SHADING_PARAMETER19 0x93
++#define MT9V111I_SHADING_PARAMETER20 0x94
++#define MT9V111I_SHADING_PARAMETER21 0x95
++#define MT9V111i_FLASH_CTRL 0x98
++#define MT9V111i_LINE_COUNTER 0x99
++#define MT9V111i_FRAME_COUNTER 0x9A
++#define MT9V111i_H_PAN 0xA5
++#define MT9V111i_H_ZOOM 0xA6
++#define MT9V111i_H_SIZE 0xA7
++#define MT9V111i_V_PAN 0xA8
++#define MT9V111i_V_ZOOM 0xA9
++#define MT9V111i_V_SIZE 0xAA
++
++#define MT9V111I_SEL_IFP 0x1
++#define MT9V111I_SEL_SCA 0x4
++#define MT9V111I_FC_RGB_OR_YUV 0x1000
++
++/*!
++ * Mt9v111 SENSOR CORE REGISTER BANK MAP
++ */
++#define MT9V111S_ADDR_SPACE_SEL 0x1
++#define MT9V111S_COLUMN_START 0x2
++#define MT9V111S_WIN_HEIGHT 0x3
++#define MT9V111S_WIN_WIDTH 0x4
++#define MT9V111S_HOR_BLANKING 0x5
++#define MT9V111S_VER_BLANKING 0x6
++#define MT9V111S_OUTPUT_CTRL 0x7
++#define MT9V111S_ROW_START 0x8
++#define MT9V111S_SHUTTER_WIDTH 0x9
++#define MT9V111S_PIXEL_CLOCK_SPEED 0xa
++#define MT9V111S_RESTART 0xb
++#define MT9V111S_SHUTTER_DELAY 0xc
++#define MT9V111S_RESET 0xd
++#define MT9V111S_COLUMN_START_IN_ZOOM 0x12
++#define MT9V111S_ROW_START_IN_ZOOM 0x13
++#define MT9V111S_DIGITAL_ZOOM 0x1e
++#define MT9V111S_READ_MODE 0x20
++#define MT9V111S_DAC_CTRL 0x27
++#define MT9V111S_GREEN1_GAIN 0x2b
++#define MT9V111S_BLUE_GAIN 0x2c
++#define MT9V111S_READ_GAIN 0x2d
++#define MT9V111S_GREEN2_GAIN 0x2e
++#define MT9V111S_ROW_NOISE_CTRL 0x30
++#define MT9V111S_DARK_TARGET_W 0x31
++#define MT9V111S_TEST_DATA 0x32
++#define MT9V111S_GLOBAL_GAIN 0x35
++#define MT9V111S_SENSOR_CORE_VERSION 0x36
++#define MT9V111S_DARK_TARGET_WO 0x37
++#define MT9V111S_VERF_DAC 0x41
++#define MT9V111S_VCM_VCL 0x42
++#define MT9V111S_DISABLE_BYPASS 0x58
++#define MT9V111S_CALIB_MEAN_TEST 0x59
++#define MT9V111S_DARK_G1_AVE 0x5B
++#define MT9V111S_DARK_G2_AVE 0x5C
++#define MT9V111S_DARK_R_AVE 0x5D
++#define MT9V111S_DARK_B_AVE 0x5E
++#define MT9V111S_CAL_THRESHOLD 0x5f
++#define MT9V111S_CAL_G1 0x60
++#define MT9V111S_CAL_G2 0x61
++#define MT9V111S_CAL_CTRL 0x62
++#define MT9V111S_CAL_R 0x63
++#define MT9V111S_CAL_B 0x64
++#define MT9V111S_CHIP_ENABLE 0xF1
++#define MT9V111S_CHIP_VERSION 0xFF
++
++// OUTPUT_CTRL
++#define MT9V111S_OUTCTRL_SYNC 0x1
++#define MT9V111S_OUTCTRL_CHIP_ENABLE 0x2
++#define MT9V111S_OUTCTRL_TEST_MODE 0x40
++
++// READ_MODE
++#define MT9V111S_RM_NOBADFRAME 0x1
++#define MT9V111S_RM_NODESTRUCT 0x2
++#define MT9V111S_RM_COLUMNSKIP 0x4
++#define MT9V111S_RM_ROWSKIP 0x8
++#define MT9V111S_RM_BOOSTEDRESET 0x1000
++#define MT9V111S_RM_COLUMN_LATE 0x10
++#define MT9V111S_RM_ROW_LATE 0x80
++#define MT9V111S_RM_RIGTH_TO_LEFT 0x4000
++#define MT9V111S_RM_BOTTOM_TO_TOP 0x8000
++
++/*! I2C Slave Address */
++#define MT9V111_I2C_ADDRESS 0x48
++
++/*!
++ * The image resolution enum for the mt9v111 sensor
++ */
++typedef enum {
++ MT9V111_OutputResolution_VGA = 0, /*!< VGA size */
++ MT9V111_OutputResolution_QVGA, /*!< QVGA size */
++ MT9V111_OutputResolution_CIF, /*!< CIF size */
++ MT9V111_OutputResolution_QCIF, /*!< QCIF size */
++ MT9V111_OutputResolution_QQVGA, /*!< QQVGA size */
++ MT9V111_OutputResolution_SXGA /*!< SXGA size */
++} MT9V111_OutputResolution;
++
++enum {
++ MT9V111_WINWIDTH = 0x287,
++ MT9V111_WINWIDTH_DEFAULT = 0x287,
++ MT9V111_WINWIDTH_MIN = 0x9,
++
++ MT9V111_WINHEIGHT = 0x1E7,
++ MT9V111_WINHEIGHT_DEFAULT = 0x1E7,
++
++ MT9V111_HORZBLANK_DEFAULT = 0x26,
++ MT9V111_HORZBLANK_MIN = 0x9,
++ MT9V111_HORZBLANK_MAX = 0x3FF,
++
++ MT9V111_VERTBLANK_DEFAULT = 0x4,
++ MT9V111_VERTBLANK_MIN = 0x3,
++ MT9V111_VERTBLANK_MAX = 0xFFF,
++};
++
++/*!
++ * Mt9v111 Core Register structure.
++ */
++typedef struct {
++ u32 addressSelect; /*!< select address bank for Core Register 0x4 */
++ u32 columnStart; /*!< Starting Column */
++ u32 windowHeight; /*!< Window Height */
++ u32 windowWidth; /*!< Window Width */
++ u32 horizontalBlanking; /*!< Horizontal Blank time, in pixels */
++ u32 verticalBlanking; /*!< Vertical Blank time, in pixels */
++ u32 outputControl; /*!< Register to control sensor output */
++ u32 rowStart; /*!< Starting Row */
++ u32 shutterWidth;
++ u32 pixelClockSpeed; /*!< pixel date rate */
++ u32 restart; /*!< Abandon the readout of current frame */
++ u32 shutterDelay;
++ u32 reset; /*!< reset the sensor to the default mode */
++ u32 zoomColStart; /*!< Column start in the Zoom mode */
++ u32 zomRowStart; /*!< Row start in the Zoom mode */
++ u32 digitalZoom; /*!< 1 means zoom by 2 */
++ u32 readMode; /*!< Readmode: aspects of the readout of the sensor */
++ u32 dACStandbyControl;
++ u32 green1Gain; /*!< Gain Settings */
++ u32 blueGain;
++ u32 redGain;
++ u32 green2Gain;
++ u32 rowNoiseControl;
++ u32 darkTargetwNC;
++ u32 testData; /*!< test mode */
++ u32 globalGain;
++ u32 chipVersion;
++ u32 darkTargetwoNC;
++ u32 vREFDACs;
++ u32 vCMandVCL;
++ u32 disableBypass;
++ u32 calibMeanTest;
++ u32 darkG1average;
++ u32 darkG2average;
++ u32 darkRaverage;
++ u32 darkBaverage;
++ u32 calibThreshold;
++ u32 calibGreen1;
++ u32 calibGreen2;
++ u32 calibControl;
++ u32 calibRed;
++ u32 calibBlue;
++ u32 chipEnable; /*!< Image core Registers written by image flow processor */
++} mt9v111_coreReg;
++
++/*!
++ * Mt9v111 IFP Register structure.
++ */
++typedef struct {
++ u32 addrSpaceSel; /*!< select address bank for Core Register 0x1 */
++ u32 baseMaxtrixSign; /*!< sign of coefficient for base color correction matrix */
++ u32 baseMaxtrixScale15; /*!< scaling of color correction coefficient K1-5 */
++ u32 baseMaxtrixScale69; /*!< scaling of color correction coefficient K6-9 */
++ u32 apertureGain; /*!< sharpening */
++ u32 modeControl; /*!< bit 7 CCIR656 sync codes are embedded in the image */
++ u32 softReset; /*!< Image processing mode: 1 reset mode, 0 operational mode */
++ u32 formatControl; /*!< bit12 1 for RGB565, 0 for YcrCb */
++ u32 baseMatrixCfk1; /*!< K1 Color correction coefficient */
++ u32 baseMatrixCfk2; /*!< K2 Color correction coefficient */
++ u32 baseMatrixCfk3; /*!< K3 Color correction coefficient */
++ u32 baseMatrixCfk4; /*!< K4 Color correction coefficient */
++ u32 baseMatrixCfk5; /*!< K5 Color correction coefficient */
++ u32 baseMatrixCfk6; /*!< K6 Color correction coefficient */
++ u32 baseMatrixCfk7; /*!< K7 Color correction coefficient */
++ u32 baseMatrixCfk8; /*!< K8 Color correction coefficient */
++ u32 baseMatrixCfk9; /*!< K9 Color correction coefficient */
++ u32 awbPosition; /*!< Current position of AWB color correction matrix */
++ u32 awbRedGain; /*!< Current value of AWB red channel gain */
++ u32 awbBlueGain; /*!< Current value of AWB blue channel gain */
++ u32 deltaMatrixCFSign; /*!< Sign of coefficients of delta color correction matrix register */
++ u32 deltaMatrixCFD1; /*!< D1 Delta coefficient */
++ u32 deltaMatrixCFD2; /*!< D2 Delta coefficient */
++ u32 deltaMatrixCFD3; /*!< D3 Delta coefficient */
++ u32 deltaMatrixCFD4; /*!< D4 Delta coefficient */
++ u32 deltaMatrixCFD5; /*!< D5 Delta coefficient */
++ u32 deltaMatrixCFD6; /*!< D6 Delta coefficient */
++ u32 deltaMatrixCFD7; /*!< D7 Delta coefficient */
++ u32 deltaMatrixCFD8; /*!< D8 Delta coefficient */
++ u32 deltaMatrixCFD9; /*!< D9 Delta coefficient */
++ u32 lumLimitWB; /*!< Luminance range of pixels considered in WB statistics */
++ u32 RBGManualWB; /*!< Red and Blue color channel gains for manual white balance */
++ u32 awbRedLimit; /*!< Limits on Red channel gain adjustment through AWB */
++ u32 awbBlueLimit; /*!< Limits on Blue channel gain adjustment through AWB */
++ u32 matrixAdjLimit; /*!< Limits on color correction matrix adjustment through AWB */
++ u32 awbSpeed; /*!< AWB speed and color saturation control */
++ u32 HBoundAE; /*!< Horizontal boundaries of AWB measurement window */
++ u32 VBoundAE; /*!< Vertical boundaries of AWB measurement window */
++ u32 HBoundAECenWin; /*!< Horizontal boundaries of AE measurement window for backlight compensation */
++ u32 VBoundAECenWin; /*!< Vertical boundaries of AE measurement window for backlight compensation */
++ u32 boundAwbWin; /*!< Boundaries of AWB measurement window */
++ u32 AEPrecisionTarget; /*!< Auto exposure target and precision control */
++ u32 AESpeed; /*!< AE speed and sensitivity control register */
++ u32 redAWBMeasure; /*!< Measure of the red channel value used by AWB */
++ u32 lumaAWBMeasure; /*!< Measure of the luminance channel value used by AWB */
++ u32 blueAWBMeasure; /*!< Measure of the blue channel value used by AWB */
++ u32 limitSharpSatuCtrl; /*!< Automatic control of sharpness and color saturation */
++ u32 lumaOffset; /*!< Luminance offset control (brightness control) */
++ u32 clipLimitOutputLumi; /*!< Clipping limits for output luminance */
++ u32 gainLimitAE; /*!< Imager gain limits for AE adjustment */
++ u32 shutterWidthLimitAE; /*!< Shutter width (exposure time) limits for AE adjustment */
++ u32 upperShutterDelayLi; /*!< Upper Shutter Delay Limit */
++ u32 outputFormatCtrl2; /*!< Output Format Control 2
++ 00 = 16-bit RGB565.
++ 01 = 15-bit RGB555.
++ 10 = 12-bit RGB444x.
++ 11 = 12-bit RGBx444. */
++ u32 ipfBlackLevelSub; /*!< IFP black level subtraction */
++ u32 ipfBlackLevelAdd; /*!< IFP black level addition */
++ u32 adcLimitAEAdj; /*!< ADC limits for AE adjustment */
++ u32 agimnThreCamAdj; /*!< Gain threshold for CCM adjustment */
++ u32 linearAE;
++ u32 thresholdEdgeDefect; /*!< Edge threshold for interpolation and defect correction */
++ u32 lumaSumMeasure; /*!< Luma measured by AE engine */
++ u32 timeAdvSumLuma; /*!< Time-averaged luminance value tracked by auto exposure */
++ u32 motion; /*!< 1 when motion is detected */
++ u32 gammaKneeY12; /*!< Gamma knee points Y1 and Y2 */
++ u32 gammaKneeY34; /*!< Gamma knee points Y3 and Y4 */
++ u32 gammaKneeY56; /*!< Gamma knee points Y5 and Y6 */
++ u32 gammaKneeY78; /*!< Gamma knee points Y7 and Y8 */
++ u32 gammaKneeY90; /*!< Gamma knee points Y9 and Y10 */
++ u32 gammaKneeY0; /*!< Gamma knee point Y0 */
++ u32 shutter_width_60;
++ u32 flickerCtrl;
++ u32 search_flicker_60;
++ u32 ratioImageGainBase;
++ u32 ratioImageGainDelta;
++ u32 signValueReg5F;
++ u32 aeGain;
++ u32 maxGainAE;
++ u32 lensCorrectCtrl;
++ u32 shadingParameter1; /*!< Shade Parameters */
++ u32 shadingParameter2;
++ u32 shadingParameter3;
++ u32 shadingParameter4;
++ u32 shadingParameter5;
++ u32 shadingParameter6;
++ u32 shadingParameter7;
++ u32 shadingParameter8;
++ u32 shadingParameter9;
++ u32 shadingParameter10;
++ u32 shadingParameter11;
++ u32 shadingParameter12;
++ u32 shadingParameter13;
++ u32 shadingParameter14;
++ u32 shadingParameter15;
++ u32 shadingParameter16;
++ u32 shadingParameter17;
++ u32 shadingParameter18;
++ u32 shadingParameter19;
++ u32 shadingParameter20;
++ u32 shadingParameter21;
++ u32 flashCtrl; /*!< Flash control */
++ u32 lineCounter; /*!< Line counter */
++ u32 frameCounter; /*!< Frame counter */
++ u32 HPan; /*!< Horizontal pan in decimation */
++ u32 HZoom; /*!< Horizontal zoom in decimation */
++ u32 HSize; /*!< Horizontal output size iIn decimation */
++ u32 VPan; /*!< Vertical pan in decimation */
++ u32 VZoom; /*!< Vertical zoom in decimation */
++ u32 VSize; /*!< Vertical output size in decimation */
++} mt9v111_IFPReg;
++
++/*!
++ * mt9v111 Config structure
++ */
++typedef struct {
++ mt9v111_coreReg *coreReg; /*!< Sensor Core Register Bank */
++ mt9v111_IFPReg *ifpReg; /*!< IFP Register Bank */
++} mt9v111_conf;
++
++typedef struct {
++ u8 index;
++ u16 width;
++ u16 height;
++} mt9v111_image_format;
++
++typedef struct {
++ u16 ae;
++ u16 awb;
++ u16 flicker;
++ u16 reserved;
++} mt9v111_ctrl_params;
++
++#endif // MT9V111_H_
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/capture/mx27_csi.c linux-2.6.28-karo/drivers/media/video/mxc/capture/mx27_csi.c
+--- linux-2.6.28/drivers/media/video/mxc/capture/mx27_csi.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/capture/mx27_csi.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,332 @@
++/*
++ * Copyright 2005-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @file mx27_csi.c
++ *
++ * @brief CMOS Sensor interface functions
++ *
++ * @ingroup CSI
++ */
++#include <linux/types.h>
++#include <linux/init.h>
++#include <linux/device.h>
++#include <linux/err.h>
++#include <linux/interrupt.h>
++#include <linux/spinlock.h>
++#include <linux/delay.h>
++#include <linux/module.h>
++#include <linux/clk.h>
++#include <asm/arch/clock.h>
++#include <asm/arch/hardware.h>
++
++#include "mx27_csi.h"
++
++static csi_config_t g_csi_cfg; /* csi hardware configuration */
++static bool gcsi_mclk_on = false;
++static csi_irq_callback_t g_callback = 0;
++static void *g_callback_data = 0;
++static struct clk csi_mclk;
++
++static irqreturn_t csi_irq_handler(int irq, void *data)
++{
++ unsigned long status = __raw_readl(CSI_CSISR);
++
++ __raw_writel(status, CSI_CSISR);
++ if (g_callback)
++ g_callback(g_callback_data, status);
++
++ pr_debug("CSI status = 0x%08lX\n", status);
++
++ return IRQ_HANDLED;
++}
++
++static void csihw_set_config(csi_config_t * cfg)
++{
++ unsigned val = 0;
++
++ /* control reg 1 */
++ val |= cfg->swap16_en ? BIT_SWAP16_EN : 0;
++ val |= cfg->ext_vsync ? BIT_EXT_VSYNC : 0;
++ val |= cfg->eof_int_en ? BIT_EOF_INT_EN : 0;
++ val |= cfg->prp_if_en ? BIT_PRP_IF_EN : 0;
++ val |= cfg->ccir_mode ? BIT_CCIR_MODE : 0;
++ val |= cfg->cof_int_en ? BIT_COF_INT_EN : 0;
++ val |= cfg->sf_or_inten ? BIT_SF_OR_INTEN : 0;
++ val |= cfg->rf_or_inten ? BIT_RF_OR_INTEN : 0;
++ val |= cfg->statff_level << SHIFT_STATFF_LEVEL;
++ val |= cfg->staff_inten ? BIT_STATFF_INTEN : 0;
++ val |= cfg->rxff_level << SHIFT_RXFF_LEVEL;
++ val |= cfg->rxff_inten ? BIT_RXFF_INTEN : 0;
++ val |= cfg->sof_pol ? BIT_SOF_POL : 0;
++ val |= cfg->sof_inten ? BIT_SOF_INTEN : 0;
++ val |= cfg->mclkdiv << SHIFT_MCLKDIV;
++ val |= cfg->hsync_pol ? BIT_HSYNC_POL : 0;
++ val |= cfg->ccir_en ? BIT_CCIR_EN : 0;
++ val |= cfg->mclken ? BIT_MCLKEN : 0;
++ val |= cfg->fcc ? BIT_FCC : 0;
++ val |= cfg->pack_dir ? BIT_PACK_DIR : 0;
++ val |= cfg->gclk_mode ? BIT_GCLK_MODE : 0;
++ val |= cfg->inv_data ? BIT_INV_DATA : 0;
++ val |= cfg->inv_pclk ? BIT_INV_PCLK : 0;
++ val |= cfg->redge ? BIT_REDGE : 0;
++
++ __raw_writel(val, CSI_CSICR1);
++
++ /* control reg 3 */
++ val = 0x0;
++ val |= cfg->csi_sup ? BIT_CSI_SUP : 0;
++ val |= cfg->zero_pack_en ? BIT_ZERO_PACK_EN : 0;
++ val |= cfg->ecc_int_en ? BIT_ECC_INT_EN : 0;
++ val |= cfg->ecc_auto_en ? BIT_ECC_AUTO_EN : 0;
++
++ __raw_writel(val, CSI_CSICR3);
++
++ /* rxfifo counter */
++ __raw_writel(cfg->rxcnt, CSI_CSIRXCNT);
++
++ /* update global config */
++ memcpy(&g_csi_cfg, cfg, sizeof(csi_config_t));
++}
++
++static void csihw_reset_frame_count(void)
++{
++ __raw_writel(__raw_readl(CSI_CSICR3) | BIT_FRMCNT_RST, CSI_CSICR3);
++}
++
++static void csihw_reset(void)
++{
++ csihw_reset_frame_count();
++ __raw_writel(CSICR1_RESET_VAL, CSI_CSICR1);
++ __raw_writel(CSICR2_RESET_VAL, CSI_CSICR2);
++ __raw_writel(CSICR3_RESET_VAL, CSI_CSICR3);
++}
++
++/*!
++ * csi_init_interface
++ * Sets initial values for the CSI registers.
++ * The width and height of the sensor and the actual frame size will be
++ * set to the same values.
++ * @param width Sensor width
++ * @param height Sensor height
++ * @param pixel_fmt pixel format
++ * @param sig csi_signal_cfg_t
++ *
++ * @return 0 for success, -EINVAL for error
++ */
++int32_t csi_init_interface(uint16_t width, uint16_t height,
++ uint32_t pixel_fmt, csi_signal_cfg_t sig)
++{
++ csi_config_t cfg;
++
++ /* Set the CSI_SENS_CONF register remaining fields */
++ cfg.swap16_en = 1;
++ cfg.ext_vsync = sig.ext_vsync;
++ cfg.eof_int_en = 0;
++ cfg.prp_if_en = 1;
++ cfg.ccir_mode = 0;
++ cfg.cof_int_en = 0;
++ cfg.sf_or_inten = 0;
++ cfg.rf_or_inten = 0;
++ cfg.statff_level = 0;
++ cfg.staff_inten = 0;
++ cfg.rxff_level = 2;
++ cfg.rxff_inten = 0;
++ cfg.sof_pol = 1;
++ cfg.sof_inten = 0;
++ cfg.mclkdiv = 0;
++ cfg.hsync_pol = 1;
++ cfg.ccir_en = 0;
++ cfg.mclken = gcsi_mclk_on ? 1 : 0;
++ cfg.fcc = 1;
++ cfg.pack_dir = 0;
++ cfg.gclk_mode = 1;
++ cfg.inv_data = sig.data_pol;
++ cfg.inv_pclk = sig.pixclk_pol;
++ cfg.redge = 1;
++ cfg.csicnt1_rsv = 0;
++
++ /* control reg 3 */
++ cfg.frmcnt = 0;
++ cfg.frame_reset = 0;
++ cfg.csi_sup = 0;
++ cfg.zero_pack_en = 0;
++ cfg.ecc_int_en = 0;
++ cfg.ecc_auto_en = 0;
++
++ csihw_set_config(&cfg);
++
++ return 0;
++}
++
++/*!
++ * csi_enable_prpif
++ * Enable or disable CSI-PrP interface
++ * @param enable Non-zero to enable, zero to disable
++ */
++void csi_enable_prpif(uint32_t enable)
++{
++ if (enable) {
++ g_csi_cfg.prp_if_en = 1;
++ g_csi_cfg.sof_inten = 0;
++ g_csi_cfg.pack_dir = 0;
++ } else {
++ g_csi_cfg.prp_if_en = 0;
++ g_csi_cfg.sof_inten = 1;
++ g_csi_cfg.pack_dir = 1;
++ }
++
++ csihw_set_config(&g_csi_cfg);
++}
++
++/*!
++ * csi_enable_mclk
++ *
++ * @param src enum define which source to control the clk
++ * CSI_MCLK_VF CSI_MCLK_ENC CSI_MCLK_RAW CSI_MCLK_I2C
++ * @param flag true to enable mclk, false to disable mclk
++ * @param wait true to wait 100ms make clock stable, false not wait
++ *
++ * @return 0 for success
++ */
++int32_t csi_enable_mclk(int src, bool flag, bool wait)
++{
++ if (flag == true) {
++ clk_enable(&csi_mclk);
++ if (wait == true)
++ msleep(10);
++ pr_debug("Enable csi clock from source %d\n", src);
++ gcsi_mclk_on = true;
++ } else {
++ clk_disable(&csi_mclk);
++ pr_debug("Disable csi clock from source %d\n", src);
++ gcsi_mclk_on = false;
++ }
++
++ return 0;
++}
++
++/*!
++ * csi_read_mclk_flag
++ *
++ * @return gcsi_mclk_source
++ */
++int csi_read_mclk_flag(void)
++{
++ return 0;
++}
++
++void csi_set_callback(csi_irq_callback_t callback, void *data)
++{
++ g_callback = callback;
++ g_callback_data = data;
++}
++
++static void _mclk_recalc(struct clk *clk)
++{
++ u32 div;
++
++ div = (__raw_readl(CSI_CSICR1) & BIT_MCLKDIV) >> SHIFT_MCLKDIV;
++ div = (div + 1) * 2;
++
++ clk->rate = clk->parent->rate / div;
++}
++
++static unsigned long _mclk_round_rate(struct clk *clk, unsigned long rate)
++{
++ /* Keep CSI divider and change parent clock */
++ if (clk->parent->round_rate) {
++ return clk->parent->round_rate(clk->parent, rate * 2);
++ }
++ return 0;
++}
++
++static int _mclk_set_rate(struct clk *clk, unsigned long rate)
++{
++ int ret = -EINVAL;
++
++ /* Keep CSI divider and change parent clock */
++ if (clk->parent->set_rate) {
++ ret = clk->parent->set_rate(clk->parent, rate * 2);
++ if (ret == 0) {
++ clk->rate = clk->parent->rate / 2;
++ }
++ }
++
++ return ret;
++}
++
++static int _mclk_enable(struct clk *clk)
++{
++ __raw_writel(__raw_readl(CSI_CSICR1) | BIT_MCLKEN, CSI_CSICR1);
++ return 0;
++}
++
++static void _mclk_disable(struct clk *clk)
++{
++ __raw_writel(__raw_readl(CSI_CSICR1) & ~BIT_MCLKEN, CSI_CSICR1);
++}
++
++static struct clk csi_mclk = {
++ .name = "csi_clk",
++ .recalc = _mclk_recalc,
++ .round_rate = _mclk_round_rate,
++ .set_rate = _mclk_set_rate,
++ .enable = _mclk_enable,
++ .disable = _mclk_disable,
++};
++
++int32_t __init csi_init_module(void)
++{
++ int ret = 0;
++ struct clk *per_clk;
++
++ per_clk = clk_get(NULL, "csi_perclk");
++ if (IS_ERR(per_clk))
++ return PTR_ERR(per_clk);
++ clk_put(per_clk);
++ csi_mclk.parent = per_clk;
++ clk_register(&csi_mclk);
++ clk_enable(per_clk);
++ csi_mclk.recalc(&csi_mclk);
++
++ csihw_reset();
++
++ /* interrupt enable */
++ ret = request_irq(INT_CSI, csi_irq_handler, 0, "csi", 0);
++ if (ret)
++ pr_debug("CSI error: irq request fail\n");
++
++ return ret;
++}
++
++void __exit csi_cleanup_module(void)
++{
++ /* free irq */
++ free_irq(INT_CSI, 0);
++
++ clk_disable(&csi_mclk);
++}
++
++module_init(csi_init_module);
++module_exit(csi_cleanup_module);
++
++EXPORT_SYMBOL(csi_init_interface);
++EXPORT_SYMBOL(csi_enable_mclk);
++EXPORT_SYMBOL(csi_read_mclk_flag);
++EXPORT_SYMBOL(csi_set_callback);
++EXPORT_SYMBOL(csi_enable_prpif);
++
++MODULE_AUTHOR("Freescale Semiconductor, Inc.");
++MODULE_DESCRIPTION("MX27 CSI driver");
++MODULE_LICENSE("GPL");
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/capture/mx27_csi.h linux-2.6.28-karo/drivers/media/video/mxc/capture/mx27_csi.h
+--- linux-2.6.28/drivers/media/video/mxc/capture/mx27_csi.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/capture/mx27_csi.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,165 @@
++/*
++ * Copyright 2005-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @file mx27_csi.h
++ *
++ * @brief CMOS Sensor interface functions
++ *
++ * @ingroup CSI
++ */
++
++#ifndef MX27_CSI_H
++#define MX27_CSI_H
++
++/* reset values */
++#define CSICR1_RESET_VAL 0x40000800
++#define CSICR2_RESET_VAL 0x0
++#define CSICR3_RESET_VAL 0x0
++
++/* csi control reg 1 */
++#define BIT_SWAP16_EN (0x1 << 31)
++#define BIT_EXT_VSYNC (0x1 << 30)
++#define BIT_EOF_INT_EN (0x1 << 29)
++#define BIT_PRP_IF_EN (0x1 << 28)
++#define BIT_CCIR_MODE (0x1 << 27)
++#define BIT_COF_INT_EN (0x1 << 26)
++#define BIT_SF_OR_INTEN (0x1 << 25)
++#define BIT_RF_OR_INTEN (0x1 << 24)
++#define BIT_STATFF_LEVEL (0x3 << 22)
++#define BIT_STATFF_INTEN (0x1 << 21)
++#define BIT_RXFF_LEVEL (0x3 << 19)
++#define BIT_RXFF_INTEN (0x1 << 18)
++#define BIT_SOF_POL (0x1 << 17)
++#define BIT_SOF_INTEN (0x1 << 16)
++#define BIT_MCLKDIV (0xF << 12)
++#define BIT_HSYNC_POL (0x1 << 11)
++#define BIT_CCIR_EN (0x1 << 10)
++#define BIT_MCLKEN (0x1 << 9)
++#define BIT_FCC (0x1 << 8)
++#define BIT_PACK_DIR (0x1 << 7)
++#define BIT_CLR_STATFIFO (0x1 << 6)
++#define BIT_CLR_RXFIFO (0x1 << 5)
++#define BIT_GCLK_MODE (0x1 << 4)
++#define BIT_INV_DATA (0x1 << 3)
++#define BIT_INV_PCLK (0x1 << 2)
++#define BIT_REDGE (0x1 << 1)
++
++#define SHIFT_STATFF_LEVEL 22
++#define SHIFT_RXFF_LEVEL 19
++#define SHIFT_MCLKDIV 12
++
++/* control reg 3 */
++#define BIT_FRMCNT (0xFFFF << 16)
++#define BIT_FRMCNT_RST (0x1 << 15)
++#define BIT_CSI_SUP (0x1 << 3)
++#define BIT_ZERO_PACK_EN (0x1 << 2)
++#define BIT_ECC_INT_EN (0x1 << 1)
++#define BIT_ECC_AUTO_EN (0x1)
++
++#define SHIFT_FRMCNT 16
++
++/* csi status reg */
++#define BIT_SFF_OR_INT (0x1 << 25)
++#define BIT_RFF_OR_INT (0x1 << 24)
++#define BIT_STATFF_INT (0x1 << 21)
++#define BIT_RXFF_INT (0x1 << 18)
++#define BIT_EOF_INT (0x1 << 17)
++#define BIT_SOF_INT (0x1 << 16)
++#define BIT_F2_INT (0x1 << 15)
++#define BIT_F1_INT (0x1 << 14)
++#define BIT_COF_INT (0x1 << 13)
++#define BIT_ECC_INT (0x1 << 1)
++#define BIT_DRDY (0x1 << 0)
++
++#define CSI_MCLK_VF 1
++#define CSI_MCLK_ENC 2
++#define CSI_MCLK_RAW 4
++#define CSI_MCLK_I2C 8
++
++#define CSI_CSICR1 (IO_ADDRESS(CSI_BASE_ADDR))
++#define CSI_CSICR2 (IO_ADDRESS(CSI_BASE_ADDR + 0x4))
++#define CSI_CSISR (IO_ADDRESS(CSI_BASE_ADDR + 0x8))
++#define CSI_STATFIFO (IO_ADDRESS(CSI_BASE_ADDR + 0xC))
++#define CSI_CSIRXFIFO (IO_ADDRESS(CSI_BASE_ADDR + 0x10))
++#define CSI_CSIRXCNT (IO_ADDRESS(CSI_BASE_ADDR + 0x14))
++#define CSI_CSICR3 (IO_ADDRESS(CSI_BASE_ADDR + 0x1C))
++
++#define CSI_CSIRXFIFO_PHYADDR (CSI_BASE_ADDR + 0x10)
++
++static __inline void csi_clear_status(unsigned long status)
++{
++ __raw_writel(status, CSI_CSISR);
++}
++
++typedef struct {
++ unsigned data_width:3;
++ unsigned clk_mode:2;
++ unsigned ext_vsync:1;
++ unsigned Vsync_pol:1;
++ unsigned Hsync_pol:1;
++ unsigned pixclk_pol:1;
++ unsigned data_pol:1;
++ unsigned sens_clksrc:1;
++} csi_signal_cfg_t;
++
++typedef struct {
++ /* control reg 1 */
++ unsigned int swap16_en:1;
++ unsigned int ext_vsync:1;
++ unsigned int eof_int_en:1;
++ unsigned int prp_if_en:1;
++ unsigned int ccir_mode:1;
++ unsigned int cof_int_en:1;
++ unsigned int sf_or_inten:1;
++ unsigned int rf_or_inten:1;
++ unsigned int statff_level:2;
++ unsigned int staff_inten:1;
++ unsigned int rxff_level:2;
++ unsigned int rxff_inten:1;
++ unsigned int sof_pol:1;
++ unsigned int sof_inten:1;
++ unsigned int mclkdiv:4;
++ unsigned int hsync_pol:1;
++ unsigned int ccir_en:1;
++ unsigned int mclken:1;
++ unsigned int fcc:1;
++ unsigned int pack_dir:1;
++ unsigned int gclk_mode:1;
++ unsigned int inv_data:1;
++ unsigned int inv_pclk:1;
++ unsigned int redge:1;
++ unsigned int csicnt1_rsv:1;
++
++ /* control reg 3 */
++ unsigned int frmcnt:16;
++ unsigned int frame_reset:1;
++ unsigned int csi_sup:1;
++ unsigned int zero_pack_en:1;
++ unsigned int ecc_int_en:1;
++ unsigned int ecc_auto_en:1;
++
++ /* fifo counter */
++ unsigned int rxcnt;
++} csi_config_t;
++
++typedef void (*csi_irq_callback_t) (void *data, unsigned long status);
++
++int32_t csi_enable_mclk(int src, bool flag, bool wait);
++int32_t csi_init_interface(uint16_t width, uint16_t height,
++ uint32_t pixel_fmt, csi_signal_cfg_t sig);
++int csi_read_mclk_flag(void);
++void csi_set_callback(csi_irq_callback_t callback, void *data);
++void csi_enable_prpif(uint32_t enable);
++
++#endif
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/capture/mx27_prp.h linux-2.6.28-karo/drivers/media/video/mxc/capture/mx27_prp.h
+--- linux-2.6.28/drivers/media/video/mxc/capture/mx27_prp.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/capture/mx27_prp.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,308 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @file mx27_prp.h
++ *
++ * @brief Header file for MX27 V4L2 capture driver
++ *
++ * @ingroup MXC_V4L2_CAPTURE
++ */
++#ifndef __MX27_PRP_H__
++#define __MX27_PRP_H__
++
++#define PRP_REG(ofs) (IO_ADDRESS(EMMA_BASE_ADDR) + ofs)
++
++/* Register definitions of PrP */
++#define PRP_CNTL PRP_REG(0x00)
++#define PRP_INTRCNTL PRP_REG(0x04)
++#define PRP_INTRSTATUS PRP_REG(0x08)
++#define PRP_SOURCE_Y_PTR PRP_REG(0x0C)
++#define PRP_SOURCE_CB_PTR PRP_REG(0x10)
++#define PRP_SOURCE_CR_PTR PRP_REG(0x14)
++#define PRP_DEST_RGB1_PTR PRP_REG(0x18)
++#define PRP_DEST_RGB2_PTR PRP_REG(0x1C)
++#define PRP_DEST_Y_PTR PRP_REG(0x20)
++#define PRP_DEST_CB_PTR PRP_REG(0x24)
++#define PRP_DEST_CR_PTR PRP_REG(0x28)
++#define PRP_SOURCE_FRAME_SIZE PRP_REG(0x2C)
++#define PRP_CH1_LINE_STRIDE PRP_REG(0x30)
++#define PRP_SRC_PIXEL_FORMAT_CNTL PRP_REG(0x34)
++#define PRP_CH1_PIXEL_FORMAT_CNTL PRP_REG(0x38)
++#define PRP_CH1_OUT_IMAGE_SIZE PRP_REG(0x3C)
++#define PRP_CH2_OUT_IMAGE_SIZE PRP_REG(0x40)
++#define PRP_SOURCE_LINE_STRIDE PRP_REG(0x44)
++#define PRP_CSC_COEF_012 PRP_REG(0x48)
++#define PRP_CSC_COEF_345 PRP_REG(0x4C)
++#define PRP_CSC_COEF_678 PRP_REG(0x50)
++#define PRP_CH1_RZ_HORI_COEF1 PRP_REG(0x54)
++#define PRP_CH1_RZ_HORI_COEF2 PRP_REG(0x58)
++#define PRP_CH1_RZ_HORI_VALID PRP_REG(0x5C)
++#define PRP_CH1_RZ_VERT_COEF1 PRP_REG(0x60)
++#define PRP_CH1_RZ_VERT_COEF2 PRP_REG(0x64)
++#define PRP_CH1_RZ_VERT_VALID PRP_REG(0x68)
++#define PRP_CH2_RZ_HORI_COEF1 PRP_REG(0x6C)
++#define PRP_CH2_RZ_HORI_COEF2 PRP_REG(0x70)
++#define PRP_CH2_RZ_HORI_VALID PRP_REG(0x74)
++#define PRP_CH2_RZ_VERT_COEF1 PRP_REG(0x78)
++#define PRP_CH2_RZ_VERT_COEF2 PRP_REG(0x7C)
++#define PRP_CH2_RZ_VERT_VALID PRP_REG(0x80)
++
++#define B_SET(b) (1 << (b))
++
++/* Bit definitions for PrP control register */
++#define PRP_CNTL_RSTVAL 0x28
++#define PRP_CNTL_CH1EN B_SET(0)
++#define PRP_CNTL_CH2EN B_SET(1)
++#define PRP_CNTL_CSI B_SET(2)
++#define PRP_CNTL_IN_32 B_SET(3)
++#define PRP_CNTL_IN_RGB B_SET(4)
++#define PRP_CNTL_IN_YUV420 0
++#define PRP_CNTL_IN_YUV422 PRP_CNTL_IN_32
++#define PRP_CNTL_IN_RGB16 PRP_CNTL_IN_RGB
++#define PRP_CNTL_IN_RGB32 (PRP_CNTL_IN_RGB | PRP_CNTL_IN_32)
++#define PRP_CNTL_CH1_RGB8 0
++#define PRP_CNTL_CH1_RGB16 B_SET(5)
++#define PRP_CNTL_CH1_RGB32 B_SET(6)
++#define PRP_CNTL_CH1_YUV422 (B_SET(5) | B_SET(6))
++#define PRP_CNTL_CH2_YUV420 0
++#define PRP_CNTL_CH2_YUV422 B_SET(7)
++#define PRP_CNTL_CH2_YUV444 B_SET(8)
++#define PRP_CNTL_CH1_LOOP B_SET(9)
++#define PRP_CNTL_CH2_LOOP B_SET(10)
++#define PRP_CNTL_AUTODROP B_SET(11)
++#define PRP_CNTL_RST B_SET(12)
++#define PRP_CNTL_CNTREN B_SET(13)
++#define PRP_CNTL_WINEN B_SET(14)
++#define PRP_CNTL_UNCHAIN B_SET(15)
++#define PRP_CNTL_IN_SKIP_NONE 0
++#define PRP_CNTL_IN_SKIP_1_2 B_SET(16)
++#define PRP_CNTL_IN_SKIP_1_3 B_SET(17)
++#define PRP_CNTL_IN_SKIP_2_3 (B_SET(16) | B_SET(17))
++#define PRP_CNTL_IN_SKIP_1_4 B_SET(18)
++#define PRP_CNTL_IN_SKIP_3_4 (B_SET(16) | B_SET(18))
++#define PRP_CNTL_IN_SKIP_2_5 (B_SET(17) | B_SET(18))
++#define PRP_CNTL_IN_SKIP_3_5 (B_SET(16) | B_SET(17) | B_SET(18))
++#define PRP_CNTL_CH1_SKIP_NONE 0
++#define PRP_CNTL_CH1_SKIP_1_2 B_SET(19)
++#define PRP_CNTL_CH1_SKIP_1_3 B_SET(20)
++#define PRP_CNTL_CH1_SKIP_2_3 (B_SET(19) | B_SET(20))
++#define PRP_CNTL_CH1_SKIP_1_4 B_SET(21)
++#define PRP_CNTL_CH1_SKIP_3_4 (B_SET(19) | B_SET(21))
++#define PRP_CNTL_CH1_SKIP_2_5 (B_SET(20) | B_SET(21))
++#define PRP_CNTL_CH1_SKIP_3_5 (B_SET(19) | B_SET(20) | B_SET(21))
++#define PRP_CNTL_CH2_SKIP_NONE 0
++#define PRP_CNTL_CH2_SKIP_1_2 B_SET(22)
++#define PRP_CNTL_CH2_SKIP_1_3 B_SET(23)
++#define PRP_CNTL_CH2_SKIP_2_3 (B_SET(22) | B_SET(23))
++#define PRP_CNTL_CH2_SKIP_1_4 B_SET(24)
++#define PRP_CNTL_CH2_SKIP_3_4 (B_SET(22) | B_SET(24))
++#define PRP_CNTL_CH2_SKIP_2_5 (B_SET(23) | B_SET(24))
++#define PRP_CNTL_CH2_SKIP_3_5 (B_SET(22) | B_SET(23) | B_SET(24))
++#define PRP_CNTL_FIFO_I128 0
++#define PRP_CNTL_FIFO_I96 B_SET(25)
++#define PRP_CNTL_FIFO_I64 B_SET(26)
++#define PRP_CNTL_FIFO_I32 (B_SET(25) | B_SET(26))
++#define PRP_CNTL_FIFO_O64 0
++#define PRP_CNTL_FIFO_O48 B_SET(27)
++#define PRP_CNTL_FIFO_O32 B_SET(28)
++#define PRP_CNTL_FIFO_O16 (B_SET(27) | B_SET(28))
++#define PRP_CNTL_CH2B1 B_SET(29)
++#define PRP_CNTL_CH2B2 B_SET(30)
++#define PRP_CNTL_CH2_FLOWEN B_SET(31)
++
++/* Bit definitions for PrP interrupt control register */
++#define PRP_INTRCNTL_RDERR B_SET(0)
++#define PRP_INTRCNTL_CH1WERR B_SET(1)
++#define PRP_INTRCNTL_CH2WERR B_SET(2)
++#define PRP_INTRCNTL_CH1FC B_SET(3)
++#define PRP_INTRCNTL_CH2FC B_SET(5)
++#define PRP_INTRCNTL_LBOVF B_SET(7)
++#define PRP_INTRCNTL_CH2OVF B_SET(8)
++
++/* Bit definitions for PrP interrupt status register */
++#define PRP_INTRSTAT_RDERR B_SET(0)
++#define PRP_INTRSTAT_CH1WERR B_SET(1)
++#define PRP_INTRSTAT_CH2WERR B_SET(2)
++#define PRP_INTRSTAT_CH2BUF2 B_SET(3)
++#define PRP_INTRSTAT_CH2BUF1 B_SET(4)
++#define PRP_INTRSTAT_CH1BUF2 B_SET(5)
++#define PRP_INTRSTAT_CH1BUF1 B_SET(6)
++#define PRP_INTRSTAT_LBOVF B_SET(7)
++#define PRP_INTRSTAT_CH2OVF B_SET(8)
++
++#define PRP_CHANNEL_1 0x1
++#define PRP_CHANNEL_2 0x2
++
++/* PRP-CSI config */
++#define PRP_CSI_EN 0x80
++#define PRP_CSI_LOOP (0x40 | PRP_CSI_EN)
++#define PRP_CSI_IRQ_FRM (0x08 | PRP_CSI_LOOP)
++#define PRP_CSI_IRQ_CH1ERR (0x10 | PRP_CSI_LOOP)
++#define PRP_CSI_IRQ_CH2ERR (0x20 | PRP_CSI_LOOP)
++#define PRP_CSI_IRQ_ALL (0x38 | PRP_CSI_LOOP)
++#define PRP_CSI_SKIP_NONE 0
++#define PRP_CSI_SKIP_1OF2 1
++#define PRP_CSI_SKIP_1OF3 2
++#define PRP_CSI_SKIP_2OF3 3
++#define PRP_CSI_SKIP_1OF4 4
++#define PRP_CSI_SKIP_3OF4 5
++#define PRP_CSI_SKIP_2OF5 6
++#define PRP_CSI_SKIP_4OF5 7
++
++#define PRP_PIXIN_RGB565 0x2CA00565
++#define PRP_PIXIN_RGB888 0x41000888
++#define PRP_PIXIN_YUV420 0
++#define PRP_PIXIN_YUYV 0x22000888
++#define PRP_PIXIN_YVYU 0x20100888
++#define PRP_PIXIN_UYVY 0x03080888
++#define PRP_PIXIN_VYUY 0x01180888
++#define PRP_PIXIN_YUV422 0x62080888
++
++#define PRP_PIX1_RGB332 0x14400322
++#define PRP_PIX1_RGB565 0x2CA00565
++#define PRP_PIX1_RGB888 0x41000888
++#define PRP_PIX1_YUYV 0x62000888
++#define PRP_PIX1_YVYU 0x60100888
++#define PRP_PIX1_UYVY 0x43080888
++#define PRP_PIX1_VYUY 0x41180888
++#define PRP_PIX1_UNUSED 0
++
++#define PRP_PIX2_YUV420 0
++#define PRP_PIX2_YUV422 1
++#define PRP_PIX2_YUV444 4
++#define PRP_PIX2_UNUSED 8
++
++#define PRP_ALGO_WIDTH_ANY 0
++#define PRP_ALGO_HEIGHT_ANY 0
++#define PRP_ALGO_WIDTH_BIL 1
++#define PRP_ALGO_WIDTH_AVG 2
++#define PRP_ALGO_HEIGHT_BIL 4
++#define PRP_ALGO_HEIGHT_AVG 8
++#define PRP_ALGO_BYPASS 0x10
++
++typedef struct _emma_prp_ratio {
++ unsigned short num;
++ unsigned short den;
++} emma_prp_ratio;
++
++/*
++ * The following definitions are for resizing. Definition values must not
++ * be changed otherwise decision logic will be wrong.
++ */
++#define BC_COEF 3
++#define MAX_TBL 20
++#define SZ_COEF (1 << BC_COEF)
++
++#define ALGO_AUTO 0
++#define ALGO_BIL 1
++#define ALGO_AVG 2
++
++typedef struct {
++ char tbl[20]; /* table entries */
++ char len; /* table length used */
++ char algo; /* ALGO_xxx */
++ char ratio[20]; /* ratios used */
++} scale_t;
++
++/*
++ * structure for prp scaling.
++ * algorithm - bilinear or averaging for each axis
++ * PRP_ALGO_WIDTH_x | PRP_ALGO_HEIGHT_x | PRP_ALGO_BYPASS
++ * PRP_ALGO_BYPASS - Ch1 will not use Ch2 scaling with this flag
++ */
++typedef struct _emma_prp_scale {
++ unsigned char algo;
++ emma_prp_ratio width;
++ emma_prp_ratio height;
++} emma_prp_scale;
++
++typedef struct emma_prp_cfg {
++ unsigned int in_pix; /* PRP_PIXIN_xxx */
++ unsigned short in_width; /* image width, 32 - 2044 */
++ unsigned short in_height; /* image height, 32 - 2044 */
++ unsigned char in_csi; /* PRP_CSI_SKIP_x | PRP_CSI_LOOP */
++ unsigned short in_line_stride; /* in_line_stride and in_line_skip */
++ unsigned short in_line_skip; /* allow cropping from CSI */
++ unsigned int in_ptr; /* bus address */
++ /*
++ * in_csc[9] = 1 -> Y-16
++ * if in_csc[1..9] == 0
++ * in_csc[0] represents YUV range 0-3 = A0,A1,B0,B1;
++ * else
++ * in_csc[0..9] represents either format
++ */
++ unsigned short in_csc[10];
++
++ unsigned char ch2_pix; /* PRP_PIX2_xxx */
++ emma_prp_scale ch2_scale; /* resizing paramters */
++ unsigned short ch2_width; /* 4-2044, 0 = scaled */
++ unsigned short ch2_height; /* 4-2044, 0 = scaled */
++ unsigned int ch2_ptr; /* bus addr */
++ unsigned int ch2_ptr2; /* bus addr for 2nd buf (loop mode) */
++ unsigned char ch2_csi; /* PRP_CSI_SKIP_x | PRP_CSI_LOOP */
++
++ unsigned int ch1_pix; /* PRP_PIX1_xxx */
++ emma_prp_scale ch1_scale; /* resizing parameters */
++ unsigned short ch1_width; /* 4-2044, 0 = scaled */
++ unsigned short ch1_height; /* 4-2044, 0 = scaled */
++ unsigned short ch1_stride; /* 4-4088, 0 = ch1_width */
++ unsigned int ch1_ptr; /* bus addr */
++ unsigned int ch1_ptr2; /* bus addr for 2nd buf (loop mode) */
++ unsigned char ch1_csi; /* PRP_CSI_SKIP_x | PRP_CSI_LOOP */
++
++ /*
++ * channel resizing coefficients
++ * scale[0] for channel 1 width
++ * scale[1] for channel 1 height
++ * scale[2] for channel 2 width
++ * scale[3] for channel 2 height
++ */
++ scale_t scale[4];
++} emma_prp_cfg;
++
++int prphw_reset(void);
++int prphw_enable(int channel);
++int prphw_disable(int channel);
++int prphw_inptr(emma_prp_cfg *);
++int prphw_ch1ptr(emma_prp_cfg *);
++int prphw_ch1ptr2(emma_prp_cfg *);
++int prphw_ch2ptr(emma_prp_cfg *);
++int prphw_ch2ptr2(emma_prp_cfg *);
++int prphw_cfg(emma_prp_cfg *);
++int prphw_isr(void);
++void prphw_init(void);
++void prphw_exit(void);
++
++/*
++ * scale out coefficient table
++ * din in scale numerator
++ * dout in scale denominator
++ * inv in pre-scale dimension
++ * vout in/out post-scale output dimension
++ * pout out post-scale internal dimension [opt]
++ * retry in retry times (round the output length) when need
++ */
++int prp_scale(scale_t * pscale, int din, int dout, int inv,
++ unsigned short *vout, unsigned short *pout, int ch);
++
++int prp_init(void *dev_id);
++void prp_exit(void *dev_id);
++int prp_enc_select(void *data);
++int prp_enc_deselect(void *data);
++int prp_vf_select(void *data);
++int prp_vf_deselect(void *data);
++int prp_still_select(void *data);
++int prp_still_deselect(void *data);
++
++#endif /* __MX27_PRP_H__ */
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/capture/mx27_prphw.c linux-2.6.28-karo/drivers/media/video/mxc/capture/mx27_prphw.c
+--- linux-2.6.28/drivers/media/video/mxc/capture/mx27_prphw.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/capture/mx27_prphw.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,1230 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @file mx27_prphw.c
++ *
++ * @brief MX27 Video For Linux 2 capture driver
++ *
++ * @ingroup MXC_V4L2_CAPTURE
++ */
++#include <linux/kernel.h>
++#include <linux/module.h>
++#include <linux/string.h>
++#include <linux/clk.h>
++#include <asm/io.h>
++#include <linux/delay.h>
++
++#include "mx27_prp.h"
++
++#define PRP_MIN_IN_WIDTH 32
++#define PRP_MAX_IN_WIDTH 2044
++#define PRP_MIN_IN_HEIGHT 32
++#define PRP_MAX_IN_HEIGHT 2044
++
++typedef struct _coeff_t {
++ unsigned long coeff[2];
++ unsigned long cntl;
++} coeff_t[2][2];
++
++static coeff_t *PRP_RSZ_COEFF = (coeff_t *) PRP_CH1_RZ_HORI_COEF1;
++
++static unsigned char scale_get(scale_t * t,
++ unsigned char *i, unsigned char *out);
++static int gcd(int x, int y);
++static int ratio(int x, int y, int *den);
++static int prp_scale_bilinear(scale_t * t, int coeff, int base, int nxt);
++static int prp_scale_ave(scale_t * t, unsigned char base);
++static int ave_scale(scale_t * t, int inv, int outv);
++static int scale(scale_t * t, int inv, int outv);
++
++/*!
++ * @param t table
++ * @param i table index
++ * @param out bilinear # input pixels to advance
++ * average whether result is ready for output
++ * @return coefficient
++*/
++static unsigned char scale_get(scale_t * t, unsigned char *i,
++ unsigned char *out)
++{
++ unsigned char c;
++
++ c = t->tbl[*i];
++ (*i)++;
++ *i %= t->len;
++
++ if (out) {
++ if (t->algo == ALGO_BIL) {
++ for ((*out) = 1;
++ (*i) && ((*i) < t->len) && !t->tbl[(*i)]; (*i)++) {
++ (*out)++;
++ }
++ if ((*i) == t->len)
++ (*i) = 0;
++ } else
++ *out = c >> BC_COEF;
++ }
++
++ c &= SZ_COEF - 1;
++
++ if (c == SZ_COEF - 1)
++ c = SZ_COEF;
++
++ return c;
++}
++
++/*!
++ * @brief Get maximum common divisor.
++ * @param x First input value
++ * @param y Second input value
++ * @return Maximum common divisor of x and y
++ */
++static int gcd(int x, int y)
++{
++ int k;
++
++ if (x < y) {
++ k = x;
++ x = y;
++ y = k;
++ }
++
++ while ((k = x % y)) {
++ x = y;
++ y = k;
++ }
++
++ return y;
++}
++
++/*!
++ * @brief Get ratio.
++ * @param x First input value
++ * @param y Second input value
++ * @param den Denominator of the ratio (corresponding to y)
++ * @return Numerator of the ratio (corresponding to x)
++ */
++static int ratio(int x, int y, int *den)
++{
++ int g;
++
++ if (!x || !y)
++ return 0;
++
++ g = gcd(x, y);
++ *den = y / g;
++
++ return x / g;
++}
++
++/*!
++ * @brief Build PrP coefficient entry based on bilinear algorithm
++ *
++ * @param t The pointer to scale_t structure
++ * @param coeff The weighting coefficient
++ * @param base The base of the coefficient
++ * @param nxt Number of pixels to be read
++ *
++ * @return The length of current coefficient table on success
++ * -1 on failure
++ */
++static int prp_scale_bilinear(scale_t * t, int coeff, int base, int nxt)
++{
++ int i;
++
++ if (t->len >= sizeof(t->tbl))
++ return -1;
++
++ coeff = ((coeff << BC_COEF) + (base >> 1)) / base;
++ if (coeff >= SZ_COEF - 1)
++ coeff--;
++
++ coeff |= SZ_COEF;
++ t->tbl[(int)t->len++] = (unsigned char)coeff;
++
++ for (i = 1; i < nxt; i++) {
++ if (t->len >= MAX_TBL)
++ return -1;
++
++ t->tbl[(int)t->len++] = 0;
++ }
++
++ return t->len;
++}
++
++#define _bary(name) static const unsigned char name[]
++
++_bary(c1) = {
++7};
++
++_bary(c2) = {
++4, 4};
++
++_bary(c3) = {
++2, 4, 2};
++
++_bary(c4) = {
++2, 2, 2, 2};
++
++_bary(c5) = {
++1, 2, 2, 2, 1};
++
++_bary(c6) = {
++1, 1, 2, 2, 1, 1};
++
++_bary(c7) = {
++1, 1, 1, 2, 1, 1, 1};
++
++_bary(c8) = {
++1, 1, 1, 1, 1, 1, 1, 1};
++
++_bary(c9) = {
++1, 1, 1, 1, 1, 1, 1, 1, 0};
++
++_bary(c10) = {
++0, 1, 1, 1, 1, 1, 1, 1, 1, 0};
++
++_bary(c11) = {
++0, 1, 1, 1, 1, 0, 1, 1, 1, 1, 0};
++
++_bary(c12) = {
++0, 1, 1, 0, 1, 1, 1, 1, 0, 1, 1, 0};
++
++_bary(c13) = {
++0, 1, 1, 0, 1, 1, 0, 1, 1, 0, 1, 1, 0};
++
++_bary(c14) = {
++0, 1, 1, 0, 1, 1, 0, 1, 1, 0, 1, 0, 1, 0};
++
++_bary(c15) = {
++0, 1, 0, 1, 0, 1, 1, 0, 1, 1, 0, 1, 0, 1, 0};
++
++_bary(c16) = {
++1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0};
++
++_bary(c17) = {
++0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0};
++
++_bary(c18) = {
++0, 1, 0, 1, 0, 1, 0, 1, 0, 0, 1, 0, 1, 0, 1, 0, 1, 0};
++
++_bary(c19) = {
++0, 1, 0, 1, 0, 1, 0, 1, 0, 0, 1, 0, 1, 0, 1, 0, 0, 1, 0};
++
++_bary(c20) = {
++0, 1, 0, 0, 1, 0, 1, 0, 1, 0, 0, 1, 0, 1, 0, 1, 0, 0, 1, 0};
++
++static const unsigned char *ave_coeff[] = {
++ c1, c2, c3, c4, c5, c6, c7, c8, c9, c10,
++ c11, c12, c13, c14, c15, c16, c17, c18, c19, c20
++};
++
++static const unsigned char coeftab[] = {
++ 1, 1,
++ 19, 20,
++ 18, 19,
++ 17, 18,
++ 16, 17,
++ 15, 16,
++ 14, 15,
++ 13, 14,
++ 12, 13,
++ 11, 12,
++ 10, 11,
++ 9, 10,
++ 17, 19,
++ 8, 9,
++ 15, 17,
++ 7, 8,
++ 13, 15,
++ 6, 7,
++ 17, 20,
++ 11, 13,
++ 16, 19,
++ 5, 6,
++ 14, 17,
++ 9, 11,
++ 13, 16,
++ 4, 5,
++ 15, 19,
++ 11, 14,
++ 7, 9,
++ 10, 13,
++ 13, 17,
++ 3, 4,
++ 14, 19,
++ 11, 15,
++ 8, 11,
++ 13, 18,
++ 5, 7,
++ 12, 17,
++ 7, 10,
++ 9, 13,
++ 11, 16,
++ 13, 19,
++ 2, 3,
++ 13, 20,
++ 11, 17,
++ 9, 14,
++ 7, 11,
++ 12, 19,
++ 5, 8,
++ 8, 13,
++ 11, 18,
++ 3, 5,
++ 10, 17,
++ 7, 12,
++ 11, 19,
++ 4, 7,
++ 9, 16,
++ 5, 9,
++ 11, 20,
++ 6, 11,
++ 7, 13,
++ 8, 15,
++ 9, 17,
++ 10, 19,
++ 1, 2,
++ 9, 19,
++ 8, 17,
++ 7, 15,
++ 6, 13,
++ 5, 11,
++ 9, 20,
++ 4, 9,
++ 7, 16,
++ 3, 7,
++ 8, 19,
++ 5, 12,
++ 7, 17,
++ 2, 5,
++ 7, 18,
++ 5, 13,
++ 3, 8,
++ 7, 19,
++ 4, 11,
++ 5, 14,
++ 6, 17,
++ 7, 20,
++ 1, 3,
++ 6, 19,
++ 5, 16,
++ 4, 13,
++ 3, 10,
++ 5, 17,
++ 2, 7,
++ 5, 18,
++ 3, 11,
++ 4, 15,
++ 5, 19,
++ 1, 4,
++ 4, 17,
++ 3, 13,
++ 2, 9,
++ 3, 14,
++ 4, 19,
++ 1, 5,
++ 3, 16,
++ 2, 11,
++ 3, 17,
++ 1, 6,
++ 3, 19,
++ 2, 13,
++ 3, 20,
++ 1, 7,
++ 2, 15,
++ 1, 8,
++ 2, 17,
++ 1, 9,
++ 2, 19,
++ 1, 10,
++ 1, 11,
++ 1, 12,
++ 1, 13,
++ 1, 14,
++ 1, 15,
++ 1, 16,
++ 1, 17,
++ 1, 18,
++ 1, 19,
++ 1, 20
++};
++
++/*!
++ * @brief Build PrP coefficient table based on average algorithm
++ *
++ * @param t The pointer to scale_t structure
++ * @param base The base of the coefficient
++ *
++ * @return The length of current coefficient table on success
++ * -1 on failure
++ */
++static int prp_scale_ave(scale_t * t, unsigned char base)
++{
++ if (t->len + base > sizeof(t->tbl))
++ return -1;
++
++ memcpy(&t->tbl[(int)t->len], ave_coeff[(int)base - 1], base);
++ t->len = (unsigned char)(t->len + base);
++ t->tbl[t->len - 1] |= SZ_COEF;
++
++ return t->len;
++}
++
++/*!
++ * @brief Build PrP coefficient table based on average algorithm
++ *
++ * @param t The pointer to scale_t structure
++ * @param inv Input resolution
++ * @param outv Output resolution
++ *
++ * @return The length of current coefficient table on success
++ * -1 on failure
++ */
++static int ave_scale(scale_t * t, int inv, int outv)
++{
++ int ratio_count;
++
++ ratio_count = 0;
++ if (outv != 1) {
++ unsigned char a[20];
++ int v;
++
++ /* split n:m into multiple n[i]:1 */
++ for (v = 0; v < outv; v++)
++ a[v] = (unsigned char)(inv / outv);
++
++ inv %= outv;
++ if (inv) {
++ /* find start of next layer */
++ v = (outv - inv) >> 1;
++ inv += v;
++ for (; v < inv; v++)
++ a[v]++;
++ }
++
++ for (v = 0; v < outv; v++) {
++ if (prp_scale_ave(t, a[v]) < 0)
++ return -1;
++
++ t->ratio[ratio_count] = a[v];
++ ratio_count++;
++ }
++ } else if (prp_scale_ave(t, inv) < 0) {
++ return -1;
++ } else {
++ t->ratio[ratio_count++] = (char)inv;
++ ratio_count++;
++ }
++
++ return t->len;
++}
++
++/*!
++ * @brief Build PrP coefficient table
++ *
++ * @param t The pointer to scale_t structure
++ * @param inv input resolution reduced ratio
++ * @param outv output resolution reduced ratio
++ *
++ * @return The length of current coefficient table on success
++ * -1 on failure
++ */
++static int scale(scale_t * t, int inv, int outv)
++{
++ int v; /* overflow counter */
++ int coeff, nxt; /* table output */
++
++ t->len = 0;
++ if (t->algo == ALGO_AUTO) {
++ /* automatic choice - bilinear for shrinking less than 2:1 */
++ t->algo = ((outv != inv) && ((2 * outv) > inv)) ?
++ ALGO_BIL : ALGO_AVG;
++ }
++
++ /* 1:1 resize must use averaging, bilinear will hang */
++ if ((inv == outv) && (t->algo == ALGO_BIL)) {
++ pr_debug("Warning: 1:1 resize must use averaging algo\n");
++ t->algo = ALGO_AVG;
++ }
++
++ memset(t->tbl, 0, sizeof(t->tbl));
++ if (t->algo == ALGO_BIL) {
++ t->ratio[0] = (char)inv;
++ t->ratio[1] = (char)outv;
++ } else
++ memset(t->ratio, 0, sizeof(t->ratio));
++
++ if (inv == outv) {
++ /* force scaling */
++ t->ratio[0] = 1;
++ if (t->algo == ALGO_BIL)
++ t->ratio[1] = 1;
++
++ return prp_scale_ave(t, 1);
++ }
++
++ if (inv < outv) {
++ pr_debug("Upscaling not supported %d:%d\n", inv, outv);
++ return -1;
++ }
++
++ if (t->algo != ALGO_BIL)
++ return ave_scale(t, inv, outv);
++
++ v = 0;
++ if (inv >= 2 * outv) {
++ /* downscale: >=2:1 bilinear approximation */
++ coeff = inv - 2 * outv;
++ v = 0;
++ nxt = 0;
++ do {
++ v += coeff;
++ nxt = 2;
++ while (v >= outv) {
++ v -= outv;
++ nxt++;
++ }
++
++ if (prp_scale_bilinear(t, 1, 2, nxt) < 0)
++ return -1;
++ } while (v);
++ } else {
++ /* downscale: bilinear */
++ int in_pos_inc = 2 * outv;
++ int out_pos = inv;
++ int out_pos_inc = 2 * inv;
++ int init_carry = inv - outv;
++ int carry = init_carry;
++
++ v = outv + in_pos_inc;
++ do {
++ coeff = v - out_pos;
++ out_pos += out_pos_inc;
++ carry += out_pos_inc;
++ for (nxt = 0; v < out_pos; nxt++) {
++ v += in_pos_inc;
++ carry -= in_pos_inc;
++ }
++ if (prp_scale_bilinear(t, coeff, in_pos_inc, nxt) < 0)
++ return -1;
++ } while (carry != init_carry);
++ }
++ return t->len;
++}
++
++/*!
++ * @brief Get approximate ratio
++ *
++ * @param pscale The pointer to scale_t structure which holdes
++ * coefficient tables
++ * @param mt Scale ratio numerator
++ * @param nt Scale ratio denominator
++ * @param *n denominator of approximate ratio
++ * @return numerator of approximate ratio
++ */
++
++static int approx_ratio(int mt, int nt, int *n)
++{
++ int index = sizeof(coeftab) / sizeof(coeftab[0]) / 2;
++ int left = 0;
++ int right = index - 1;
++ int nom, den;
++ while (index > 0) {
++ nom = coeftab[(((right + left) >> 1) << 1) + 1];
++ den = coeftab[(((right + left) >> 1) << 1)];
++ if ((nom * nt - mt * den) < 0) {
++ left = (right + left) >> 1;
++ } else {
++ right = (right + left) >> 1;
++ }
++ index = index >> 1;
++ }
++ *n = coeftab[left * 2];
++ nom = coeftab[left * 2 + 1];
++ return nom;
++}
++
++/*!
++ * @brief Build PrP coefficient table
++ *
++ * @param pscale The pointer to scale_t structure which holdes
++ * coefficient tables
++ * @param din Scale ratio numerator
++ * @param dout Scale ratio denominator
++ * @param inv Input resolution
++ * @param vout Output resolution
++ * @param pout Internal output resolution
++ * @param retry Retry times (round the output length) when need
++ *
++ * @return Zero on success, others on failure
++ */
++int prp_scale(scale_t * pscale, int din, int dout, int inv,
++ unsigned short *vout, unsigned short *pout, int ch)
++{
++ int num, new_num;
++ int den, new_den;
++ unsigned short outv;
++
++ /* auto-generation of values */
++ if (!(dout && din)) {
++ if (!*vout)
++ dout = din = 1;
++ else {
++ din = inv;
++ dout = *vout;
++ }
++ }
++
++ if (din < dout) {
++ pr_debug("Scale err, unsupported ratio %d : %d\n", din, dout);
++ return -1;
++ }
++
++ num = ratio(din, dout, &den);
++ if (!num) {
++ pr_debug("Scale err, unsupported ratio %d : %d\n", din, dout);
++ return -1;
++ }
++
++ if (num > MAX_TBL) {
++ if (num / den <= MAX_TBL) {
++ new_num = approx_ratio(num, den, &new_den);
++ num = new_num;
++ den = new_den;
++ } else if (ch == PRP_CHANNEL_2) {
++ pr_debug("Unsupported ch_2 resize ratio %d : %d\n", num,
++ den);
++ return -1;
++ } else if (num / den > MAX_TBL * MAX_TBL) {
++ pr_debug("Unsupported ch_1 resize ratio %d : %d\n", num,
++ den);
++ return -1;
++ }
++ }
++
++ if ((num > MAX_TBL * MAX_TBL) || scale(pscale, num, den) < 0) {
++ pr_debug("Scale err, unsupported ratio %d : %d\n", num, den);
++ return -1;
++ }
++
++ if (pscale->algo == ALGO_BIL) {
++ unsigned char i, j, k;
++
++ outv =
++ (unsigned short)(inv / pscale->ratio[0] * pscale->ratio[1]);
++ inv %= pscale->ratio[0];
++ for (i = j = 0; inv > 0; j++) {
++ unsigned char nxt;
++
++ k = scale_get(pscale, &i, &nxt);
++ if (inv == 1 && k < SZ_COEF) {
++ /* needs 2 pixels for this output */
++ break;
++ }
++ inv -= nxt;
++ }
++ outv = outv + j;
++ } else {
++ unsigned char i, tot;
++
++ for (tot = i = 0; pscale->ratio[i]; i++)
++ tot = tot + pscale->ratio[i];
++
++ outv = (unsigned short)(inv / tot) * i;
++ inv %= tot;
++ for (i = 0; inv > 0; i++, outv++)
++ inv -= pscale->ratio[i];
++ }
++
++ if (!(*vout) || ((*vout) > outv))
++ *vout = outv;
++
++ if (pout)
++ *pout = outv;
++
++ return 0;
++}
++
++/*!
++ * @brief Reset PrP block
++ */
++int prphw_reset(void)
++{
++ unsigned long val;
++ unsigned long flag;
++ int i;
++
++ flag = PRP_CNTL_RST;
++ val = PRP_CNTL_RSTVAL;
++
++ __raw_writel(flag, PRP_CNTL);
++
++ /* timeout */
++ for (i = 0; i < 1000; i++) {
++ if (!(__raw_readl(PRP_CNTL) & flag)) {
++ pr_debug("PrP reset over\n");
++ break;
++ }
++ msleep(1);
++ }
++
++ /* verify reset value */
++ if (__raw_readl(PRP_CNTL) != val) {
++ pr_info("PrP reset err, val = 0x%08X\n", __raw_readl(PRP_CNTL));
++ return -1;
++ }
++
++ return 0;
++}
++
++/*!
++ * @brief Enable PrP channel.
++ * @param channel Channel number to be enabled
++ * @return Zero on success, others on failure
++ */
++int prphw_enable(int channel)
++{
++ unsigned long val;
++
++ val = __raw_readl(PRP_CNTL);
++ if (channel & PRP_CHANNEL_1)
++ val |= PRP_CNTL_CH1EN;
++ if (channel & PRP_CHANNEL_2)
++ val |=
++ (PRP_CNTL_CH2EN | PRP_CNTL_CH2_FLOWEN | PRP_CNTL_AUTODROP);
++
++ __raw_writel(val, PRP_CNTL);
++
++ return 0;
++}
++
++/*!
++ * @brief Disable PrP channel.
++ * @param channel Channel number to be disable
++ * @return Zero on success, others on failure
++ */
++int prphw_disable(int channel)
++{
++ unsigned long val;
++
++ val = __raw_readl(PRP_CNTL);
++ if (channel & PRP_CHANNEL_1)
++ val &= ~PRP_CNTL_CH1EN;
++ if (channel & PRP_CHANNEL_2)
++ val &= ~(PRP_CNTL_CH2EN | PRP_CNTL_CH2_FLOWEN);
++
++ __raw_writel(val, PRP_CNTL);
++
++ return 0;
++}
++
++/*!
++ * @brief Set PrP input buffer address.
++ * @param cfg Pointer to PrP configuration parameter
++ * @return Zero on success, others on failure
++ */
++int prphw_inptr(emma_prp_cfg * cfg)
++{
++ if (cfg->in_csi & PRP_CSI_EN)
++ return -1;
++
++ __raw_writel(cfg->in_ptr, PRP_SOURCE_Y_PTR);
++ if (cfg->in_pix == PRP_PIXIN_YUV420) {
++ u32 size;
++
++ size = cfg->in_line_stride * cfg->in_height;
++ __raw_writel(cfg->in_ptr + size, PRP_SOURCE_CB_PTR);
++ __raw_writel(cfg->in_ptr + size + (size >> 2),
++ PRP_SOURCE_CR_PTR);
++ }
++ return 0;
++}
++
++/*!
++ * @brief Set PrP channel 1 output buffer 1 address.
++ * @param cfg Pointer to PrP configuration parameter
++ * @return Zero on success, others on failure
++ */
++int prphw_ch1ptr(emma_prp_cfg * cfg)
++{
++ if (cfg->ch1_pix == PRP_PIX1_UNUSED)
++ return -1;
++
++ __raw_writel(cfg->ch1_ptr, PRP_DEST_RGB1_PTR);
++
++ /* support double buffer in loop mode only */
++ if ((cfg->in_csi & PRP_CSI_LOOP) == PRP_CSI_LOOP) {
++ if (cfg->ch1_ptr2)
++ __raw_writel(cfg->ch1_ptr2, PRP_DEST_RGB2_PTR);
++ else
++ __raw_writel(cfg->ch1_ptr, PRP_DEST_RGB2_PTR);
++ }
++
++ return 0;
++}
++
++/*!
++ * @brief Set PrP channel 1 output buffer 2 address.
++ * @param cfg Pointer to PrP configuration parameter
++ * @return Zero on success, others on failure
++ */
++int prphw_ch1ptr2(emma_prp_cfg * cfg)
++{
++ if (cfg->ch1_pix == PRP_PIX1_UNUSED ||
++ (cfg->in_csi & PRP_CSI_LOOP) != PRP_CSI_LOOP)
++ return -1;
++
++ if (cfg->ch1_ptr2)
++ __raw_writel(cfg->ch1_ptr2, PRP_DEST_RGB2_PTR);
++ else
++ return -1;
++
++ return 0;
++}
++
++/*!
++ * @brief Set PrP channel 2 output buffer 1 address.
++ * @param cfg Pointer to PrP configuration parameter
++ * @return Zero on success, others on failure
++ */
++int prphw_ch2ptr(emma_prp_cfg * cfg)
++{
++ u32 size;
++
++ if (cfg->ch2_pix == PRP_PIX2_UNUSED)
++ return -1;
++
++ __raw_writel(cfg->ch2_ptr, PRP_DEST_Y_PTR);
++
++ if (cfg->ch2_pix == PRP_PIX2_YUV420) {
++ size = cfg->ch2_width * cfg->ch2_height;
++ __raw_writel(cfg->ch2_ptr + size, PRP_DEST_CB_PTR);
++ __raw_writel(cfg->ch2_ptr + size + (size >> 2),
++ PRP_DEST_CR_PTR);
++ }
++
++ __raw_writel(__raw_readl(PRP_CNTL) | PRP_CNTL_CH2B1, PRP_CNTL);
++ return 0;
++}
++
++/*!
++ * @brief Set PrP channel 2 output buffer 2 address.
++ * @param cfg Pointer to PrP configuration parameter
++ * @return Zero on success, others on failure
++ */
++int prphw_ch2ptr2(emma_prp_cfg * cfg)
++{
++ u32 size;
++
++ if (cfg->ch2_pix == PRP_PIX2_UNUSED ||
++ (cfg->in_csi & PRP_CSI_LOOP) != PRP_CSI_LOOP)
++ return -1;
++
++ __raw_writel(cfg->ch2_ptr2, PRP_SOURCE_Y_PTR);
++ if (cfg->ch2_pix == PRP_PIX2_YUV420) {
++ size = cfg->ch2_width * cfg->ch2_height;
++ __raw_writel(cfg->ch2_ptr2 + size, PRP_SOURCE_CB_PTR);
++ __raw_writel(cfg->ch2_ptr2 + size + (size >> 2),
++ PRP_SOURCE_CR_PTR);
++ }
++
++ __raw_writel(__raw_readl(PRP_CNTL) | PRP_CNTL_CH2B2, PRP_CNTL);
++ return 0;
++}
++
++/*!
++ * @brief Build CSC table
++ * @param csc CSC table
++ * in csc[0]=index 0..3 : A.1 A.0 B.1 B.0
++ * csc[1]=direction 0 : YUV2RGB 1 : RGB2YUV
++ * out csc[0..4] are coefficients c[9] is offset
++ * csc[0..8] are coefficients c[9] is offset
++ */
++void csc_tbl(short csc[10])
++{
++ static const unsigned short _r2y[][9] = {
++ {0x4D, 0x4B, 0x3A, 0x57, 0x55, 0x40, 0x40, 0x6B, 0x29},
++ {0x42, 0x41, 0x32, 0x4C, 0x4A, 0x38, 0x38, 0x5E, 0x24},
++ {0x36, 0x5C, 0x25, 0x3B, 0x63, 0x40, 0x40, 0x74, 0x18},
++ {0x2F, 0x4F, 0x20, 0x34, 0x57, 0x38, 0x38, 0x66, 0x15},
++ };
++ static const unsigned short _y2r[][5] = {
++ {0x80, 0xb4, 0x2c, 0x5b, 0x0e4},
++ {0x95, 0xcc, 0x32, 0x68, 0x104},
++ {0x80, 0xca, 0x18, 0x3c, 0x0ec},
++ {0x95, 0xe5, 0x1b, 0x44, 0x1e0},
++ };
++ unsigned short *_csc;
++ int _csclen;
++
++ csc[9] = csc[0] & 1;
++ _csclen = csc[0] & 3;
++
++ if (csc[1]) {
++ _csc = (unsigned short *)_r2y[_csclen];
++ _csclen = sizeof(_r2y[0]);
++ } else {
++ _csc = (unsigned short *)_y2r[_csclen];
++ _csclen = sizeof(_y2r[0]);
++ memset(csc + 5, 0, sizeof(short) * 4);
++ }
++ memcpy(csc, _csc, _csclen);
++}
++
++/*!
++ * @brief Setup PrP resize coefficient registers
++ *
++ * @param ch PrP channel number
++ * @param dir Direction, 0 - horizontal, 1 - vertical
++ * @param scale The pointer to scale_t structure
++ */
++static void prp_set_scaler(int ch, int dir, scale_t * scale)
++{
++ int i;
++ unsigned int coeff[2];
++ unsigned int valid;
++
++ for (coeff[0] = coeff[1] = valid = 0, i = 19; i >= 0; i--) {
++ int j;
++
++ j = i > 9 ? 1 : 0;
++ coeff[j] = (coeff[j] << BC_COEF) |
++ (scale->tbl[i] & (SZ_COEF - 1));
++
++ if (i == 5 || i == 15)
++ coeff[j] <<= 1;
++
++ valid = (valid << 1) | (scale->tbl[i] >> BC_COEF);
++ }
++
++ valid |= (scale->len << 24) | ((2 - scale->algo) << 31);
++
++ for (i = 0; i < 2; i++)
++ (*PRP_RSZ_COEFF)[1 - ch][dir].coeff[i] = coeff[i];
++
++ (*PRP_RSZ_COEFF)[1 - ch][dir].cntl = valid;
++}
++
++/*!
++ * @brief Setup PrP registers relevant to input.
++ * @param cfg Pointer to PrP configuration parameter
++ * @param prp_cntl Holds the value for PrP control register
++ * @return Zero on success, others on failure
++ */
++static int prphw_input_cfg(emma_prp_cfg * cfg, unsigned long *prp_cntl)
++{
++ unsigned long mask;
++
++ switch (cfg->in_pix) {
++ case PRP_PIXIN_YUV420:
++ *prp_cntl |= PRP_CNTL_IN_YUV420;
++ mask = 0x7;
++ break;
++ case PRP_PIXIN_YUYV:
++ case PRP_PIXIN_YVYU:
++ case PRP_PIXIN_UYVY:
++ case PRP_PIXIN_VYUY:
++ *prp_cntl |= PRP_CNTL_IN_YUV422;
++ mask = 0x1;
++ break;
++ case PRP_PIXIN_RGB565:
++ *prp_cntl |= PRP_CNTL_IN_RGB16;
++ mask = 0x1;
++ break;
++ case PRP_PIXIN_RGB888:
++ *prp_cntl |= PRP_CNTL_IN_RGB32;
++ mask = 0;
++ break;
++ default:
++ pr_debug("Unsupported input pix format 0x%08X\n", cfg->in_pix);
++ return -1;
++ }
++
++ /* align the input image width */
++ if (cfg->in_width & mask) {
++ pr_debug("in_width misaligned. in_width=%d\n", cfg->in_width);
++ return -1;
++ }
++
++ if ((cfg->in_width < PRP_MIN_IN_WIDTH)
++ || (cfg->in_width > PRP_MAX_IN_WIDTH)) {
++ pr_debug("Unsupported input width %d\n", cfg->in_width);
++ return -1;
++ }
++
++ cfg->in_height &= ~1; /* truncate to make even */
++
++ if ((cfg->in_height < PRP_MIN_IN_HEIGHT)
++ || (cfg->in_height > PRP_MAX_IN_HEIGHT)) {
++ pr_debug("Unsupported input height %d\n", cfg->in_height);
++ return -1;
++ }
++
++ if (!(cfg->in_csi & PRP_CSI_EN))
++ if (!cfg->in_line_stride)
++ cfg->in_line_stride = cfg->in_width;
++
++ __raw_writel(cfg->in_pix, PRP_SRC_PIXEL_FORMAT_CNTL);
++ __raw_writel((cfg->in_width << 16) | cfg->in_height,
++ PRP_SOURCE_FRAME_SIZE);
++ __raw_writel((cfg->in_line_skip << 16) | cfg->in_line_stride,
++ PRP_SOURCE_LINE_STRIDE);
++
++ if (!(cfg->in_csi & PRP_CSI_EN)) {
++ __raw_writel(cfg->in_ptr, PRP_SOURCE_Y_PTR);
++ if (cfg->in_pix == PRP_PIXIN_YUV420) {
++ unsigned int size;
++
++ size = cfg->in_line_stride * cfg->in_height;
++ __raw_writel(cfg->in_ptr + size, PRP_SOURCE_CB_PTR);
++ __raw_writel(cfg->in_ptr + size + (size >> 2),
++ PRP_SOURCE_CR_PTR);
++ }
++ }
++
++ /* always cropping */
++ *prp_cntl |= PRP_CNTL_WINEN;
++
++ /* color space conversion */
++ if (!cfg->in_csc[1]) {
++ if (cfg->in_csc[0] > 3) {
++ pr_debug("in_csc invalid 0x%X\n", cfg->in_csc[0]);
++ return -1;
++ }
++ if ((cfg->in_pix == PRP_PIXIN_RGB565)
++ || (cfg->in_pix == PRP_PIXIN_RGB888))
++ cfg->in_csc[1] = 1;
++ else
++ cfg->in_csc[0] = 0;
++ csc_tbl(cfg->in_csc);
++ }
++
++ __raw_writel((cfg->in_csc[0] << 21) | (cfg->in_csc[1] << 11)
++ | cfg->in_csc[2], PRP_CSC_COEF_012);
++ __raw_writel((cfg->in_csc[3] << 21) | (cfg->in_csc[4] << 11)
++ | cfg->in_csc[5], PRP_CSC_COEF_345);
++ __raw_writel((cfg->in_csc[6] << 21) | (cfg->in_csc[7] << 11)
++ | cfg->in_csc[8] | (cfg->in_csc[9] << 31),
++ PRP_CSC_COEF_678);
++
++ if (cfg->in_csi & PRP_CSI_EN) {
++ *prp_cntl |= PRP_CNTL_CSI;
++
++ /* loop mode enable, ch1 ch2 together */
++ if ((cfg->in_csi & PRP_CSI_LOOP) == PRP_CSI_LOOP)
++ *prp_cntl |= (PRP_CNTL_CH1_LOOP | PRP_CNTL_CH2_LOOP);
++ }
++
++ return 0;
++}
++
++/*!
++ * @brief Setup PrP registers relevant to channel 2.
++ * @param cfg Pointer to PrP configuration parameter
++ * @param prp_cntl Holds the value for PrP control register
++ * @return Zero on success, others on failure
++ */
++static int prphw_ch2_cfg(emma_prp_cfg * cfg, unsigned long *prp_cntl)
++{
++ switch (cfg->ch2_pix) {
++ case PRP_PIX2_YUV420:
++ *prp_cntl |= PRP_CNTL_CH2_YUV420;
++ break;
++ case PRP_PIX2_YUV422:
++ *prp_cntl |= PRP_CNTL_CH2_YUV422;
++ break;
++ case PRP_PIX2_YUV444:
++ *prp_cntl |= PRP_CNTL_CH2_YUV444;
++ break;
++ case PRP_PIX2_UNUSED:
++ return 0;
++ default:
++ pr_debug("Unsupported channel 2 pix format 0x%08X\n",
++ cfg->ch2_pix);
++ return -1;
++ }
++
++ if (cfg->ch2_pix == PRP_PIX2_YUV420) {
++ cfg->ch2_height &= ~1; /* ensure U/V presence */
++ cfg->ch2_width &= ~7; /* ensure U/V word aligned */
++ } else if (cfg->ch2_pix == PRP_PIX2_YUV422) {
++ cfg->ch2_width &= ~1; /* word aligned */
++ }
++
++ __raw_writel((cfg->ch2_width << 16) | cfg->ch2_height,
++ PRP_CH2_OUT_IMAGE_SIZE);
++
++ return 0;
++}
++
++/*!
++ * @brief Setup PrP registers relevant to channel 1.
++ * @param cfg Pointer to PrP configuration parameter
++ * @param prp_cntl Holds the value for PrP control register
++ * @return Zero on success, others on failure
++ */
++static int prphw_ch1_cfg(emma_prp_cfg * cfg, unsigned long *prp_cntl)
++{
++ int ch1_bpp = 0;
++
++ switch (cfg->ch1_pix) {
++ case PRP_PIX1_RGB332:
++ *prp_cntl |= PRP_CNTL_CH1_RGB8;
++ ch1_bpp = 1;
++ break;
++ case PRP_PIX1_RGB565:
++ *prp_cntl |= PRP_CNTL_CH1_RGB16;
++ ch1_bpp = 2;
++ break;
++ case PRP_PIX1_RGB888:
++ *prp_cntl |= PRP_CNTL_CH1_RGB32;
++ ch1_bpp = 4;
++ break;
++ case PRP_PIX1_YUYV:
++ case PRP_PIX1_YVYU:
++ case PRP_PIX1_UYVY:
++ case PRP_PIX1_VYUY:
++ *prp_cntl |= PRP_CNTL_CH1_YUV422;
++ ch1_bpp = 2;
++ break;
++ case PRP_PIX1_UNUSED:
++ return 0;
++ default:
++ pr_debug("Unsupported channel 1 pix format 0x%08X\n",
++ cfg->ch1_pix);
++ return -1;
++ }
++
++ /* parallel or cascade resize */
++ if (cfg->ch1_scale.algo & PRP_ALGO_BYPASS)
++ *prp_cntl |= PRP_CNTL_UNCHAIN;
++
++ /* word align */
++ if (ch1_bpp == 2)
++ cfg->ch1_width &= ~1;
++ else if (ch1_bpp == 1)
++ cfg->ch1_width &= ~3;
++
++ if (!cfg->ch1_stride)
++ cfg->ch1_stride = cfg->ch1_width;
++
++ __raw_writel(cfg->ch1_pix, PRP_CH1_PIXEL_FORMAT_CNTL);
++ __raw_writel((cfg->ch1_width << 16) | cfg->ch1_height,
++ PRP_CH1_OUT_IMAGE_SIZE);
++ __raw_writel(cfg->ch1_stride * ch1_bpp, PRP_CH1_LINE_STRIDE);
++ __raw_writel(cfg->ch1_ptr, PRP_DEST_RGB1_PTR);
++
++ /* double buffer for loop mode */
++ if ((cfg->in_csi & PRP_CSI_LOOP) == PRP_CSI_LOOP) {
++ if (cfg->ch1_ptr2)
++ __raw_writel(cfg->ch1_ptr2, PRP_DEST_RGB2_PTR);
++ else
++ __raw_writel(cfg->ch1_ptr, PRP_DEST_RGB2_PTR);
++ }
++
++ return 0;
++}
++
++/*!
++ * @brief Setup PrP registers.
++ * @param cfg Pointer to PrP configuration parameter
++ * @return Zero on success, others on failure
++ */
++int prphw_cfg(emma_prp_cfg * cfg)
++{
++ unsigned long prp_cntl = 0;
++ unsigned long val;
++
++ /* input pixel format checking */
++ if (prphw_input_cfg(cfg, &prp_cntl))
++ return -1;
++
++ if (prphw_ch2_cfg(cfg, &prp_cntl))
++ return -1;
++
++ if (prphw_ch1_cfg(cfg, &prp_cntl))
++ return -1;
++
++ /* register setting */
++ __raw_writel(prp_cntl, PRP_CNTL);
++
++ /* interrupt configuration */
++ val = PRP_INTRCNTL_RDERR | PRP_INTRCNTL_LBOVF;
++ if (cfg->ch1_pix != PRP_PIX1_UNUSED)
++ val |= PRP_INTRCNTL_CH1FC | PRP_INTRCNTL_CH1WERR;
++ if (cfg->ch2_pix != PRP_PIX2_UNUSED)
++ val |=
++ PRP_INTRCNTL_CH2FC | PRP_INTRCNTL_CH2WERR |
++ PRP_INTRCNTL_CH2OVF;
++ __raw_writel(val, PRP_INTRCNTL);
++
++ prp_set_scaler(1, 0, &cfg->scale[0]); /* Channel 1 width */
++ prp_set_scaler(1, 1, &cfg->scale[1]); /* Channel 1 height */
++ prp_set_scaler(0, 0, &cfg->scale[2]); /* Channel 2 width */
++ prp_set_scaler(0, 1, &cfg->scale[3]); /* Channel 2 height */
++
++ return 0;
++}
++
++/*!
++ * @brief Check PrP interrupt status.
++ * @return PrP interrupt status
++ */
++int prphw_isr(void)
++{
++ int status;
++
++ status = __raw_readl(PRP_INTRSTATUS) & 0x1FF;
++
++ if (status & (PRP_INTRSTAT_RDERR | PRP_INTRSTAT_CH1WERR |
++ PRP_INTRSTAT_CH2WERR))
++ pr_debug("isr bus error. status= 0x%08X\n", status);
++ else if (status & PRP_INTRSTAT_CH2OVF)
++ pr_debug("isr ch 2 buffer overflow. status= 0x%08X\n", status);
++ else if (status & PRP_INTRSTAT_LBOVF)
++ pr_debug("isr line buffer overflow. status= 0x%08X\n", status);
++
++ /* silicon bug?? enable bit does not self clear? */
++ if (!(__raw_readl(PRP_CNTL) & PRP_CNTL_CH1_LOOP))
++ __raw_writel(__raw_readl(PRP_CNTL) & (~PRP_CNTL_CH1EN),
++ PRP_CNTL);
++ if (!(__raw_readl(PRP_CNTL) & PRP_CNTL_CH2_LOOP))
++ __raw_writel(__raw_readl(PRP_CNTL) & (~PRP_CNTL_CH2EN),
++ PRP_CNTL);
++
++ __raw_writel(status, PRP_INTRSTATUS); /* clr irq */
++
++ return status;
++}
++
++static struct clk *emma_clk;
++
++/*!
++ * @brief PrP module clock enable
++ */
++void prphw_init(void)
++{
++ emma_clk = clk_get(NULL, "emma_clk");
++ clk_enable(emma_clk);
++}
++
++/*!
++ * @brief PrP module clock disable
++ */
++void prphw_exit(void)
++{
++ clk_disable(emma_clk);
++ clk_put(emma_clk);
++}
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/capture/mx27_prpsw.c linux-2.6.28-karo/drivers/media/video/mxc/capture/mx27_prpsw.c
+--- linux-2.6.28/drivers/media/video/mxc/capture/mx27_prpsw.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/capture/mx27_prpsw.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,1050 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @file mx27_prpsw.c
++ *
++ * @brief MX27 Video For Linux 2 capture driver
++ *
++ * @ingroup MXC_V4L2_CAPTURE
++ */
++#include <linux/kernel.h>
++#include <linux/module.h>
++#include <linux/string.h>
++#include <linux/fb.h>
++#include <linux/pci.h>
++#include <asm/cacheflush.h>
++#include <asm/io.h>
++#include <asm/irq.h>
++
++#include "mxc_v4l2_capture.h"
++#include "mx27_prp.h"
++#include "mx27_csi.h"
++#include "../drivers/video/mxc/mx2fb.h"
++#include "../opl/opl.h"
++
++#define MEAN_COEF (SZ_COEF >> 1)
++
++static char prp_dev[] = "emma_prp";
++static int g_still_on = 0;
++static emma_prp_cfg g_prp_cfg;
++static int g_vfbuf, g_rotbuf;
++static struct tasklet_struct prp_vf_tasklet;
++
++/*
++ * The following variables represents the virtual address for the cacheable
++ * buffers accessed by SW rotation/mirroring. The rotation/mirroring in
++ * cacheable buffers has significant performance improvement than it in
++ * non-cacheable buffers.
++ */
++static char *g_vaddr_vfbuf[2] = { 0, 0 };
++static char *g_vaddr_rotbuf[2] = { 0, 0 };
++static char *g_vaddr_fb = 0;
++
++static int set_ch1_addr(emma_prp_cfg * cfg, cam_data * cam);
++static int prp_v4l2_cfg(emma_prp_cfg * cfg, cam_data * cam);
++static int prp_vf_mem_alloc(cam_data * cam);
++static void prp_vf_mem_free(cam_data * cam);
++static int prp_rot_mem_alloc(cam_data * cam);
++static void prp_rot_mem_free(cam_data * cam);
++static int prp_enc_update_eba(u32 eba, int *buffer_num);
++static int prp_enc_enable(void *private);
++static int prp_enc_disable(void *private);
++static int prp_vf_start(void *private);
++static int prp_vf_stop(void *private);
++static int prp_still_start(void *private);
++static int prp_still_stop(void *private);
++static irqreturn_t prp_isr(int irq, void *dev_id);
++static void rotation(unsigned long private);
++static int prp_resize_check_ch1(emma_prp_cfg * cfg);
++static int prp_resize_check_ch2(emma_prp_cfg * cfg);
++
++#define PRP_DUMP(val) pr_debug("%s\t = 0x%08X\t%d\n", #val, val, val)
++
++/*!
++ * @brief Dump PrP configuration parameters.
++ * @param cfg The pointer to PrP configuration parameter
++ */
++static void prp_cfg_dump(emma_prp_cfg * cfg)
++{
++ PRP_DUMP(cfg->in_pix);
++ PRP_DUMP(cfg->in_width);
++ PRP_DUMP(cfg->in_height);
++ PRP_DUMP(cfg->in_csi);
++ PRP_DUMP(cfg->in_line_stride);
++ PRP_DUMP(cfg->in_line_skip);
++ PRP_DUMP(cfg->in_ptr);
++
++ PRP_DUMP(cfg->ch1_pix);
++ PRP_DUMP(cfg->ch1_width);
++ PRP_DUMP(cfg->ch1_height);
++ PRP_DUMP(cfg->ch1_scale.algo);
++ PRP_DUMP(cfg->ch1_scale.width.num);
++ PRP_DUMP(cfg->ch1_scale.width.den);
++ PRP_DUMP(cfg->ch1_scale.height.num);
++ PRP_DUMP(cfg->ch1_scale.height.den);
++ PRP_DUMP(cfg->ch1_stride);
++ PRP_DUMP(cfg->ch1_ptr);
++ PRP_DUMP(cfg->ch1_ptr2);
++ PRP_DUMP(cfg->ch1_csi);
++
++ PRP_DUMP(cfg->ch2_pix);
++ PRP_DUMP(cfg->ch2_width);
++ PRP_DUMP(cfg->ch2_height);
++ PRP_DUMP(cfg->ch2_scale.algo);
++ PRP_DUMP(cfg->ch2_scale.width.num);
++ PRP_DUMP(cfg->ch2_scale.width.den);
++ PRP_DUMP(cfg->ch2_scale.height.num);
++ PRP_DUMP(cfg->ch2_scale.height.den);
++ PRP_DUMP(cfg->ch2_ptr);
++ PRP_DUMP(cfg->ch2_ptr2);
++ PRP_DUMP(cfg->ch2_csi);
++}
++
++/*!
++ * @brief Set PrP channel 1 output address.
++ * @param cfg Pointer to emma_prp_cfg structure
++ * @param cam Pointer to cam_data structure
++ * @return Zero on success, others on failure
++ */
++static int set_ch1_addr(emma_prp_cfg * cfg, cam_data * cam)
++{
++ if (cam->rotation != V4L2_MXC_ROTATE_NONE) {
++ cfg->ch1_ptr = (unsigned int)cam->rot_vf_bufs[0];
++ cfg->ch1_ptr2 = (unsigned int)cam->rot_vf_bufs[1];
++ if ((cam->rotation == V4L2_MXC_ROTATE_90_RIGHT)
++ || (cam->rotation == V4L2_MXC_ROTATE_90_RIGHT_VFLIP)
++ || (cam->rotation == V4L2_MXC_ROTATE_90_RIGHT_HFLIP)
++ || (cam->rotation == V4L2_MXC_ROTATE_90_LEFT))
++ cfg->ch1_stride = cam->win.w.height;
++ else
++ cfg->ch1_stride = cam->win.w.width;
++
++ if (cam->v4l2_fb.flags != V4L2_FBUF_FLAG_OVERLAY) {
++ struct fb_info *fb = cam->overlay_fb;
++ if (!fb)
++ return -1;
++ if (g_vaddr_fb)
++ iounmap(g_vaddr_fb);
++ g_vaddr_fb = ioremap_cached(fb->fix.smem_start,
++ fb->fix.smem_len);
++ if (!g_vaddr_fb)
++ return -1;
++ }
++ } else if (cam->v4l2_fb.flags == V4L2_FBUF_FLAG_OVERLAY) {
++ cfg->ch1_ptr = (unsigned int)cam->vf_bufs[0];
++ cfg->ch1_ptr2 = (unsigned int)cam->vf_bufs[1];
++ cfg->ch1_stride = cam->win.w.width;
++ } else {
++ struct fb_info *fb = cam->overlay_fb;
++
++ if (!fb)
++ return -1;
++
++ cfg->ch1_ptr = fb->fix.smem_start;
++ cfg->ch1_ptr += cam->win.w.top * fb->var.xres_virtual
++ * (fb->var.bits_per_pixel >> 3)
++ + cam->win.w.left * (fb->var.bits_per_pixel >> 3);
++ cfg->ch1_ptr2 = cfg->ch1_ptr;
++ cfg->ch1_stride = fb->var.xres_virtual;
++ }
++
++ return 0;
++}
++
++/*!
++ * @brief Setup PrP configuration parameters.
++ * @param cfg Pointer to emma_prp_cfg structure
++ * @param cam Pointer to cam_data structure
++ * @return Zero on success, others on failure
++ */
++static int prp_v4l2_cfg(emma_prp_cfg * cfg, cam_data * cam)
++{
++ cfg->in_pix = PRP_PIXIN_YUYV;
++ cfg->in_width = cam->crop_current.width;
++ cfg->in_height = cam->crop_current.height;
++ cfg->in_line_stride = cam->crop_current.left;
++ cfg->in_line_skip = cam->crop_current.top;
++ cfg->in_ptr = 0;
++ cfg->in_csi = PRP_CSI_LOOP;
++ memset(cfg->in_csc, 0, sizeof(cfg->in_csc));
++
++ if (cam->overlay_on) {
++ /* Convert V4L2 pixel format to PrP pixel format */
++ switch (cam->v4l2_fb.fmt.pixelformat) {
++ case V4L2_PIX_FMT_RGB332:
++ cfg->ch1_pix = PRP_PIX1_RGB332;
++ break;
++ case V4L2_PIX_FMT_RGB32:
++ case V4L2_PIX_FMT_BGR32:
++ cfg->ch1_pix = PRP_PIX1_RGB888;
++ break;
++ case V4L2_PIX_FMT_YUYV:
++ cfg->ch1_pix = PRP_PIX1_YUYV;
++ break;
++ case V4L2_PIX_FMT_UYVY:
++ cfg->ch1_pix = PRP_PIX1_UYVY;
++ break;
++ case V4L2_PIX_FMT_RGB565:
++ default:
++ cfg->ch1_pix = PRP_PIX1_RGB565;
++ break;
++ }
++ if ((cam->rotation == V4L2_MXC_ROTATE_90_RIGHT)
++ || (cam->rotation == V4L2_MXC_ROTATE_90_RIGHT_VFLIP)
++ || (cam->rotation == V4L2_MXC_ROTATE_90_RIGHT_HFLIP)
++ || (cam->rotation == V4L2_MXC_ROTATE_90_LEFT)) {
++ cfg->ch1_width = cam->win.w.height;
++ cfg->ch1_height = cam->win.w.width;
++ } else {
++ cfg->ch1_width = cam->win.w.width;
++ cfg->ch1_height = cam->win.w.height;
++ }
++
++ if (set_ch1_addr(cfg, cam))
++ return -1;
++ } else {
++ cfg->ch1_pix = PRP_PIX1_UNUSED;
++ cfg->ch1_width = cfg->in_width;
++ cfg->ch1_height = cfg->in_height;
++ }
++ cfg->ch1_scale.algo = 0;
++ cfg->ch1_scale.width.num = cfg->in_width;
++ cfg->ch1_scale.width.den = cfg->ch1_width;
++ cfg->ch1_scale.height.num = cfg->in_height;
++ cfg->ch1_scale.height.den = cfg->ch1_height;
++ cfg->ch1_csi = PRP_CSI_EN;
++
++ if (cam->capture_on || g_still_on) {
++ switch (cam->v2f.fmt.pix.pixelformat) {
++ case V4L2_PIX_FMT_YUYV:
++ cfg->ch2_pix = PRP_PIX2_YUV422;
++ break;
++ case V4L2_PIX_FMT_YUV420:
++ cfg->ch2_pix = PRP_PIX2_YUV420;
++ break;
++ /*
++ * YUV444 is not defined by V4L2.
++ * We support it in default case.
++ */
++ default:
++ cfg->ch2_pix = PRP_PIX2_YUV444;
++ break;
++ }
++ cfg->ch2_width = cam->v2f.fmt.pix.width;
++ cfg->ch2_height = cam->v2f.fmt.pix.height;
++ } else {
++ cfg->ch2_pix = PRP_PIX2_UNUSED;
++ cfg->ch2_width = cfg->in_width;
++ cfg->ch2_height = cfg->in_height;
++ }
++ cfg->ch2_scale.algo = 0;
++ cfg->ch2_scale.width.num = cfg->in_width;
++ cfg->ch2_scale.width.den = cfg->ch2_width;
++ cfg->ch2_scale.height.num = cfg->in_height;
++ cfg->ch2_scale.height.den = cfg->ch2_height;
++ cfg->ch2_csi = PRP_CSI_EN;
++
++ memset(cfg->scale, 0, sizeof(cfg->scale));
++ cfg->scale[0].algo = cfg->ch1_scale.algo & 3;
++ cfg->scale[1].algo = (cfg->ch1_scale.algo >> 2) & 3;
++ cfg->scale[2].algo = cfg->ch2_scale.algo & 3;
++ cfg->scale[3].algo = (cfg->ch2_scale.algo >> 2) & 3;
++
++ prp_cfg_dump(cfg);
++
++ if (prp_resize_check_ch2(cfg))
++ return -1;
++
++ if (prp_resize_check_ch1(cfg))
++ return -1;
++
++ return 0;
++}
++
++/*!
++ * @brief PrP interrupt handler
++ */
++static irqreturn_t prp_isr(int irq, void *dev_id)
++{
++ int status;
++ cam_data *cam = (cam_data *) dev_id;
++
++ status = prphw_isr();
++
++ if (g_still_on && (status & PRP_INTRSTAT_CH2BUF1)) {
++ prp_still_stop(cam);
++ cam->still_counter++;
++ wake_up_interruptible(&cam->still_queue);
++ /*
++ * Still & video capture use the same PrP channel 2.
++ * They are execlusive.
++ */
++ } else if (cam->capture_on) {
++ if (status & PRP_INTRSTAT_CH2OVF) {
++ prphw_disable(PRP_CHANNEL_2);
++ cam->enc_callback(1, cam);
++ } else if (status &
++ (PRP_INTRSTAT_CH2BUF1 | PRP_INTRSTAT_CH2BUF2)) {
++ if (cam->overflow != 1)
++ cam->enc_callback(0, cam);
++ }
++ }
++ if (cam->overlay_on
++ && (status & (PRP_INTRSTAT_CH1BUF1 | PRP_INTRSTAT_CH1BUF2))) {
++ if (cam->rotation != V4L2_MXC_ROTATE_NONE) {
++ g_rotbuf = (status & PRP_INTRSTAT_CH1BUF1) ? 0 : 1;
++ tasklet_schedule(&prp_vf_tasklet);
++ } else if (cam->v4l2_fb.flags == V4L2_FBUF_FLAG_OVERLAY) {
++ struct fb_gwinfo gwinfo;
++
++ gwinfo.enabled = 1;
++ gwinfo.alpha_value = 255;
++ gwinfo.ck_enabled = 0;
++ gwinfo.xpos = cam->win.w.left;
++ gwinfo.ypos = cam->win.w.top;
++ gwinfo.xres = cam->win.w.width;
++ gwinfo.yres = cam->win.w.height;
++ gwinfo.xres_virtual = cam->win.w.width;
++ gwinfo.vs_reversed = 0;
++ if (status & PRP_INTRSTAT_CH1BUF1)
++ gwinfo.base = (unsigned long)cam->vf_bufs[0];
++ else
++ gwinfo.base = (unsigned long)cam->vf_bufs[1];
++
++ mx2_gw_set(&gwinfo);
++ }
++ }
++
++ return IRQ_HANDLED;
++}
++
++/*!
++ * @brief PrP initialization.
++ * @param dev_id Pointer to cam_data structure
++ * @return Zero on success, others on failure
++ */
++int prp_init(void *dev_id)
++{
++ enable_irq(INT_EMMAPRP);
++ if (request_irq(INT_EMMAPRP, prp_isr, 0, prp_dev, dev_id))
++ return -1;
++ prphw_init();
++
++ return 0;
++}
++
++/*!
++ * @brief PrP initialization.
++ * @param dev_id Pointer to cam_data structure
++ */
++void prp_exit(void *dev_id)
++{
++ prphw_exit();
++ disable_irq(INT_EMMAPRP);
++ free_irq(INT_EMMAPRP, dev_id);
++}
++
++/*!
++ * @brief Update PrP channel 2 output buffer address.
++ * @param eba Physical address for PrP output buffer
++ * @param buffer_num The PrP channel 2 buffer number to be updated
++ * @return Zero on success, others on failure
++ */
++static int prp_enc_update_eba(u32 eba, int *buffer_num)
++{
++ if (*buffer_num) {
++ g_prp_cfg.ch2_ptr2 = eba;
++ prphw_ch2ptr2(&g_prp_cfg);
++ *buffer_num = 0;
++ } else {
++ g_prp_cfg.ch2_ptr = eba;
++ prphw_ch2ptr(&g_prp_cfg);
++ *buffer_num = 1;
++ }
++
++ return 0;
++}
++
++/*!
++ * @brief Enable PrP for encoding.
++ * @param private Pointer to cam_data structure
++ * @return Zero on success, others on failure
++ */
++static int prp_enc_enable(void *private)
++{
++ cam_data *cam = (cam_data *) private;
++
++ if (prp_v4l2_cfg(&g_prp_cfg, cam))
++ return -1;
++
++ csi_enable_mclk(CSI_MCLK_ENC, true, true);
++ prphw_reset();
++
++ if (prphw_cfg(&g_prp_cfg))
++ return -1;
++
++ prphw_enable(cam->overlay_on ? (PRP_CHANNEL_1 | PRP_CHANNEL_2)
++ : PRP_CHANNEL_2);
++
++ return 0;
++}
++
++/*!
++ * @brief Disable PrP for encoding.
++ * @param private Pointer to cam_data structure
++ * @return Zero on success, others on failure
++ */
++static int prp_enc_disable(void *private)
++{
++ prphw_disable(PRP_CHANNEL_2);
++ csi_enable_mclk(CSI_MCLK_ENC, false, false);
++
++ return 0;
++}
++
++/*!
++ * @brief Setup encoding functions.
++ * @param private Pointer to cam_data structure
++ * @return Zero on success, others on failure
++ */
++int prp_enc_select(void *private)
++{
++ int ret = 0;
++ cam_data *cam = (cam_data *) private;
++
++ if (cam) {
++ cam->enc_update_eba = prp_enc_update_eba;
++ cam->enc_enable = prp_enc_enable;
++ cam->enc_disable = prp_enc_disable;
++ } else
++ ret = -EIO;
++
++ return ret;
++}
++
++/*!
++ * @brief Uninstall encoding functions.
++ * @param private Pointer to cam_data structure
++ * @return Zero on success, others on failure
++ */
++int prp_enc_deselect(void *private)
++{
++ int ret = 0;
++ cam_data *cam = (cam_data *) private;
++
++ ret = prp_enc_disable(private);
++
++ if (cam) {
++ cam->enc_update_eba = NULL;
++ cam->enc_enable = NULL;
++ cam->enc_disable = NULL;
++ }
++
++ return ret;
++}
++
++/*!
++ * @brief Allocate memory for overlay.
++ * @param cam Pointer to cam_data structure
++ * @return Zero on success, others on failure
++ */
++static int prp_vf_mem_alloc(cam_data * cam)
++{
++ int i;
++
++ for (i = 0; i < 2; i++) {
++ cam->vf_bufs_size[i] = cam->win.w.width * cam->win.w.height * 2;
++ cam->vf_bufs_vaddr[i] = dma_alloc_coherent(0,
++ cam->vf_bufs_size[i],
++ &cam->vf_bufs[i],
++ GFP_DMA |
++ GFP_KERNEL);
++ if (!cam->vf_bufs_vaddr[i]) {
++ pr_debug("Failed to alloc memory for vf.\n");
++ prp_vf_mem_free(cam);
++ return -1;
++ }
++
++ g_vaddr_vfbuf[i] =
++ ioremap_cached(cam->vf_bufs[i], cam->vf_bufs_size[i]);
++ if (!g_vaddr_vfbuf[i]) {
++ pr_debug("Failed to ioremap_cached() for vf.\n");
++ prp_vf_mem_free(cam);
++ return -1;
++ }
++ }
++
++ return 0;
++}
++
++/*!
++ * @brief Free memory for overlay.
++ * @param cam Pointer to cam_data structure
++ * @return Zero on success, others on failure
++ */
++static void prp_vf_mem_free(cam_data * cam)
++{
++ int i;
++
++ for (i = 0; i < 2; i++) {
++ if (cam->vf_bufs_vaddr[i]) {
++ dma_free_coherent(0,
++ cam->vf_bufs_size[i],
++ cam->vf_bufs_vaddr[i],
++ cam->vf_bufs[i]);
++ }
++ cam->vf_bufs[i] = 0;
++ cam->vf_bufs_vaddr[i] = 0;
++ cam->vf_bufs_size[i] = 0;
++ if (g_vaddr_vfbuf[i]) {
++ iounmap(g_vaddr_vfbuf[i]);
++ g_vaddr_vfbuf[i] = 0;
++ }
++ }
++}
++
++/*!
++ * @brief Allocate intermediate memory for overlay rotation/mirroring.
++ * @param cam Pointer to cam_data structure
++ * @return Zero on success, others on failure
++ */
++static int prp_rot_mem_alloc(cam_data * cam)
++{
++ int i;
++
++ for (i = 0; i < 2; i++) {
++ cam->rot_vf_buf_size[i] =
++ cam->win.w.width * cam->win.w.height * 2;
++ cam->rot_vf_bufs_vaddr[i] =
++ dma_alloc_coherent(0, cam->rot_vf_buf_size[i],
++ &cam->rot_vf_bufs[i],
++ GFP_DMA | GFP_KERNEL);
++ if (!cam->rot_vf_bufs_vaddr[i]) {
++ pr_debug("Failed to alloc memory for vf rotation.\n");
++ prp_rot_mem_free(cam);
++ return -1;
++ }
++
++ g_vaddr_rotbuf[i] =
++ ioremap_cached(cam->rot_vf_bufs[i],
++ cam->rot_vf_buf_size[i]);
++ if (!g_vaddr_rotbuf[i]) {
++ pr_debug
++ ("Failed to ioremap_cached() for rotation buffer.\n");
++ prp_rot_mem_free(cam);
++ return -1;
++ }
++ }
++
++ return 0;
++}
++
++/*!
++ * @brief Free intermedaite memory for overlay rotation/mirroring.
++ * @param cam Pointer to cam_data structure
++ * @return Zero on success, others on failure
++ */
++static void prp_rot_mem_free(cam_data * cam)
++{
++ int i;
++
++ for (i = 0; i < 2; i++) {
++ if (cam->rot_vf_bufs_vaddr[i]) {
++ dma_free_coherent(0,
++ cam->rot_vf_buf_size[i],
++ cam->rot_vf_bufs_vaddr[i],
++ cam->rot_vf_bufs[i]);
++ }
++ cam->rot_vf_bufs[i] = 0;
++ cam->rot_vf_bufs_vaddr[i] = 0;
++ cam->rot_vf_buf_size[i] = 0;
++ if (g_vaddr_rotbuf[i]) {
++ iounmap(g_vaddr_rotbuf[i]);
++ g_vaddr_rotbuf[i] = 0;
++ }
++ }
++}
++
++/*!
++ * @brief Start overlay (view finder).
++ * @param private Pointer to cam_data structure
++ * @return Zero on success, others on failure
++ */
++static int prp_vf_start(void *private)
++{
++ cam_data *cam = (cam_data *) private;
++
++ if (cam->v4l2_fb.flags == V4L2_FBUF_FLAG_OVERLAY) {
++ prp_vf_mem_free(cam);
++ if (prp_vf_mem_alloc(cam)) {
++ pr_info("Error to allocate vf buffer\n");
++ return -ENOMEM;
++ }
++ }
++
++ if (cam->rotation != V4L2_MXC_ROTATE_NONE) {
++ prp_rot_mem_free(cam);
++ if (prp_rot_mem_alloc(cam)) {
++ pr_info("Error to allocate rotation buffer\n");
++ prp_vf_mem_free(cam);
++ return -ENOMEM;
++ }
++ }
++
++ if (prp_v4l2_cfg(&g_prp_cfg, cam)) {
++ prp_vf_mem_free(cam);
++ prp_rot_mem_free(cam);
++ return -1;
++ }
++
++ csi_enable_mclk(CSI_MCLK_VF, true, true);
++ prphw_reset();
++
++ if (prphw_cfg(&g_prp_cfg)) {
++ prp_vf_mem_free(cam);
++ prp_rot_mem_free(cam);
++ return -1;
++ }
++ g_vfbuf = g_rotbuf = 0;
++ tasklet_init(&prp_vf_tasklet, rotation, (unsigned long)private);
++
++ prphw_enable(cam->capture_on ? (PRP_CHANNEL_1 | PRP_CHANNEL_2)
++ : PRP_CHANNEL_1);
++
++ return 0;
++}
++
++/*!
++ * @brief Stop overlay (view finder).
++ * @param private Pointer to cam_data structure
++ * @return Zero on success, others on failure
++ */
++static int prp_vf_stop(void *private)
++{
++ cam_data *cam = (cam_data *) private;
++
++ prphw_disable(PRP_CHANNEL_1);
++
++ csi_enable_mclk(CSI_MCLK_VF, false, false);
++ tasklet_kill(&prp_vf_tasklet);
++
++ if (cam->v4l2_fb.flags == V4L2_FBUF_FLAG_OVERLAY) {
++ struct fb_gwinfo gwinfo;
++
++ /* Disable graphic window */
++ gwinfo.enabled = 0;
++ mx2_gw_set(&gwinfo);
++
++ prp_vf_mem_free(cam);
++ }
++ prp_rot_mem_free(cam);
++ if (g_vaddr_fb) {
++ iounmap(g_vaddr_fb);
++ g_vaddr_fb = 0;
++ }
++
++ return 0;
++}
++
++/*!
++ * @brief Setup overlay functions.
++ * @param private Pointer to cam_data structure
++ * @return Zero on success, others on failure
++ */
++int prp_vf_select(void *private)
++{
++ int ret = 0;
++ cam_data *cam = (cam_data *) private;
++
++ if (cam) {
++ cam->vf_start_sdc = prp_vf_start;
++ cam->vf_stop_sdc = prp_vf_stop;
++ cam->overlay_active = false;
++ } else
++ ret = -EIO;
++
++ return ret;
++}
++
++/*!
++ * @brief Uninstall overlay functions.
++ * @param private Pointer to cam_data structure
++ * @return Zero on success, others on failure
++ */
++int prp_vf_deselect(void *private)
++{
++ int ret = 0;
++ cam_data *cam = (cam_data *) private;
++
++ ret = prp_vf_stop(private);
++
++ if (cam) {
++ cam->vf_start_sdc = NULL;
++ cam->vf_stop_sdc = NULL;
++ }
++
++ return ret;
++}
++
++/*!
++ * @brief Start still picture capture.
++ * @param private Pointer to cam_data structure
++ * @return Zero on success, others on failure
++ */
++static int prp_still_start(void *private)
++{
++ cam_data *cam = (cam_data *) private;
++
++ g_still_on = 1;
++ g_prp_cfg.ch2_ptr = (unsigned int)cam->still_buf;
++ g_prp_cfg.ch2_ptr2 = 0;
++
++ if (prp_v4l2_cfg(&g_prp_cfg, cam))
++ return -1;
++
++ csi_enable_mclk(CSI_MCLK_RAW, true, true);
++ prphw_reset();
++
++ if (prphw_cfg(&g_prp_cfg)) {
++ g_still_on = 0;
++ return -1;
++ }
++
++ prphw_enable(cam->overlay_on ? (PRP_CHANNEL_1 | PRP_CHANNEL_2)
++ : PRP_CHANNEL_2);
++
++ return 0;
++}
++
++/*!
++ * @brief Stop still picture capture.
++ * @param private Pointer to cam_data structure
++ * @return Zero on success, others on failure
++ */
++static int prp_still_stop(void *private)
++{
++ prphw_disable(PRP_CHANNEL_2);
++
++ csi_enable_mclk(CSI_MCLK_RAW, false, false);
++
++ g_still_on = 0;
++
++ return 0;
++}
++
++/*!
++ * @brief Setup functions for still picture capture.
++ * @param private Pointer to cam_data structure
++ * @return Zero on success, others on failure
++ */
++int prp_still_select(void *private)
++{
++ cam_data *cam = (cam_data *) private;
++
++ if (cam) {
++ cam->csi_start = prp_still_start;
++ cam->csi_stop = prp_still_stop;
++ }
++
++ return 0;
++}
++
++/*!
++ * @brief Uninstall functions for still picture capture.
++ * @param private Pointer to cam_data structure
++ * @return Zero on success, others on failure
++ */
++int prp_still_deselect(void *private)
++{
++ cam_data *cam = (cam_data *) private;
++ int err = 0;
++
++ err = prp_still_stop(cam);
++
++ if (cam) {
++ cam->csi_start = NULL;
++ cam->csi_stop = NULL;
++ }
++
++ return err;
++}
++
++/*!
++ * @brief Perform software rotation or mirroring
++ * @param private Argument passed to the tasklet
++ */
++static void rotation(unsigned long private)
++{
++ char *src, *dst;
++ int width, height, s_stride, d_stride;
++ int size;
++ cam_data *cam = (cam_data *) private;
++
++ src = g_vaddr_rotbuf[g_rotbuf];
++ size = cam->rot_vf_buf_size[g_rotbuf];
++
++ if ((cam->rotation == V4L2_MXC_ROTATE_90_RIGHT)
++ || (cam->rotation == V4L2_MXC_ROTATE_90_RIGHT_VFLIP)
++ || (cam->rotation == V4L2_MXC_ROTATE_90_RIGHT_HFLIP)
++ || (cam->rotation == V4L2_MXC_ROTATE_90_LEFT)) {
++ width = cam->win.w.height;
++ height = cam->win.w.width;
++ s_stride = cam->win.w.height << 1;
++ } else {
++ width = cam->win.w.width;
++ height = cam->win.w.height;
++ s_stride = cam->win.w.width << 1;
++ }
++
++ if (cam->v4l2_fb.flags == V4L2_FBUF_FLAG_OVERLAY) {
++ dst = g_vaddr_vfbuf[g_vfbuf];
++ d_stride = cam->win.w.width << 1;
++ } else { /* The destination is the framebuffer */
++ struct fb_info *fb = cam->overlay_fb;
++ if (!fb)
++ return;
++ dst = g_vaddr_fb;
++ dst += cam->win.w.top * fb->var.xres_virtual
++ * (fb->var.bits_per_pixel >> 3)
++ + cam->win.w.left * (fb->var.bits_per_pixel >> 3);
++ d_stride = fb->var.xres_virtual << 1;
++ }
++
++ /*
++ * Invalidate the data in cache before performing the SW rotaion
++ * or mirroring in case the image size is less than QVGA. For image
++ * larger than QVGA it is not invalidated becase the invalidation
++ * will consume much time while we don't see any artifacts on the
++ * output if we don't perform invalidation for them.
++ * Similarly we don't flush the data after SW rotation/mirroring.
++ */
++ if (size < 320 * 240 * 2)
++ dmac_inv_range(src, src + size);
++ switch (cam->rotation) {
++ case V4L2_MXC_ROTATE_VERT_FLIP:
++ opl_vmirror_u16(src, s_stride, width, height, dst, d_stride);
++ break;
++ case V4L2_MXC_ROTATE_HORIZ_FLIP:
++ opl_hmirror_u16(src, s_stride, width, height, dst, d_stride);
++ break;
++ case V4L2_MXC_ROTATE_180:
++ opl_rotate180_u16(src, s_stride, width, height, dst, d_stride);
++ break;
++ case V4L2_MXC_ROTATE_90_RIGHT:
++ opl_rotate90_u16(src, s_stride, width, height, dst, d_stride);
++ break;
++ case V4L2_MXC_ROTATE_90_RIGHT_VFLIP:
++ opl_rotate90_vmirror_u16(src, s_stride, width, height, dst,
++ d_stride);
++ break;
++ case V4L2_MXC_ROTATE_90_RIGHT_HFLIP:
++ /* ROTATE_90_RIGHT_HFLIP = ROTATE_270_RIGHT_VFLIP */
++ opl_rotate270_vmirror_u16(src, s_stride, width, height, dst,
++ d_stride);
++ break;
++ case V4L2_MXC_ROTATE_90_LEFT:
++ opl_rotate270_u16(src, s_stride, width, height, dst, d_stride);
++ break;
++ default:
++ return;
++ }
++
++ /* Config and display the graphic window */
++ if (cam->v4l2_fb.flags == V4L2_FBUF_FLAG_OVERLAY) {
++ struct fb_gwinfo gwinfo;
++
++ gwinfo.enabled = 1;
++ gwinfo.alpha_value = 255;
++ gwinfo.ck_enabled = 0;
++ gwinfo.xpos = cam->win.w.left;
++ gwinfo.ypos = cam->win.w.top;
++ gwinfo.xres = cam->win.w.width;
++ gwinfo.yres = cam->win.w.height;
++ gwinfo.xres_virtual = cam->win.w.width;
++ gwinfo.vs_reversed = 0;
++ gwinfo.base = (unsigned long)cam->vf_bufs[g_vfbuf];
++ mx2_gw_set(&gwinfo);
++
++ g_vfbuf = g_vfbuf ? 0 : 1;
++ }
++}
++
++/*
++ * @brief Check if the resize ratio is supported based on the input and output
++ * dimension
++ * @param input input dimension
++ * @param output output dimension
++ * @return output dimension (should equal the parameter *output*)
++ * -1 on failure
++ */
++static int check_simple(scale_t * scale, int input, int output, int ch)
++{
++ unsigned short int_out; /* PrP internel width or height */
++ unsigned short orig_out = output;
++
++ if (prp_scale(scale, input, output, input, &orig_out, &int_out, ch))
++ return -1; /* resize failed */
++ else
++ return int_out;
++}
++
++/*!
++ * @brief Check if the resize ratio is supported by PrP channel 1
++ * @param cfg Pointer to emma_prp_cfg structure
++ * @return Zero on success, others on failure
++ */
++static int prp_resize_check_ch1(emma_prp_cfg * cfg)
++{
++ int in_w, in_h, ch1_w, ch1_h, ch2_w, ch2_h, w, h;
++ scale_t *pscale = &cfg->scale[0]; /* Ch1 width resize coeff */
++
++ if (cfg->ch1_pix == PRP_PIX1_UNUSED)
++ return 0;
++
++ in_w = cfg->in_width;
++ in_h = cfg->in_height;
++ ch1_w = cfg->ch1_width;
++ ch1_h = cfg->ch1_height;
++ ch2_w = cfg->ch2_width;
++ ch2_h = cfg->ch2_height;
++
++ /*
++ * For channel 1, try parallel resize first. If the resize
++ * ratio is not exactly supported, try cascade resize. If it
++ * still fails, use parallel resize but with rounded value.
++ */
++ w = check_simple(pscale, in_w, ch1_w, PRP_CHANNEL_1);
++ h = check_simple(pscale + 1, in_h, ch1_h, PRP_CHANNEL_1);
++
++ if ((in_w <= ch1_w * MAX_TBL) && (in_h <= MAX_TBL * ch1_h))
++ goto exit_parallel;
++
++ if (cfg->ch2_pix != PRP_PIX2_UNUSED) {
++ /*
++ * Channel 2 is already used. The pscale is still pointing
++ * to ch1 resize coeff for temporary use.
++ */
++ if ((ch2_w * MAX_TBL <= ch1_w) && (ch2_h * MAX_TBL <= ch1_h)) {
++ w = check_simple(pscale, ch2_w, ch1_w, PRP_CHANNEL_1);
++ h = check_simple(pscale + 1, ch2_h, ch1_h,
++ PRP_CHANNEL_1);
++ goto exit_cascade;
++ }
++ } else {
++ /*
++ * Try cascade resize for width, width is multiple of 2.
++ * Channel 2 is not used. So we have more values to pick
++ * for channel 2 resize.
++ */
++ if (in_w * MAX_TBL > ch1_w) {
++ for (w = in_w / 2; w > ch1_w; w /= 2) {
++ /* Ch1 width resize */
++ if (check_simple
++ (pscale, w, ch1_w, PRP_CHANNEL_1) < 0)
++ continue;
++ /* Ch2 width resize */
++ ch2_w =
++ check_simple(pscale + 2, in_w, w,
++ PRP_CHANNEL_1);
++ if (ch2_w < 0) {
++ w = in_w / MAX_TBL;
++ continue;
++ }
++ check_simple(pscale, ch2_w, ch1_w,
++ PRP_CHANNEL_1);
++ break;
++ }
++ } else {
++ w = check_simple(pscale, in_w, ch1_w, PRP_CHANNEL_1);
++ ch2_w = check_simple(pscale + 2, w, w, PRP_CHANNEL_1);
++ }
++ if (ch2_w >= ch1_w) {
++ if (in_h * MAX_TBL > ch1_h) {
++ /* try cascade resize for height */
++ for (h = in_h / 2; h > ch1_h; h /= 2) {
++ /* Ch2 height resize */
++ if (check_simple
++ (pscale + 1, h, ch1_h,
++ PRP_CHANNEL_1) < 0)
++ continue;
++ /* Ch1 height resize */
++ ch2_h =
++ check_simple(pscale + 3, in_h, h,
++ PRP_CHANNEL_1);
++ if (ch2_w < 0) {
++ h = in_h / MAX_TBL;
++ continue;
++ }
++ check_simple(pscale + 1, ch2_h, ch1_h,
++ PRP_CHANNEL_1);
++ break;
++ }
++ } else {
++ h = check_simple(pscale + 1, in_h, ch1_h,
++ PRP_CHANNEL_1);
++ ch2_h =
++ check_simple(pscale + 3, h, h,
++ PRP_CHANNEL_1);
++ }
++
++ goto exit_cascade;
++ }
++ }
++
++ pr_debug("Ch1 resize error.\n");
++ return -1;
++
++ exit_parallel:
++ cfg->ch1_scale.algo |= PRP_ALGO_BYPASS;
++ pr_debug("ch1 parallel resize.\n");
++ pr_debug("original width = %d internel width = %d\n", ch1_w, w);
++ pr_debug("original height = %d internel height = %d\n", ch1_h, h);
++ return 0;
++
++ exit_cascade:
++ cfg->ch1_scale.algo &= ~PRP_ALGO_BYPASS;
++ pr_debug("ch1 cascade resize.\n");
++ pr_debug("[width] in : ch2 : ch1=%d : %d : %d\n", in_w, ch2_w, ch1_w);
++ pr_debug("[height] in : ch2 : ch1=%d : %d : %d\n", in_h, ch2_h, ch1_h);
++ return 0;
++}
++
++/*!
++ * @brief Check if the resize ratio is supported by PrP channel 2
++ * @param cfg Pointer to emma_prp_cfg structure
++ * @return Zero on success, others on failure
++ */
++static int prp_resize_check_ch2(emma_prp_cfg * cfg)
++{
++ int w, h;
++ scale_t *pscale = &cfg->scale[2]; /* Ch2 width resize coeff */
++
++ if (cfg->ch2_pix == PRP_PIX2_UNUSED)
++ return 0;
++
++ w = check_simple(pscale, cfg->in_width, cfg->ch2_width, PRP_CHANNEL_2);
++ h = check_simple(pscale + 1, cfg->in_height, cfg->ch2_height,
++ PRP_CHANNEL_2);
++ if ((w != -1) && (h != -1)) {
++ pr_debug("Ch2 resize.\n");
++ pr_debug("Original width = %d internel width = %d\n",
++ cfg->ch2_width, w);
++ pr_debug("Original height = %d internel height = %d\n",
++ cfg->ch2_height, h);
++ return 0;
++ } else {
++ pr_debug("Ch2 resize error.\n");
++ return -1;
++ }
++}
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/capture/mx27_v4l2_capture.c linux-2.6.28-karo/drivers/media/video/mxc/capture/mx27_v4l2_capture.c
+--- linux-2.6.28/drivers/media/video/mxc/capture/mx27_v4l2_capture.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/capture/mx27_v4l2_capture.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,2288 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @file mx27_v4l2_capture.c
++ *
++ * @brief MX27 Video For Linux 2 driver
++ *
++ * @ingroup MXC_V4L2_CAPTURE
++ */
++
++#include <linux/module.h>
++#include <linux/init.h>
++#include <linux/fs.h>
++#include <linux/slab.h>
++#include <linux/ctype.h>
++#include <linux/pagemap.h>
++#include <linux/vmalloc.h>
++#include <linux/types.h>
++#include <linux/fb.h>
++#include <linux/pci.h>
++#include <linux/platform_device.h>
++#include <linux/version.h>
++#include <media/v4l2-dev.h>
++#include <asm/io.h>
++#include <asm/semaphore.h>
++
++#include "mxc_v4l2_capture.h"
++#include "mx27_prp.h"
++#include "mx27_csi.h"
++
++static int csi_mclk_flag_backup;
++static int video_nr = -1;
++cam_data *g_cam;
++EXPORT_SYMBOL(g_cam);
++
++static int dq_intr_cnt = 0;
++static int dq_timeout_cnt = 0;
++static int empty_wq_cnt = 0;
++struct workqueue_struct *v4l2_work;
++
++static void prp_reset(struct work_struct *unused)
++{
++ struct mxc_v4l_frame *done_frame, *ready_frame, *temp_frame;
++
++ g_cam->ping_pong_csi = 0;
++ g_cam->enc_enable(g_cam);
++ if (!list_empty(&g_cam->working_q)) {
++ done_frame =
++ list_entry(g_cam->working_q.next, struct mxc_v4l_frame,
++ queue);
++ list_del(g_cam->working_q.next);
++ if (!list_empty(&g_cam->working_q)) {
++ temp_frame =
++ list_entry(g_cam->working_q.next,
++ struct mxc_v4l_frame, queue);
++ list_del(g_cam->working_q.next);
++ list_add_tail(&temp_frame->queue, &g_cam->working_q);
++ g_cam->enc_update_eba(temp_frame->paddress,
++ &g_cam->ping_pong_csi);
++ }
++ list_add_tail(&done_frame->queue, &g_cam->working_q);
++ g_cam->enc_update_eba(done_frame->paddress,
++ &g_cam->ping_pong_csi);
++ pr_debug("prp_reset - working_q\n");
++ } else if (list_empty(&g_cam->ready_q)) {
++ prphw_disable(PRP_CHANNEL_2);
++ g_cam->skip_frame++;
++ } else {
++ ready_frame =
++ list_entry(g_cam->ready_q.next, struct mxc_v4l_frame,
++ queue);
++ list_del(g_cam->ready_q.next);
++ list_add_tail(&ready_frame->queue, &g_cam->working_q);
++ g_cam->enc_update_eba(ready_frame->paddress,
++ &g_cam->ping_pong_csi);
++ }
++ g_cam->overflow = 0;
++ wake_up_interruptible(&g_cam->overflow_queue);
++}
++
++DECLARE_WORK(prp_reset_work, prp_reset);
++/*!
++ * Free frame buffers
++ *
++ * @param cam Structure cam_data *
++ *
++ * @return status 0 success.
++ */
++static int mxc_free_frame_buf(cam_data * cam)
++{
++ int i;
++
++ for (i = 0; i < FRAME_NUM; i++) {
++ if (cam->frame[i].vaddress != 0) {
++ dma_free_coherent(0,
++ cam->frame[i].buffer.length,
++ cam->frame[i].vaddress,
++ cam->frame[i].paddress);
++ cam->frame[i].vaddress = 0;
++ }
++ }
++
++ return 0;
++}
++
++/*!
++ * Allocate frame buffers
++ *
++ * @param cam Structure cam_data *
++ *
++ * @param count int number of buffer need to allocated
++ *
++ * @return status -0 Successfully allocated a buffer, -ENOBUFS failed.
++ */
++static int mxc_allocate_frame_buf(cam_data * cam, int count)
++{
++ int i;
++
++ for (i = 0; i < count; i++) {
++ cam->frame[i].vaddress = dma_alloc_coherent(0,
++ PAGE_ALIGN(cam->v2f.
++ fmt.pix.
++ sizeimage),
++ &cam->frame[i].
++ paddress,
++ GFP_DMA |
++ GFP_KERNEL);
++ if (cam->frame[i].vaddress == 0) {
++ pr_debug("mxc_allocate_frame_buf failed.\n");
++ mxc_free_frame_buf(cam);
++ return -ENOBUFS;
++ }
++ cam->frame[i].buffer.index = i;
++ cam->frame[i].buffer.flags = V4L2_BUF_FLAG_MAPPED;
++ cam->frame[i].buffer.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
++ cam->frame[i].buffer.length =
++ PAGE_ALIGN(cam->v2f.fmt.pix.sizeimage);
++ cam->frame[i].buffer.memory = V4L2_MEMORY_MMAP;
++ cam->frame[i].buffer.m.offset = cam->frame[i].paddress;
++ cam->frame[i].index = i;
++ }
++
++ return 0;
++}
++
++/*!
++ * Free frame buffers status
++ *
++ * @param cam Structure cam_data *
++ *
++ * @return none
++ */
++static void mxc_free_frames(cam_data * cam)
++{
++ int i;
++
++ for (i = 0; i < FRAME_NUM; i++) {
++ cam->frame[i].buffer.flags = V4L2_BUF_FLAG_MAPPED;
++ }
++
++ cam->enc_counter = 0;
++ cam->skip_frame = 0;
++ INIT_LIST_HEAD(&cam->ready_q);
++ INIT_LIST_HEAD(&cam->working_q);
++ INIT_LIST_HEAD(&cam->done_q);
++}
++
++/*!
++ * Return the buffer status
++ *
++ * @param cam Structure cam_data *
++ * @param buf Structure v4l2_buffer *
++ *
++ * @return status 0 success, EINVAL failed.
++ */
++static int mxc_v4l2_buffer_status(cam_data * cam, struct v4l2_buffer *buf)
++{
++ /* check range */
++ if (buf->index < 0 || buf->index >= FRAME_NUM) {
++ pr_debug("mxc_v4l2_buffer_status buffers not allocated\n");
++ return -EINVAL;
++ }
++
++ memcpy(buf, &(cam->frame[buf->index].buffer), sizeof(*buf));
++ return 0;
++}
++
++/*!
++ * start the encoder job
++ *
++ * @param cam structure cam_data *
++ *
++ * @return status 0 Success
++ */
++static int mxc_streamon(cam_data * cam)
++{
++ struct mxc_v4l_frame *frame;
++ int err = 0;
++
++ if (!cam)
++ return -EIO;
++
++ if (list_empty(&cam->ready_q)) {
++ printk(KERN_ERR "mxc_streamon buffer not been queued yet\n");
++ return -EINVAL;
++ }
++
++ cam->capture_pid = current->pid;
++
++ if (cam->enc_enable) {
++ err = cam->enc_enable(cam);
++ if (err != 0) {
++ return err;
++ }
++ }
++
++ cam->ping_pong_csi = 0;
++ cam->overflow = 0;
++ if (cam->enc_update_eba) {
++ frame =
++ list_entry(cam->ready_q.next, struct mxc_v4l_frame, queue);
++ list_del(cam->ready_q.next);
++ list_add_tail(&frame->queue, &cam->working_q);
++ err = cam->enc_update_eba(frame->paddress, &cam->ping_pong_csi);
++
++ frame =
++ list_entry(cam->ready_q.next, struct mxc_v4l_frame, queue);
++ list_del(cam->ready_q.next);
++ list_add_tail(&frame->queue, &cam->working_q);
++ err |=
++ cam->enc_update_eba(frame->paddress, &cam->ping_pong_csi);
++ } else {
++ return -EINVAL;
++ }
++
++ return err;
++}
++
++/*!
++ * Shut down the encoder job
++ *
++ * @param cam structure cam_data *
++ *
++ * @return status 0 Success
++ */
++static int mxc_streamoff(cam_data * cam)
++{
++ int err = 0;
++
++ if (!cam)
++ return -EIO;
++
++ if (cam->enc_disable) {
++ err = cam->enc_disable(cam);
++ }
++ mxc_free_frames(cam);
++ return err;
++}
++
++/*!
++ * Valid whether the palette is supported
++ *
++ * @param palette pixel format
++ *
++ * @return 0 if failed
++ */
++static inline int valid_mode(u32 palette)
++{
++ /*
++ * MX27 PrP channel 2 supports YUV444, but YUV444 is not
++ * defined by V4L2 :(
++ */
++ return ((palette == V4L2_PIX_FMT_YUYV) ||
++ (palette == V4L2_PIX_FMT_YUV420));
++}
++
++/*!
++ * Valid and adjust the overlay window size, position
++ *
++ * @param cam structure cam_data *
++ * @param win struct v4l2_window *
++ *
++ * @return 0
++ */
++static int verify_preview(cam_data * cam, struct v4l2_window *win)
++{
++ if (cam->output >= num_registered_fb) {
++ pr_debug("verify_preview No matched.\n");
++ return -EINVAL;
++ }
++ cam->overlay_fb = (struct fb_info *)registered_fb[cam->output];
++
++ /* TODO: suppose 16bpp, 4 bytes alignment */
++ win->w.left &= ~0x1;
++
++ if (win->w.width + win->w.left > cam->overlay_fb->var.xres)
++ win->w.width = cam->overlay_fb->var.xres - win->w.left;
++ if (win->w.height + win->w.top > cam->overlay_fb->var.yres)
++ win->w.height = cam->overlay_fb->var.yres - win->w.top;
++
++ /*
++ * TODO: suppose 16bpp. Rounded down to a multiple of 2 pixels for
++ * width according to PrP limitations.
++ */
++ if ((cam->rotation == V4L2_MXC_ROTATE_90_RIGHT)
++ || (cam->rotation == V4L2_MXC_ROTATE_90_RIGHT_VFLIP)
++ || (cam->rotation == V4L2_MXC_ROTATE_90_RIGHT_HFLIP)
++ || (cam->rotation == V4L2_MXC_ROTATE_90_LEFT))
++ win->w.height &= ~0x1;
++ else
++ win->w.width &= ~0x1;
++
++ return 0;
++}
++
++/*!
++ * start the viewfinder job
++ *
++ * @param cam structure cam_data *
++ *
++ * @return status 0 Success
++ */
++static int start_preview(cam_data * cam)
++{
++ int err = 0;
++
++ err = prp_vf_select(cam);
++ if (err != 0)
++ return err;
++
++ cam->overlay_pid = current->pid;
++ err = cam->vf_start_sdc(cam);
++
++ return err;
++}
++
++/*!
++ * shut down the viewfinder job
++ *
++ * @param cam structure cam_data *
++ *
++ * @return status 0 Success
++ */
++static int stop_preview(cam_data * cam)
++{
++ int err = 0;
++
++ err = prp_vf_deselect(cam);
++ return err;
++}
++
++/*!
++ * V4L2 - mxc_v4l2_g_fmt function
++ *
++ * @param cam structure cam_data *
++ *
++ * @param f structure v4l2_format *
++ *
++ * @return status 0 success, EINVAL failed
++ */
++static int mxc_v4l2_g_fmt(cam_data * cam, struct v4l2_format *f)
++{
++ int retval = 0;
++
++ switch (f->type) {
++ case V4L2_BUF_TYPE_VIDEO_CAPTURE:
++ f->fmt.pix.width = cam->v2f.fmt.pix.width;
++ f->fmt.pix.height = cam->v2f.fmt.pix.height;
++ f->fmt.pix.sizeimage = cam->v2f.fmt.pix.sizeimage;
++ f->fmt.pix.pixelformat = cam->v2f.fmt.pix.pixelformat;
++ f->fmt.pix.bytesperline = cam->v2f.fmt.pix.bytesperline;
++ f->fmt.pix.colorspace = V4L2_COLORSPACE_JPEG;
++ retval = 0;
++ break;
++ case V4L2_BUF_TYPE_VIDEO_OVERLAY:
++ f->fmt.win = cam->win;
++ break;
++ default:
++ retval = -EINVAL;
++ }
++ return retval;
++}
++
++/*!
++ * V4L2 - mxc_v4l2_s_fmt function
++ *
++ * @param cam structure cam_data *
++ *
++ * @param f structure v4l2_format *
++ *
++ * @return status 0 success, EINVAL failed
++ */
++static int mxc_v4l2_s_fmt(cam_data * cam, struct v4l2_format *f)
++{
++ int retval = 0;
++ int size = 0;
++ int bytesperline = 0;
++
++ switch (f->type) {
++ case V4L2_BUF_TYPE_VIDEO_CAPTURE:
++ if (!valid_mode(f->fmt.pix.pixelformat)) {
++ pr_debug("mxc_v4l2_s_fmt: format not supported\n");
++ retval = -EINVAL;
++ }
++
++ if (cam->rotation != V4L2_MXC_ROTATE_NONE)
++ pr_debug("mxc_v4l2_s_fmt: capture rotation ignored\n");
++
++ switch (f->fmt.pix.pixelformat) {
++ case V4L2_PIX_FMT_YUYV:
++ f->fmt.pix.width &= ~0x1; /* Multiple of 2 */
++ size = f->fmt.pix.width * f->fmt.pix.height * 2;
++ bytesperline = f->fmt.pix.width * 2;
++ break;
++ case V4L2_PIX_FMT_YUV420:
++ f->fmt.pix.width &= ~0x7; /* Multiple of 8 */
++ f->fmt.pix.height &= ~0x1; /* Multiple of 2 */
++ size = f->fmt.pix.width * f->fmt.pix.height * 3 / 2;
++ bytesperline = f->fmt.pix.width * 3 / 2;
++ break;
++ default:
++ /* Suppose it's YUV444 or 32bpp */
++ size = f->fmt.pix.width * f->fmt.pix.height * 4;
++ bytesperline = f->fmt.pix.width * 4;
++ pr_info("mxc_v4l2_s_fmt: default assume"
++ " to be YUV444 interleaved.\n");
++ break;
++ }
++
++ if (f->fmt.pix.bytesperline < bytesperline) {
++ f->fmt.pix.bytesperline = bytesperline;
++ } else {
++ bytesperline = f->fmt.pix.bytesperline;
++ }
++
++ if (f->fmt.pix.sizeimage > size) {
++ pr_debug("mxc_v4l2_s_fmt: sizeimage bigger than"
++ " needed.\n");
++ size = f->fmt.pix.sizeimage;
++ }
++ f->fmt.pix.sizeimage = size;
++
++ cam->v2f.fmt.pix.sizeimage = size;
++ cam->v2f.fmt.pix.bytesperline = bytesperline;
++ cam->v2f.fmt.pix.width = f->fmt.pix.width;
++ cam->v2f.fmt.pix.height = f->fmt.pix.height;
++ cam->v2f.fmt.pix.pixelformat = f->fmt.pix.pixelformat;
++ retval = 0;
++ break;
++ case V4L2_BUF_TYPE_VIDEO_OVERLAY:
++ retval = verify_preview(cam, &f->fmt.win);
++ cam->win = f->fmt.win;
++ break;
++ default:
++ retval = -EINVAL;
++ }
++ return retval;
++}
++
++/*!
++ * get control param
++ *
++ * @param cam structure cam_data *
++ *
++ * @param c structure v4l2_control *
++ *
++ * @return status 0 success, EINVAL failed
++ */
++static int mxc_get_v42l_control(cam_data * cam, struct v4l2_control *c)
++{
++ int status = 0;
++
++ switch (c->id) {
++ case V4L2_CID_HFLIP:
++ c->value = cam->rotation;
++ break;
++ case V4L2_CID_VFLIP:
++ c->value = cam->rotation;
++ break;
++ case V4L2_CID_MXC_ROT:
++ c->value = cam->rotation;
++ break;
++ case V4L2_CID_BRIGHTNESS:
++ c->value = cam->bright;
++ break;
++ case V4L2_CID_HUE:
++ c->value = cam->hue;
++ break;
++ case V4L2_CID_CONTRAST:
++ c->value = cam->contrast;
++ break;
++ case V4L2_CID_SATURATION:
++ c->value = cam->saturation;
++ break;
++ case V4L2_CID_RED_BALANCE:
++ c->value = cam->red;
++ break;
++ case V4L2_CID_BLUE_BALANCE:
++ c->value = cam->blue;
++ break;
++ case V4L2_CID_BLACK_LEVEL:
++ c->value = cam->ae_mode;
++ break;
++ case V4L2_CID_AUTO_WHITE_BALANCE:
++ c->value = cam->awb_enable;
++ break;
++ case V4L2_CID_AUTOGAIN:
++ c->value = cam->ae_enable;
++ break;
++ case V4L2_CID_MXC_FLICKER:
++ c->value = cam->flicker_ctrl;
++ break;
++ default:
++ status = -EINVAL;
++ }
++ return status;
++}
++
++/*!
++ * V4L2 - set_control function
++ * V4L2_CID_MXC_ROT is the extention for rotation/mirroring.
++ *
++ * @param cam structure cam_data *
++ *
++ * @param c structure v4l2_control *
++ *
++ * @return status 0 success, EINVAL failed
++ */
++static int mxc_set_v42l_control(cam_data * cam, struct v4l2_control *c)
++{
++ switch (c->id) {
++ case V4L2_CID_HFLIP:
++ if (c->value == 1) {
++ if ((cam->rotation != V4L2_MXC_ROTATE_VERT_FLIP) &&
++ (cam->rotation != V4L2_MXC_ROTATE_180))
++ cam->rotation = V4L2_MXC_ROTATE_HORIZ_FLIP;
++ else
++ cam->rotation = V4L2_MXC_ROTATE_180;
++ } else {
++ if (cam->rotation == V4L2_MXC_ROTATE_HORIZ_FLIP)
++ cam->rotation = V4L2_MXC_ROTATE_NONE;
++ else if (cam->rotation == V4L2_MXC_ROTATE_180)
++ cam->rotation = V4L2_MXC_ROTATE_VERT_FLIP;
++ }
++ break;
++ case V4L2_CID_VFLIP:
++ if (c->value == 1) {
++ if ((cam->rotation != V4L2_MXC_ROTATE_HORIZ_FLIP) &&
++ (cam->rotation != V4L2_MXC_ROTATE_180))
++ cam->rotation = V4L2_MXC_ROTATE_VERT_FLIP;
++ else
++ cam->rotation = V4L2_MXC_ROTATE_180;
++ } else {
++ if (cam->rotation == V4L2_MXC_ROTATE_VERT_FLIP)
++ cam->rotation = V4L2_MXC_ROTATE_NONE;
++ if (cam->rotation == V4L2_MXC_ROTATE_180)
++ cam->rotation = V4L2_MXC_ROTATE_HORIZ_FLIP;
++ }
++ break;
++ case V4L2_CID_MXC_ROT:
++ switch (c->value) {
++ case V4L2_MXC_ROTATE_NONE:
++ case V4L2_MXC_ROTATE_VERT_FLIP:
++ case V4L2_MXC_ROTATE_HORIZ_FLIP:
++ case V4L2_MXC_ROTATE_180:
++ case V4L2_MXC_ROTATE_90_RIGHT:
++ case V4L2_MXC_ROTATE_90_RIGHT_VFLIP:
++ case V4L2_MXC_ROTATE_90_RIGHT_HFLIP:
++ case V4L2_MXC_ROTATE_90_LEFT:
++ cam->rotation = c->value;
++ break;
++ default:
++ return -EINVAL;
++ }
++ break;
++ case V4L2_CID_HUE:
++ cam->hue = c->value;
++ break;
++ case V4L2_CID_CONTRAST:
++ cam->contrast = c->value;
++ break;
++ case V4L2_CID_BRIGHTNESS:
++ cam->bright = c->value;
++ case V4L2_CID_SATURATION:
++ cam->saturation = c->value;
++ case V4L2_CID_RED_BALANCE:
++ cam->red = c->value;
++ case V4L2_CID_BLUE_BALANCE:
++ cam->blue = c->value;
++ csi_enable_mclk(CSI_MCLK_I2C, true, true);
++ cam->cam_sensor->set_color(cam->bright, cam->saturation,
++ cam->red, cam->green, cam->blue);
++ csi_enable_mclk(CSI_MCLK_I2C, false, false);
++ break;
++ case V4L2_CID_BLACK_LEVEL:
++ cam->ae_mode = c->value & 0x03;
++ csi_enable_mclk(CSI_MCLK_I2C, true, true);
++ if (cam->cam_sensor->set_ae_mode)
++ cam->cam_sensor->set_ae_mode(cam->ae_mode);
++ csi_enable_mclk(CSI_MCLK_I2C, false, false);
++ break;
++ case V4L2_CID_MXC_FLASH:
++ break;
++ case V4L2_CID_AUTOGAIN:
++ cam->ae_enable = c->value;
++ csi_enable_mclk(CSI_MCLK_I2C, true, true);
++ if (cam->cam_sensor->set_ae)
++ cam->cam_sensor->set_ae(cam->ae_enable);
++ csi_enable_mclk(CSI_MCLK_I2C, false, false);
++ break;
++ case V4L2_CID_MXC_GAIN_LIMIT:
++ cam->ae_limit = c->value;
++ csi_enable_mclk(CSI_MCLK_I2C, true, true);
++ if (cam->cam_sensor->set_ae_limit)
++ cam->cam_sensor->set_ae_limit(cam->ae_limit);
++ csi_enable_mclk(CSI_MCLK_I2C, false, false);
++ break;
++ case V4L2_CID_AUTO_WHITE_BALANCE:
++ cam->awb_enable = c->value;
++ csi_enable_mclk(CSI_MCLK_I2C, true, true);
++ if (cam->cam_sensor->set_awb)
++ cam->cam_sensor->set_awb(cam->awb_enable);
++ csi_enable_mclk(CSI_MCLK_I2C, false, false);
++ break;
++ case V4L2_CID_MXC_FLICKER:
++ cam->flicker_ctrl = c->value;
++ csi_enable_mclk(CSI_MCLK_I2C, true, true);
++ if (cam->cam_sensor->flicker_control)
++ cam->cam_sensor->flicker_control(cam->flicker_ctrl);
++ csi_enable_mclk(CSI_MCLK_I2C, false, false);
++ break;
++ default:
++ return -EINVAL;
++ }
++ return 0;
++}
++
++/*!
++ * V4L2 - mxc_v4l2_s_param function
++ *
++ * @param cam structure cam_data *
++ *
++ * @param parm structure v4l2_streamparm *
++ *
++ * @return status 0 success, EINVAL failed
++ */
++static int mxc_v4l2_s_param(cam_data * cam, struct v4l2_streamparm *parm)
++{
++ sensor_interface *param;
++ csi_signal_cfg_t csi_param;
++
++ if (parm->type != V4L2_BUF_TYPE_VIDEO_CAPTURE) {
++ pr_debug("mxc_v4l2_s_param invalid type\n");
++ return -EINVAL;
++ }
++
++ if (parm->parm.capture.timeperframe.denominator >
++ cam->standard.frameperiod.denominator) {
++ pr_debug("mxc_v4l2_s_param frame rate %d larger "
++ "than standard supported %d\n",
++ parm->parm.capture.timeperframe.denominator,
++ cam->standard.frameperiod.denominator);
++ return -EINVAL;
++ }
++
++ cam->streamparm.parm.capture.capability = V4L2_CAP_TIMEPERFRAME;
++
++ csi_enable_mclk(CSI_MCLK_I2C, true, true);
++ param = cam->cam_sensor->config
++ (&parm->parm.capture.timeperframe.denominator,
++ parm->parm.capture.capturemode);
++ csi_enable_mclk(CSI_MCLK_I2C, false, false);
++
++ cam->streamparm.parm.capture.timeperframe =
++ parm->parm.capture.timeperframe;
++
++ if ((parm->parm.capture.capturemode != 0) &&
++ (parm->parm.capture.capturemode != V4L2_MODE_HIGHQUALITY)) {
++ pr_debug("mxc_v4l2_s_param frame un-supported capture mode\n");
++ return -EINVAL;
++ }
++
++ if (parm->parm.capture.capturemode ==
++ cam->streamparm.parm.capture.capturemode) {
++ return 0;
++ }
++
++ /* resolution changed, so need to re-program the CSI */
++ csi_param.sens_clksrc = 0;
++ csi_param.clk_mode = param->clk_mode;
++ csi_param.pixclk_pol = param->pixclk_pol;
++ csi_param.data_width = param->data_width;
++ csi_param.data_pol = param->data_pol;
++ csi_param.ext_vsync = param->ext_vsync;
++ csi_param.Vsync_pol = param->Vsync_pol;
++ csi_param.Hsync_pol = param->Hsync_pol;
++ csi_init_interface(param->width, param->height, param->pixel_fmt,
++ csi_param);
++
++ if (parm->parm.capture.capturemode != V4L2_MODE_HIGHQUALITY) {
++ cam->streamparm.parm.capture.capturemode = 0;
++ } else {
++ cam->streamparm.parm.capture.capturemode =
++ V4L2_MODE_HIGHQUALITY;
++ cam->streamparm.parm.capture.extendedmode =
++ parm->parm.capture.extendedmode;
++ cam->streamparm.parm.capture.readbuffers = 1;
++ }
++ return 0;
++}
++
++/*!
++ * Dequeue one V4L capture buffer
++ *
++ * @param cam structure cam_data *
++ * @param buf structure v4l2_buffer *
++ *
++ * @return status 0 success, EINVAL invalid frame number,
++ * ETIME timeout, ERESTARTSYS interrupted by user
++ */
++static int mxc_v4l_dqueue(cam_data * cam, struct v4l2_buffer *buf)
++{
++ int retval = 0;
++ struct mxc_v4l_frame *frame;
++
++ if (!wait_event_interruptible_timeout(cam->enc_queue,
++ cam->enc_counter != 0, 10 * HZ)) {
++ if ((dq_timeout_cnt & 0x1f) == 0)
++ printk(KERN_ERR
++ "mxc_v4l_dqueue timeout enc_counter %x\n",
++ cam->enc_counter);
++ dq_timeout_cnt++;
++ return -ETIME;
++ } else if (signal_pending(current)) {
++ if (dq_intr_cnt == 0)
++ printk(KERN_ERR
++ "mxc_v4l_dqueue() interrupt received %d\n",
++ dq_intr_cnt);
++ dq_intr_cnt++;
++ return -ERESTARTSYS;
++ }
++
++ cam->enc_counter--;
++
++ frame = list_entry(cam->done_q.next, struct mxc_v4l_frame, queue);
++ list_del(cam->done_q.next);
++ if (frame->buffer.flags & V4L2_BUF_FLAG_DONE) {
++ frame->buffer.flags &= ~V4L2_BUF_FLAG_DONE;
++ } else if (frame->buffer.flags & V4L2_BUF_FLAG_QUEUED) {
++ printk(KERN_ERR "VIDIOC_DQBUF: Buffer not filled.\n");
++ frame->buffer.flags &= ~V4L2_BUF_FLAG_QUEUED;
++ retval = -EINVAL;
++ } else if ((frame->buffer.flags & 0x7) == V4L2_BUF_FLAG_MAPPED) {
++ printk(KERN_ERR "VIDIOC_DQBUF: Buffer not queued.\n");
++ retval = -EINVAL;
++ }
++
++ buf->bytesused = cam->v2f.fmt.pix.sizeimage;
++ buf->index = frame->index;
++ buf->flags = frame->buffer.flags;
++ buf->m = cam->frame[frame->index].buffer.m;
++ return retval;
++}
++
++/*!
++ * Get the current attached camera device
++ *
++ * @param inode struct i2c_client *
++ *
++ * @param int int * p_input_index
++ *
++ * @return 0 success, ENODEV for invalid device instance,
++ * -1 for other errors.
++ */
++static int mxc_get_video_input(cam_data * cam)
++{
++ int retval = 0;
++ csi_enable_mclk(CSI_MCLK_I2C, true, true);
++ retval = cam->cam_sensor->get_status();
++ csi_enable_mclk(CSI_MCLK_I2C, false, false);
++ return retval;
++}
++
++/*!
++ * V4L interface - open function
++ *
++ * @param inode structure inode *
++ * @param file structure file *
++ *
++ * @return status 0 success, ENODEV invalid device instance,
++ * ENODEV timeout, ERESTARTSYS interrupted by user
++ */
++static int mxc_v4l_open(struct inode *inode, struct file *file)
++{
++ sensor_interface *param;
++ csi_signal_cfg_t csi_param;
++ struct video_device *dev = video_devdata(file);
++ cam_data *cam = dev->priv;
++ int err = 0;
++
++ dq_intr_cnt = 0;
++ dq_timeout_cnt = 0;
++ empty_wq_cnt = 0;
++ if (!cam) {
++ pr_info("Internal error, cam_data not found!\n");
++ return -ENODEV;
++ }
++
++ err = mxc_get_video_input(cam);
++ if (0 != err)
++ return -ENODEV;
++
++ if (down_interruptible(&cam->busy_lock))
++ return -EINTR;
++
++ if (signal_pending(current))
++ goto oops;
++
++ if (cam->open_count++ == 0) {
++ wait_event_interruptible(cam->power_queue,
++ cam->low_power == false);
++
++ err = prp_enc_select(cam);
++
++ cam->enc_counter = 0;
++ cam->skip_frame = 0;
++ INIT_LIST_HEAD(&cam->ready_q);
++ INIT_LIST_HEAD(&cam->working_q);
++ INIT_LIST_HEAD(&cam->done_q);
++
++ csi_enable_mclk(CSI_MCLK_I2C, true, true);
++ param = cam->cam_sensor->reset();
++ if (param == NULL) {
++ cam->open_count--;
++ csi_enable_mclk(CSI_MCLK_I2C, false, false);
++ err = -ENODEV;
++ goto oops;
++ }
++ csi_param.sens_clksrc = 0;
++ csi_param.clk_mode = param->clk_mode;
++ csi_param.pixclk_pol = param->pixclk_pol;
++ csi_param.data_width = param->data_width;
++ csi_param.data_pol = param->data_pol;
++ csi_param.ext_vsync = param->ext_vsync;
++ csi_param.Vsync_pol = param->Vsync_pol;
++ csi_param.Hsync_pol = param->Hsync_pol;
++ csi_init_interface(param->width, param->height,
++ param->pixel_fmt, csi_param);
++ cam->cam_sensor->get_color(&cam->bright, &cam->saturation,
++ &cam->red, &cam->green, &cam->blue);
++ if (cam->cam_sensor->get_ae_mode)
++ cam->cam_sensor->get_ae_mode(&cam->ae_mode);
++ cam->cam_sensor->get_control_params(&cam->ae_enable,
++ &cam->awb_enable,
++ &cam->flicker_ctrl);
++ csi_enable_mclk(CSI_MCLK_I2C, false, false);
++ prp_init(cam);
++
++ }
++
++ file->private_data = dev;
++ oops:
++ up(&cam->busy_lock);
++ return err;
++}
++
++/*!
++ * V4L interface - close function
++ *
++ * @param inode struct inode *
++ * @param file struct file *
++ *
++ * @return 0 success
++ */
++static int mxc_v4l_close(struct inode *inode, struct file *file)
++{
++ struct video_device *dev = video_devdata(file);
++ int err = 0;
++ cam_data *cam = dev->priv;
++
++ /* for the case somebody hit the ctrl C */
++ if (cam->overlay_pid == current->pid) {
++ err = stop_preview(cam);
++ cam->overlay_on = false;
++ }
++ if (cam->capture_pid == current->pid) {
++ err |= mxc_streamoff(cam);
++ cam->capture_on = false;
++ wake_up_interruptible(&cam->enc_queue);
++ wake_up_interruptible(&cam->overflow_queue);
++ }
++
++ if (--cam->open_count == 0) {
++ wait_event_interruptible(cam->power_queue,
++ cam->low_power == false);
++ pr_debug("mxc_v4l_close: release resource\n");
++
++ err |= prp_enc_deselect(cam);
++
++ mxc_free_frame_buf(cam);
++ file->private_data = NULL;
++
++ /* capture off */
++ wake_up_interruptible(&cam->enc_queue);
++ wake_up_interruptible(&cam->overflow_queue);
++ mxc_free_frames(cam);
++ cam->enc_counter++;
++ prp_exit(cam);
++ }
++
++ return err;
++}
++
++#ifdef CONFIG_VIDEO_MXC_CSI_DMA
++#include <asm/arch/dma.h>
++
++#define CSI_DMA_STATUS_IDLE 0 /* DMA is not started */
++#define CSI_DMA_STATUS_WORKING 1 /* DMA is transfering the data */
++#define CSI_DMA_STATUS_DONE 2 /* One frame completes successfully */
++#define CSI_DMA_STATUS_ERROR 3 /* Error occurs during the DMA */
++
++/*
++ * Sometimes the start of the DMA is not synchronized with the CSI
++ * SOF (Start of Frame) interrupt which will lead to incorrect
++ * captured image. In this case the driver will re-try capturing
++ * another frame. The following macro defines the maximum re-try
++ * times.
++ */
++#define CSI_DMA_RETRY 8
++
++/*
++ * Size of the physical contiguous memory area used to hold image data
++ * transfered by DMA. It can be less than the size of the image data.
++ */
++#define CSI_MEM_SIZE (1024 * 600)
++
++/* Number of bytes for one DMA transfer */
++#define CSI_DMA_LENGTH (1024 * 200)
++
++static int g_dma_channel = 0;
++static int g_dma_status = CSI_DMA_STATUS_DONE;
++static volatile int g_dma_completed; /* number of completed DMA transfers */
++static volatile int g_dma_copied; /* number of copied DMA transfers */
++static struct tasklet_struct g_dma_tasklet;
++static char *g_user_buf; /* represents the buf passed by read() */
++static int g_user_count; /* represents the count passed by read() */
++
++/*!
++ * @brief setup the DMA to transfer data
++ * There may be more than one DMA to transfer the whole image. Those
++ * DMAs work like chain. This function is used to setup the DMA in
++ * case there is enough space to hold the data.
++ * @param data pointer to the cam structure
++ */
++static void mxc_csi_dma_chaining(void *data)
++{
++ cam_data *cam = (cam_data *) data;
++ int count, chained = 0;
++ int max_dma = CSI_MEM_SIZE / CSI_DMA_LENGTH;
++ mxc_dma_requestbuf_t dma_request;
++
++ while (chained * CSI_DMA_LENGTH < g_user_count) {
++ /*
++ * Calculate how many bytes the DMA should transfer. It may
++ * be less than CSI_DMA_LENGTH if the DMA is the last one.
++ */
++ if ((chained + 1) * CSI_DMA_LENGTH > g_user_count)
++ count = g_user_count - chained * CSI_DMA_LENGTH;
++ else
++ count = CSI_DMA_LENGTH;
++ pr_debug("%s() DMA chained count = %d\n", __FUNCTION__, count);
++
++ /* Config DMA */
++ memset(&dma_request, 0, sizeof(mxc_dma_requestbuf_t));
++ dma_request.dst_addr = cam->still_buf
++ + (chained % max_dma) * CSI_DMA_LENGTH;
++ dma_request.src_addr = (dma_addr_t) CSI_CSIRXFIFO_PHYADDR;
++ dma_request.num_of_bytes = count;
++ mxc_dma_config(g_dma_channel, &dma_request, 1,
++ MXC_DMA_MODE_READ);
++
++ chained++;
++ }
++}
++
++/*!
++ * @brief Copy image data from physical contiguous memory to user space buffer
++ * Once the data are copied, there will be more spare space in the
++ * physical contiguous memory to receive data from DMA.
++ * @param data pointer to the cam structure
++ */
++static void mxc_csi_dma_task(unsigned long data)
++{
++ cam_data *cam = (cam_data *) data;
++ int count;
++ int max_dma = CSI_MEM_SIZE / CSI_DMA_LENGTH;
++
++ while (g_dma_copied < g_dma_completed) {
++ /*
++ * Calculate how many bytes the DMA has transfered. It may
++ * be less than CSI_DMA_LENGTH if the DMA is the last one.
++ */
++ if ((g_dma_copied + 1) * CSI_DMA_LENGTH > g_user_count)
++ count = g_user_count - g_dma_copied * CSI_DMA_LENGTH;
++ else
++ count = CSI_DMA_LENGTH;
++ if (copy_to_user(g_user_buf + g_dma_copied * CSI_DMA_LENGTH,
++ cam->still_buf_vaddr + (g_dma_copied % max_dma)
++ * CSI_DMA_LENGTH, count))
++ pr_debug("Warning: some bytes not copied\n");
++
++ g_dma_copied++;
++ }
++
++ /* If the whole image has been captured */
++ if (g_dma_copied * CSI_DMA_LENGTH >= g_user_count) {
++ cam->still_counter++;
++ wake_up_interruptible(&cam->still_queue);
++ }
++
++ pr_debug("%s() DMA completed = %d copied = %d\n",
++ __FUNCTION__, g_dma_completed, g_dma_copied);
++}
++
++/*!
++ * @brief DMA interrupt callback function
++ * @param data pointer to the cam structure
++ * @param error DMA error flag
++ * @param count number of bytes transfered by the DMA
++ */
++static void mxc_csi_dma_callback(void *data, int error, unsigned int count)
++{
++ cam_data *cam = (cam_data *) data;
++ int max_dma = CSI_MEM_SIZE / CSI_DMA_LENGTH;
++ unsigned long lock_flags;
++
++ spin_lock_irqsave(&cam->int_lock, lock_flags);
++
++ g_dma_completed++;
++
++ if (error != MXC_DMA_DONE) {
++ g_dma_status = CSI_DMA_STATUS_ERROR;
++ pr_debug("%s() DMA error\n", __FUNCTION__);
++ }
++
++ /* If the whole image has been captured */
++ if ((g_dma_status != CSI_DMA_STATUS_ERROR)
++ && (g_dma_completed * CSI_DMA_LENGTH >= g_user_count))
++ g_dma_status = CSI_DMA_STATUS_DONE;
++
++ if ((g_dma_status == CSI_DMA_STATUS_WORKING) &&
++ (g_dma_completed >= g_dma_copied + max_dma)) {
++ g_dma_status = CSI_DMA_STATUS_ERROR;
++ pr_debug("%s() Previous buffer over written\n", __FUNCTION__);
++ }
++
++ /* Schedule the tasklet */
++ tasklet_schedule(&g_dma_tasklet);
++
++ spin_unlock_irqrestore(&cam->int_lock, lock_flags);
++
++ pr_debug("%s() count = %d bytes\n", __FUNCTION__, count);
++}
++
++/*!
++ * @brief CSI interrupt callback function
++ * @param data pointer to the cam structure
++ * @param status CSI interrupt status
++ */
++static void mxc_csi_irq_callback(void *data, unsigned long status)
++{
++ cam_data *cam = (cam_data *) data;
++ unsigned long lock_flags;
++
++ spin_lock_irqsave(&cam->int_lock, lock_flags);
++
++ /* Wait for SOF (Start of Frame) interrupt to sync the image */
++ if (status & BIT_SOF_INT) {
++ if (g_dma_status == CSI_DMA_STATUS_IDLE) {
++ /* Start DMA transfer to capture image */
++ mxc_dma_enable(g_dma_channel);
++ g_dma_status = CSI_DMA_STATUS_WORKING;
++ pr_debug("%s() DMA started.\n", __FUNCTION__);
++ } else if (g_dma_status == CSI_DMA_STATUS_WORKING) {
++ /*
++ * Another SOF occurs during DMA transfer. In this
++ * case the image is not synchronized so need to
++ * report error and probably try again.
++ */
++ g_dma_status = CSI_DMA_STATUS_ERROR;
++ pr_debug("%s() Image is not synchronized with DMA - "
++ "SOF before DMA completes\n", __FUNCTION__);
++ }
++ }
++
++ spin_unlock_irqrestore(&cam->int_lock, lock_flags);
++
++ pr_debug("%s() g_dma_status = %d\n", __FUNCTION__, g_dma_status);
++}
++
++/*!
++ * V4L interface - read function
++ *
++ * @param file struct file *
++ * @param read buf char *
++ * @param count size_t
++ * @param ppos structure loff_t *
++ *
++ * @return bytes read
++ */
++static ssize_t
++mxc_v4l_read(struct file *file, char *buf, size_t count, loff_t * ppos)
++{
++ int err = 0;
++ struct video_device *dev = video_devdata(file);
++ cam_data *cam = dev->priv;
++ int retry = CSI_DMA_RETRY;
++
++ g_user_buf = buf;
++
++ if (down_interruptible(&cam->busy_lock))
++ return -EINTR;
++
++ /* Video capture and still image capture are exclusive */
++ if (cam->capture_on == true) {
++ err = -EBUSY;
++ goto exit0;
++ }
++
++ /* The CSI-DMA can not do CSC */
++ if (cam->v2f.fmt.pix.pixelformat != V4L2_PIX_FMT_YUYV) {
++ pr_info("mxc_v4l_read support YUYV pixel format only\n");
++ err = -EINVAL;
++ goto exit0;
++ }
++
++ /* The CSI-DMA can not do resize or crop */
++ if ((cam->v2f.fmt.pix.width != cam->crop_bounds.width)
++ || (cam->v2f.fmt.pix.height != cam->crop_bounds.height)) {
++ pr_info("mxc_v4l_read resize is not supported\n");
++ pr_info("supported image size width = %d height = %d\n",
++ cam->crop_bounds.width, cam->crop_bounds.height);
++ err = -EINVAL;
++ goto exit0;
++ }
++ if ((cam->crop_current.left != cam->crop_bounds.left)
++ || (cam->crop_current.width != cam->crop_bounds.width)
++ || (cam->crop_current.top != cam->crop_bounds.top)
++ || (cam->crop_current.height != cam->crop_bounds.height)) {
++ pr_info("mxc_v4l_read cropping is not supported\n");
++ err = -EINVAL;
++ goto exit0;
++ }
++
++ cam->still_buf_vaddr = dma_alloc_coherent(0,
++ PAGE_ALIGN(CSI_MEM_SIZE),
++ &cam->still_buf,
++ GFP_DMA | GFP_KERNEL);
++
++ if (!cam->still_buf_vaddr) {
++ pr_info("mxc_v4l_read failed at allocate still_buf\n");
++ err = -ENOBUFS;
++ goto exit0;
++ }
++
++ /* Initialize DMA */
++ g_dma_channel = mxc_dma_request(MXC_DMA_CSI_RX, "CSI RX DMA");
++ if (g_dma_channel < 0) {
++ pr_debug("mxc_v4l_read failed to request DMA channel\n");
++ err = -EIO;
++ goto exit1;
++ }
++
++ err = mxc_dma_callback_set(g_dma_channel,
++ mxc_csi_dma_callback,
++ cam);
++ if (err != 0) {
++ pr_debug("mxc_v4l_read failed to set DMA callback\n");
++ err = -EIO;
++ goto exit2;
++ }
++
++ g_user_buf = buf;
++ if (cam->v2f.fmt.pix.sizeimage < count)
++ g_user_count = cam->v2f.fmt.pix.sizeimage;
++ else
++ g_user_count = count & ~0x3;
++
++ tasklet_init(&g_dma_tasklet, mxc_csi_dma_task, (unsigned long)cam);
++ g_dma_status = CSI_DMA_STATUS_DONE;
++ csi_set_callback(mxc_csi_irq_callback, cam);
++ csi_enable_prpif(0);
++
++ /* clear current SOF first */
++ csi_clear_status(BIT_SOF_INT);
++ csi_enable_mclk(CSI_MCLK_RAW, true, true);
++
++ do {
++ g_dma_completed = g_dma_copied = 0;
++ mxc_csi_dma_chaining(cam);
++ cam->still_counter = 0;
++ g_dma_status = CSI_DMA_STATUS_IDLE;
++
++ if (!wait_event_interruptible_timeout(cam->still_queue,
++ cam->still_counter != 0,
++ 10 * HZ)) {
++ pr_info("mxc_v4l_read timeout counter %x\n",
++ cam->still_counter);
++ err = -ETIME;
++ goto exit3;
++ }
++
++ if (g_dma_status == CSI_DMA_STATUS_DONE)
++ break;
++
++ if (retry-- == 0)
++ break;
++
++ pr_debug("Now retry image capture\n");
++ } while (1);
++
++ if (g_dma_status != CSI_DMA_STATUS_DONE)
++ err = -EIO;
++
++ exit3:
++ csi_enable_prpif(1);
++ g_dma_status = CSI_DMA_STATUS_DONE;
++ csi_set_callback(0, 0);
++ csi_enable_mclk(CSI_MCLK_RAW, false, false);
++ tasklet_kill(&g_dma_tasklet);
++
++ exit2:
++ mxc_dma_free(g_dma_channel);
++
++ exit1:
++ dma_free_coherent(0, PAGE_ALIGN(CSI_MEM_SIZE),
++ cam->still_buf_vaddr, cam->still_buf);
++ cam->still_buf = 0;
++
++ exit0:
++ up(&cam->busy_lock);
++ if (err < 0)
++ return err;
++ else
++ return g_user_count;
++}
++#else
++/*!
++ * V4L interface - read function
++ *
++ * @param file struct file *
++ * @param read buf char *
++ * @param count size_t
++ * @param ppos structure loff_t *
++ *
++ * @return bytes read
++ */
++static ssize_t
++mxc_v4l_read(struct file *file, char *buf, size_t count, loff_t * ppos)
++{
++ int err = 0;
++ u8 *v_address;
++ struct video_device *dev = video_devdata(file);
++ cam_data *cam = dev->priv;
++
++ if (down_interruptible(&cam->busy_lock))
++ return -EINTR;
++
++ /* Video capture and still image capture are exclusive */
++ if (cam->capture_on == true) {
++ err = -EBUSY;
++ goto exit0;
++ }
++
++ v_address = dma_alloc_coherent(0,
++ PAGE_ALIGN(cam->v2f.fmt.pix.sizeimage),
++ &cam->still_buf, GFP_DMA | GFP_KERNEL);
++
++ if (!v_address) {
++ pr_info("mxc_v4l_read failed at allocate still_buf\n");
++ err = -ENOBUFS;
++ goto exit0;
++ }
++
++ if (prp_still_select(cam)) {
++ err = -EIO;
++ goto exit1;
++ }
++
++ cam->still_counter = 0;
++ if (cam->csi_start(cam)) {
++ err = -EIO;
++ goto exit2;
++ }
++
++ if (!wait_event_interruptible_timeout(cam->still_queue,
++ cam->still_counter != 0,
++ 10 * HZ)) {
++ pr_info("mxc_v4l_read timeout counter %x\n",
++ cam->still_counter);
++ err = -ETIME;
++ goto exit2;
++ }
++ err = copy_to_user(buf, v_address, cam->v2f.fmt.pix.sizeimage);
++
++ exit2:
++ prp_still_deselect(cam);
++
++ exit1:
++ dma_free_coherent(0, cam->v2f.fmt.pix.sizeimage, v_address,
++ cam->still_buf);
++ cam->still_buf = 0;
++
++ exit0:
++ up(&cam->busy_lock);
++ if (err < 0)
++ return err;
++ else
++ return (cam->v2f.fmt.pix.sizeimage - err);
++}
++#endif /* CONFIG_VIDEO_MXC_CSI_DMA */
++
++/*!
++ * V4L interface - ioctl function
++ *
++ * @param inode struct inode *
++ *
++ * @param file struct file *
++ *
++ * @param ioctlnr unsigned int
++ *
++ * @param arg void *
++ *
++ * @return 0 success, ENODEV for invalid device instance,
++ * -1 for other errors.
++ */
++static int
++mxc_v4l_do_ioctl(struct inode *inode, struct file *file,
++ unsigned int ioctlnr, void *arg)
++{
++ struct video_device *dev = video_devdata(file);
++ cam_data *cam = dev->priv;
++ int retval = 0;
++ unsigned long lock_flags;
++
++ if (!cam)
++ return -EBADF;
++
++ wait_event_interruptible(cam->power_queue, cam->low_power == false);
++ /* make this _really_ smp-safe */
++ if (down_interruptible(&cam->busy_lock))
++ return -EBUSY;
++
++ switch (ioctlnr) {
++ /*!
++ * V4l2 VIDIOC_QUERYCAP ioctl
++ */
++ case VIDIOC_QUERYCAP:{
++ struct v4l2_capability *cap = arg;
++ strcpy(cap->driver, "mxc_v4l2");
++ cap->version = KERNEL_VERSION(0, 1, 11);
++ cap->capabilities = V4L2_CAP_VIDEO_CAPTURE |
++ V4L2_CAP_VIDEO_OVERLAY | V4L2_CAP_STREAMING
++ | V4L2_CAP_READWRITE;
++ cap->card[0] = '\0';
++ cap->bus_info[0] = '\0';
++ retval = 0;
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_G_FMT ioctl
++ */
++ case VIDIOC_G_FMT:{
++ struct v4l2_format *gf = arg;
++ retval = mxc_v4l2_g_fmt(cam, gf);
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_S_FMT ioctl
++ */
++ case VIDIOC_S_FMT:{
++ struct v4l2_format *sf = arg;
++ retval = mxc_v4l2_s_fmt(cam, sf);
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_REQBUFS ioctl
++ */
++ case VIDIOC_REQBUFS:{
++ struct v4l2_requestbuffers *req = arg;
++ int i;
++ if (req->count > FRAME_NUM) {
++ pr_info("VIDIOC_REQBUFS: not enough buffer\n");
++ req->count = FRAME_NUM;
++ }
++
++ if ((req->type != V4L2_BUF_TYPE_VIDEO_CAPTURE) ||
++ ((req->memory != V4L2_MEMORY_MMAP)
++ && (req->memory != V4L2_MEMORY_USERPTR))) {
++ pr_debug("VIDIOC_REQBUFS: wrong buffer type\n");
++ retval = -EINVAL;
++ break;
++ }
++
++ mxc_streamoff(cam);
++ mxc_free_frame_buf(cam);
++
++ if (req->memory == V4L2_MEMORY_MMAP)
++ retval =
++ mxc_allocate_frame_buf(cam, req->count);
++ else if (req->memory == V4L2_MEMORY_USERPTR) {
++ for (i = 0; i < req->count; i++) {
++ cam->frame[i].vaddress = 0;
++ cam->frame[i].buffer.index = i;
++ cam->frame[i].buffer.type =
++ V4L2_BUF_TYPE_VIDEO_CAPTURE;
++ cam->frame[i].buffer.flags = 0;
++ cam->frame[i].buffer.length = 0;
++ cam->frame[i].buffer.memory =
++ V4L2_MEMORY_USERPTR;
++ cam->frame[i].buffer.m.offset = 0;
++ cam->frame[i].index = i;
++ }
++ }
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_QUERYBUF ioctl
++ */
++ case VIDIOC_QUERYBUF:{
++ struct v4l2_buffer *buf = arg;
++ int index = buf->index;
++
++ if (buf->type != V4L2_BUF_TYPE_VIDEO_CAPTURE) {
++ pr_debug
++ ("VIDIOC_QUERYBUFS: wrong buffer type\n");
++ retval = -EINVAL;
++ break;
++ }
++
++ memset(buf, 0, sizeof(buf));
++ buf->index = index;
++
++ down(&cam->param_lock);
++ retval = mxc_v4l2_buffer_status(cam, buf);
++ up(&cam->param_lock);
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_QBUF ioctl
++ */
++ case VIDIOC_QBUF:{
++ struct v4l2_buffer *buf = arg;
++ int index = buf->index;
++
++ pr_debug("VIDIOC_QBUF: %d\n", buf->index);
++ wait_event_interruptible(cam->overflow_queue,
++ cam->overflow == 0);
++ spin_lock_irqsave(&cam->int_lock, lock_flags);
++ if (cam->frame[index].buffer.memory == V4L2_MEMORY_MMAP) {
++ if ((cam->frame[index].buffer.flags & 0x7) ==
++ V4L2_BUF_FLAG_MAPPED) {
++ cam->frame[index].buffer.flags |=
++ V4L2_BUF_FLAG_QUEUED;
++ if (cam->skip_frame > 0) {
++ prphw_enable(PRP_CHANNEL_2);
++ list_add_tail(&cam->
++ frame[index].
++ queue,
++ &cam->working_q);
++ retval =
++ cam->enc_update_eba(cam->
++ frame
++ [index].
++ paddress,
++ &cam->
++ ping_pong_csi);
++ cam->skip_frame = 0;
++ } else {
++ list_add_tail(&cam->
++ frame[index].
++ queue,
++ &cam->ready_q);
++ }
++ } else if (cam->frame[index].buffer.flags &
++ V4L2_BUF_FLAG_QUEUED) {
++ pr_debug
++ ("VIDIOC_QBUF: buffer already queued\n");
++ } else if (cam->frame[index].buffer.
++ flags & V4L2_BUF_FLAG_DONE) {
++ pr_debug
++ ("VIDIOC_QBUF: overwrite done buffer.\n");
++ cam->frame[index].buffer.flags &=
++ ~V4L2_BUF_FLAG_DONE;
++ cam->frame[index].buffer.flags |=
++ V4L2_BUF_FLAG_QUEUED;
++ }
++ } else {
++ cam->frame[index].buffer.flags =
++ (V4L2_BUF_FLAG_MAPPED |
++ V4L2_BUF_FLAG_QUEUED);
++ cam->frame[index].buffer.m.offset =
++ buf->m.offset;
++ cam->frame[index].paddress = buf->m.offset;
++ cam->frame[index].buffer.length = buf->length;
++ if (cam->skip_frame > 0) {
++ prphw_enable(PRP_CHANNEL_2);
++ list_add_tail(&cam->frame[index].queue,
++ &cam->working_q);
++ retval =
++ cam->enc_update_eba(cam->
++ frame[index].
++ paddress,
++ &cam->
++ ping_pong_csi);
++ cam->skip_frame = 0;
++ } else {
++ list_add_tail(&cam->frame[index].queue,
++ &cam->ready_q);
++ }
++ }
++ buf->flags = cam->frame[index].buffer.flags;
++ spin_unlock_irqrestore(&cam->int_lock, lock_flags);
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_DQBUF ioctl
++ */
++ case VIDIOC_DQBUF:{
++ struct v4l2_buffer *buf = arg;
++
++ retval = mxc_v4l_dqueue(cam, buf);
++
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_STREAMON ioctl
++ */
++ case VIDIOC_STREAMON:{
++ cam->capture_on = true;
++ retval = mxc_streamon(cam);
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_STREAMOFF ioctl
++ */
++ case VIDIOC_STREAMOFF:{
++ retval = mxc_streamoff(cam);
++ cam->capture_on = false;
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_G_CTRL ioctl
++ */
++ case VIDIOC_G_CTRL:{
++ retval = mxc_get_v42l_control(cam, arg);
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_S_CTRL ioctl
++ */
++ case VIDIOC_S_CTRL:{
++ retval = mxc_set_v42l_control(cam, arg);
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_CROPCAP ioctl
++ */
++ case VIDIOC_CROPCAP:{
++ struct v4l2_cropcap *cap = arg;
++
++ if (cap->type != V4L2_BUF_TYPE_VIDEO_CAPTURE &&
++ cap->type != V4L2_BUF_TYPE_VIDEO_OVERLAY) {
++ retval = -EINVAL;
++ break;
++ }
++ cap->bounds = cam->crop_bounds;
++ cap->defrect = cam->crop_defrect;
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_G_CROP ioctl
++ */
++ case VIDIOC_G_CROP:{
++ struct v4l2_crop *crop = arg;
++
++ if (crop->type != V4L2_BUF_TYPE_VIDEO_CAPTURE &&
++ crop->type != V4L2_BUF_TYPE_VIDEO_OVERLAY) {
++ retval = -EINVAL;
++ break;
++ }
++ crop->c = cam->crop_current;
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_S_CROP ioctl
++ */
++ case VIDIOC_S_CROP:{
++ struct v4l2_crop *crop = arg;
++ struct v4l2_rect *b = &cam->crop_bounds;
++ int i;
++
++ if (crop->type != V4L2_BUF_TYPE_VIDEO_CAPTURE &&
++ crop->type != V4L2_BUF_TYPE_VIDEO_OVERLAY) {
++ retval = -EINVAL;
++ break;
++ }
++
++ crop->c.top = (crop->c.top < b->top) ? b->top
++ : crop->c.top;
++ if (crop->c.top > b->top + b->height)
++ crop->c.top = b->top + b->height - 1;
++ if (crop->c.height > b->top + b->height - crop->c.top)
++ crop->c.height =
++ b->top + b->height - crop->c.top;
++
++ crop->c.left = (crop->c.left < b->left) ? b->left
++ : crop->c.left;
++ if (crop->c.left > b->left + b->width)
++ crop->c.left = b->left + b->width - 1;
++ if (crop->c.width > b->left - crop->c.left + b->width)
++ crop->c.width =
++ b->left - crop->c.left + b->width;
++
++ crop->c.width &= ~0x1;
++
++ /*
++ * MX27 PrP limitation:
++ * The right spare space (CSI_FRAME_X_SIZE
++ * - SOURCE_LINE_STRIDE - PICTURE_X_SIZE)) must be
++ * multiple of 32.
++ * So we tune the crop->c.left value to the closest
++ * desired cropping value and meet the PrP requirement.
++ */
++ i = ((b->left + b->width)
++ - (crop->c.left + crop->c.width)) % 32;
++ if (i <= 16) {
++ if (crop->c.left + crop->c.width + i
++ <= b->left + b->width)
++ crop->c.left += i;
++ else if (crop->c.left - (32 - i) >= b->left)
++ crop->c.left -= 32 - i;
++ else {
++ retval = -EINVAL;
++ break;
++ }
++ } else {
++ if (crop->c.left - (32 - i) >= b->left)
++ crop->c.left -= 32 - i;
++ else if (crop->c.left + crop->c.width + i
++ <= b->left + b->width)
++ crop->c.left += i;
++ else {
++ retval = -EINVAL;
++ break;
++ }
++ }
++
++ cam->crop_current = crop->c;
++
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_OVERLAY ioctl
++ */
++ case VIDIOC_OVERLAY:{
++ int *on = arg;
++ if (*on) {
++ cam->overlay_on = true;
++ retval = start_preview(cam);
++ }
++ if (!*on) {
++ retval = stop_preview(cam);
++ cam->overlay_on = false;
++ }
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_G_FBUF ioctl
++ */
++ case VIDIOC_G_FBUF:{
++ struct v4l2_framebuffer *fb = arg;
++ struct fb_var_screeninfo *var;
++
++ if (cam->output >= num_registered_fb) {
++ retval = -EINVAL;
++ break;
++ }
++
++ var = &registered_fb[cam->output]->var;
++ cam->v4l2_fb.fmt.width = var->xres;
++ cam->v4l2_fb.fmt.height = var->yres;
++ cam->v4l2_fb.fmt.bytesperline =
++ var->xres_virtual * var->bits_per_pixel;
++ cam->v4l2_fb.fmt.colorspace = V4L2_COLORSPACE_SRGB;
++ *fb = cam->v4l2_fb;
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_S_FBUF ioctl
++ */
++ case VIDIOC_S_FBUF:{
++ struct v4l2_framebuffer *fb = arg;
++ cam->v4l2_fb.flags = fb->flags;
++ cam->v4l2_fb.fmt.pixelformat = fb->fmt.pixelformat;
++ break;
++ }
++
++ case VIDIOC_G_PARM:{
++ struct v4l2_streamparm *parm = arg;
++ if (parm->type != V4L2_BUF_TYPE_VIDEO_CAPTURE) {
++ pr_debug("VIDIOC_G_PARM invalid type\n");
++ retval = -EINVAL;
++ break;
++ }
++ parm->parm.capture = cam->streamparm.parm.capture;
++ break;
++ }
++ case VIDIOC_S_PARM:{
++ struct v4l2_streamparm *parm = arg;
++ retval = mxc_v4l2_s_param(cam, parm);
++ break;
++ }
++
++ /* linux v4l2 bug, kernel c0485619 user c0405619 */
++ case VIDIOC_ENUMSTD:{
++ struct v4l2_standard *e = arg;
++ *e = cam->standard;
++ pr_debug("VIDIOC_ENUMSTD call\n");
++ retval = 0;
++ break;
++ }
++
++ case VIDIOC_G_STD:{
++ v4l2_std_id *e = arg;
++ *e = cam->standard.id;
++ break;
++ }
++
++ case VIDIOC_S_STD:{
++ break;
++ }
++
++ case VIDIOC_ENUMOUTPUT:
++ {
++ struct v4l2_output *output = arg;
++
++ if (output->index >= num_registered_fb) {
++ retval = -EINVAL;
++ break;
++ }
++
++ strncpy(output->name,
++ registered_fb[output->index]->fix.id, 31);
++ output->type = V4L2_OUTPUT_TYPE_ANALOG;
++ output->audioset = 0;
++ output->modulator = 0;
++ output->std = V4L2_STD_UNKNOWN;
++
++ break;
++ }
++ case VIDIOC_G_OUTPUT:
++ {
++ int *p_output_num = arg;
++
++ *p_output_num = cam->output;
++ break;
++ }
++ case VIDIOC_S_OUTPUT:
++ {
++ int *p_output_num = arg;
++
++ if (*p_output_num >= num_registered_fb) {
++ retval = -EINVAL;
++ break;
++ }
++
++ cam->output = *p_output_num;
++ break;
++ }
++
++ case VIDIOC_G_INPUT:
++ {
++ int *p_input_index = arg;
++
++ retval = mxc_get_video_input(cam);
++ if (0 == retval)
++ *p_input_index = 1;
++ else
++ *p_input_index = -ENODEV;
++
++ break;
++ }
++
++ case VIDIOC_ENUM_FMT:
++ case VIDIOC_TRY_FMT:
++ case VIDIOC_QUERYCTRL:
++ case VIDIOC_ENUMINPUT:
++ case VIDIOC_S_INPUT:
++ case VIDIOC_G_TUNER:
++ case VIDIOC_S_TUNER:
++ case VIDIOC_G_FREQUENCY:
++ case VIDIOC_S_FREQUENCY:
++ default:
++ retval = -EINVAL;
++ break;
++ }
++
++ up(&cam->busy_lock);
++ return retval;
++}
++
++/*
++ * V4L interface - ioctl function
++ *
++ * @return None
++ */
++static int
++mxc_v4l_ioctl(struct inode *inode, struct file *file,
++ unsigned int cmd, unsigned long arg)
++{
++ return video_usercopy(inode, file, cmd, arg, mxc_v4l_do_ioctl);
++}
++
++/*!
++ * V4L interface - mmap function
++ *
++ * @param file structure file *
++ *
++ * @param vma structure vm_area_struct *
++ *
++ * @return status 0 Success, EINTR busy lock error, ENOBUFS remap_page error
++ */
++static int mxc_mmap(struct file *file, struct vm_area_struct *vma)
++{
++ struct video_device *dev = video_devdata(file);
++ unsigned long size;
++ int res = 0;
++ cam_data *cam = dev->priv;
++
++ pr_debug("pgoff=0x%lx, start=0x%lx, end=0x%lx\n",
++ vma->vm_pgoff, vma->vm_start, vma->vm_end);
++
++ /* make this _really_ smp-safe */
++ if (down_interruptible(&cam->busy_lock))
++ return -EINTR;
++
++ size = vma->vm_end - vma->vm_start;
++ vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot);
++
++ if (remap_pfn_range(vma, vma->vm_start,
++ vma->vm_pgoff, size, vma->vm_page_prot)) {
++ pr_debug("mxc_mmap: remap_pfn_range failed\n");
++ res = -ENOBUFS;
++ goto mxc_mmap_exit;
++ }
++
++ vma->vm_flags &= ~VM_IO; /* using shared anonymous pages */
++
++ mxc_mmap_exit:
++ up(&cam->busy_lock);
++ return res;
++}
++
++/*!
++ * V4L interface - poll function
++ *
++ * @param file structure file *
++ *
++ * @param wait structure poll_table *
++ *
++ * @return status POLLIN | POLLRDNORM
++ */
++static unsigned int mxc_poll(struct file *file, poll_table * wait)
++{
++ struct video_device *dev = video_devdata(file);
++ cam_data *cam = dev->priv;
++ wait_queue_head_t *queue = NULL;
++ int res = POLLIN | POLLRDNORM;
++
++ if (down_interruptible(&cam->busy_lock))
++ return -EINTR;
++
++ queue = &cam->enc_queue;
++ poll_wait(file, queue, wait);
++
++ up(&cam->busy_lock);
++ return res;
++}
++
++static struct
++file_operations mxc_v4l_fops = {
++ .owner = THIS_MODULE,
++ .open = mxc_v4l_open,
++ .release = mxc_v4l_close,
++ .read = mxc_v4l_read,
++ .ioctl = mxc_v4l_ioctl,
++ .mmap = mxc_mmap,
++ .poll = mxc_poll,
++};
++
++static struct video_device mxc_v4l_template = {
++ .owner = THIS_MODULE,
++ .name = "Mxc Camera",
++ .type = 0,
++ .type2 = VID_TYPE_CAPTURE,
++ .hardware = 0,
++ .fops = &mxc_v4l_fops,
++ .release = video_device_release,
++};
++
++static void camera_platform_release(struct device *device)
++{
++}
++
++/*! Device Definition for Mt9v111 devices */
++static struct platform_device mxc_v4l2_devices = {
++ .name = "mxc_v4l2",
++ .dev = {
++ .release = camera_platform_release,
++ },
++ .id = 0,
++};
++
++extern struct camera_sensor camera_sensor_if;
++
++/*!
++* Camera V4l2 callback function.
++*
++* @return status
++*/
++static void camera_callback(u32 mask, void *dev)
++{
++ struct mxc_v4l_frame *done_frame;
++ struct mxc_v4l_frame *ready_frame;
++
++ cam_data *cam = (cam_data *) dev;
++ if (cam == NULL)
++ return;
++
++ if (mask == 1) {
++ if (cam->overflow == 0) {
++ cam->overflow = 1;
++ queue_work(v4l2_work, &prp_reset_work);
++ }
++ return;
++ }
++
++ if (list_empty(&cam->working_q)) {
++ if (empty_wq_cnt == 0) {
++ printk(KERN_ERR
++ "camera_callback: working queue empty %d\n",
++ empty_wq_cnt);
++ }
++ empty_wq_cnt++;
++ if (list_empty(&cam->ready_q)) {
++ prphw_disable(PRP_CHANNEL_2);
++ cam->skip_frame++;
++ } else {
++ ready_frame =
++ list_entry(cam->ready_q.next, struct mxc_v4l_frame,
++ queue);
++ list_del(cam->ready_q.next);
++ list_add_tail(&ready_frame->queue, &cam->working_q);
++ cam->enc_update_eba(ready_frame->paddress,
++ &cam->ping_pong_csi);
++ }
++ return;
++ }
++
++ done_frame =
++ list_entry(cam->working_q.next, struct mxc_v4l_frame, queue);
++ if (done_frame->buffer.flags & V4L2_BUF_FLAG_QUEUED) {
++ done_frame->buffer.flags |= V4L2_BUF_FLAG_DONE;
++ done_frame->buffer.flags &= ~V4L2_BUF_FLAG_QUEUED;
++
++ list_del(cam->working_q.next);
++ if (list_empty(&cam->ready_q)) {
++ if (list_empty(&cam->working_q))
++ prphw_disable(PRP_CHANNEL_2);
++ cam->skip_frame++;
++ } else {
++ ready_frame =
++ list_entry(cam->ready_q.next, struct mxc_v4l_frame,
++ queue);
++ list_del(cam->ready_q.next);
++ list_add_tail(&ready_frame->queue, &cam->working_q);
++ cam->enc_update_eba(ready_frame->paddress,
++ &cam->ping_pong_csi);
++ }
++
++ /* Added to the done queue */
++ list_add_tail(&done_frame->queue, &cam->done_q);
++
++ /* Wake up the queue */
++ cam->enc_counter++;
++ wake_up_interruptible(&cam->enc_queue);
++ } else {
++ printk(KERN_ERR "camera_callback :buffer not queued\n");
++ }
++}
++
++/*!
++ * initialize cam_data structure
++ *
++ * @param cam structure cam_data *
++ *
++ * @return status 0 Success
++ */
++static void init_camera_struct(cam_data * cam)
++{
++ int i;
++
++ /* Default everything to 0 */
++ memset(cam, 0, sizeof(cam_data));
++
++ init_MUTEX(&cam->param_lock);
++ init_MUTEX(&cam->busy_lock);
++
++ cam->video_dev = video_device_alloc();
++ if (cam->video_dev == NULL)
++ return;
++
++ *(cam->video_dev) = mxc_v4l_template;
++
++ video_set_drvdata(cam->video_dev, cam);
++ dev_set_drvdata(&mxc_v4l2_devices.dev, cam);
++ cam->video_dev->minor = -1;
++
++ for (i = 0; i < FRAME_NUM; i++) {
++ cam->frame[i].width = 0;
++ cam->frame[i].height = 0;
++ cam->frame[i].paddress = 0;
++ }
++
++ init_waitqueue_head(&cam->enc_queue);
++ init_waitqueue_head(&cam->still_queue);
++ init_waitqueue_head(&cam->overflow_queue);
++ cam->overflow = 0;
++
++ /* setup cropping */
++ cam->crop_bounds.left = 0;
++ cam->crop_bounds.width = 640;
++ cam->crop_bounds.top = 0;
++ cam->crop_bounds.height = 480;
++ cam->crop_current = cam->crop_defrect = cam->crop_bounds;
++ cam->streamparm.parm.capture.capturemode = 0;
++
++ cam->standard.index = 0;
++ cam->standard.id = V4L2_STD_UNKNOWN;
++ cam->standard.frameperiod.denominator = 30;
++ cam->standard.frameperiod.numerator = 1;
++ cam->standard.framelines = 480;
++ cam->streamparm.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
++ cam->streamparm.parm.capture.timeperframe = cam->standard.frameperiod;
++ cam->streamparm.parm.capture.capability = V4L2_CAP_TIMEPERFRAME;
++ cam->overlay_on = false;
++ cam->capture_on = false;
++ cam->skip_frame = 0;
++ cam->v4l2_fb.capability = V4L2_FBUF_CAP_EXTERNOVERLAY;
++ cam->v4l2_fb.flags = V4L2_FBUF_FLAG_PRIMARY;
++
++ cam->v2f.fmt.pix.sizeimage = 352 * 288 * 3 / 2;
++ cam->v2f.fmt.pix.bytesperline = 288 * 3 / 2;
++ cam->v2f.fmt.pix.width = 288;
++ cam->v2f.fmt.pix.height = 352;
++ cam->v2f.fmt.pix.pixelformat = V4L2_PIX_FMT_YUV420;
++ cam->win.w.width = 160;
++ cam->win.w.height = 160;
++ cam->win.w.left = 0;
++ cam->win.w.top = 0;
++
++ cam->cam_sensor = &camera_sensor_if;
++ cam->enc_callback = camera_callback;
++
++ init_waitqueue_head(&cam->power_queue);
++ cam->int_lock = SPIN_LOCK_UNLOCKED;
++ spin_lock_init(&cam->int_lock);
++}
++
++extern void gpio_sensor_active(void);
++extern void gpio_sensor_inactive(void);
++
++/*!
++ * camera_power function
++ * Turn Sensor power On/Off
++ *
++ * @param cameraOn true to turn camera on, otherwise shut down
++ *
++ * @return status
++ */
++static u8 camera_power(bool cameraOn)
++{
++ if (cameraOn == true) {
++ gpio_sensor_active();
++ csi_enable_mclk(csi_mclk_flag_backup, true, true);
++ } else {
++ csi_mclk_flag_backup = csi_read_mclk_flag();
++ csi_enable_mclk(csi_mclk_flag_backup, false, false);
++ gpio_sensor_inactive();
++ }
++ return 0;
++}
++
++/*!
++ * This function is called to put the sensor in a low power state. Refer to the
++ * document driver-model/driver.txt in the kernel source tree for more
++ * information.
++ *
++ * @param pdev the device structure used to give information on which I2C
++ * to suspend
++ * @param state the power state the device is entering
++ *
++ * @return The function returns 0 on success and -1 on failure.
++ */
++static int mxc_v4l2_suspend(struct platform_device *pdev, pm_message_t state)
++{
++ cam_data *cam = platform_get_drvdata(pdev);
++
++ cam->low_power = true;
++
++ if (cam->overlay_on == true)
++ stop_preview(cam);
++ if ((cam->capture_on == true) && cam->enc_disable) {
++ cam->enc_disable(cam);
++ }
++ camera_power(false);
++
++ return 0;
++}
++
++/*!
++ * This function is called to bring the sensor back from a low power state.Refer
++ * to the document driver-model/driver.txt in the kernel source tree for more
++ * information.
++ *
++ * @param pdev the device structure
++ *
++ * @return The function returns 0 on success and -1 on failure
++ */
++static int mxc_v4l2_resume(struct platform_device *pdev)
++{
++ cam_data *cam = platform_get_drvdata(pdev);
++
++ cam->low_power = false;
++ wake_up_interruptible(&cam->power_queue);
++
++ if (cam->overlay_on == true)
++ start_preview(cam);
++ if (cam->capture_on == true)
++ mxc_streamon(cam);
++ camera_power(true);
++
++ return 0;
++}
++
++/*!
++ * This structure contains pointers to the power management callback functions.
++ */
++static struct platform_driver mxc_v4l2_driver = {
++ .driver = {
++ .name = "mxc_v4l2",
++ .owner = THIS_MODULE,
++ .bus = &platform_bus_type,
++ },
++ .probe = NULL,
++ .remove = NULL,
++ .suspend = mxc_v4l2_suspend,
++ .resume = mxc_v4l2_resume,
++ .shutdown = NULL,
++};
++
++/*!
++ * Entry point for the V4L2
++ *
++ * @return Error code indicating success or failure
++ */
++static __init int camera_init(void)
++{
++ u8 err = 0;
++ cam_data *cam;
++
++ if ((g_cam = cam = kzalloc(sizeof(cam_data), GFP_KERNEL)) == NULL) {
++ pr_debug("failed to mxc_v4l_register_camera\n");
++ return -ENOMEM;
++ }
++
++ init_camera_struct(cam);
++
++ v4l2_work = create_singlethread_workqueue("v4l2_emma");
++
++ /* Register the I2C device */
++ err = platform_device_register(&mxc_v4l2_devices);
++ if (err != 0) {
++ pr_debug("camera_init: platform_device_register failed.\n");
++ video_device_release(cam->video_dev);
++ kfree(cam);
++ g_cam = NULL;
++ }
++
++ /* Register the device driver structure. */
++ err = platform_driver_register(&mxc_v4l2_driver);
++ if (err != 0) {
++ platform_device_unregister(&mxc_v4l2_devices);
++ pr_debug("camera_init: driver_register failed.\n");
++ video_device_release(cam->video_dev);
++ kfree(cam);
++ g_cam = NULL;
++ return err;
++ }
++
++ /* register v4l device */
++ err = video_register_device(cam->video_dev, VFL_TYPE_GRABBER, video_nr);
++ if (err != 0) {
++ platform_driver_unregister(&mxc_v4l2_driver);
++ platform_device_unregister(&mxc_v4l2_devices);
++ video_device_release(cam->video_dev);
++ kfree(cam);
++ g_cam = NULL;
++ pr_debug("video_register_device failed\n");
++ return err;
++ }
++
++ return err;
++}
++
++/*!
++ * Exit and cleanup for the V4L2
++ *
++ */
++static void __exit camera_exit(void)
++{
++ pr_debug("unregistering video\n");
++
++ flush_workqueue(v4l2_work);
++ destroy_workqueue(v4l2_work);
++
++ video_unregister_device(g_cam->video_dev);
++
++ platform_driver_unregister(&mxc_v4l2_driver);
++ platform_device_unregister(&mxc_v4l2_devices);
++
++ if (g_cam->open_count) {
++ pr_debug("camera open -- setting ops to NULL\n");
++ } else {
++ pr_debug("freeing camera\n");
++ mxc_free_frame_buf(g_cam);
++ kfree(g_cam);
++ g_cam = NULL;
++ }
++}
++
++module_init(camera_init);
++module_exit(camera_exit);
++
++module_param(video_nr, int, 0444);
++
++MODULE_AUTHOR("Freescale Semiconductor, Inc.");
++MODULE_DESCRIPTION("V4L2 capture driver for Mxc based cameras");
++MODULE_LICENSE("GPL");
++MODULE_SUPPORTED_DEVICE("video");
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/capture/mxc_v4l2_capture.c linux-2.6.28-karo/drivers/media/video/mxc/capture/mxc_v4l2_capture.c
+--- linux-2.6.28/drivers/media/video/mxc/capture/mxc_v4l2_capture.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/capture/mxc_v4l2_capture.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,1867 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @file drivers/media/video/mxc/capture/mxc_v4l2_capture.c
++ *
++ * @brief Mxc Video For Linux 2 driver
++ *
++ * @ingroup MXC_V4L2_CAPTURE
++ */
++
++#include <linux/version.h>
++#include <linux/module.h>
++#include <linux/init.h>
++#include <linux/platform_device.h>
++#include <linux/fs.h>
++#include <linux/slab.h>
++#include <linux/ctype.h>
++#include <asm/io.h>
++#include <asm/semaphore.h>
++#include <linux/pagemap.h>
++#include <linux/vmalloc.h>
++#include <linux/types.h>
++#include <linux/fb.h>
++#include <linux/dma-mapping.h>
++
++#include <asm/arch/mxcfb.h>
++#include "mxc_v4l2_capture.h"
++#include "ipu_prp_sw.h"
++
++static int csi_mclk_flag_backup;
++static int video_nr = -1;
++cam_data *g_cam;
++EXPORT_SYMBOL(g_cam);
++
++#define MXC_V4L2_CAPTURE_NUM_OUTPUTS 2
++static struct v4l2_output mxc_capture_outputs[MXC_V4L2_CAPTURE_NUM_OUTPUTS] = {
++ {
++ .index = 0,
++ .name = "DISP3",
++ .type = V4L2_OUTPUT_TYPE_ANALOG,
++ .audioset = 0,
++ .modulator = 0,
++ .std = V4L2_STD_UNKNOWN,
++ },
++ {
++ .index = 1,
++ .name = "DISP0",
++ .type = V4L2_OUTPUT_TYPE_ANALOG,
++ .audioset = 0,
++ .modulator = 0,
++ .std = V4L2_STD_UNKNOWN,
++ }
++};
++
++/*!
++ * Free frame buffers
++ *
++ * @param cam Structure cam_data *
++ *
++ * @return status 0 success.
++ */
++static int mxc_free_frame_buf(cam_data * cam)
++{
++ int i;
++
++ for (i = 0; i < FRAME_NUM; i++) {
++ if (cam->frame[i].vaddress != 0) {
++ dma_free_coherent(0, cam->frame[i].buffer.length,
++ cam->frame[i].vaddress,
++ cam->frame[i].paddress);
++ cam->frame[i].vaddress = 0;
++ }
++ }
++
++ return 0;
++}
++
++/*!
++ * Allocate frame buffers
++ *
++ * @param cam Structure cam_data *
++ *
++ * @param count int number of buffer need to allocated
++ *
++ * @return status -0 Successfully allocated a buffer, -ENOBUFS failed.
++ */
++static int mxc_allocate_frame_buf(cam_data * cam, int count)
++{
++ int i;
++
++ for (i = 0; i < count; i++) {
++ cam->frame[i].vaddress =
++ dma_alloc_coherent(0,
++ PAGE_ALIGN(cam->v2f.fmt.pix.sizeimage),
++ &cam->frame[i].paddress,
++ GFP_DMA | GFP_KERNEL);
++ if (cam->frame[i].vaddress == 0) {
++ printk(KERN_ERR "mxc_allocate_frame_buf failed.\n");
++ mxc_free_frame_buf(cam);
++ return -ENOBUFS;
++ }
++ cam->frame[i].buffer.index = i;
++ cam->frame[i].buffer.flags = V4L2_BUF_FLAG_MAPPED;
++ cam->frame[i].buffer.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
++ cam->frame[i].buffer.length =
++ PAGE_ALIGN(cam->v2f.fmt.pix.sizeimage);
++ cam->frame[i].buffer.memory = V4L2_MEMORY_MMAP;
++ cam->frame[i].buffer.m.offset = cam->frame[i].paddress;
++ cam->frame[i].index = i;
++ }
++
++ return 0;
++}
++
++/*!
++ * Free frame buffers status
++ *
++ * @param cam Structure cam_data *
++ *
++ * @return none
++ */
++static void mxc_free_frames(cam_data * cam)
++{
++ int i;
++
++ for (i = 0; i < FRAME_NUM; i++) {
++ cam->frame[i].buffer.flags = V4L2_BUF_FLAG_MAPPED;
++ }
++
++ cam->enc_counter = 0;
++ cam->skip_frame = 0;
++ INIT_LIST_HEAD(&cam->ready_q);
++ INIT_LIST_HEAD(&cam->working_q);
++ INIT_LIST_HEAD(&cam->done_q);
++}
++
++/*!
++ * Return the buffer status
++ *
++ * @param cam Structure cam_data *
++ * @param buf Structure v4l2_buffer *
++ *
++ * @return status 0 success, EINVAL failed.
++ */
++static int mxc_v4l2_buffer_status(cam_data * cam, struct v4l2_buffer *buf)
++{
++ if (buf->index < 0 || buf->index >= FRAME_NUM) {
++ printk(KERN_ERR
++ "mxc_v4l2_buffer_status buffers not allocated\n");
++ return -EINVAL;
++ }
++
++ memcpy(buf, &(cam->frame[buf->index].buffer), sizeof(*buf));
++ return 0;
++}
++
++/*!
++ * start the encoder job
++ *
++ * @param cam structure cam_data *
++ *
++ * @return status 0 Success
++ */
++static int mxc_streamon(cam_data * cam)
++{
++ struct mxc_v4l_frame *frame;
++ int err = 0;
++
++ if (list_empty(&cam->ready_q)) {
++ printk(KERN_ERR "mxc_streamon buffer not been queued yet\n");
++ return -EINVAL;
++ }
++
++ cam->capture_pid = current->pid;
++
++ if (cam->enc_enable) {
++ err = cam->enc_enable(cam);
++ if (err != 0) {
++ return err;
++ }
++ }
++
++ cam->ping_pong_csi = 0;
++ if (cam->enc_update_eba) {
++ frame =
++ list_entry(cam->ready_q.next, struct mxc_v4l_frame, queue);
++ list_del(cam->ready_q.next);
++ list_add_tail(&frame->queue, &cam->working_q);
++ err =
++ cam->enc_update_eba(frame->buffer.m.offset,
++ &cam->ping_pong_csi);
++
++ frame =
++ list_entry(cam->ready_q.next, struct mxc_v4l_frame, queue);
++ list_del(cam->ready_q.next);
++ list_add_tail(&frame->queue, &cam->working_q);
++ err |=
++ cam->enc_update_eba(frame->buffer.m.offset,
++ &cam->ping_pong_csi);
++ } else {
++ return -EINVAL;
++ }
++
++ cam->capture_on = true;
++ return err;
++}
++
++/*!
++ * Shut down the encoder job
++ *
++ * @param cam structure cam_data *
++ *
++ * @return status 0 Success
++ */
++static int mxc_streamoff(cam_data * cam)
++{
++ int err = 0;
++
++ if (cam->capture_on == false)
++ return 0;
++
++ if (cam->enc_disable) {
++ err = cam->enc_disable(cam);
++ }
++ mxc_free_frames(cam);
++ cam->capture_on = false;
++ return err;
++}
++
++/*!
++ * Valid whether the palette is supported
++ *
++ * @param palette V4L2_PIX_FMT_RGB565, V4L2_PIX_FMT_BGR24 or V4L2_PIX_FMT_BGR32
++ *
++ * @return 0 if failed
++ */
++static inline int valid_mode(u32 palette)
++{
++ return ((palette == V4L2_PIX_FMT_RGB565) ||
++ (palette == V4L2_PIX_FMT_BGR24) ||
++ (palette == V4L2_PIX_FMT_RGB24) ||
++ (palette == V4L2_PIX_FMT_BGR32) ||
++ (palette == V4L2_PIX_FMT_RGB32) ||
++ (palette == V4L2_PIX_FMT_YUV422P) ||
++ (palette == V4L2_PIX_FMT_UYVY) ||
++ (palette == V4L2_PIX_FMT_YUV420));
++}
++
++/*!
++ * Valid and adjust the overlay window size, position
++ *
++ * @param cam structure cam_data *
++ * @param win struct v4l2_window *
++ *
++ * @return 0
++ */
++static int verify_preview(cam_data * cam, struct v4l2_window *win)
++{
++ int i = 0;
++ int *width, *height;
++
++ do {
++ cam->overlay_fb = (struct fb_info *)registered_fb[i];
++ if (cam->overlay_fb == NULL) {
++ printk(KERN_ERR "verify_preview No matched.\n");
++ return -1;
++ }
++ if (strncmp(cam->overlay_fb->fix.id,
++ mxc_capture_outputs[cam->output].name, 5) == 0) {
++ break;
++ }
++ } while (++i < FB_MAX);
++
++ /* 4 bytes alignment for both FG and BG */
++ if (cam->overlay_fb->var.bits_per_pixel == 24) {
++ win->w.left -= win->w.left % 4;
++ } else if (cam->overlay_fb->var.bits_per_pixel == 16) {
++ win->w.left -= win->w.left % 2;
++ }
++
++ if (win->w.width + win->w.left > cam->overlay_fb->var.xres)
++ win->w.width = cam->overlay_fb->var.xres - win->w.left;
++ if (win->w.height + win->w.top > cam->overlay_fb->var.yres)
++ win->w.height = cam->overlay_fb->var.yres - win->w.top;
++
++ /* stride line limitation */
++ win->w.height -= win->w.height % 8;
++ win->w.width -= win->w.width % 8;
++
++ if (cam->rotation >= IPU_ROTATE_90_RIGHT) {
++ height = &win->w.width;
++ width = &win->w.height;
++ } else {
++ width = &win->w.width;
++ height = &win->w.height;
++ }
++
++ if ((cam->crop_bounds.width / *width > 8) ||
++ ((cam->crop_bounds.width / *width == 8) &&
++ (cam->crop_bounds.width % *width))) {
++ *width = cam->crop_bounds.width / 8;
++ if (*width % 8)
++ *width += 8 - *width % 8;
++ if (*width + win->w.left > cam->overlay_fb->var.xres) {
++ printk(KERN_ERR "width exceed resize limit.\n");
++ return -1;
++ }
++ printk(KERN_ERR "width exceed limit resize to %d.\n", *width);
++ }
++
++ if ((cam->crop_bounds.height / *height > 8) ||
++ ((cam->crop_bounds.height / *height == 8) &&
++ (cam->crop_bounds.height % *height))) {
++ *height = cam->crop_bounds.height / 8;
++ if (*height % 8)
++ *height += 8 - *height % 8;
++ if (*height + win->w.top > cam->overlay_fb->var.yres) {
++ printk(KERN_ERR "height exceed resize limit.\n");
++ return -1;
++ }
++ printk(KERN_ERR "height exceed limit resize to %d.\n", *height);
++ }
++
++ return 0;
++}
++
++/*!
++ * start the viewfinder job
++ *
++ * @param cam structure cam_data *
++ *
++ * @return status 0 Success
++ */
++static int start_preview(cam_data * cam)
++{
++ int err = 0;
++#if defined(CONFIG_MXC_IPU_PRP_VF_SDC) || defined(CONFIG_MXC_IPU_PRP_VF_SDC_MODULE)
++ if (cam->output == 0) {
++ if (cam->v4l2_fb.flags == V4L2_FBUF_FLAG_OVERLAY)
++ err = prp_vf_sdc_select(cam);
++ else if (cam->v4l2_fb.flags == V4L2_FBUF_FLAG_PRIMARY)
++ err = prp_vf_sdc_select_bg(cam);
++ if (err != 0)
++ return err;
++
++ err = cam->vf_start_sdc(cam);
++ }
++#endif
++
++#if defined(CONFIG_MXC_IPU_PRP_VF_ADC) || defined(CONFIG_MXC_IPU_PRP_VF_ADC_MODULE)
++ if (cam->output == 1) {
++ err = prp_vf_adc_select(cam);
++ if (err != 0)
++ return err;
++
++ err = cam->vf_start_adc(cam);
++ }
++#endif
++
++ return err;
++}
++
++/*!
++ * shut down the viewfinder job
++ *
++ * @param cam structure cam_data *
++ *
++ * @return status 0 Success
++ */
++static int stop_preview(cam_data * cam)
++{
++ int err = 0;
++
++#if defined(CONFIG_MXC_IPU_PRP_VF_ADC) || defined(CONFIG_MXC_IPU_PRP_VF_ADC_MODULE)
++ if (cam->output == 1) {
++ err = prp_vf_adc_deselect(cam);
++ }
++#endif
++
++#if defined(CONFIG_MXC_IPU_PRP_VF_SDC) || defined(CONFIG_MXC_IPU_PRP_VF_SDC_MODULE)
++ if (cam->output == 0) {
++ if (cam->v4l2_fb.flags == V4L2_FBUF_FLAG_OVERLAY)
++ err = prp_vf_sdc_deselect(cam);
++ else if (cam->v4l2_fb.flags == V4L2_FBUF_FLAG_PRIMARY)
++ err = prp_vf_sdc_deselect_bg(cam);
++ }
++#endif
++
++ return err;
++}
++
++/*!
++ * V4L2 - mxc_v4l2_g_fmt function
++ *
++ * @param cam structure cam_data *
++ *
++ * @param f structure v4l2_format *
++ *
++ * @return status 0 success, EINVAL failed
++ */
++static int mxc_v4l2_g_fmt(cam_data * cam, struct v4l2_format *f)
++{
++ int retval = 0;
++
++ switch (f->type) {
++ case V4L2_BUF_TYPE_VIDEO_CAPTURE:
++ f->fmt.pix = cam->v2f.fmt.pix;
++ retval = 0;
++ break;
++ case V4L2_BUF_TYPE_VIDEO_OVERLAY:
++ f->fmt.win = cam->win;
++ break;
++ default:
++ retval = -EINVAL;
++ }
++ return retval;
++}
++
++/*!
++ * V4L2 - mxc_v4l2_s_fmt function
++ *
++ * @param cam structure cam_data *
++ *
++ * @param f structure v4l2_format *
++ *
++ * @return status 0 success, EINVAL failed
++ */
++static int mxc_v4l2_s_fmt(cam_data * cam, struct v4l2_format *f)
++{
++ int retval = 0;
++ int size = 0;
++ int bytesperline = 0;
++ int *width, *height;
++
++ switch (f->type) {
++ case V4L2_BUF_TYPE_VIDEO_CAPTURE:
++ if (!valid_mode(f->fmt.pix.pixelformat)) {
++ printk(KERN_ERR
++ "mxc_v4l2_s_fmt: format not supported\n");
++ return -EINVAL;
++ }
++
++ if (cam->rotation >= IPU_ROTATE_90_RIGHT) {
++ height = &f->fmt.pix.width;
++ width = &f->fmt.pix.height;
++ } else {
++ width = &f->fmt.pix.width;
++ height = &f->fmt.pix.height;
++ }
++
++ /* stride line limitation */
++ *width -= *width % 8;
++ *height -= *height % 8;
++
++ if ((cam->crop_bounds.width / *width > 8) ||
++ ((cam->crop_bounds.width / *width == 8) &&
++ (cam->crop_bounds.width % *width))) {
++ *width = cam->crop_bounds.width / 8;
++ if (*width % 8)
++ *width += 8 - *width % 8;
++ printk(KERN_ERR "width exceed limit resize to %d.\n",
++ *width);
++ }
++
++ if ((cam->crop_bounds.height / *height > 8) ||
++ ((cam->crop_bounds.height / *height == 8) &&
++ (cam->crop_bounds.height % *height))) {
++ *height = cam->crop_bounds.height / 8;
++ if (*height % 8)
++ *height += 8 - *height % 8;
++ printk(KERN_ERR "height exceed limit resize to %d.\n",
++ *height);
++ }
++
++ switch (f->fmt.pix.pixelformat) {
++ case V4L2_PIX_FMT_RGB565:
++ size = f->fmt.pix.width * f->fmt.pix.height * 2;
++ bytesperline = f->fmt.pix.width * 2;
++ break;
++ case V4L2_PIX_FMT_BGR24:
++ size = f->fmt.pix.width * f->fmt.pix.height * 3;
++ bytesperline = f->fmt.pix.width * 3;
++ break;
++ case V4L2_PIX_FMT_RGB24:
++ size = f->fmt.pix.width * f->fmt.pix.height * 3;
++ bytesperline = f->fmt.pix.width * 3;
++ break;
++ case V4L2_PIX_FMT_BGR32:
++ size = f->fmt.pix.width * f->fmt.pix.height * 4;
++ bytesperline = f->fmt.pix.width * 4;
++ break;
++ case V4L2_PIX_FMT_RGB32:
++ size = f->fmt.pix.width * f->fmt.pix.height * 4;
++ bytesperline = f->fmt.pix.width * 4;
++ break;
++ case V4L2_PIX_FMT_YUV422P:
++ size = f->fmt.pix.width * f->fmt.pix.height * 2;
++ bytesperline = f->fmt.pix.width;
++ break;
++ case V4L2_PIX_FMT_UYVY:
++ size = f->fmt.pix.width * f->fmt.pix.height * 2;
++ bytesperline = f->fmt.pix.width * 2;
++ break;
++ case V4L2_PIX_FMT_YUV420:
++ size = f->fmt.pix.width * f->fmt.pix.height * 3 / 2;
++ bytesperline = f->fmt.pix.width;
++ break;
++ default:
++ break;
++ }
++
++ if (f->fmt.pix.bytesperline < bytesperline) {
++ f->fmt.pix.bytesperline = bytesperline;
++ } else {
++ bytesperline = f->fmt.pix.bytesperline;
++ }
++
++ if (f->fmt.pix.sizeimage < size) {
++ f->fmt.pix.sizeimage = size;
++ } else {
++ size = f->fmt.pix.sizeimage;
++ }
++
++ cam->v2f.fmt.pix = f->fmt.pix;
++
++ if (cam->v2f.fmt.pix.priv != 0) {
++ if (copy_from_user(&cam->offset,
++ (void *)cam->v2f.fmt.pix.priv,
++ sizeof(cam->offset))) {
++ retval = -EFAULT;
++ break;
++ }
++ }
++ retval = 0;
++ break;
++ case V4L2_BUF_TYPE_VIDEO_OVERLAY:
++ retval = verify_preview(cam, &f->fmt.win);
++ cam->win = f->fmt.win;
++ break;
++ default:
++ retval = -EINVAL;
++ }
++ return retval;
++}
++
++/*!
++ * get control param
++ *
++ * @param cam structure cam_data *
++ *
++ * @param c structure v4l2_control *
++ *
++ * @return status 0 success, EINVAL failed
++ */
++static int mxc_get_v42l_control(cam_data * cam, struct v4l2_control *c)
++{
++ int status = 0;
++
++ switch (c->id) {
++ case V4L2_CID_HFLIP:
++ if (cam->rotation == IPU_ROTATE_HORIZ_FLIP)
++ c->value = 1;
++ break;
++ case V4L2_CID_VFLIP:
++ if (cam->rotation == IPU_ROTATE_VERT_FLIP)
++ c->value = 1;
++ break;
++ case V4L2_CID_MXC_ROT:
++ c->value = cam->rotation;
++ break;
++ case V4L2_CID_BRIGHTNESS:
++ c->value = cam->bright;
++ break;
++ case V4L2_CID_HUE:
++ c->value = cam->hue;
++ break;
++ case V4L2_CID_CONTRAST:
++ c->value = cam->contrast;
++ break;
++ case V4L2_CID_SATURATION:
++ c->value = cam->saturation;
++ break;
++ case V4L2_CID_RED_BALANCE:
++ c->value = cam->red;
++ break;
++ case V4L2_CID_BLUE_BALANCE:
++ c->value = cam->blue;
++ break;
++ case V4L2_CID_BLACK_LEVEL:
++ c->value = cam->ae_mode;
++ break;
++ default:
++ status = -EINVAL;
++ }
++ return status;
++}
++
++/*!
++ * V4L2 - set_control function
++ * V4L2_CID_PRIVATE_BASE is the extention for IPU preprocessing.
++ * 0 for normal operation
++ * 1 for vertical flip
++ * 2 for horizontal flip
++ * 3 for horizontal and vertical flip
++ * 4 for 90 degree rotation
++ * @param cam structure cam_data *
++ *
++ * @param c structure v4l2_control *
++ *
++ * @return status 0 success, EINVAL failed
++ */
++static int mxc_set_v42l_control(cam_data * cam, struct v4l2_control *c)
++{
++ switch (c->id) {
++ case V4L2_CID_HFLIP:
++ if (c->value == 1) {
++ if ((cam->rotation != IPU_ROTATE_VERT_FLIP) &&
++ (cam->rotation != IPU_ROTATE_180))
++ cam->rotation = IPU_ROTATE_HORIZ_FLIP;
++ else
++ cam->rotation = IPU_ROTATE_180;
++ } else {
++ if (cam->rotation == IPU_ROTATE_HORIZ_FLIP)
++ cam->rotation = IPU_ROTATE_NONE;
++ if (cam->rotation == IPU_ROTATE_180)
++ cam->rotation = IPU_ROTATE_VERT_FLIP;
++ }
++ break;
++ case V4L2_CID_VFLIP:
++ if (c->value == 1) {
++ if ((cam->rotation != IPU_ROTATE_HORIZ_FLIP) &&
++ (cam->rotation != IPU_ROTATE_180))
++ cam->rotation = IPU_ROTATE_VERT_FLIP;
++ else
++ cam->rotation = IPU_ROTATE_180;
++ } else {
++ if (cam->rotation == IPU_ROTATE_VERT_FLIP)
++ cam->rotation = IPU_ROTATE_NONE;
++ if (cam->rotation == IPU_ROTATE_180)
++ cam->rotation = IPU_ROTATE_HORIZ_FLIP;
++ }
++ break;
++ case V4L2_CID_MXC_ROT:
++ switch (c->value) {
++ case V4L2_MXC_ROTATE_NONE:
++ cam->rotation = IPU_ROTATE_NONE;
++ break;
++ case V4L2_MXC_ROTATE_VERT_FLIP:
++ cam->rotation = IPU_ROTATE_VERT_FLIP;
++ break;
++ case V4L2_MXC_ROTATE_HORIZ_FLIP:
++ cam->rotation = IPU_ROTATE_HORIZ_FLIP;
++ break;
++ case V4L2_MXC_ROTATE_180:
++ cam->rotation = IPU_ROTATE_180;
++ break;
++ case V4L2_MXC_ROTATE_90_RIGHT:
++ cam->rotation = IPU_ROTATE_90_RIGHT;
++ break;
++ case V4L2_MXC_ROTATE_90_RIGHT_VFLIP:
++ cam->rotation = IPU_ROTATE_90_RIGHT_VFLIP;
++ break;
++ case V4L2_MXC_ROTATE_90_RIGHT_HFLIP:
++ cam->rotation = IPU_ROTATE_90_RIGHT_HFLIP;
++ break;
++ case V4L2_MXC_ROTATE_90_LEFT:
++ cam->rotation = IPU_ROTATE_90_LEFT;
++ break;
++ default:
++ return -EINVAL;
++ }
++ break;
++ case V4L2_CID_HUE:
++ cam->hue = c->value;
++ break;
++ case V4L2_CID_CONTRAST:
++ cam->contrast = c->value;
++ break;
++ case V4L2_CID_BRIGHTNESS:
++ cam->bright = c->value;
++ case V4L2_CID_SATURATION:
++ cam->saturation = c->value;
++ case V4L2_CID_RED_BALANCE:
++ cam->red = c->value;
++ case V4L2_CID_BLUE_BALANCE:
++ cam->blue = c->value;
++ ipu_csi_enable_mclk(CSI_MCLK_I2C, true, true);
++ cam->cam_sensor->set_color(cam->bright, cam->saturation,
++ cam->red, cam->green, cam->blue);
++ ipu_csi_enable_mclk(CSI_MCLK_I2C, false, false);
++ break;
++ case V4L2_CID_BLACK_LEVEL:
++ cam->ae_mode = c->value & 0x03;
++ ipu_csi_enable_mclk(CSI_MCLK_I2C, true, true);
++ if (cam->cam_sensor->set_ae_mode)
++ cam->cam_sensor->set_ae_mode(cam->ae_mode);
++ ipu_csi_enable_mclk(CSI_MCLK_I2C, false, false);
++ break;
++ case V4L2_CID_MXC_FLASH:
++ ipu_csi_flash_strobe(true);
++ break;
++ default:
++ return -EINVAL;
++ }
++ return 0;
++}
++
++/*!
++ * V4L2 - mxc_v4l2_s_param function
++ *
++ * @param cam structure cam_data *
++ *
++ * @param parm structure v4l2_streamparm *
++ *
++ * @return status 0 success, EINVAL failed
++ */
++static int mxc_v4l2_s_param(cam_data * cam, struct v4l2_streamparm *parm)
++{
++ sensor_interface *param;
++ ipu_csi_signal_cfg_t csi_param;
++ int err = 0;
++
++ if (parm->type != V4L2_BUF_TYPE_VIDEO_CAPTURE) {
++ printk(KERN_ERR "mxc_v4l2_s_param invalid type\n");
++ return -EINVAL;
++ }
++
++ if (parm->parm.capture.timeperframe.denominator >
++ cam->standard.frameperiod.denominator) {
++ printk(KERN_ERR "mxc_v4l2_s_param frame rate %d larger "
++ "than standard supported %d\n",
++ parm->parm.capture.timeperframe.denominator,
++ cam->standard.frameperiod.denominator);
++ return -EINVAL;
++ }
++
++ /* Stop the viewfinder */
++ if (cam->overlay_on == true) {
++ stop_preview(cam);
++ }
++
++ cam->streamparm.parm.capture.capability = V4L2_CAP_TIMEPERFRAME;
++
++ ipu_csi_enable_mclk(CSI_MCLK_I2C, true, true);
++ param = cam->cam_sensor->config
++ (&parm->parm.capture.timeperframe.denominator,
++ parm->parm.capture.capturemode);
++ ipu_csi_enable_mclk(CSI_MCLK_I2C, false, false);
++ cam->streamparm.parm.capture.timeperframe =
++ parm->parm.capture.timeperframe;
++
++ if ((parm->parm.capture.capturemode != 0) &&
++ (parm->parm.capture.capturemode != V4L2_MODE_HIGHQUALITY)) {
++ printk(KERN_ERR
++ "mxc_v4l2_s_param frame un-supported capture mode\n");
++ err = -EINVAL;
++ goto exit;
++ }
++
++ if (parm->parm.capture.capturemode ==
++ cam->streamparm.parm.capture.capturemode) {
++ goto exit;
++ }
++
++ /* resolution changed, so need to re-program the CSI */
++ csi_param.sens_clksrc = 0;
++ csi_param.clk_mode = param->clk_mode;
++ csi_param.pixclk_pol = param->pixclk_pol;
++ csi_param.data_width = param->data_width;
++ csi_param.data_pol = param->data_pol;
++ csi_param.ext_vsync = param->ext_vsync;
++ csi_param.Vsync_pol = param->Vsync_pol;
++ csi_param.Hsync_pol = param->Hsync_pol;
++ ipu_csi_init_interface(param->width, param->height,
++ param->pixel_fmt, csi_param);
++ ipu_csi_set_window_size(param->width + 1, param->height + 1);
++
++ if (parm->parm.capture.capturemode != V4L2_MODE_HIGHQUALITY) {
++ cam->streamparm.parm.capture.capturemode = 0;
++ } else {
++ cam->streamparm.parm.capture.capturemode =
++ V4L2_MODE_HIGHQUALITY;
++ cam->streamparm.parm.capture.extendedmode =
++ parm->parm.capture.extendedmode;
++ cam->streamparm.parm.capture.readbuffers = 1;
++ }
++
++ exit:
++ if (cam->overlay_on == true) {
++ start_preview(cam);
++ }
++
++ return err;
++}
++
++/*!
++ * Dequeue one V4L capture buffer
++ *
++ * @param cam structure cam_data *
++ * @param buf structure v4l2_buffer *
++ *
++ * @return status 0 success, EINVAL invalid frame number,
++ * ETIME timeout, ERESTARTSYS interrupted by user
++ */
++static int mxc_v4l_dqueue(cam_data * cam, struct v4l2_buffer *buf)
++{
++ int retval = 0;
++ struct mxc_v4l_frame *frame;
++
++ if (!wait_event_interruptible_timeout(cam->enc_queue,
++ cam->enc_counter != 0, 10 * HZ)) {
++ printk(KERN_ERR "mxc_v4l_dqueue timeout enc_counter %x\n",
++ cam->enc_counter);
++ return -ETIME;
++ } else if (signal_pending(current)) {
++ printk(KERN_ERR "mxc_v4l_dqueue() interrupt received\n");
++ mxc_free_frames(cam);
++ return -ERESTARTSYS;
++ }
++
++ cam->enc_counter--;
++
++ frame = list_entry(cam->done_q.next, struct mxc_v4l_frame, queue);
++ list_del(cam->done_q.next);
++ if (frame->buffer.flags & V4L2_BUF_FLAG_DONE) {
++ frame->buffer.flags &= ~V4L2_BUF_FLAG_DONE;
++ } else if (frame->buffer.flags & V4L2_BUF_FLAG_QUEUED) {
++ printk(KERN_ERR "VIDIOC_DQBUF: Buffer not filled.\n");
++ frame->buffer.flags &= ~V4L2_BUF_FLAG_QUEUED;
++ retval = -EINVAL;
++ } else if ((frame->buffer.flags & 0x7) == V4L2_BUF_FLAG_MAPPED) {
++ printk(KERN_ERR "VIDIOC_DQBUF: Buffer not queued.\n");
++ retval = -EINVAL;
++ }
++
++ buf->bytesused = cam->v2f.fmt.pix.sizeimage;
++ buf->index = frame->index;
++ buf->flags = frame->buffer.flags;
++ buf->m = cam->frame[frame->index].buffer.m;
++
++ return retval;
++}
++
++/*!
++ * V4L interface - open function
++ *
++ * @param inode structure inode *
++ * @param file structure file *
++ *
++ * @return status 0 success, ENODEV invalid device instance,
++ * ENODEV timeout, ERESTARTSYS interrupted by user
++ */
++static int mxc_v4l_open(struct inode *inode, struct file *file)
++{
++ sensor_interface *param;
++ ipu_csi_signal_cfg_t csi_param;
++ struct video_device *dev = video_devdata(file);
++ cam_data *cam = dev->priv;
++ int err = 0;
++
++ if (!cam) {
++ printk(KERN_ERR "Internal error, cam_data not found!\n");
++ return -EBADF;
++ }
++
++ down(&cam->busy_lock);
++
++ err = 0;
++ if (signal_pending(current))
++ goto oops;
++
++ if (cam->open_count++ == 0) {
++ wait_event_interruptible(cam->power_queue,
++ cam->low_power == false);
++
++#if defined(CONFIG_MXC_IPU_PRP_ENC) || defined(CONFIG_MXC_IPU_PRP_ENC_MODULE)
++ err = prp_enc_select(cam);
++#endif
++
++ cam->enc_counter = 0;
++ cam->skip_frame = 0;
++ INIT_LIST_HEAD(&cam->ready_q);
++ INIT_LIST_HEAD(&cam->working_q);
++ INIT_LIST_HEAD(&cam->done_q);
++
++ ipu_csi_enable_mclk(CSI_MCLK_I2C, true, true);
++ param = cam->cam_sensor->reset();
++ if (param == NULL) {
++ cam->open_count--;
++ ipu_csi_enable_mclk(CSI_MCLK_I2C, false, false);
++ err = -ENODEV;
++ goto oops;
++ }
++
++ csi_param.sens_clksrc = 0;
++ csi_param.clk_mode = param->clk_mode;
++ csi_param.pixclk_pol = param->pixclk_pol;
++ csi_param.data_width = param->data_width;
++ csi_param.data_pol = param->data_pol;
++ csi_param.ext_vsync = param->ext_vsync;
++ csi_param.Vsync_pol = param->Vsync_pol;
++ csi_param.Hsync_pol = param->Hsync_pol;
++ ipu_csi_init_interface(param->width, param->height,
++ param->pixel_fmt, csi_param);
++
++ cam->cam_sensor->get_color(&cam->bright, &cam->saturation,
++ &cam->red, &cam->green, &cam->blue);
++ if (cam->cam_sensor->get_ae_mode)
++ cam->cam_sensor->get_ae_mode(&cam->ae_mode);
++
++ /* pr_info("mxc_v4l_open saturation %x ae_mode %x\n",
++ cam->saturation, cam->ae_mode); */
++
++ ipu_csi_enable_mclk(CSI_MCLK_I2C, false, false);
++ }
++
++ file->private_data = dev;
++ oops:
++ up(&cam->busy_lock);
++ return err;
++}
++
++/*!
++ * V4L interface - close function
++ *
++ * @param inode struct inode *
++ * @param file struct file *
++ *
++ * @return 0 success
++ */
++static int mxc_v4l_close(struct inode *inode, struct file *file)
++{
++ struct video_device *dev = video_devdata(file);
++ int err = 0;
++ cam_data *cam = dev->priv;
++
++ if (!cam) {
++ printk(KERN_ERR "Internal error, cam_data not found!\n");
++ return -EBADF;
++ }
++
++ /* for the case somebody hit the ctrl C */
++ if (cam->overlay_pid == current->pid) {
++ err = stop_preview(cam);
++ cam->overlay_on = false;
++ }
++ if (cam->capture_pid == current->pid) {
++ err |= mxc_streamoff(cam);
++ wake_up_interruptible(&cam->enc_queue);
++ }
++
++ if (--cam->open_count == 0) {
++ wait_event_interruptible(cam->power_queue,
++ cam->low_power == false);
++ pr_info("mxc_v4l_close: release resource\n");
++
++#if defined(CONFIG_MXC_IPU_PRP_ENC) || defined(CONFIG_MXC_IPU_PRP_ENC_MODULE)
++ err |= prp_enc_deselect(cam);
++#endif
++ mxc_free_frame_buf(cam);
++ file->private_data = NULL;
++
++ /* capture off */
++ wake_up_interruptible(&cam->enc_queue);
++ mxc_free_frames(cam);
++ cam->enc_counter++;
++ }
++ return err;
++}
++
++#if defined(CONFIG_MXC_IPU_PRP_ENC) || defined(CONFIG_MXC_IPU_PRP_ENC_MODULE)
++/*
++ * V4L interface - read function
++ *
++ * @param file struct file *
++ * @param read buf char *
++ * @param count size_t
++ * @param ppos structure loff_t *
++ *
++ * @return bytes read
++ */
++static ssize_t
++mxc_v4l_read(struct file *file, char *buf, size_t count, loff_t * ppos)
++{
++ int err = 0;
++ u8 *v_address;
++ struct video_device *dev = video_devdata(file);
++ cam_data *cam = dev->priv;
++
++ if (down_interruptible(&cam->busy_lock))
++ return -EINTR;
++
++ /* Stop the viewfinder */
++ if (cam->overlay_on == true)
++ stop_preview(cam);
++
++ v_address = dma_alloc_coherent(0,
++ PAGE_ALIGN(cam->v2f.fmt.pix.sizeimage),
++ &cam->still_buf, GFP_DMA | GFP_KERNEL);
++
++ if (!v_address) {
++ err = -ENOBUFS;
++ goto exit0;
++ }
++
++ err = prp_still_select(cam);
++ if (err != 0) {
++ err = -EIO;
++ goto exit1;
++ }
++
++ cam->still_counter = 0;
++ err = cam->csi_start(cam);
++ if (err != 0) {
++ err = -EIO;
++ goto exit2;
++ }
++
++ if (!wait_event_interruptible_timeout(cam->still_queue,
++ cam->still_counter != 0,
++ 10 * HZ)) {
++ printk(KERN_ERR "mxc_v4l_read timeout counter %x\n",
++ cam->still_counter);
++ err = -ETIME;
++ goto exit2;
++ }
++ err = copy_to_user(buf, v_address, cam->v2f.fmt.pix.sizeimage);
++
++ exit2:
++ prp_still_deselect(cam);
++
++ exit1:
++ dma_free_coherent(0, cam->v2f.fmt.pix.sizeimage, v_address,
++ cam->still_buf);
++ cam->still_buf = 0;
++
++ exit0:
++ if (cam->overlay_on == true) {
++ start_preview(cam);
++ }
++
++ up(&cam->busy_lock);
++ if (err < 0)
++ return err;
++
++ return (cam->v2f.fmt.pix.sizeimage - err);
++}
++#endif
++
++/*!
++ * V4L interface - ioctl function
++ *
++ * @param inode struct inode *
++ *
++ * @param file struct file *
++ *
++ * @param ioctlnr unsigned int
++ *
++ * @param arg void *
++ *
++ * @return 0 success, ENODEV for invalid device instance,
++ * -1 for other errors.
++ */
++static int
++mxc_v4l_do_ioctl(struct inode *inode, struct file *file,
++ unsigned int ioctlnr, void *arg)
++{
++ struct video_device *dev = video_devdata(file);
++ cam_data *cam = dev->priv;
++ int retval = 0;
++ unsigned long lock_flags;
++
++ wait_event_interruptible(cam->power_queue, cam->low_power == false);
++ /* make this _really_ smp-safe */
++ if (down_interruptible(&cam->busy_lock))
++ return -EBUSY;
++
++ switch (ioctlnr) {
++ /*!
++ * V4l2 VIDIOC_QUERYCAP ioctl
++ */
++ case VIDIOC_QUERYCAP:{
++ struct v4l2_capability *cap = arg;
++ strcpy(cap->driver, "mxc_v4l2");
++ cap->version = KERNEL_VERSION(0, 1, 11);
++ cap->capabilities = V4L2_CAP_VIDEO_CAPTURE |
++ V4L2_CAP_VIDEO_OVERLAY | V4L2_CAP_STREAMING
++ | V4L2_CAP_READWRITE;
++ cap->card[0] = '\0';
++ cap->bus_info[0] = '\0';
++ retval = 0;
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_G_FMT ioctl
++ */
++ case VIDIOC_G_FMT:{
++ struct v4l2_format *gf = arg;
++ retval = mxc_v4l2_g_fmt(cam, gf);
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_S_FMT ioctl
++ */
++ case VIDIOC_S_FMT:{
++ struct v4l2_format *sf = arg;
++ retval = mxc_v4l2_s_fmt(cam, sf);
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_REQBUFS ioctl
++ */
++ case VIDIOC_REQBUFS:{
++ struct v4l2_requestbuffers *req = arg;
++ if (req->count > FRAME_NUM) {
++ printk(KERN_ERR
++ "VIDIOC_REQBUFS: not enough buffer\n");
++ req->count = FRAME_NUM;
++ }
++
++ if ((req->type != V4L2_BUF_TYPE_VIDEO_CAPTURE) ||
++ (req->memory != V4L2_MEMORY_MMAP)) {
++ printk(KERN_ERR
++ "VIDIOC_REQBUFS: wrong buffer type\n");
++ retval = -EINVAL;
++ break;
++ }
++
++ mxc_streamoff(cam);
++ mxc_free_frame_buf(cam);
++
++ retval = mxc_allocate_frame_buf(cam, req->count);
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_QUERYBUF ioctl
++ */
++ case VIDIOC_QUERYBUF:{
++ struct v4l2_buffer *buf = arg;
++ int index = buf->index;
++
++ if (buf->type != V4L2_BUF_TYPE_VIDEO_CAPTURE) {
++ printk(KERN_ERR
++ "VIDIOC_QUERYBUFS: wrong buffer type\n");
++ retval = -EINVAL;
++ break;
++ }
++
++ memset(buf, 0, sizeof(buf));
++ buf->index = index;
++
++ down(&cam->param_lock);
++ retval = mxc_v4l2_buffer_status(cam, buf);
++ up(&cam->param_lock);
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_QBUF ioctl
++ */
++ case VIDIOC_QBUF:{
++ struct v4l2_buffer *buf = arg;
++ int index = buf->index;
++
++ spin_lock_irqsave(&cam->int_lock, lock_flags);
++ cam->frame[index].buffer.m.offset = buf->m.offset;
++ if ((cam->frame[index].buffer.flags & 0x7) ==
++ V4L2_BUF_FLAG_MAPPED) {
++ cam->frame[index].buffer.flags |=
++ V4L2_BUF_FLAG_QUEUED;
++ if (cam->skip_frame > 0) {
++ list_add_tail(&cam->frame[index].queue,
++ &cam->working_q);
++ retval =
++ cam->enc_update_eba(cam->
++ frame[index].
++ buffer.m.offset,
++ &cam->
++ ping_pong_csi);
++ cam->skip_frame = 0;
++ } else {
++ list_add_tail(&cam->frame[index].queue,
++ &cam->ready_q);
++ }
++ } else if (cam->frame[index].buffer.
++ flags & V4L2_BUF_FLAG_QUEUED) {
++ printk(KERN_ERR
++ "VIDIOC_QBUF: buffer already queued\n");
++ } else if (cam->frame[index].buffer.
++ flags & V4L2_BUF_FLAG_DONE) {
++ printk(KERN_ERR
++ "VIDIOC_QBUF: overwrite done buffer.\n");
++ cam->frame[index].buffer.flags &=
++ ~V4L2_BUF_FLAG_DONE;
++ cam->frame[index].buffer.flags |=
++ V4L2_BUF_FLAG_QUEUED;
++ }
++
++ buf->flags = cam->frame[index].buffer.flags;
++ spin_unlock_irqrestore(&cam->int_lock, lock_flags);
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_DQBUF ioctl
++ */
++ case VIDIOC_DQBUF:{
++ struct v4l2_buffer *buf = arg;
++
++ retval = mxc_v4l_dqueue(cam, buf);
++
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_STREAMON ioctl
++ */
++ case VIDIOC_STREAMON:{
++ retval = mxc_streamon(cam);
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_STREAMOFF ioctl
++ */
++ case VIDIOC_STREAMOFF:{
++ retval = mxc_streamoff(cam);
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_G_CTRL ioctl
++ */
++ case VIDIOC_G_CTRL:{
++ retval = mxc_get_v42l_control(cam, arg);
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_S_CTRL ioctl
++ */
++ case VIDIOC_S_CTRL:{
++ retval = mxc_set_v42l_control(cam, arg);
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_CROPCAP ioctl
++ */
++ case VIDIOC_CROPCAP:{
++ struct v4l2_cropcap *cap = arg;
++
++ if (cap->type != V4L2_BUF_TYPE_VIDEO_CAPTURE &&
++ cap->type != V4L2_BUF_TYPE_VIDEO_OVERLAY) {
++ retval = -EINVAL;
++ break;
++ }
++ cap->bounds = cam->crop_bounds;
++ cap->defrect = cam->crop_defrect;
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_G_CROP ioctl
++ */
++ case VIDIOC_G_CROP:{
++ struct v4l2_crop *crop = arg;
++
++ if (crop->type != V4L2_BUF_TYPE_VIDEO_CAPTURE &&
++ crop->type != V4L2_BUF_TYPE_VIDEO_OVERLAY) {
++ retval = -EINVAL;
++ break;
++ }
++ crop->c = cam->crop_current;
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_S_CROP ioctl
++ */
++ case VIDIOC_S_CROP:{
++ struct v4l2_crop *crop = arg;
++ struct v4l2_rect *b = &cam->crop_bounds;
++
++ if (crop->type != V4L2_BUF_TYPE_VIDEO_CAPTURE &&
++ crop->type != V4L2_BUF_TYPE_VIDEO_OVERLAY) {
++ retval = -EINVAL;
++ break;
++ }
++
++ crop->c.top = (crop->c.top < b->top) ? b->top
++ : crop->c.top;
++ if (crop->c.top > b->top + b->height)
++ crop->c.top = b->top + b->height - 1;
++ if (crop->c.height > b->top + b->height - crop->c.top)
++ crop->c.height =
++ b->top + b->height - crop->c.top;
++
++ crop->c.left = (crop->c.left < b->left) ? b->left
++ : crop->c.left;
++ if (crop->c.left > b->left + b->width)
++ crop->c.left = b->left + b->width - 1;
++ if (crop->c.width > b->left - crop->c.left + b->width)
++ crop->c.width =
++ b->left - crop->c.left + b->width;
++
++ crop->c.width -= crop->c.width % 8;
++ crop->c.left -= crop->c.left % 4;
++ cam->crop_current = crop->c;
++
++ ipu_csi_set_window_size(cam->crop_current.width,
++ cam->crop_current.height);
++ ipu_csi_set_window_pos(cam->crop_current.left,
++ cam->crop_current.top);
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_OVERLAY ioctl
++ */
++ case VIDIOC_OVERLAY:{
++ int *on = arg;
++ if (*on) {
++ cam->overlay_on = true;
++ cam->overlay_pid = current->pid;
++ retval = start_preview(cam);
++ }
++ if (!*on) {
++ retval = stop_preview(cam);
++ cam->overlay_on = false;
++ }
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_G_FBUF ioctl
++ */
++ case VIDIOC_G_FBUF:{
++ struct v4l2_framebuffer *fb = arg;
++ *fb = cam->v4l2_fb;
++ fb->capability = V4L2_FBUF_CAP_EXTERNOVERLAY;
++ break;
++ }
++
++ /*!
++ * V4l2 VIDIOC_S_FBUF ioctl
++ */
++ case VIDIOC_S_FBUF:{
++ struct v4l2_framebuffer *fb = arg;
++ cam->v4l2_fb = *fb;
++ break;
++ }
++
++ case VIDIOC_G_PARM:{
++ struct v4l2_streamparm *parm = arg;
++ if (parm->type != V4L2_BUF_TYPE_VIDEO_CAPTURE) {
++ printk(KERN_ERR "VIDIOC_G_PARM invalid type\n");
++ retval = -EINVAL;
++ break;
++ }
++ parm->parm.capture = cam->streamparm.parm.capture;
++ break;
++ }
++ case VIDIOC_S_PARM:{
++ struct v4l2_streamparm *parm = arg;
++ retval = mxc_v4l2_s_param(cam, parm);
++ break;
++ }
++
++ /* linux v4l2 bug, kernel c0485619 user c0405619 */
++ case VIDIOC_ENUMSTD:{
++ struct v4l2_standard *e = arg;
++ *e = cam->standard;
++ printk(KERN_ERR "VIDIOC_ENUMSTD call\n");
++ retval = 0;
++ break;
++ }
++
++ case VIDIOC_G_STD:{
++ v4l2_std_id *e = arg;
++ *e = cam->standard.id;
++ break;
++ }
++
++ case VIDIOC_S_STD:{
++ break;
++ }
++
++ case VIDIOC_ENUMOUTPUT:
++ {
++ struct v4l2_output *output = arg;
++
++ if (output->index >= MXC_V4L2_CAPTURE_NUM_OUTPUTS) {
++ retval = -EINVAL;
++ break;
++ }
++
++ *output = mxc_capture_outputs[output->index];
++
++ break;
++ }
++ case VIDIOC_G_OUTPUT:
++ {
++ int *p_output_num = arg;
++
++ *p_output_num = cam->output;
++ break;
++ }
++ case VIDIOC_S_OUTPUT:
++ {
++ int *p_output_num = arg;
++
++ if (*p_output_num >= MXC_V4L2_CAPTURE_NUM_OUTPUTS) {
++ retval = -EINVAL;
++ break;
++ }
++
++ cam->output = *p_output_num;
++ break;
++ }
++
++ case VIDIOC_ENUM_FMT:
++ case VIDIOC_TRY_FMT:
++ case VIDIOC_QUERYCTRL:
++ case VIDIOC_ENUMINPUT:
++ case VIDIOC_G_INPUT:
++ case VIDIOC_S_INPUT:
++ case VIDIOC_G_TUNER:
++ case VIDIOC_S_TUNER:
++ case VIDIOC_G_FREQUENCY:
++ case VIDIOC_S_FREQUENCY:
++ default:
++ retval = -EINVAL;
++ break;
++ }
++
++ up(&cam->busy_lock);
++ return retval;
++}
++
++/*
++ * V4L interface - ioctl function
++ *
++ * @return None
++ */
++static int
++mxc_v4l_ioctl(struct inode *inode, struct file *file,
++ unsigned int cmd, unsigned long arg)
++{
++ return video_usercopy(inode, file, cmd, arg, mxc_v4l_do_ioctl);
++}
++
++/*!
++ * V4L interface - mmap function
++ *
++ * @param file structure file *
++ *
++ * @param vma structure vm_area_struct *
++ *
++ * @return status 0 Success, EINTR busy lock error, ENOBUFS remap_page error
++ */
++static int mxc_mmap(struct file *file, struct vm_area_struct *vma)
++{
++ struct video_device *dev = video_devdata(file);
++ unsigned long size;
++ int res = 0;
++ cam_data *cam = dev->priv;
++
++ pr_debug("pgoff=0x%lx, start=0x%lx, end=0x%lx\n",
++ vma->vm_pgoff, vma->vm_start, vma->vm_end);
++
++ /* make this _really_ smp-safe */
++ if (down_interruptible(&cam->busy_lock))
++ return -EINTR;
++
++ size = vma->vm_end - vma->vm_start;
++ vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot);
++
++ if (remap_pfn_range(vma, vma->vm_start,
++ vma->vm_pgoff, size, vma->vm_page_prot)) {
++ printk(KERN_ERR "mxc_mmap: remap_pfn_range failed\n");
++ res = -ENOBUFS;
++ goto mxc_mmap_exit;
++ }
++
++ vma->vm_flags &= ~VM_IO; /* using shared anonymous pages */
++
++ mxc_mmap_exit:
++ up(&cam->busy_lock);
++ return res;
++}
++
++/*!
++ * V4L interface - poll function
++ *
++ * @param file structure file *
++ *
++ * @param wait structure poll_table *
++ *
++ * @return status POLLIN | POLLRDNORM
++ */
++static unsigned int mxc_poll(struct file *file, poll_table * wait)
++{
++ struct video_device *dev = video_devdata(file);
++ cam_data *cam = dev->priv;
++ wait_queue_head_t *queue = NULL;
++ int res = POLLIN | POLLRDNORM;
++
++ if (down_interruptible(&cam->busy_lock))
++ return -EINTR;
++
++ queue = &cam->enc_queue;
++ poll_wait(file, queue, wait);
++
++ up(&cam->busy_lock);
++ return res;
++}
++
++static struct
++file_operations mxc_v4l_fops = {
++ .owner = THIS_MODULE,
++ .open = mxc_v4l_open,
++ .release = mxc_v4l_close,
++ .read = mxc_v4l_read,
++ .ioctl = mxc_v4l_ioctl,
++ .mmap = mxc_mmap,
++ .poll = mxc_poll,
++};
++
++static struct video_device mxc_v4l_template = {
++ .owner = THIS_MODULE,
++ .name = "Mxc Camera",
++ .type = 0,
++ .type2 = VID_TYPE_CAPTURE,
++ .hardware = 0,
++ .fops = &mxc_v4l_fops,
++ .release = video_device_release,
++};
++
++static void camera_platform_release(struct device *device)
++{
++}
++
++/*! Device Definition for Mt9v111 devices */
++static struct platform_device mxc_v4l2_devices = {
++ .name = "mxc_v4l2",
++ .dev = {
++ .release = camera_platform_release,
++ },
++ .id = 0,
++};
++
++extern struct camera_sensor camera_sensor_if;
++
++/*!
++* Camera V4l2 callback function.
++*
++* @param mask u32
++*
++* @param dev void device structure
++*
++* @return status
++*/
++static void camera_callback(u32 mask, void *dev)
++{
++ struct mxc_v4l_frame *done_frame;
++ struct mxc_v4l_frame *ready_frame;
++
++ cam_data *cam = (cam_data *) dev;
++ if (cam == NULL)
++ return;
++
++ if (list_empty(&cam->working_q)) {
++ printk(KERN_ERR "camera_callback: working queue empty\n");
++ return;
++ }
++
++ done_frame =
++ list_entry(cam->working_q.next, struct mxc_v4l_frame, queue);
++ if (done_frame->buffer.flags & V4L2_BUF_FLAG_QUEUED) {
++ done_frame->buffer.flags |= V4L2_BUF_FLAG_DONE;
++ done_frame->buffer.flags &= ~V4L2_BUF_FLAG_QUEUED;
++
++ if (list_empty(&cam->ready_q)) {
++ cam->skip_frame++;
++ } else {
++ ready_frame =
++ list_entry(cam->ready_q.next, struct mxc_v4l_frame,
++ queue);
++ list_del(cam->ready_q.next);
++ list_add_tail(&ready_frame->queue, &cam->working_q);
++ cam->enc_update_eba(ready_frame->buffer.m.offset,
++ &cam->ping_pong_csi);
++ }
++
++ /* Added to the done queue */
++ list_del(cam->working_q.next);
++ list_add_tail(&done_frame->queue, &cam->done_q);
++
++ /* Wake up the queue */
++ cam->enc_counter++;
++ wake_up_interruptible(&cam->enc_queue);
++ } else {
++ printk(KERN_ERR "camera_callback :buffer not queued\n");
++ }
++}
++
++/*!
++ * initialize cam_data structure
++ *
++ * @param cam structure cam_data *
++ *
++ * @return status 0 Success
++ */
++static void init_camera_struct(cam_data * cam)
++{
++ /* Default everything to 0 */
++ memset(cam, 0, sizeof(cam_data));
++
++ init_MUTEX(&cam->param_lock);
++ init_MUTEX(&cam->busy_lock);
++
++ cam->video_dev = video_device_alloc();
++ if (cam->video_dev == NULL)
++ return;
++
++ *(cam->video_dev) = mxc_v4l_template;
++
++ video_set_drvdata(cam->video_dev, cam);
++ dev_set_drvdata(&mxc_v4l2_devices.dev, (void *)cam);
++ cam->video_dev->minor = -1;
++
++ init_waitqueue_head(&cam->enc_queue);
++ init_waitqueue_head(&cam->still_queue);
++
++ /* setup cropping */
++ cam->crop_bounds.left = 0;
++ cam->crop_bounds.width = 640;
++ cam->crop_bounds.top = 0;
++ cam->crop_bounds.height = 480;
++ cam->crop_current = cam->crop_defrect = cam->crop_bounds;
++ ipu_csi_set_window_size(cam->crop_current.width,
++ cam->crop_current.height);
++ ipu_csi_set_window_pos(cam->crop_current.left, cam->crop_current.top);
++ cam->streamparm.parm.capture.capturemode = 0;
++
++ cam->standard.index = 0;
++ cam->standard.id = V4L2_STD_UNKNOWN;
++ cam->standard.frameperiod.denominator = 30;
++ cam->standard.frameperiod.numerator = 1;
++ cam->standard.framelines = 480;
++ cam->streamparm.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
++ cam->streamparm.parm.capture.timeperframe = cam->standard.frameperiod;
++ cam->streamparm.parm.capture.capability = V4L2_CAP_TIMEPERFRAME;
++ cam->overlay_on = false;
++ cam->capture_on = false;
++ cam->skip_frame = 0;
++ cam->v4l2_fb.flags = V4L2_FBUF_FLAG_OVERLAY;
++
++ cam->v2f.fmt.pix.sizeimage = 352 * 288 * 3 / 2;
++ cam->v2f.fmt.pix.bytesperline = 288 * 3 / 2;
++ cam->v2f.fmt.pix.width = 288;
++ cam->v2f.fmt.pix.height = 352;
++ cam->v2f.fmt.pix.pixelformat = V4L2_PIX_FMT_YUV420;
++ cam->win.w.width = 160;
++ cam->win.w.height = 160;
++ cam->win.w.left = 0;
++ cam->win.w.top = 0;
++
++ cam->cam_sensor = &camera_sensor_if;
++ cam->enc_callback = camera_callback;
++ init_waitqueue_head(&cam->power_queue);
++ cam->int_lock = SPIN_LOCK_UNLOCKED;
++ spin_lock_init(&cam->int_lock);
++}
++
++extern void gpio_sensor_active(void);
++extern void gpio_sensor_inactive(void);
++
++/*!
++ * camera_power function
++ * Turn Sensor power On/Off
++ *
++ * @param cameraOn true to turn camera on, otherwise shut down
++ *
++ * @return status
++ */
++static u8 camera_power(bool cameraOn)
++{
++ if (cameraOn == true) {
++ gpio_sensor_active();
++ ipu_csi_enable_mclk(csi_mclk_flag_backup, true, true);
++ } else {
++ csi_mclk_flag_backup = ipu_csi_read_mclk_flag();
++ ipu_csi_enable_mclk(csi_mclk_flag_backup, false, false);
++ gpio_sensor_inactive();
++ }
++ return 0;
++}
++
++/*!
++ * This function is called to put the sensor in a low power state. Refer to the
++ * document driver-model/driver.txt in the kernel source tree for more
++ * information.
++ *
++ * @param pdev the device structure used to give information on which I2C
++ * to suspend
++ * @param state the power state the device is entering
++ *
++ * @return The function returns 0 on success and -1 on failure.
++ */
++static int mxc_v4l2_suspend(struct platform_device *pdev, pm_message_t state)
++{
++ cam_data *cam = platform_get_drvdata(pdev);
++
++ if (cam == NULL) {
++ return -1;
++ }
++
++ cam->low_power = true;
++
++ if (cam->overlay_on == true)
++ stop_preview(cam);
++ if ((cam->capture_on == true) && cam->enc_disable) {
++ cam->enc_disable(cam);
++ }
++ camera_power(false);
++
++ return 0;
++}
++
++/*!
++ * This function is called to bring the sensor back from a low power state.Refer
++ * to the document driver-model/driver.txt in the kernel source tree for more
++ * information.
++ *
++ * @param pdev the device structure
++ *
++ * @return The function returns 0 on success and -1 on failure
++ */
++static int mxc_v4l2_resume(struct platform_device *pdev)
++{
++ cam_data *cam = platform_get_drvdata(pdev);
++
++ if (cam == NULL) {
++ return -1;
++ }
++
++ cam->low_power = false;
++ wake_up_interruptible(&cam->power_queue);
++
++ if (cam->overlay_on == true)
++ start_preview(cam);
++ if (cam->capture_on == true)
++ mxc_streamon(cam);
++ camera_power(true);
++
++ return 0;
++}
++
++/*!
++ * This structure contains pointers to the power management callback functions.
++ */
++static struct platform_driver mxc_v4l2_driver = {
++ .driver = {
++ .name = "mxc_v4l2",
++ },
++ .probe = NULL,
++ .remove = NULL,
++ .suspend = mxc_v4l2_suspend,
++ .resume = mxc_v4l2_resume,
++ .shutdown = NULL,
++};
++
++/*!
++ * Entry point for the V4L2
++ *
++ * @return Error code indicating success or failure
++ */
++static __init int camera_init(void)
++{
++ u8 err = 0;
++
++ /* Register the device driver structure. */
++ err = platform_driver_register(&mxc_v4l2_driver);
++ if (err != 0) {
++ printk("camera_init: platform_driver_register failed.\n");
++ return err;
++ }
++
++ if ((g_cam = kmalloc(sizeof(cam_data), GFP_KERNEL)) == NULL) {
++ printk(KERN_ERR "failed to mxc_v4l_register_camera\n");
++ return -1;
++ }
++
++ init_camera_struct(g_cam);
++
++ /* Register the I2C device */
++ err = platform_device_register(&mxc_v4l2_devices);
++ if (err != 0) {
++ printk(KERN_ERR
++ "camera_init: platform_device_register failed.\n");
++ video_device_release(g_cam->video_dev);
++ kfree(g_cam);
++ g_cam = NULL;
++ }
++
++ /* register v4l device */
++ if (video_register_device(g_cam->video_dev, VFL_TYPE_GRABBER, video_nr)
++ == -1) {
++ platform_device_unregister(&mxc_v4l2_devices);
++ platform_driver_unregister(&mxc_v4l2_driver);
++ video_device_release(g_cam->video_dev);
++ kfree(g_cam);
++ g_cam = NULL;
++ printk(KERN_ERR "video_register_device failed\n");
++ return -1;
++ }
++
++ return err;
++}
++
++/*!
++ * Exit and cleanup for the V4L2
++ *
++ */
++static void __exit camera_exit(void)
++{
++ pr_info("unregistering video\n");
++ video_unregister_device(g_cam->video_dev);
++
++ platform_driver_unregister(&mxc_v4l2_driver);
++ platform_device_unregister(&mxc_v4l2_devices);
++
++ if (g_cam->open_count) {
++ printk(KERN_ERR "camera open -- setting ops to NULL\n");
++ } else {
++ pr_info("freeing camera\n");
++ mxc_free_frame_buf(g_cam);
++ kfree(g_cam);
++ g_cam = NULL;
++ }
++}
++
++module_init(camera_init);
++module_exit(camera_exit);
++
++module_param(video_nr, int, 0444);
++MODULE_AUTHOR("Freescale Semiconductor, Inc.");
++MODULE_DESCRIPTION("V4L2 capture driver for Mxc based cameras");
++MODULE_LICENSE("GPL");
++MODULE_SUPPORTED_DEVICE("video");
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/capture/mxc_v4l2_capture.h linux-2.6.28-karo/drivers/media/video/mxc/capture/mxc_v4l2_capture.h
+--- linux-2.6.28/drivers/media/video/mxc/capture/mxc_v4l2_capture.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/capture/mxc_v4l2_capture.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,190 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @defgroup MXC_V4L2_CAPTURE MXC V4L2 Video Capture Driver
++ */
++/*!
++ * @file mxc_v4l2_capture.h
++ *
++ * @brief mxc V4L2 capture device API Header file
++ *
++ * It include all the defines for frame operations, also three structure defines
++ * use case ops structure, common v4l2 driver structure and frame structure.
++ *
++ * @ingroup MXC_V4L2_CAPTURE
++ */
++#ifndef __MXC_V4L2_CAPTURE_H__
++#define __MXC_V4L2_CAPTURE_H__
++
++#include <asm/uaccess.h>
++#include <linux/list.h>
++#include <linux/smp_lock.h>
++
++#include <media/v4l2-dev.h>
++#include <asm/arch/ipu.h>
++#include <asm/arch/mxc_v4l2.h>
++
++#define FRAME_NUM 3
++
++/*!
++ * v4l2 frame structure.
++ */
++struct mxc_v4l_frame {
++ u32 paddress;
++ void *vaddress;
++ int count;
++ int width;
++ int height;
++
++ struct v4l2_buffer buffer;
++ struct list_head queue;
++ int index;
++};
++
++typedef struct {
++ u8 clk_mode;
++ u8 ext_vsync;
++ u8 Vsync_pol;
++ u8 Hsync_pol;
++ u8 pixclk_pol;
++ u8 data_pol;
++ u8 data_width;
++ u16 width;
++ u16 height;
++ u32 pixel_fmt;
++ u32 mclk;
++} sensor_interface;
++
++/* Sensor control function */
++struct camera_sensor {
++ struct module *owner;
++ void (*set_color) (int bright, int saturation, int red, int green,
++ int blue);
++ void (*get_color) (int *bright, int *saturation, int *red, int *green,
++ int *blue);
++ void (*set_ae_mode) (int ae_mode);
++ void (*get_ae_mode) (int *ae_mode);
++ void (*set_ae) (int active);
++ void (*set_ae_limit) (int limit);
++ void (*set_awb) (int active);
++ void (*flicker_control) (int control);
++ void (*get_control_params) (int *ae, int *awb, int *flicker);
++ sensor_interface *(*config) (int *frame_rate, int high_quality);
++ sensor_interface *(*reset) (void);
++ int (*get_status) (void);
++};
++
++/*!
++ * common v4l2 driver structure.
++ */
++typedef struct _cam_data {
++ struct video_device *video_dev;
++
++ /* semaphore guard against SMP multithreading */
++ struct semaphore busy_lock;
++
++ int open_count;
++
++ /* params lock for this camera */
++ struct semaphore param_lock;
++
++ /* Encorder */
++ struct list_head ready_q;
++ struct list_head done_q;
++ struct list_head working_q;
++ int ping_pong_csi;
++ spinlock_t int_lock;
++ struct mxc_v4l_frame frame[FRAME_NUM];
++ int skip_frame;
++ wait_queue_head_t overflow_queue;
++ int overflow;
++ wait_queue_head_t enc_queue;
++ int enc_counter;
++ dma_addr_t rot_enc_bufs[2];
++ void *rot_enc_bufs_vaddr[2];
++ int rot_enc_buf_size[2];
++ enum v4l2_buf_type type;
++
++ /* still image capture */
++ wait_queue_head_t still_queue;
++ int still_counter;
++ dma_addr_t still_buf;
++ void *still_buf_vaddr;
++
++ /* overlay */
++ struct v4l2_window win;
++ struct v4l2_framebuffer v4l2_fb;
++ dma_addr_t vf_bufs[2];
++ void *vf_bufs_vaddr[2];
++ int vf_bufs_size[2];
++ dma_addr_t rot_vf_bufs[2];
++ void *rot_vf_bufs_vaddr[2];
++ int rot_vf_buf_size[2];
++ bool overlay_active;
++ int output;
++ struct fb_info *overlay_fb;
++
++ /* v4l2 format */
++ struct v4l2_format v2f;
++ int rotation;
++ struct v4l2_mxc_offset offset;
++
++ /* V4l2 control bit */
++ int bright;
++ int hue;
++ int contrast;
++ int saturation;
++ int red;
++ int green;
++ int blue;
++ int ae_mode;
++ int ae_enable;
++ int ae_limit;
++ int awb_enable;
++ int flicker_ctrl;
++
++ /* standart */
++ struct v4l2_streamparm streamparm;
++ struct v4l2_standard standard;
++
++ /* crop */
++ struct v4l2_rect crop_bounds;
++ struct v4l2_rect crop_defrect;
++ struct v4l2_rect crop_current;
++
++ int (*enc_update_eba) (dma_addr_t eba, int *bufferNum);
++ int (*enc_enable) (void *private);
++ int (*enc_disable) (void *private);
++ void (*enc_callback) (u32 mask, void *dev);
++ int (*vf_start_adc) (void *private);
++ int (*vf_stop_adc) (void *private);
++ int (*vf_start_sdc) (void *private);
++ int (*vf_stop_sdc) (void *private);
++ int (*csi_start) (void *private);
++ int (*csi_stop) (void *private);
++
++ /* misc status flag */
++ bool overlay_on;
++ bool capture_on;
++ int overlay_pid;
++ int capture_pid;
++ bool low_power;
++ wait_queue_head_t power_queue;
++
++ /* camera sensor interface */
++ struct camera_sensor *cam_sensor;
++} cam_data;
++
++void set_mclk_rate(uint32_t * p_mclk_freq);
++#endif /* __MXC_V4L2_CAPTURE_H__ */
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/capture/ov2640.c linux-2.6.28-karo/drivers/media/video/mxc/capture/ov2640.c
+--- linux-2.6.28/drivers/media/video/mxc/capture/ov2640.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/capture/ov2640.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,817 @@
++/*
++ * Copyright 2005-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++#include <linux/module.h>
++#include <linux/init.h>
++#include <linux/slab.h>
++#include <linux/ctype.h>
++#include <linux/types.h>
++#include <linux/delay.h>
++#include <linux/device.h>
++#include <linux/i2c.h>
++#include <asm/arch/mxc_i2c.h>
++
++#ifdef NONSENSE
++#include <asm/arch/pmic_power.h>
++#endif
++
++#include "mxc_v4l2_capture.h"
++
++#define OV2640_DEBUG
++#ifdef OV2640_DEBUG
++#define PRINTK(fmt...) printk(KERN_DEBUG fmt)
++#else
++#define PRINTK(fmt...)
++#endif
++
++#define OV2640_I2C_ADDRESS 0x30
++
++static sensor_interface *interface_param = NULL;
++static int reset_frame_rate = 30;
++static int ov2640_attach(struct i2c_adapter *adapter);
++static int ov2640_detach(struct i2c_client *client);
++
++static struct i2c_driver ov2640_i2c_driver = {
++ .driver = {
++ .owner = THIS_MODULE,
++ .name = "OV2640 Client",
++ },
++ .attach_adapter = ov2640_attach,
++ .detach_client = ov2640_detach,
++};
++
++static struct i2c_client ov2640_i2c_client = {
++ .name = "ov2640 I2C dev",
++ .addr = OV2640_I2C_ADDRESS,
++ .driver = &ov2640_i2c_driver,
++};
++
++/*!
++ * ov2640 I2C attach function
++ *
++ * @param adapter struct i2c_adapter *
++ * @return Error code indicating success or failure
++ */
++static int ov2640_attach(struct i2c_adapter *adapter)
++{
++ int ret;
++
++ if (strcmp(adapter->name, MXC_ADAPTER_NAME) != 0) {
++ PRINTK("%s: %s\n", __func__, adapter->name);
++ return -ENODEV;
++ }
++ ov2640_i2c_client.adapter = adapter;
++
++ ret = i2c_attach_client(&ov2640_i2c_client);
++ if (ret != 0) {
++ ov2640_i2c_client.adapter = NULL;
++ PRINTK("%s: i2c_attach_client failed\n", __func__);
++ return ret;
++ }
++
++ interface_param = kzalloc(sizeof(sensor_interface), GFP_KERNEL);
++ if (!interface_param) {
++ PRINTK("%s: kmalloc failed\n", __func__);
++ return -ENOMEM;
++ }
++ return 0;
++}
++
++/*!
++ * ov2640 I2C detach function
++ *
++ * @param client struct i2c_client *
++ * @return Error code indicating success or failure
++ */
++static int ov2640_detach(struct i2c_client *client)
++{
++ int err;
++#ifdef NONSENSE
++ PMIC_STATUS ret;
++#endif
++ err = i2c_detach_client(&ov2640_i2c_client);
++ ov2640_i2c_client.adapter = NULL;
++
++ kfree(interface_param);
++ interface_param = NULL;
++
++#ifdef NONSENSE
++ /* DOVDD */
++ if ((ret = pmic_power_regulator_off(REGU_GPO3)) < 0) {
++ PRINTK("%s:REGU_GPO3 power off error:%d\n", __func__, ret);
++ return 0;
++ } else {
++ PRINTK("%s:REGU_GPO3 power off ok\n", __func__);
++ }
++ PRINTK("%s:OV2640 power off ok\n", __func__);
++#endif
++ return err;
++}
++
++static int
++ov2640_i2c_client_xfer(int addr, u8 reg, char *buf, int num, int tran_flag)
++{
++ struct i2c_msg msg[2];
++ int ret;
++
++ msg[0].addr = addr;
++ msg[0].len = 1;
++ msg[0].buf = &reg;
++ msg[0].flags = tran_flag;
++ msg[0].flags &= ~I2C_M_RD;
++
++ msg[1].addr = addr;
++ msg[1].len = num;
++ msg[1].buf = buf;
++ msg[1].flags = tran_flag;
++
++ if (tran_flag & MXC_I2C_FLAG_READ) {
++ msg[1].flags |= I2C_M_RD;
++ } else {
++ msg[1].flags &= ~I2C_M_RD;
++ }
++
++#if 1
++ BUG_ON(ov2640_i2c_client.adapter == NULL);
++#else
++ if (ov2640_i2c_client.adapter == NULL) {
++ PRINTK("%s: adapter error\n", __func__);
++ return -ENODEV;
++ }
++#endif
++ ret = i2c_transfer(ov2640_i2c_client.adapter, msg, 2);
++ if (ret >= 0) {
++ /* PRINTK("%s:i2c transfer num:%d\n", __func__, ret); */
++ return 0;
++ }
++ PRINTK("%s: i2c transfer error: %d\n", __func__, ret);
++ return ret;
++}
++
++static int ov2640_write_reg(u8 reg, u8 val)
++{
++ int ret;
++
++ ret = ov2640_i2c_client_xfer(OV2640_I2C_ADDRESS, reg, &val, 1, 0);
++ if (ret < 0) {
++ PRINTK("%s: write reg error: reg=%02x, val=%02x ret=%d\n",
++ __func__, reg, val, ret);
++ return ret;
++ } else if (ret != 1) {
++ PRINTK("%s: write reg error: reg=%02x, val=%02x ret=%d\n",
++ __func__, reg, val, ret);
++ return -EIO;
++ }
++ return 0;
++}
++
++/* should be replaced by width and height version. */
++static int ov2640_init_1600_1120(void)
++{
++ int ret;
++
++ ret = ov2640_write_reg(0xff, 1);
++ if (ret != 0) {
++ return ret;
++ }
++ ov2640_write_reg(0x12, 0x80);
++ udelay(1000);
++ ov2640_write_reg(0xff, 0x00);
++ ov2640_write_reg(0x2c, 0xff);
++ ov2640_write_reg(0x2e, 0xdf);
++ ov2640_write_reg(0xff, 0x01);
++ ov2640_write_reg(0x3c, 0x32);
++ ov2640_write_reg(0x11, 0x01);
++ ov2640_write_reg(0x09, 0x00);
++ ov2640_write_reg(0x04, 0x28);
++ ov2640_write_reg(0x13, 0xe5);
++ ov2640_write_reg(0x14, 0x48);
++ ov2640_write_reg(0x2c, 0x0c);
++ ov2640_write_reg(0x33, 0x78);
++ ov2640_write_reg(0x3a, 0x33);
++ ov2640_write_reg(0x3b, 0xfb);
++ ov2640_write_reg(0x3e, 0x00);
++ ov2640_write_reg(0x43, 0x11);
++ ov2640_write_reg(0x16, 0x10);
++ ov2640_write_reg(0x39, 0x82);
++ ov2640_write_reg(0x35, 0x88);
++ ov2640_write_reg(0x22, 0x0a);
++ ov2640_write_reg(0x37, 0x40);
++ ov2640_write_reg(0x23, 0x00);
++ ov2640_write_reg(0x34, 0xa0);
++ ov2640_write_reg(0x36, 0x1a);
++ ov2640_write_reg(0x06, 0x02);
++ ov2640_write_reg(0x07, 0xc0);
++ ov2640_write_reg(0x0d, 0xb7);
++ ov2640_write_reg(0x0e, 0x01);
++ ov2640_write_reg(0x4c, 0x00);
++ ov2640_write_reg(0x4a, 0x81);
++ ov2640_write_reg(0x21, 0x99);
++ ov2640_write_reg(0x24, 0x40);
++ ov2640_write_reg(0x25, 0x38);
++ ov2640_write_reg(0x26, 0x82);
++ ov2640_write_reg(0x5c, 0x00);
++ ov2640_write_reg(0x63, 0x00);
++ ov2640_write_reg(0x46, 0x3f);
++ ov2640_write_reg(0x0c, 0x3c);
++ ov2640_write_reg(0x5d, 0x55);
++ ov2640_write_reg(0x5e, 0x7d);
++ ov2640_write_reg(0x5f, 0x7d);
++ ov2640_write_reg(0x60, 0x55);
++ ov2640_write_reg(0x61, 0x70);
++ ov2640_write_reg(0x62, 0x80);
++ ov2640_write_reg(0x7c, 0x05);
++ ov2640_write_reg(0x20, 0x80);
++ ov2640_write_reg(0x28, 0x30);
++ ov2640_write_reg(0x6c, 0x00);
++ ov2640_write_reg(0x6d, 0x80);
++ ov2640_write_reg(0x6e, 0x00);
++ ov2640_write_reg(0x70, 0x02);
++ ov2640_write_reg(0x71, 0x94);
++ ov2640_write_reg(0x73, 0xc1);
++ ov2640_write_reg(0x3d, 0x34);
++ ov2640_write_reg(0x5a, 0x57);
++ ov2640_write_reg(0x4f, 0xbb);
++ ov2640_write_reg(0x50, 0x9c);
++ ov2640_write_reg(0xff, 0x00);
++ ov2640_write_reg(0xe5, 0x7f);
++ ov2640_write_reg(0xf9, 0xc0);
++ ov2640_write_reg(0x41, 0x24);
++ ov2640_write_reg(0x44, 0x06);
++ ov2640_write_reg(0xe0, 0x14);
++ ov2640_write_reg(0x76, 0xff);
++ ov2640_write_reg(0x33, 0xa0);
++ ov2640_write_reg(0x42, 0x20);
++ ov2640_write_reg(0x43, 0x18);
++ ov2640_write_reg(0x4c, 0x00);
++ ov2640_write_reg(0x87, 0xd0);
++ ov2640_write_reg(0xd7, 0x03);
++ ov2640_write_reg(0xd9, 0x10);
++ ov2640_write_reg(0xd3, 0x82);
++ ov2640_write_reg(0xc8, 0x08);
++ ov2640_write_reg(0xc9, 0x80);
++ ov2640_write_reg(0x7c, 0x00);
++ ov2640_write_reg(0x7d, 0x00);
++ ov2640_write_reg(0x7c, 0x03);
++ ov2640_write_reg(0x7d, 0x48);
++ ov2640_write_reg(0x7d, 0x48);
++ ov2640_write_reg(0x7c, 0x08);
++ ov2640_write_reg(0x7d, 0x20);
++ ov2640_write_reg(0x7d, 0x10);
++ ov2640_write_reg(0x7d, 0x0e);
++ ov2640_write_reg(0x90, 0x00);
++ ov2640_write_reg(0x91, 0x0e);
++ ov2640_write_reg(0x91, 0x1a);
++ ov2640_write_reg(0x91, 0x31);
++ ov2640_write_reg(0x91, 0x5a);
++ ov2640_write_reg(0x91, 0x69);
++ ov2640_write_reg(0x91, 0x75);
++ ov2640_write_reg(0x91, 0x7e);
++ ov2640_write_reg(0x91, 0x88);
++ ov2640_write_reg(0x91, 0x8f);
++ ov2640_write_reg(0x91, 0x96);
++ ov2640_write_reg(0x91, 0xa3);
++ ov2640_write_reg(0x91, 0xaf);
++ ov2640_write_reg(0x91, 0xc4);
++ ov2640_write_reg(0x91, 0xd7);
++ ov2640_write_reg(0x91, 0xe8);
++ ov2640_write_reg(0x91, 0x20);
++
++ ov2640_write_reg(0x92, 0x00);
++ ov2640_write_reg(0x93, 0x06);
++ ov2640_write_reg(0x93, 0xe3);
++ ov2640_write_reg(0x93, 0x03);
++ ov2640_write_reg(0x93, 0x03);
++ ov2640_write_reg(0x93, 0x00);
++ ov2640_write_reg(0x93, 0x02);
++ ov2640_write_reg(0x93, 0x00);
++ ov2640_write_reg(0x93, 0x00);
++ ov2640_write_reg(0x93, 0x00);
++ ov2640_write_reg(0x93, 0x00);
++ ov2640_write_reg(0x93, 0x00);
++ ov2640_write_reg(0x93, 0x00);
++ ov2640_write_reg(0x93, 0x00);
++
++ ov2640_write_reg(0x96, 0x00);
++ ov2640_write_reg(0x97, 0x08);
++ ov2640_write_reg(0x97, 0x19);
++ ov2640_write_reg(0x97, 0x02);
++ ov2640_write_reg(0x97, 0x0c);
++ ov2640_write_reg(0x97, 0x24);
++ ov2640_write_reg(0x97, 0x30);
++ ov2640_write_reg(0x97, 0x28);
++ ov2640_write_reg(0x97, 0x26);
++ ov2640_write_reg(0x97, 0x02);
++ ov2640_write_reg(0x97, 0x98);
++ ov2640_write_reg(0x97, 0x80);
++ ov2640_write_reg(0x97, 0x00);
++ ov2640_write_reg(0x97, 0x00);
++
++ ov2640_write_reg(0xa4, 0x00);
++ ov2640_write_reg(0xa8, 0x00);
++ ov2640_write_reg(0xc5, 0x11);
++ ov2640_write_reg(0xc6, 0x51);
++ ov2640_write_reg(0xbf, 0x80);
++ ov2640_write_reg(0xc7, 0x10);
++ ov2640_write_reg(0xb6, 0x66);
++ ov2640_write_reg(0xb8, 0xa5);
++ ov2640_write_reg(0xb7, 0x64);
++ ov2640_write_reg(0xb9, 0x7c);
++ ov2640_write_reg(0xb3, 0xaf);
++ ov2640_write_reg(0xb4, 0x97);
++ ov2640_write_reg(0xb5, 0xff);
++ ov2640_write_reg(0xb0, 0xc5);
++ ov2640_write_reg(0xb1, 0x94);
++ ov2640_write_reg(0xb2, 0x0f);
++ ov2640_write_reg(0xc4, 0x5c);
++
++ ov2640_write_reg(0xa6, 0x00);
++ ov2640_write_reg(0xa7, 0x20);
++ ov2640_write_reg(0xa7, 0xd8);
++ ov2640_write_reg(0xa7, 0x1b);
++ ov2640_write_reg(0xa7, 0x31);
++ ov2640_write_reg(0xa7, 0x00);
++ ov2640_write_reg(0xa7, 0x18);
++ ov2640_write_reg(0xa7, 0x20);
++ ov2640_write_reg(0xa7, 0xd8);
++ ov2640_write_reg(0xa7, 0x19);
++ ov2640_write_reg(0xa7, 0x31);
++ ov2640_write_reg(0xa7, 0x00);
++ ov2640_write_reg(0xa7, 0x18);
++ ov2640_write_reg(0xa7, 0x20);
++ ov2640_write_reg(0xa7, 0xd8);
++ ov2640_write_reg(0xa7, 0x19);
++ ov2640_write_reg(0xa7, 0x31);
++ ov2640_write_reg(0xa7, 0x00);
++ ov2640_write_reg(0xa7, 0x18);
++
++ ov2640_write_reg(0xc0, 0xc8);
++ ov2640_write_reg(0xc1, 0x96);
++ ov2640_write_reg(0x86, 0x3d);
++ ov2640_write_reg(0x50, 0x00);
++ ov2640_write_reg(0x51, 0x90);
++ ov2640_write_reg(0x52, 0x18);
++ ov2640_write_reg(0x53, 0x00);
++ ov2640_write_reg(0x54, 0x00);
++ ov2640_write_reg(0x55, 0x88);
++ ov2640_write_reg(0x57, 0x00);
++ ov2640_write_reg(0x5a, 0x90);
++ ov2640_write_reg(0x5b, 0x18);
++ ov2640_write_reg(0x5c, 0x05);
++ ov2640_write_reg(0xc3, 0xef);
++ ov2640_write_reg(0x7f, 0x00);
++ ov2640_write_reg(0xda, 0x01);
++ ov2640_write_reg(0xe5, 0x1f);
++ ov2640_write_reg(0xe1, 0x67);
++ ov2640_write_reg(0xe0, 0x00);
++ ov2640_write_reg(0xdd, 0x7f);
++ ov2640_write_reg(0x05, 0x00);
++
++ return 0;
++}
++
++static int ov2640_init_800_600(void)
++{
++ int ret;
++ ret = ov2640_write_reg(0xff, 0x00);
++ if (ret != 0) {
++ return ret;
++ }
++ ov2640_write_reg(0xff, 0x01);
++ ov2640_write_reg(0x12, 0x80);
++ udelay(1000);
++ ov2640_write_reg(0xff, 0x00);
++ ov2640_write_reg(0x2c, 0xff);
++ ov2640_write_reg(0x2e, 0xdf);
++ ov2640_write_reg(0xff, 0x01);
++ ov2640_write_reg(0x3c, 0x32);
++ ov2640_write_reg(0x11, 0x01);
++ ov2640_write_reg(0x09, 0x00);
++ ov2640_write_reg(0x04, 0x28);
++ ov2640_write_reg(0x13, 0xe5);
++ ov2640_write_reg(0x14, 0x48);
++ ov2640_write_reg(0x2c, 0x0c);
++ ov2640_write_reg(0x33, 0x78);
++ ov2640_write_reg(0x3a, 0x33);
++ ov2640_write_reg(0x3b, 0xfb);
++ ov2640_write_reg(0x3e, 0x00);
++ ov2640_write_reg(0x43, 0x11);
++ ov2640_write_reg(0x16, 0x10);
++ ov2640_write_reg(0x39, 0x92);
++ ov2640_write_reg(0x35, 0xda);
++ ov2640_write_reg(0x22, 0x1a);
++ ov2640_write_reg(0x37, 0xc3);
++ ov2640_write_reg(0x23, 0x00);
++ ov2640_write_reg(0x34, 0xc0);
++ ov2640_write_reg(0x36, 0x1a);
++ ov2640_write_reg(0x06, 0x88);
++ ov2640_write_reg(0x07, 0xc0);
++ ov2640_write_reg(0x0d, 0x87);
++ ov2640_write_reg(0x0e, 0x41);
++ ov2640_write_reg(0x4c, 0x00);
++ ov2640_write_reg(0x4a, 0x81);
++ ov2640_write_reg(0x21, 0x99);
++ ov2640_write_reg(0x24, 0x40);
++ ov2640_write_reg(0x25, 0x38);
++ ov2640_write_reg(0x26, 0x82);
++ ov2640_write_reg(0x5c, 0x00);
++ ov2640_write_reg(0x63, 0x00);
++ ov2640_write_reg(0x46, 0x22);
++ ov2640_write_reg(0x0c, 0x3c);
++ ov2640_write_reg(0x5d, 0x55);
++ ov2640_write_reg(0x5e, 0x7d);
++ ov2640_write_reg(0x5f, 0x7d);
++ ov2640_write_reg(0x60, 0x55);
++ ov2640_write_reg(0x61, 0x70);
++ ov2640_write_reg(0x62, 0x80);
++ ov2640_write_reg(0x7c, 0x05);
++ ov2640_write_reg(0x20, 0x80);
++ ov2640_write_reg(0x28, 0x30);
++ ov2640_write_reg(0x6c, 0x00);
++ ov2640_write_reg(0x6d, 0x80);
++ ov2640_write_reg(0x6e, 0x00);
++ ov2640_write_reg(0x70, 0x02);
++ ov2640_write_reg(0x71, 0x94);
++ ov2640_write_reg(0x73, 0xc1);
++ ov2640_write_reg(0x12, 0x40);
++ ov2640_write_reg(0x17, 0x11);
++ ov2640_write_reg(0x18, 0x43);
++ ov2640_write_reg(0x19, 0x00);
++ ov2640_write_reg(0x1a, 0x4b);
++ ov2640_write_reg(0x32, 0x09);
++ ov2640_write_reg(0x37, 0xc0);
++ ov2640_write_reg(0x4f, 0xca);
++ ov2640_write_reg(0x50, 0xa8);
++ ov2640_write_reg(0x6d, 0x00);
++ ov2640_write_reg(0x3d, 0x38);
++ ov2640_write_reg(0xff, 0x00);
++ ov2640_write_reg(0xe5, 0x7f);
++ ov2640_write_reg(0xf9, 0xc0);
++ ov2640_write_reg(0x41, 0x24);
++ ov2640_write_reg(0x44, 0x06);
++ ov2640_write_reg(0xe0, 0x14);
++ ov2640_write_reg(0x76, 0xff);
++ ov2640_write_reg(0x33, 0xa0);
++ ov2640_write_reg(0x42, 0x20);
++ ov2640_write_reg(0x43, 0x18);
++ ov2640_write_reg(0x4c, 0x00);
++ ov2640_write_reg(0x87, 0xd0);
++ ov2640_write_reg(0x88, 0x3f);
++ ov2640_write_reg(0xd7, 0x03);
++ ov2640_write_reg(0xd9, 0x10);
++ ov2640_write_reg(0xd3, 0x82);
++ ov2640_write_reg(0xc8, 0x08);
++ ov2640_write_reg(0xc9, 0x80);
++ ov2640_write_reg(0x7c, 0x00);
++ ov2640_write_reg(0x7d, 0x00);
++ ov2640_write_reg(0x7c, 0x03);
++ ov2640_write_reg(0x7d, 0x48);
++ ov2640_write_reg(0x7d, 0x48);
++ ov2640_write_reg(0x7c, 0x08);
++ ov2640_write_reg(0x7d, 0x20);
++ ov2640_write_reg(0x7d, 0x10);
++ ov2640_write_reg(0x7d, 0x0e);
++ ov2640_write_reg(0x90, 0x00);
++ ov2640_write_reg(0x91, 0x0e);
++ ov2640_write_reg(0x91, 0x1a);
++ ov2640_write_reg(0x91, 0x31);
++ ov2640_write_reg(0x91, 0x5a);
++ ov2640_write_reg(0x91, 0x69);
++ ov2640_write_reg(0x91, 0x75);
++ ov2640_write_reg(0x91, 0x7e);
++ ov2640_write_reg(0x91, 0x88);
++ ov2640_write_reg(0x91, 0x8f);
++ ov2640_write_reg(0x91, 0x96);
++ ov2640_write_reg(0x91, 0xa3);
++ ov2640_write_reg(0x91, 0xaf);
++ ov2640_write_reg(0x91, 0xc4);
++ ov2640_write_reg(0x91, 0xd7);
++ ov2640_write_reg(0x91, 0xe8);
++ ov2640_write_reg(0x91, 0x20);
++
++ ov2640_write_reg(0x92, 0x00);
++ ov2640_write_reg(0x93, 0x06);
++ ov2640_write_reg(0x93, 0xe3);
++ ov2640_write_reg(0x93, 0x03);
++ ov2640_write_reg(0x93, 0x03);
++ ov2640_write_reg(0x93, 0x00);
++ ov2640_write_reg(0x93, 0x02);
++ ov2640_write_reg(0x93, 0x00);
++ ov2640_write_reg(0x93, 0x00);
++ ov2640_write_reg(0x93, 0x00);
++ ov2640_write_reg(0x93, 0x00);
++ ov2640_write_reg(0x93, 0x00);
++ ov2640_write_reg(0x93, 0x00);
++ ov2640_write_reg(0x93, 0x00);
++
++ ov2640_write_reg(0x96, 0x00);
++ ov2640_write_reg(0x97, 0x08);
++ ov2640_write_reg(0x97, 0x19);
++ ov2640_write_reg(0x97, 0x02);
++ ov2640_write_reg(0x97, 0x0c);
++ ov2640_write_reg(0x97, 0x24);
++ ov2640_write_reg(0x97, 0x30);
++ ov2640_write_reg(0x97, 0x28);
++ ov2640_write_reg(0x97, 0x26);
++ ov2640_write_reg(0x97, 0x02);
++ ov2640_write_reg(0x97, 0x98);
++ ov2640_write_reg(0x97, 0x80);
++ ov2640_write_reg(0x97, 0x00);
++ ov2640_write_reg(0x97, 0x00);
++
++ ov2640_write_reg(0xa4, 0x00);
++ ov2640_write_reg(0xa8, 0x00);
++ ov2640_write_reg(0xc5, 0x11);
++ ov2640_write_reg(0xc6, 0x51);
++ ov2640_write_reg(0xbf, 0x80);
++ ov2640_write_reg(0xc7, 0x10);
++ ov2640_write_reg(0xb6, 0x66);
++ ov2640_write_reg(0xb8, 0xa5);
++ ov2640_write_reg(0xb7, 0x64);
++ ov2640_write_reg(0xb9, 0x7c);
++ ov2640_write_reg(0xb3, 0xaf);
++ ov2640_write_reg(0xb4, 0x97);
++ ov2640_write_reg(0xb5, 0xff);
++ ov2640_write_reg(0xb0, 0xc5);
++ ov2640_write_reg(0xb1, 0x94);
++ ov2640_write_reg(0xb2, 0x0f);
++ ov2640_write_reg(0xc4, 0x5c);
++
++ ov2640_write_reg(0xa6, 0x00);
++ ov2640_write_reg(0xa7, 0x20);
++ ov2640_write_reg(0xa7, 0xd8);
++ ov2640_write_reg(0xa7, 0x1b);
++ ov2640_write_reg(0xa7, 0x31);
++ ov2640_write_reg(0xa7, 0x00);
++ ov2640_write_reg(0xa7, 0x18);
++ ov2640_write_reg(0xa7, 0x20);
++ ov2640_write_reg(0xa7, 0xd8);
++ ov2640_write_reg(0xa7, 0x19);
++ ov2640_write_reg(0xa7, 0x31);
++ ov2640_write_reg(0xa7, 0x00);
++ ov2640_write_reg(0xa7, 0x18);
++ ov2640_write_reg(0xa7, 0x20);
++ ov2640_write_reg(0xa7, 0xd8);
++ ov2640_write_reg(0xa7, 0x19);
++ ov2640_write_reg(0xa7, 0x31);
++ ov2640_write_reg(0xa7, 0x00);
++ ov2640_write_reg(0xa7, 0x18);
++
++ ov2640_write_reg(0xc0, 0x64);
++ ov2640_write_reg(0xc1, 0x4b);
++ ov2640_write_reg(0x86, 0x1d);
++ ov2640_write_reg(0x50, 0x00);
++ ov2640_write_reg(0x51, 0xc8);
++ ov2640_write_reg(0x52, 0x96);
++ ov2640_write_reg(0x53, 0x00);
++ ov2640_write_reg(0x54, 0x00);
++ ov2640_write_reg(0x55, 0x00);
++ ov2640_write_reg(0x57, 0x00);
++ ov2640_write_reg(0x5a, 0xc8);
++ ov2640_write_reg(0x5b, 0x96);
++ ov2640_write_reg(0x5c, 0x00);
++ ov2640_write_reg(0xc3, 0xef);
++ ov2640_write_reg(0x7f, 0x00);
++ ov2640_write_reg(0xda, 0x01);
++ ov2640_write_reg(0xe5, 0x1f);
++ ov2640_write_reg(0xe1, 0x67);
++ ov2640_write_reg(0xe0, 0x00);
++ ov2640_write_reg(0xdd, 0x7f);
++ ov2640_write_reg(0x05, 0x00);
++
++ return 0;
++}
++
++/*!
++ * ov2640 sensor interface Initialization
++ * @param param sensor_interface *
++ * @param width u32
++ * @param height u32
++ * @return None
++ */
++static void ov2640_interface(sensor_interface * param, u32 width, u32 height)
++{
++ param->Vsync_pol = 0x0;
++ param->clk_mode = 0x0; /* gated */
++ param->pixclk_pol = 0x0;
++ param->data_width = 0x1;
++ param->data_pol = 0x0;
++ param->ext_vsync = 0x0;
++ param->Vsync_pol = 0x0;
++ param->Hsync_pol = 0x0;
++ param->width = width - 1;
++ param->height = height - 1;
++ param->pixel_fmt = IPU_PIX_FMT_UYVY;
++ param->mclk = 27000000;
++}
++
++static void
++ov2640_set_color(int bright, int saturation, int red, int green, int blue)
++{
++ return;
++}
++
++static void
++ov2640_get_color(int *bright, int *saturation, int *red, int *green, int *blue)
++{
++ return;
++}
++
++static void ov2640_set_ae_mode(int ae_mode)
++{
++ return;
++}
++
++static void ov2640_get_ae_mode(int *ae_mode)
++{
++ return;
++}
++
++extern void gpio_sensor_active(void);
++extern cam_data *g_cam;
++
++static sensor_interface *ov2640_config(int *frame_rate, int high_quality)
++{
++ u32 out_width, out_height;
++
++ if (interface_param == NULL) {
++ return NULL;
++ }
++#ifdef NONSENSE
++ PMIC_STATUS ret;
++ t_regulator_voltage voltage;
++
++ /* AVDD--2.8v */
++ voltage.vmmc1 = VMMC1_2_8V;
++ if ((ret = pmic_power_regulator_set_voltage(REGU_VMMC1, voltage)) < 0) {
++ PRINTK("%s:vmmc1 set voltage error:%d\n", __func__, ret);
++ return NULL;
++ } else {
++ PRINTK("%s:vmmc1 set voltage ok\n", __func__);
++ }
++
++ if ((ret = pmic_power_regulator_on(REGU_VMMC1)) < 0) {
++ PRINTK("%s:vmmc1 power on error:%d\n", __func__, ret);
++ return NULL;
++ } else {
++ PRINTK("%s:vmmc1 power on ok\n", __func__);
++ }
++
++ /* DVDD--1.3v */
++ voltage.vvib = VVIB_1_3V;
++ if ((ret = pmic_power_regulator_set_voltage(REGU_VVIB, voltage)) < 0) {
++ PRINTK("%s:VVIB set voltage error:%d\n", __func__, ret);
++ return NULL;
++ } else {
++ PRINTK("%s:VVIB set voltage ok\n", __func__);
++ }
++ if ((ret = pmic_power_regulator_on(REGU_VVIB)) < 0) {
++ PRINTK("%s:VVIB power regulator on error:%d\n", __func__, ret);
++ return NULL;
++ } else {
++ PRINTK("%s:VVIB power on ok\n", __func__);
++ }
++
++ /* DOVDD--2v(1.8-3.0) */
++ voltage.sw2b = SW2B_2V;
++ if ((ret = pmic_power_regulator_set_voltage(SW_SW2B, voltage)) < 0) {
++ PRINTK("%s:SW2B set voltage error:%d\n", __func__, ret);
++ return NULL;
++ } else {
++ PRINTK("%s:SW2B set voltage ok\n", __func__);
++ }
++ if (pmic_power_set_regen_assig(REGU_GPO3, 1) < 0) {
++ PRINTK("%s:set_regen_assig error\n", __func__);
++ return NULL;
++ } else {
++ PRINTK("%s:set_regen_assig ok\n", __func__);
++ }
++ if ((ret = pmic_power_regulator_on(REGU_GPO3)) < 0) {
++ PRINTK("%s:REGU_GPO3 power on error:%d\n", __func__, ret);
++ return NULL;
++ } else {
++ PRINTK("%s:REGU_GPO3 power on ok\n", __func__);
++ }
++ PRINTK("%s:OV2640 power on ok\n", __func__);
++#endif
++ if (high_quality) {
++ out_width = 1600;
++ out_height = 1120;
++ g_cam->crop_bounds.left = 0;
++ g_cam->crop_bounds.width = 1600;
++ g_cam->crop_bounds.top = 0;
++ g_cam->crop_bounds.height = 1120;
++ g_cam->crop_current = g_cam->crop_defrect = g_cam->crop_bounds;
++#ifdef CONFIG_ARCH_MX3
++ ipu_csi_set_window_size(g_cam->crop_current.width,
++ g_cam->crop_current.height);
++ ipu_csi_set_window_pos(g_cam->crop_current.left,
++ g_cam->crop_current.top);
++#endif
++ g_cam->streamparm.parm.capture.capturemode = 1;
++ } else {
++ out_width = 640;
++ out_height = 480;
++ g_cam->crop_bounds.left = 0;
++ g_cam->crop_bounds.width = 640;
++ g_cam->crop_bounds.top = 0;
++ g_cam->crop_bounds.height = 480;
++ g_cam->crop_current = g_cam->crop_defrect = g_cam->crop_bounds;
++#ifdef CONFIG_ARCH_MX3
++ ipu_csi_set_window_size(g_cam->crop_current.width,
++ g_cam->crop_current.height);
++ ipu_csi_set_window_pos(g_cam->crop_current.left,
++ g_cam->crop_current.top);
++#endif
++ g_cam->streamparm.parm.capture.capturemode = 0;
++ }
++ ov2640_interface(interface_param, out_width, out_height);
++ set_mclk_rate(&interface_param->mclk);
++
++ if (high_quality) {
++ if (ov2640_init_1600_1120() != 0) {
++ return NULL;
++ }
++ } else {
++ if (ov2640_init_800_600() != 0) {
++ return NULL;
++ }
++ }
++ return interface_param;
++}
++
++static void ov2640_get_control_params(int *ae, int *awb, int *flicker)
++{
++ *ae = 0;
++ *awb = 0;
++ *flicker = 0;
++}
++
++static sensor_interface *ov2640_reset(void)
++{
++ return ov2640_config(&reset_frame_rate, 0);
++}
++
++static int ov2640_get_status(void)
++{
++ return 0;
++}
++
++struct camera_sensor camera_sensor_if = {
++ .set_color = ov2640_set_color,
++ .get_color = ov2640_get_color,
++ .set_ae_mode = ov2640_set_ae_mode,
++ .get_ae_mode = ov2640_get_ae_mode,
++ .get_control_params = ov2640_get_control_params,
++ .config = ov2640_config,
++ .reset = ov2640_reset,
++ .get_status = ov2640_get_status,
++};
++
++EXPORT_SYMBOL(camera_sensor_if);
++
++/*!
++ * ov2640 init function
++ *
++ * @return Error code indicating success or failure
++ */
++static __init int ov2640_init(void)
++{
++ u8 err;
++
++ gpio_sensor_active();
++
++ err = i2c_add_driver(&ov2640_i2c_driver);
++
++ return err;
++}
++
++extern void gpio_sensor_inactive(void);
++/*!
++ * OV2640 cleanup function
++ *
++ * @return Error code indicating success or failure
++ */
++static void __exit ov2640_clean(void)
++{
++ i2c_del_driver(&ov2640_i2c_driver);
++
++ gpio_sensor_inactive();
++}
++
++module_init(ov2640_init);
++module_exit(ov2640_clean);
++
++MODULE_AUTHOR("Freescale Semiconductor, Inc.");
++MODULE_DESCRIPTION("OV2640 Camera Driver");
++MODULE_LICENSE("GPL");
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/capture/sensor_clock.c linux-2.6.28-karo/drivers/media/video/mxc/capture/sensor_clock.c
+--- linux-2.6.28/drivers/media/video/mxc/capture/sensor_clock.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/capture/sensor_clock.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,56 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @file sensor_clock.c
++ *
++ * @brief camera clock function
++ *
++ * @ingroup Camera
++ */
++#include <linux/init.h>
++#include <linux/ctype.h>
++#include <linux/types.h>
++#include <linux/device.h>
++#include <linux/clk.h>
++
++/*
++ * set_mclk_rate
++ *
++ * @param p_mclk_freq mclk frequence
++ *
++ */
++void set_mclk_rate(uint32_t * p_mclk_freq)
++{
++ struct clk *clk;
++ int i;
++ uint32_t freq = 0;
++ uint32_t step = *p_mclk_freq / 8;
++
++ clk = clk_get(NULL, "csi_clk");
++
++ for (i = 0; i <= 8; i++) {
++ freq = clk_round_rate(clk, *p_mclk_freq - (i * step));
++ if (freq <= *p_mclk_freq)
++ break;
++ }
++ clk_set_rate(clk, freq);
++
++ *p_mclk_freq = freq;
++
++ clk_put(clk);
++ pr_debug("mclk frequency = %d\n", *p_mclk_freq);
++}
++
++/* Exported symbols for modules. */
++EXPORT_SYMBOL(set_mclk_rate);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/opl/Makefile linux-2.6.28-karo/drivers/media/video/mxc/opl/Makefile
+--- linux-2.6.28/drivers/media/video/mxc/opl/Makefile 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/opl/Makefile 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,5 @@
++opl-objs := opl_mod.o rotate90_u16.o rotate270_u16.o \
++ rotate90_u16_qcif.o rotate270_u16_qcif.o \
++ vmirror_u16.o hmirror_rotate180_u16.o
++
++obj-$(CONFIG_VIDEO_MXC_OPL) += opl.o
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/opl/hmirror_rotate180_u16.c linux-2.6.28-karo/drivers/media/video/mxc/opl/hmirror_rotate180_u16.c
+--- linux-2.6.28/drivers/media/video/mxc/opl/hmirror_rotate180_u16.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/opl/hmirror_rotate180_u16.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,259 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++#include <linux/module.h>
++#include "opl.h"
++
++static inline u32 rot_left_u16(u16 x, unsigned int n)
++{
++ return (x << n) | (x >> (16 - n));
++}
++
++static inline u32 rot_left_u32(u32 x, unsigned int n)
++{
++ return (x << n) | (x >> (32 - n));
++}
++
++static inline u32 byte_swap_u32(u32 x)
++{
++ u32 t1, t2, t3;
++
++ t1 = x ^ ((x << 16) | x >> 16);
++ t2 = t1 & 0xff00ffff;
++ t3 = (x >> 8) | (x << 24);
++ return t3 ^ (t2 >> 8);
++}
++
++static int opl_hmirror_u16_by1(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride,
++ int vmirror);
++static int opl_hmirror_u16_by2(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride,
++ int vmirror);
++static int opl_hmirror_u16_by4(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride,
++ int vmirror);
++static int opl_hmirror_u16_by8(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride,
++ int vmirror);
++
++int opl_hmirror_u16(const u8 * src, int src_line_stride, int width, int height,
++ u8 * dst, int dst_line_stride)
++{
++ if (!src || !dst)
++ return OPLERR_NULL_PTR;
++
++ if (width == 0 || height == 0 || src_line_stride == 0
++ || dst_line_stride == 0)
++ return OPLERR_BAD_ARG;
++
++ if (width % 8 == 0)
++ return opl_hmirror_u16_by8(src, src_line_stride, width, height,
++ dst, dst_line_stride, 0);
++ else if (width % 4 == 0)
++ return opl_hmirror_u16_by4(src, src_line_stride, width, height,
++ dst, dst_line_stride, 0);
++ else if (width % 2 == 0)
++ return opl_hmirror_u16_by2(src, src_line_stride, width, height,
++ dst, dst_line_stride, 0);
++ else /* (width % 1) */
++ return opl_hmirror_u16_by1(src, src_line_stride, width, height,
++ dst, dst_line_stride, 0);
++}
++
++int opl_rotate180_u16(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride)
++{
++ if (!src || !dst)
++ return OPLERR_NULL_PTR;
++
++ if (width == 0 || height == 0 || src_line_stride == 0
++ || dst_line_stride == 0)
++ return OPLERR_BAD_ARG;
++
++ if (width % 8 == 0)
++ return opl_hmirror_u16_by8(src, src_line_stride, width, height,
++ dst, dst_line_stride, 1);
++ else if (width % 4 == 0)
++ return opl_hmirror_u16_by4(src, src_line_stride, width, height,
++ dst, dst_line_stride, 1);
++ else if (width % 2 == 0)
++ return opl_hmirror_u16_by2(src, src_line_stride, width, height,
++ dst, dst_line_stride, 1);
++ else /* (width % 1) */
++ return opl_hmirror_u16_by1(src, src_line_stride, width, height,
++ dst, dst_line_stride, 1);
++}
++
++static int opl_hmirror_u16_by1(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride,
++ int vmirror)
++{
++ const u8 *src_row_addr;
++ const u8 *psrc;
++ u8 *dst_row_addr, *pdst;
++ int i, j;
++ u16 pixel;
++
++ src_row_addr = src;
++ if (vmirror) {
++ dst_row_addr = dst + dst_line_stride * (height - 1);
++ dst_line_stride = -dst_line_stride;
++ } else
++ dst_row_addr = dst;
++
++ /* Loop over all rows */
++ for (i = 0; i < height; i++) {
++ /* Loop over each pixel */
++ psrc = src_row_addr;
++ pdst = dst_row_addr + (width - 1) * BYTES_PER_PIXEL
++ - (BYTES_PER_PIXEL - BYTES_PER_PIXEL);
++ for (j = 0; j < width; j++) {
++ pixel = *(u16 *) psrc;
++ *(u16 *) pdst = pixel;
++ psrc += BYTES_PER_PIXEL;
++ pdst -= BYTES_PER_PIXEL;
++ }
++ src_row_addr += src_line_stride;
++ dst_row_addr += dst_line_stride;
++ }
++
++ return OPLERR_SUCCESS;
++}
++
++static int opl_hmirror_u16_by2(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride,
++ int vmirror)
++{
++ const u8 *src_row_addr;
++ const u8 *psrc;
++ u8 *dst_row_addr, *pdst;
++ int i, j;
++ u32 pixelsin, pixelsout;
++
++ src_row_addr = src;
++ if (vmirror) {
++ dst_row_addr = dst + dst_line_stride * (height - 1);
++ dst_line_stride = -dst_line_stride;
++ } else
++ dst_row_addr = dst;
++
++ /* Loop over all rows */
++ for (i = 0; i < height; i++) {
++ /* Loop over each pixel */
++ psrc = src_row_addr;
++ pdst = dst_row_addr + (width - 2) * BYTES_PER_PIXEL;
++ for (j = 0; j < (width >> 1); j++) {
++ pixelsin = *(u32 *) psrc;
++ pixelsout = rot_left_u32(pixelsin, 16);
++ *(u32 *) pdst = pixelsout;
++ psrc += BYTES_PER_2PIXEL;
++ pdst -= BYTES_PER_2PIXEL;
++ }
++ src_row_addr += src_line_stride;
++ dst_row_addr += dst_line_stride;
++ }
++
++ return OPLERR_SUCCESS;
++}
++
++static int opl_hmirror_u16_by4(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride,
++ int vmirror)
++{
++ const u8 *src_row_addr;
++ const u8 *psrc;
++ u8 *dst_row_addr, *pdst;
++ int i, j;
++
++ union doubleword {
++ u64 dw;
++ u32 w[2];
++ };
++
++ union doubleword inbuf;
++ union doubleword outbuf;
++
++ src_row_addr = src;
++ if (vmirror) {
++ dst_row_addr = dst + dst_line_stride * (height - 1);
++ dst_line_stride = -dst_line_stride;
++ } else
++ dst_row_addr = dst;
++
++ /* Loop over all rows */
++ for (i = 0; i < height; i++) {
++ /* Loop over each pixel */
++ psrc = src_row_addr;
++ pdst = dst_row_addr + (width - 4) * BYTES_PER_PIXEL;
++ for (j = 0; j < (width >> 2); j++) {
++ inbuf.dw = *(u64 *) psrc;
++ outbuf.w[0] = rot_left_u32(inbuf.w[1], 16);
++ outbuf.w[1] = rot_left_u32(inbuf.w[0], 16);
++ *(u64 *) pdst = outbuf.dw;
++ psrc += BYTES_PER_4PIXEL;
++ pdst -= BYTES_PER_4PIXEL;
++ }
++ src_row_addr += src_line_stride;
++ dst_row_addr += dst_line_stride;
++ }
++ return OPLERR_SUCCESS;
++}
++
++static int opl_hmirror_u16_by8(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride,
++ int vmirror)
++{
++ const u8 *src_row_addr;
++ const u8 *psrc;
++ u8 *dst_row_addr, *pdst;
++ int i, j;
++
++ src_row_addr = src;
++ if (vmirror) {
++ dst_row_addr = dst + dst_line_stride * (height - 1);
++ dst_line_stride = -dst_line_stride;
++ } else
++ dst_row_addr = dst;
++
++ /* Loop over all rows */
++ for (i = 0; i < height; i++) {
++ /* Loop over each pixel */
++ psrc = src_row_addr;
++ pdst = dst_row_addr + (width - 1) * BYTES_PER_PIXEL - 2;
++ for (j = (width >> 3); j > 0; j--) {
++ __asm__ volatile (
++ "ldmia %0!,{r2-r5}\n\t"
++ "mov r6, r2\n\t"
++ "mov r7, r3\n\t"
++ "mov r2, r5, ROR #16\n\t"
++ "mov r3, r4, ROR #16\n\t"
++ "mov r4, r7, ROR #16\n\t"
++ "mov r5, r6, ROR #16\n\t"
++ "stmda %1!,{r2-r5}\n\t"
++
++ :"+r"(psrc), "+r"(pdst)
++ :"0"(psrc), "1"(pdst)
++ :"r2", "r3", "r4", "r5", "r6", "r7",
++ "memory"
++ );
++ }
++ src_row_addr += src_line_stride;
++ dst_row_addr += dst_line_stride;
++ }
++
++ return OPLERR_SUCCESS;
++}
++
++EXPORT_SYMBOL(opl_hmirror_u16);
++EXPORT_SYMBOL(opl_rotate180_u16);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/opl/opl.h linux-2.6.28-karo/drivers/media/video/mxc/opl/opl.h
+--- linux-2.6.28/drivers/media/video/mxc/opl/opl.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/opl/opl.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,162 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @defgroup OPLIP OPL Image Processing
++ */
++/*!
++ * @file opl.h
++ *
++ * @brief The OPL (Open Primitives Library) Image Processing library defines
++ * efficient functions for rotation and mirroring.
++ *
++ * It includes ARM9-optimized rotation and mirroring functions. It is derived
++ * from the original OPL project which is found at sourceforge.freescale.net.
++ *
++ * @ingroup OPLIP
++ */
++#ifndef __OPL_H__
++#define __OPL_H__
++
++#include <linux/types.h>
++
++#define BYTES_PER_PIXEL 2
++#define CACHE_LINE_WORDS 8
++#define BYTES_PER_WORD 4
++
++#define BYTES_PER_2PIXEL (BYTES_PER_PIXEL * 2)
++#define BYTES_PER_4PIXEL (BYTES_PER_PIXEL * 4)
++#define BYTES_PER_8PIXEL (BYTES_PER_PIXEL * 8)
++
++#define QCIF_Y_WIDTH 176
++#define QCIF_Y_HEIGHT 144
++
++/*! Enumerations of opl error code */
++enum opl_error {
++ OPLERR_SUCCESS = 0,
++ OPLERR_NULL_PTR,
++ OPLERR_BAD_ARG,
++ OPLERR_DIV_BY_ZERO,
++ OPLERR_OVER_FLOW,
++ OPLERR_UNDER_FLOW,
++ OPLERR_MISALIGNED,
++};
++
++/*!
++ * @brief Rotate a 16bbp buffer 90 degrees clockwise.
++ *
++ * @param src Pointer to the input buffer
++ * @param src_line_stride Length in bytes of a raster line of the input buffer
++ * @param width Width in pixels of the region in the input buffer
++ * @param height Height in pixels of the region in the input buffer
++ * @param dst Pointer to the output buffer
++ * @param dst_line_stride Length in bytes of a raster line of the output buffer
++ *
++ * @return Standard OPL error code. See enumeration for possible result codes.
++ */
++int opl_rotate90_u16(const u8 * src, int src_line_stride, int width, int height,
++ u8 * dst, int dst_line_stride);
++
++/*!
++ * @brief Rotate a 16bbp buffer 180 degrees clockwise.
++ *
++ * @param src Pointer to the input buffer
++ * @param src_line_stride Length in bytes of a raster line of the input buffer
++ * @param width Width in pixels of the region in the input buffer
++ * @param height Height in pixels of the region in the input buffer
++ * @param dst Pointer to the output buffer
++ * @param dst_line_stride Length in bytes of a raster line of the output buffer
++ *
++ * @return Standard OPL error code. See enumeration for possible result codes.
++ */
++int opl_rotate180_u16(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride);
++
++/*!
++ * @brief Rotate a 16bbp buffer 270 degrees clockwise
++ *
++ * @param src Pointer to the input buffer
++ * @param src_line_stride Length in bytes of a raster line of the input buffer
++ * @param width Width in pixels of the region in the input buffer
++ * @param height Height in pixels of the region in the input buffer
++ * @param dst Pointer to the output buffer
++ * @param dst_line_stride Length in bytes of a raster line of the output buffer
++ *
++ * @return Standard OPL error code. See enumeration for possible result codes.
++ */
++int opl_rotate270_u16(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride);
++
++/*!
++ * @brief Mirror a 16bpp buffer horizontally
++ *
++ * @param src Pointer to the input buffer
++ * @param src_line_stride Length in bytes of a raster line of the input buffer
++ * @param width Width in pixels of the region in the input buffer
++ * @param height Height in pixels of the region in the input buffer
++ * @param dst Pointer to the output buffer
++ * @param dst_line_stride Length in bytes of a raster line of the output buffer
++ *
++ * @return Standard OPL error code. See enumeration for possible result codes.
++ */
++int opl_hmirror_u16(const u8 * src, int src_line_stride, int width, int height,
++ u8 * dst, int dst_line_stride);
++
++/*!
++ * @brief Mirror a 16bpp buffer vertically
++ *
++ * @param src Pointer to the input buffer
++ * @param src_line_stride Length in bytes of a raster line of the input buffer
++ * @param width Width in pixels of the region in the input buffer
++ * @param height Height in pixels of the region in the input buffer
++ * @param dst Pointer to the output buffer
++ * @param dst_line_stride Length in bytes of a raster line of the output buffer
++ *
++ * @return Standard OPL error code. See enumeration for possible result codes.
++ */
++int opl_vmirror_u16(const u8 * src, int src_line_stride, int width, int height,
++ u8 * dst, int dst_line_stride);
++
++/*!
++ * @brief Rotate a 16bbp buffer 90 degrees clockwise and mirror vertically
++ * It is equivalent to rotate 270 degree and mirror horizontally
++ *
++ * @param src Pointer to the input buffer
++ * @param src_line_stride Length in bytes of a raster line of the input buffer
++ * @param width Width in pixels of the region in the input buffer
++ * @param height Height in pixels of the region in the input buffer
++ * @param dst Pointer to the output buffer
++ * @param dst_line_stride Length in bytes of a raster line of the output buffer
++ *
++ * @return Standard OPL error code. See enumeration for possible result codes.
++ */
++int opl_rotate90_vmirror_u16(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride);
++
++/*!
++ * @brief Rotate a 16bbp buffer 270 degrees clockwise and mirror vertically
++ * It is equivalent to rotate 90 degree and mirror horizontally
++ *
++ * @param src Pointer to the input buffer
++ * @param src_line_stride Length in bytes of a raster line of the input buffer
++ * @param width Width in pixels of the region in the input buffer
++ * @param height Height in pixels of the region in the input buffer
++ * @param dst Pointer to the output buffer
++ * @param dst_line_stride Length in bytes of a raster line of the output buffer
++ *
++ * @return Standard OPL error code. See enumeration for possible result codes.
++ */
++int opl_rotate270_vmirror_u16(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride);
++
++#endif /* __OPL_H__ */
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/opl/opl_mod.c linux-2.6.28-karo/drivers/media/video/mxc/opl/opl_mod.c
+--- linux-2.6.28/drivers/media/video/mxc/opl/opl_mod.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/opl/opl_mod.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,30 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++#include <linux/module.h>
++
++static __init int opl_init(void)
++{
++ return 0;
++}
++
++static void __exit opl_exit(void)
++{
++}
++
++module_init(opl_init);
++module_exit(opl_exit);
++
++MODULE_AUTHOR("Freescale Semiconductor, Inc.");
++MODULE_DESCRIPTION("OPL Software Rotation/Mirroring");
++MODULE_LICENSE("GPL");
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/opl/rotate270_u16.c linux-2.6.28-karo/drivers/media/video/mxc/opl/rotate270_u16.c
+--- linux-2.6.28/drivers/media/video/mxc/opl/rotate270_u16.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/opl/rotate270_u16.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,285 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++#include <linux/module.h>
++#include "opl.h"
++
++static int opl_rotate270_u16_by16(const u8 * src, int src_line_stride,
++ int width, int height, u8 * dst,
++ int dst_line_stride, int vmirror);
++static int opl_rotate270_u16_by4(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride,
++ int vmirror);
++static int opl_rotate270_vmirror_u16_both(const u8 * src, int src_line_stride,
++ int width, int height, u8 * dst,
++ int dst_line_stride, int vmirror);
++int opl_rotate270_u16_qcif(const u8 * src, u8 * dst);
++
++int opl_rotate270_u16(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride)
++{
++ return opl_rotate270_vmirror_u16_both(src, src_line_stride, width,
++ height, dst, dst_line_stride, 0);
++}
++
++int opl_rotate270_vmirror_u16(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride)
++{
++ return opl_rotate270_vmirror_u16_both(src, src_line_stride, width,
++ height, dst, dst_line_stride, 1);
++}
++
++static int opl_rotate270_vmirror_u16_both(const u8 * src, int src_line_stride,
++ int width, int height, u8 * dst,
++ int dst_line_stride, int vmirror)
++{
++ const int BLOCK_SIZE_PIXELS = CACHE_LINE_WORDS * BYTES_PER_WORD
++ / BYTES_PER_PIXEL;
++ const int BLOCK_SIZE_PIXELS_BY4 = CACHE_LINE_WORDS * BYTES_PER_WORD
++ / BYTES_PER_PIXEL / 4;
++
++ if (!src || !dst)
++ return OPLERR_NULL_PTR;
++
++ if (width == 0 || height == 0 || src_line_stride == 0
++ || dst_line_stride == 0)
++ return OPLERR_BAD_ARG;
++
++ /* The QCIF algorithm doesn't support vertical mirroring */
++ if (vmirror == 0 && width == QCIF_Y_WIDTH && height == QCIF_Y_HEIGHT
++ && src_line_stride == QCIF_Y_WIDTH * 2
++ && src_line_stride == QCIF_Y_HEIGHT * 2)
++ return opl_rotate270_u16_qcif(src, dst);
++ else if (width % BLOCK_SIZE_PIXELS == 0
++ && height % BLOCK_SIZE_PIXELS == 0)
++ return opl_rotate270_u16_by16(src, src_line_stride, width,
++ height, dst, dst_line_stride,
++ vmirror);
++ else if (width % BLOCK_SIZE_PIXELS_BY4 == 0
++ && height % BLOCK_SIZE_PIXELS_BY4 == 0)
++ return opl_rotate270_u16_by4(src, src_line_stride, width,
++ height, dst, dst_line_stride,
++ vmirror);
++ else
++ return OPLERR_BAD_ARG;
++}
++
++/*
++ * Rotate Counter Clockwise, divide RGB component into 16 row strips, read
++ * non sequentially and write sequentially. This is done in 16 line strips
++ * so that the cache is used better. Cachelines are 8 words = 32 bytes. Pixels
++ * are 2 bytes. The 16 reads will be cache misses, but the next 240 should
++ * be from cache. The writes to the output buffer will be sequential for 16
++ * writes.
++ *
++ * Example:
++ * Input data matrix: output matrix
++ *
++ * 0 | 1 | 2 | 3 | 4 | 4 | 0 | 0 | 3 |
++ * 4 | 3 | 2 | 1 | 0 | 3 | 1 | 9 | 6 |
++ * 6 | 7 | 8 | 9 | 0 | 2 | 2 | 8 | 2 |
++ * 5 | 3 | 2 | 6 | 3 | 1 | 3 | 7 | 3 |
++ * ^ 0 | 4 | 6 | 5 | < Write the input data sequentially
++ * Read first column
++ * Start at the bottom
++ * Move to next column and repeat
++ *
++ * Loop over k decreasing (blocks)
++ * in_block_ptr = src + (((RGB_HEIGHT_PIXELS / BLOCK_SIZE_PIXELS) - k)
++ * * BLOCK_SIZE_PIXELS) * (RGB_WIDTH_BYTES)
++ * out_block_ptr = dst + (((RGB_HEIGHT_PIXELS / BLOCK_SIZE_PIXELS) - k)
++ * * BLOCK_SIZE_BYTES) + (RGB_WIDTH_PIXELS - 1)
++ * * RGB_HEIGHT_PIXELS * BYTES_PER_PIXEL
++ *
++ * Loop over i decreasing (width)
++ * Each pix:
++ * in_block_ptr += RGB_WIDTH_BYTES
++ * out_block_ptr += 4
++ *
++ * Each row of block:
++ * in_block_ptr -= RGB_WIDTH_BYTES * BLOCK_SIZE_PIXELS - 2
++ * out_block_ptr -= RGB_HEIGHT_PIXELS * BYTES_PER_PIXEL + 2 * BLOCK_SIZE_PIXELS;
++ *
++ * It may perform vertical mirroring too depending on the vmirror flag.
++ */
++static int opl_rotate270_u16_by16(const u8 * src, int src_line_stride,
++ int width, int height, u8 * dst,
++ int dst_line_stride, int vmirror)
++{
++ const int BLOCK_SIZE_PIXELS = CACHE_LINE_WORDS * BYTES_PER_WORD
++ / BYTES_PER_PIXEL;
++ const int IN_INDEX = src_line_stride * BLOCK_SIZE_PIXELS
++ - BYTES_PER_PIXEL;
++ const int OUT_INDEX = vmirror ?
++ -dst_line_stride + BYTES_PER_PIXEL * BLOCK_SIZE_PIXELS
++ : dst_line_stride + BYTES_PER_PIXEL * BLOCK_SIZE_PIXELS;
++ const u8 *in_block_ptr;
++ u8 *out_block_ptr;
++ int i, k;
++
++ for (k = height / BLOCK_SIZE_PIXELS; k > 0; k--) {
++ in_block_ptr = src + (((height / BLOCK_SIZE_PIXELS) - k)
++ * BLOCK_SIZE_PIXELS) * src_line_stride;
++ out_block_ptr = dst + (((height / BLOCK_SIZE_PIXELS) - k)
++ * BLOCK_SIZE_PIXELS * BYTES_PER_PIXEL) +
++ (width - 1) * dst_line_stride;
++
++ /*
++ * For vertical mirroring the writing starts from the
++ * first line
++ */
++ if (vmirror)
++ out_block_ptr -= dst_line_stride * (width - 1);
++
++ for (i = width; i > 0; i--) {
++ __asm__ volatile (
++ "ldrh r2, [%0], %4\n\t"
++ "ldrh r3, [%0], %4\n\t"
++ "ldrh r4, [%0], %4\n\t"
++ "ldrh r5, [%0], %4\n\t"
++ "orr r2, r2, r3, lsl #16\n\t"
++ "orr r4, r4, r5, lsl #16\n\t"
++ "str r2, [%1], #4\n\t"
++ "str r4, [%1], #4\n\t"
++
++ "ldrh r2, [%0], %4\n\t"
++ "ldrh r3, [%0], %4\n\t"
++ "ldrh r4, [%0], %4\n\t"
++ "ldrh r5, [%0], %4\n\t"
++ "orr r2, r2, r3, lsl #16\n\t"
++ "orr r4, r4, r5, lsl #16\n\t"
++ "str r2, [%1], #4\n\t"
++ "str r4, [%1], #4\n\t"
++
++ "ldrh r2, [%0], %4\n\t"
++ "ldrh r3, [%0], %4\n\t"
++ "ldrh r4, [%0], %4\n\t"
++ "ldrh r5, [%0], %4\n\t"
++ "orr r2, r2, r3, lsl #16\n\t"
++ "orr r4, r4, r5, lsl #16\n\t"
++ "str r2, [%1], #4\n\t"
++ "str r4, [%1], #4\n\t"
++
++ "ldrh r2, [%0], %4\n\t"
++ "ldrh r3, [%0], %4\n\t"
++ "ldrh r4, [%0], %4\n\t"
++ "ldrh r5, [%0], %4\n\t"
++ "orr r2, r2, r3, lsl #16\n\t"
++ "orr r4, r4, r5, lsl #16\n\t"
++ "str r2, [%1], #4\n\t"
++ "str r4, [%1], #4\n\t"
++
++ :"+r" (in_block_ptr), "+r"(out_block_ptr) /* output */
++ :"0"(in_block_ptr), "1"(out_block_ptr), "r"(src_line_stride) /* input */
++ :"r2", "r3", "r4", "r5", "memory" /* modify */
++ );
++ in_block_ptr -= IN_INDEX;
++ out_block_ptr -= OUT_INDEX;
++ }
++ }
++
++ return OPLERR_SUCCESS;
++}
++
++/*
++ * Rotate Counter Clockwise, divide RGB component into 4 row strips, read
++ * non sequentially and write sequentially. This is done in 4 line strips
++ * so that the cache is used better. Cachelines are 8 words = 32 bytes. Pixels
++ * are 2 bytes. The 4 reads will be cache misses, but the next 60 should
++ * be from cache. The writes to the output buffer will be sequential for 4
++ * writes.
++ *
++ * Example:
++ * Input data matrix: output matrix
++ *
++ * 0 | 1 | 2 | 3 | 4 | 4 | 0 | 0 | 3 |
++ * 4 | 3 | 2 | 1 | 0 | 3 | 1 | 9 | 6 |
++ * 6 | 7 | 8 | 9 | 0 | 2 | 2 | 8 | 2 |
++ * 5 | 3 | 2 | 6 | 3 | 1 | 3 | 7 | 3 |
++ * ^ 0 | 4 | 6 | 5 | < Write the input data sequentially
++ * Read first column
++ * Start at the bottom
++ * Move to next column and repeat
++ *
++ * Loop over k decreasing (blocks)
++ * in_block_ptr = src + (((RGB_HEIGHT_PIXELS / BLOCK_SIZE_PIXELS) - k)
++ * * BLOCK_SIZE_PIXELS) * (RGB_WIDTH_BYTES)
++ * out_block_ptr = dst + (((RGB_HEIGHT_PIXELS / BLOCK_SIZE_PIXELS) - k)
++ * * BLOCK_SIZE_BYTES) + (RGB_WIDTH_PIXELS - 1)
++ * * RGB_HEIGHT_PIXELS * BYTES_PER_PIXEL
++ *
++ * Loop over i decreasing (width)
++ * Each pix:
++ * in_block_ptr += RGB_WIDTH_BYTES
++ * out_block_ptr += 4
++ *
++ * Each row of block:
++ * in_block_ptr -= RGB_WIDTH_BYTES * BLOCK_SIZE_PIXELS - 2
++ * out_block_ptr -= RGB_HEIGHT_PIXELS * BYTES_PER_PIXEL + 2 * BLOCK_SIZE_PIXELS;
++ *
++ * It may perform vertical mirroring too depending on the vmirror flag.
++ */
++static int opl_rotate270_u16_by4(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride,
++ int vmirror)
++{
++ const int BLOCK_SIZE_PIXELS = CACHE_LINE_WORDS * BYTES_PER_WORD
++ / BYTES_PER_PIXEL / 4;
++ const int IN_INDEX = src_line_stride * BLOCK_SIZE_PIXELS
++ - BYTES_PER_PIXEL;
++ const int OUT_INDEX = vmirror ?
++ -dst_line_stride + BYTES_PER_PIXEL * BLOCK_SIZE_PIXELS
++ : dst_line_stride + BYTES_PER_PIXEL * BLOCK_SIZE_PIXELS;
++ const u8 *in_block_ptr;
++ u8 *out_block_ptr;
++ int i, k;
++
++ for (k = height / BLOCK_SIZE_PIXELS; k > 0; k--) {
++ in_block_ptr = src + (((height / BLOCK_SIZE_PIXELS) - k)
++ * BLOCK_SIZE_PIXELS) * src_line_stride;
++ out_block_ptr = dst + (((height / BLOCK_SIZE_PIXELS) - k)
++ * BLOCK_SIZE_PIXELS * BYTES_PER_PIXEL)
++ + (width - 1) * dst_line_stride;
++
++ /*
++ * For vertical mirroring the writing starts from the
++ * first line
++ */
++ if (vmirror)
++ out_block_ptr -= dst_line_stride * (width - 1);
++
++ for (i = width; i > 0; i--) {
++ __asm__ volatile (
++ "ldrh r2, [%0], %4\n\t"
++ "ldrh r3, [%0], %4\n\t"
++ "ldrh r4, [%0], %4\n\t"
++ "ldrh r5, [%0], %4\n\t"
++ "orr r2, r2, r3, lsl #16\n\t"
++ "orr r4, r4, r5, lsl #16\n\t"
++ "str r2, [%1], #4\n\t"
++ "str r4, [%1], #4\n\t"
++
++ :"+r" (in_block_ptr), "+r"(out_block_ptr) /* output */
++ :"0"(in_block_ptr), "1"(out_block_ptr), "r"(src_line_stride) /* input */
++ :"r2", "r3", "r4", "r5", "memory" /* modify */
++ );
++ in_block_ptr -= IN_INDEX;
++ out_block_ptr -= OUT_INDEX;
++ }
++ }
++
++ return OPLERR_SUCCESS;
++}
++
++EXPORT_SYMBOL(opl_rotate270_u16);
++EXPORT_SYMBOL(opl_rotate270_vmirror_u16);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/opl/rotate270_u16_qcif.S linux-2.6.28-karo/drivers/media/video/mxc/opl/rotate270_u16_qcif.S
+--- linux-2.6.28/drivers/media/video/mxc/opl/rotate270_u16_qcif.S 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/opl/rotate270_u16_qcif.S 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,70 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++#include <linux/linkage.h>
++
++ .text
++ .align 2
++ENTRY(opl_rotate270_u16_qcif)
++ STMFD sp!,{r4-r10}
++ MOV r12,#0x160
++ MOV r10,#0x90
++ MOV r3,r10,LSR #4
++.L1.16:
++ RSB r2,r3,r10,LSR #4
++ MOV r5,r2,LSL #5
++ MOV r4,r12,LSR #1
++ SMULBB r4,r5,r4
++ ADD r2,r1,r2,LSL #5
++ ADD r5,r2,#0xc000
++ ADD r5,r5,#0x4e0
++ MOV r2,r12,LSR #1
++ ADD r4,r0,r4
++.L1.52:
++ LDRH r6,[r4],r12
++ LDRH r7,[r4],r12
++ LDRH r8,[r4],r12
++ LDRH r9,[r4],r12
++ ORR r6,r6,r7,LSL #16
++ ORR r7,r8,r9,LSL #16
++ STMIA r5!,{r6,r7}
++ SUBS r2,r2,#1
++ LDRH r6,[r4],r12
++ LDRH r7,[r4],r12
++ LDRH r8,[r4],r12
++ LDRH r9,[r4],r12
++ ORR r6,r6,r7,LSL #16
++ ORR r7,r8,r9,LSL #16
++ STMIA r5!,{r6,r7}
++ LDRH r6,[r4],r12
++ LDRH r7,[r4],r12
++ LDRH r8,[r4],r12
++ LDRH r9,[r4],r12
++ ORR r6,r6,r7,LSL #16
++ ORR r7,r8,r9,LSL #16
++ STMIA r5!,{r6,r7}
++ LDRH r6,[r4],r12
++ LDRH r7,[r4],r12
++ LDRH r8,[r4],r12
++ LDRH r9,[r4],r12
++ ORR r6,r6,r7,LSL #16
++ ORR r7,r8,r9,LSL #16
++ SUB r4,r4,#0x1500
++ STMIA r5,{r6,r7}
++ SUB r5,r5,#0x138
++ SUB r4,r4,#0xfe
++ BGT .L1.52
++ SUBS r3,r3,#1
++ BGT .L1.16
++ LDMFD sp!,{r4-r10}
++ BX lr
++ .size opl_rotate270_u16_qcif, . - opl_rotate270_u16_qcif
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/opl/rotate90_u16.c linux-2.6.28-karo/drivers/media/video/mxc/opl/rotate90_u16.c
+--- linux-2.6.28/drivers/media/video/mxc/opl/rotate90_u16.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/opl/rotate90_u16.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,220 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++#include <linux/module.h>
++#include "opl.h"
++
++static int opl_rotate90_u16_by16(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride,
++ int vmirror);
++static int opl_rotate90_u16_by4(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride,
++ int vmirror);
++static int opl_rotate90_vmirror_u16_both(const u8 * src, int src_line_stride,
++ int width, int height, u8 * dst,
++ int dst_line_stride, int vmirror);
++int opl_rotate90_u16_qcif(const u8 * src, u8 * dst);
++
++int opl_rotate90_u16(const u8 * src, int src_line_stride, int width, int height,
++ u8 * dst, int dst_line_stride)
++{
++ return opl_rotate90_vmirror_u16_both(src, src_line_stride, width,
++ height, dst, dst_line_stride, 0);
++}
++
++int opl_rotate90_vmirror_u16(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride)
++{
++ return opl_rotate90_vmirror_u16_both(src, src_line_stride, width,
++ height, dst, dst_line_stride, 1);
++}
++
++static int opl_rotate90_vmirror_u16_both(const u8 * src, int src_line_stride,
++ int width, int height, u8 * dst,
++ int dst_line_stride, int vmirror)
++{
++ const int BLOCK_SIZE_PIXELS = CACHE_LINE_WORDS * BYTES_PER_WORD
++ / BYTES_PER_PIXEL;
++ const int BLOCK_SIZE_PIXELS_BY4 = CACHE_LINE_WORDS * BYTES_PER_WORD
++ / BYTES_PER_PIXEL / 4;
++
++ if (!src || !dst)
++ return OPLERR_NULL_PTR;
++
++ if (width == 0 || height == 0 || src_line_stride == 0
++ || dst_line_stride == 0)
++ return OPLERR_BAD_ARG;
++
++ /* The QCIF algorithm doesn't support vertical mirroring */
++ if (vmirror == 0 && width == QCIF_Y_WIDTH && height == QCIF_Y_HEIGHT
++ && src_line_stride == QCIF_Y_WIDTH * 2
++ && src_line_stride == QCIF_Y_HEIGHT * 2)
++ return opl_rotate90_u16_qcif(src, dst);
++ else if (width % BLOCK_SIZE_PIXELS == 0
++ && height % BLOCK_SIZE_PIXELS == 0)
++ return opl_rotate90_u16_by16(src, src_line_stride, width,
++ height, dst, dst_line_stride,
++ vmirror);
++ else if (width % BLOCK_SIZE_PIXELS_BY4 == 0
++ && height % BLOCK_SIZE_PIXELS_BY4 == 0)
++ return opl_rotate90_u16_by4(src, src_line_stride, width, height,
++ dst, dst_line_stride, vmirror);
++ else
++ return OPLERR_BAD_ARG;
++}
++
++/*
++ * Performs clockwise rotation (and possibly vertical mirroring depending
++ * on the vmirror flag) using block sizes of 16x16
++ * The algorithm is similar to 270 degree clockwise rotation algorithm
++ */
++static int opl_rotate90_u16_by16(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride,
++ int vmirror)
++{
++ const int BLOCK_SIZE_PIXELS = CACHE_LINE_WORDS * BYTES_PER_WORD
++ / BYTES_PER_PIXEL;
++ const int BLOCK_SIZE_BYTES = BYTES_PER_PIXEL * BLOCK_SIZE_PIXELS;
++ const int IN_INDEX = src_line_stride * BLOCK_SIZE_PIXELS
++ + BYTES_PER_PIXEL;
++ const int OUT_INDEX = vmirror ?
++ -dst_line_stride - BLOCK_SIZE_BYTES
++ : dst_line_stride - BLOCK_SIZE_BYTES;
++ const u8 *in_block_ptr;
++ u8 *out_block_ptr;
++ int i, k;
++
++ for (k = height / BLOCK_SIZE_PIXELS; k > 0; k--) {
++ in_block_ptr = src + src_line_stride * (height - 1)
++ - (src_line_stride * BLOCK_SIZE_PIXELS *
++ (height / BLOCK_SIZE_PIXELS - k));
++ out_block_ptr = dst + BYTES_PER_PIXEL * BLOCK_SIZE_PIXELS *
++ ((height / BLOCK_SIZE_PIXELS) - k);
++
++ /*
++ * For vertical mirroring the writing starts from the
++ * bottom line
++ */
++ if (vmirror)
++ out_block_ptr += dst_line_stride * (width - 1);
++
++ for (i = width; i > 0; i--) {
++ __asm__ volatile (
++ "ldrh r2, [%0], -%4\n\t"
++ "ldrh r3, [%0], -%4\n\t"
++ "ldrh r4, [%0], -%4\n\t"
++ "ldrh r5, [%0], -%4\n\t"
++ "orr r2, r2, r3, lsl #16\n\t"
++ "orr r4, r4, r5, lsl #16\n\t"
++ "str r2, [%1], #4\n\t"
++ "str r4, [%1], #4\n\t"
++
++ "ldrh r2, [%0], -%4\n\t"
++ "ldrh r3, [%0], -%4\n\t"
++ "ldrh r4, [%0], -%4\n\t"
++ "ldrh r5, [%0], -%4\n\t"
++ "orr r2, r2, r3, lsl #16\n\t"
++ "orr r4, r4, r5, lsl #16\n\t"
++ "str r2, [%1], #4\n\t"
++ "str r4, [%1], #4\n\t"
++
++ "ldrh r2, [%0], -%4\n\t"
++ "ldrh r3, [%0], -%4\n\t"
++ "ldrh r4, [%0], -%4\n\t"
++ "ldrh r5, [%0], -%4\n\t"
++ "orr r2, r2, r3, lsl #16\n\t"
++ "orr r4, r4, r5, lsl #16\n\t"
++ "str r2, [%1], #4\n\t"
++ "str r4, [%1], #4\n\t"
++
++ "ldrh r2, [%0], -%4\n\t"
++ "ldrh r3, [%0], -%4\n\t"
++ "ldrh r4, [%0], -%4\n\t"
++ "ldrh r5, [%0], -%4\n\t"
++ "orr r2, r2, r3, lsl #16\n\t"
++ "orr r4, r4, r5, lsl #16\n\t"
++ "str r2, [%1], #4\n\t"
++ "str r4, [%1], #4\n\t"
++
++ :"+r" (in_block_ptr), "+r"(out_block_ptr) /* output */
++ :"0"(in_block_ptr), "1"(out_block_ptr), "r"(src_line_stride) /* input */
++ :"r2", "r3", "r4", "r5", "memory" /* modify */
++ );
++ in_block_ptr += IN_INDEX;
++ out_block_ptr += OUT_INDEX;
++ }
++ }
++
++ return OPLERR_SUCCESS;
++}
++
++/*
++ * Performs clockwise rotation (and possibly vertical mirroring depending
++ * on the vmirror flag) using block sizes of 4x4
++ * The algorithm is similar to 270 degree clockwise rotation algorithm
++ */
++static int opl_rotate90_u16_by4(const u8 * src, int src_line_stride, int width,
++ int height, u8 * dst, int dst_line_stride,
++ int vmirror)
++{
++ const int BLOCK_SIZE_PIXELS = CACHE_LINE_WORDS * BYTES_PER_WORD
++ / BYTES_PER_PIXEL / 4;
++ const int BLOCK_SIZE_BYTES = BYTES_PER_PIXEL * BLOCK_SIZE_PIXELS;
++ const int IN_INDEX = src_line_stride * BLOCK_SIZE_PIXELS
++ + BYTES_PER_PIXEL;
++ const int OUT_INDEX = vmirror ?
++ -dst_line_stride - BLOCK_SIZE_BYTES
++ : dst_line_stride - BLOCK_SIZE_BYTES;
++ const u8 *in_block_ptr;
++ u8 *out_block_ptr;
++ int i, k;
++
++ for (k = height / BLOCK_SIZE_PIXELS; k > 0; k--) {
++ in_block_ptr = src + src_line_stride * (height - 1)
++ - (src_line_stride * BLOCK_SIZE_PIXELS *
++ (height / BLOCK_SIZE_PIXELS - k));
++ out_block_ptr = dst + BYTES_PER_PIXEL * BLOCK_SIZE_PIXELS
++ * ((height / BLOCK_SIZE_PIXELS) - k);
++
++ /*
++ * For horizontal mirroring the writing starts from the
++ * bottom line
++ */
++ if (vmirror)
++ out_block_ptr += dst_line_stride * (width - 1);
++
++ for (i = width; i > 0; i--) {
++ __asm__ volatile (
++ "ldrh r2, [%0], -%4\n\t"
++ "ldrh r3, [%0], -%4\n\t"
++ "ldrh r4, [%0], -%4\n\t"
++ "ldrh r5, [%0], -%4\n\t"
++ "orr r2, r2, r3, lsl #16\n\t"
++ "orr r4, r4, r5, lsl #16\n\t"
++ "str r2, [%1], #4\n\t"
++ "str r4, [%1], #4\n\t"
++
++ :"+r" (in_block_ptr), "+r"(out_block_ptr) /* output */
++ :"0"(in_block_ptr), "1"(out_block_ptr), "r"(src_line_stride) /* input */
++ :"r2", "r3", "r4", "r5", "memory" /* modify */
++ );
++ in_block_ptr += IN_INDEX;
++ out_block_ptr += OUT_INDEX;
++ }
++ }
++
++ return OPLERR_SUCCESS;
++}
++
++EXPORT_SYMBOL(opl_rotate90_u16);
++EXPORT_SYMBOL(opl_rotate90_vmirror_u16);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/opl/rotate90_u16_qcif.S linux-2.6.28-karo/drivers/media/video/mxc/opl/rotate90_u16_qcif.S
+--- linux-2.6.28/drivers/media/video/mxc/opl/rotate90_u16_qcif.S 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/opl/rotate90_u16_qcif.S 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,71 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++#include <linux/linkage.h>
++
++ .text
++ .align 2
++ENTRY(opl_rotate90_u16_qcif)
++ STMFD sp!,{r4-r10}
++ MOV r12,#0x160
++ MOV r10,#0x90
++ MOV r3,r10,LSR #4
++.L1.216:
++ RSB r2,r3,r10,LSR #4
++ MOV r4,#0x20
++ SMULBB r5,r4,r2
++ MOV r4,#0x1600
++ SMULBB r2,r4,r2
++ ADD r4,r0,#0xc000
++ ADD r4,r4,#0x4a0
++ SUB r4,r4,r2
++ MOV r2,r12,LSR #1
++ ADD r5,r1,r5
++.L1.256:
++ LDRH r6,[r4],-r12
++ LDRH r7,[r4],-r12
++ LDRH r8,[r4],-r12
++ LDRH r9,[r4],-r12
++ ORR r6,r6,r7,LSL #16
++ ORR r7,r8,r9,LSL #16
++ STMIA r5!,{r6,r7}
++ SUBS r2,r2,#1
++ LDRH r6,[r4],-r12
++ LDRH r7,[r4],-r12
++ LDRH r8,[r4],-r12
++ LDRH r9,[r4],-r12
++ ORR r6,r6,r7,LSL #16
++ ORR r7,r8,r9,LSL #16
++ STMIA r5!,{r6,r7}
++ LDRH r6,[r4],-r12
++ LDRH r7,[r4],-r12
++ LDRH r8,[r4],-r12
++ LDRH r9,[r4],-r12
++ ORR r6,r6,r7,LSL #16
++ ORR r7,r8,r9,LSL #16
++ STMIA r5!,{r6,r7}
++ LDRH r6,[r4],-r12
++ LDRH r7,[r4],-r12
++ LDRH r8,[r4],-r12
++ LDRH r9,[r4],-r12
++ ORR r6,r6,r7,LSL #16
++ ORR r7,r8,r9,LSL #16
++ ADD r4,r4,#0x1600
++ STMIA r5!,{r6,r7}
++ ADD r5,r5,#0x100
++ ADD r4,r4,#2
++ BGT .L1.256
++ SUBS r3,r3,#1
++ BGT .L1.216
++ LDMFD sp!,{r4-r10}
++ BX lr
++ .size opl_rotate90_u16_qcif, . - opl_rotate90_u16_qcif
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/opl/vmirror_u16.c linux-2.6.28-karo/drivers/media/video/mxc/opl/vmirror_u16.c
+--- linux-2.6.28/drivers/media/video/mxc/opl/vmirror_u16.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/opl/vmirror_u16.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,46 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++#include <linux/module.h>
++#include <linux/string.h>
++#include "opl.h"
++
++int opl_vmirror_u16(const u8 * src, int src_line_stride, int width, int height,
++ u8 * dst, int dst_line_stride)
++{
++ const u8 *src_row_addr;
++ u8 *dst_row_addr;
++ int i;
++
++ if (!src || !dst)
++ return OPLERR_NULL_PTR;
++
++ if (width == 0 || height == 0 || src_line_stride == 0
++ || dst_line_stride == 0)
++ return OPLERR_BAD_ARG;
++
++ src_row_addr = src;
++ dst_row_addr = dst + (height - 1) * dst_line_stride;
++
++ /* Loop over all rows */
++ for (i = 0; i < height; i++) {
++ /* memcpy each row */
++ memcpy(dst_row_addr, src_row_addr, BYTES_PER_PIXEL * width);
++ src_row_addr += src_line_stride;
++ dst_row_addr -= dst_line_stride;
++ }
++
++ return OPLERR_SUCCESS;
++}
++
++EXPORT_SYMBOL(opl_vmirror_u16);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/output/Kconfig linux-2.6.28-karo/drivers/media/video/mxc/output/Kconfig
+--- linux-2.6.28/drivers/media/video/mxc/output/Kconfig 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/output/Kconfig 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,19 @@
++config VIDEO_MXC_OUTPUT_DEBUG
++ bool "Verbose MXC video output debugging"
++ depends on VIDEO_MXC_OUTPUT
++ default n
++
++config VIDEO_MXC_IPU_OUTPUT
++ bool
++ depends on VIDEO_MXC_OUTPUT && MXC_IPU
++ default y
++ ---help---
++ This is the video4linux2 driver for IPU post processing video output.
++
++config VIDEO_MXC_EMMA_OUTPUT
++ tristate "EMMA video output postprocessor"
++ depends on VIDEO_MXC_OUTPUT && MXC_EMMA && FB_IMX
++ default y
++ ---help---
++ This is the video4linux2 driver for EMMA post processing video output.
++
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/output/Makefile linux-2.6.28-karo/drivers/media/video/mxc/output/Makefile
+--- linux-2.6.28/drivers/media/video/mxc/output/Makefile 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/output/Makefile 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,8 @@
++ifneq ($(CONFIG_VIDEO_MXC_OUTPUT_DEBUG),)
++ EXTRA_CFLAGS += -DDEBUG
++endif
++
++mx27_output-objs := mx27_v4l2_output.o mx27_pp.o
++obj-$(CONFIG_VIDEO_MXC_EMMA_OUTPUT) += mx27_output.o
++
++obj-$(CONFIG_VIDEO_MXC_IPU_OUTPUT) += mxc_v4l2_output.o
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/output/mx27_pp.c linux-2.6.28-karo/drivers/media/video/mxc/output/mx27_pp.c
+--- linux-2.6.28/drivers/media/video/mxc/output/mx27_pp.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/output/mx27_pp.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,1114 @@
++/*
++ * Copyright 2005-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @file mx27_pp.c
++ *
++ * @brief MX27 V4L2 Video Output Driver
++ *
++ * Video4Linux2 Output Device using MX27 eMMA Post-processing functionality.
++ *
++ * @ingroup MXC_V4L2_OUTPUT
++ */
++#include <linux/kernel.h>
++#include <linux/string.h>
++#include <linux/module.h>
++#include <linux/fb.h>
++#include <linux/clk.h>
++#include <linux/interrupt.h>
++#include <asm/io.h>
++#include <mach/mx27.h>
++
++#include "mx27_pp.h"
++#include "mxc_v4l2_output.h"
++
++#define SCALE_RETRY 32 /* to be more relax, less precise */
++#define PP_SKIP 1
++#define PP_TBL_MAX 40
++
++static unsigned short scale_tbl[PP_TBL_MAX];
++static int g_hlen, g_vlen;
++
++static emma_pp_cfg g_pp_cfg;
++static int g_disp_num = 0;
++static char pp_dev[] = "emma_pp";
++
++/*!
++ * @brief PP resizing routines
++ */
++static int scale_2d(emma_pp_scale *sz);
++
++static irqreturn_t pp_isr(int irq, void *dev_id);
++static int set_output_addr(emma_pp_cfg *cfg, vout_data *vout);
++static int pphw_reset(void);
++static int pphw_enable(int flag);
++static int pphw_ptr(emma_pp_cfg *cfg);
++static int pphw_outptr(emma_pp_cfg *cfg);
++static int pphw_cfg(emma_pp_cfg *cfg);
++static int pphw_isr(void);
++static int pphw_init(void);
++static void pphw_exit(void);
++
++#define PP_DUMP(reg) pr_debug("%s[%08lx]\t = 0x%08x\n", #reg, \
++ reg - IO_ADDRESS(EMMA_PP_BASE_ADDR) + EMMA_PP_BASE_ADDR, __raw_readl(reg))
++void pp_dump(void)
++{
++ PP_DUMP(PP_CNTL);
++ PP_DUMP(PP_INTRCNTL);
++ PP_DUMP(PP_INTRSTATUS);
++ PP_DUMP(PP_SOURCE_Y_PTR);
++ PP_DUMP(PP_SOURCE_CB_PTR);
++ PP_DUMP(PP_SOURCE_CR_PTR);
++ PP_DUMP(PP_DEST_RGB_PTR);
++ PP_DUMP(PP_QUANTIZER_PTR);
++ PP_DUMP(PP_PROCESS_FRAME_PARA);
++ PP_DUMP(PP_SOURCE_FRAME_WIDTH);
++ PP_DUMP(PP_DEST_DISPLAY_WIDTH);
++ PP_DUMP(PP_DEST_IMAGE_SIZE);
++ PP_DUMP(PP_DEST_FRAME_FMT_CNTL);
++ PP_DUMP(PP_RESIZE_INDEX);
++ PP_DUMP(PP_CSC_COEF_0123);
++ PP_DUMP(PP_CSC_COEF_4);
++}
++
++static const unsigned char pp_coeftab[] = {
++ 2, 1,
++ 19, 10,
++ 17, 9,
++ 15, 8,
++ 13, 7,
++ 11, 6,
++ 20, 11,
++ 9, 5,
++ 16, 9,
++ 7, 4,
++ 19, 11,
++ 12, 7,
++ 17, 10,
++ 5, 3,
++ 18, 11,
++ 13, 8,
++ 8, 5,
++ 19, 12,
++ 11, 7,
++ 14, 9,
++ 17, 11,
++ 20, 13,
++ 3, 2,
++ 19, 13,
++ 16, 11,
++ 13, 9,
++ 10, 7,
++ 17, 12,
++ 7, 5,
++ 18, 13,
++ 11, 8,
++ 15, 11,
++ 19, 14,
++ 4, 3,
++ 17, 13,
++ 13, 10,
++ 9, 7,
++ 14, 11,
++ 19, 15,
++ 5, 4,
++ 16, 13,
++ 11, 9,
++ 17, 14,
++ 6, 5,
++ 19, 16,
++ 13, 11,
++ 20, 17,
++ 7, 6,
++ 15, 13,
++ 8, 7,
++ 17, 15,
++ 9, 8,
++ 19, 17,
++ 10, 9,
++ 11, 10,
++ 12, 11,
++ 13, 12,
++ 14, 13,
++ 15, 14,
++ 16, 15,
++ 17, 16,
++ 18, 17,
++ 19, 18,
++ 20, 19,
++ 1, 1,
++ 19, 20,
++ 18, 19,
++ 17, 18,
++ 16, 17,
++ 15, 16,
++ 14, 15,
++ 13, 14,
++ 12, 13,
++ 11, 12,
++ 10, 11,
++ 9, 10,
++ 17, 19,
++ 8, 9,
++ 15, 17,
++ 7, 8,
++ 13, 15,
++ 6, 7,
++ 17, 20,
++ 11, 13,
++ 16, 19,
++ 5, 6,
++ 14, 17,
++ 9, 11,
++ 13, 16,
++ 4, 5,
++ 15, 19,
++ 11, 14,
++ 7, 9,
++ 10, 13,
++ 13, 17,
++ 3, 4,
++ 14, 19,
++ 11, 15,
++ 8, 11,
++ 13, 18,
++ 5, 7,
++ 12, 17,
++ 7, 10,
++ 9, 13,
++ 11, 16,
++ 13, 19,
++ 2, 3,
++ 13, 20,
++ 11, 17,
++ 9, 14,
++ 7, 11,
++ 12, 19,
++ 5, 8,
++ 8, 13,
++ 11, 18,
++ 3, 5,
++ 10, 17,
++ 7, 12,
++ 11, 19,
++ 4, 7,
++ 9, 16,
++ 5, 9,
++ 11, 20,
++ 6, 11,
++ 7, 13,
++ 8, 15,
++ 9, 17,
++ 10, 19,
++ 1, 2,
++ 9, 19,
++ 8, 17,
++ 7, 15,
++ 6, 13,
++ 5, 11,
++ 9, 20,
++ 4, 9,
++ 7, 16,
++ 3, 7,
++ 8, 19,
++ 5, 12,
++ 7, 17,
++ 2, 5,
++ 7, 18,
++ 5, 13,
++ 3, 8,
++ 7, 19,
++ 4, 11,
++ 5, 14,
++ 6, 17,
++ 7, 20,
++ 1, 3,
++ 6, 19,
++ 5, 16,
++ 4, 13,
++ 3, 10,
++ 5, 17,
++ 2, 7,
++ 5, 18,
++ 3, 11,
++ 4, 15,
++ 5, 19,
++ 1, 4
++};
++
++/*!
++ * @brief Set PP input address.
++ * @param ptr The pointer to the Y value of input
++ * @return Zero on success, others on failure
++ */
++int pp_ptr(unsigned long ptr, struct v4l2_mxc_offset offset)
++{
++ g_pp_cfg.ptr.y = ptr + offset.y_offset;
++ g_pp_cfg.ptr.u = ptr + offset.u_offset;
++ g_pp_cfg.ptr.v = ptr + offset.v_offset;
++ g_pp_cfg.ptr.qp = ptr + offset.qp_offset;
++
++ return pphw_ptr(&g_pp_cfg);
++}
++
++/*!
++ * @brief Enable or disable PP.
++ * @param flag Zero to disable PP, others to enable PP
++ * @return Zero on success, others on failure
++ */
++int pp_enable(int flag)
++{
++ return pphw_enable(flag);
++}
++
++/*!
++ * @brief Get the display No. of last completed PP frame.
++ * @return The display No. of last completed PP frame.
++ */
++int pp_num_last(void)
++{
++ return g_disp_num ? 0 : 1;
++}
++
++/*!
++ * @brief Initialize PP.
++ * @param vout Pointer to _vout_data structure
++ * @return Zero on success, others on failure
++ */
++static int interrupts;
++
++int pp_init(vout_data *vout)
++{
++ int ret = pphw_init();
++ if (ret) {
++ return ret;
++ }
++ ret = pphw_enable(0);
++ if (ret) {
++ pphw_exit();
++ return ret;
++ }
++ interrupts = 0;
++ ret = request_irq(MXC_INT_EMMAPP, pp_isr, 0, pp_dev, vout);
++ if (ret) {
++ pphw_exit();
++ return ret;
++ }
++ return 0;
++}
++
++/*!
++ * @brief Deinitialize PP.
++ * @param vout Pointer to _vout_data structure
++ */
++void pp_exit(vout_data *vout)
++{
++ free_irq(MXC_INT_EMMAPP, vout);
++ pr_debug("%s: Interrupts: %d\n", __FUNCTION__, interrupts);
++ pphw_enable(0);
++ pphw_exit();
++}
++
++/*!
++ * @brief Configure PP.
++ * @param vout Pointer to _vout_data structure
++ * @return Zero on success, others on failure
++ */
++int pp_cfg(vout_data *vout)
++{
++ unsigned long flags;
++
++ /* PP accepts YUV420 input only */
++ if ((vout->v2f.fmt.pix.pixelformat != V4L2_PIX_FMT_YUV420) &&
++ (vout->v2f.fmt.pix.pixelformat != V4L2_PIX_FMT_YUV422P)) {
++// if (vout->v2f.fmt.pix.pixelformat != V4L2_PIX_FMT_YUV420) {
++ pr_debug("unsupported pixel format: %08x\n", vout->v2f.fmt.pix.pixelformat);
++ return -EINVAL;
++ }
++
++ g_pp_cfg.operation = 0;
++
++ memset(g_pp_cfg.csc_table, 0, sizeof(g_pp_cfg.csc_table));
++
++ /* Convert output pixel format to PP required format */
++ switch (vout->v4l2_fb.fmt.pixelformat) {
++ case V4L2_PIX_FMT_BGR32:
++ g_pp_cfg.red_width = 8;
++ g_pp_cfg.green_width = 8;
++ g_pp_cfg.blue_width = 8;
++ g_pp_cfg.red_offset = 8;
++ g_pp_cfg.green_offset = 16;
++ g_pp_cfg.blue_offset = 24;
++ g_pp_cfg.rgb_resolution = 32;
++ break;
++ case V4L2_PIX_FMT_RGB32:
++ g_pp_cfg.red_width = 8;
++ g_pp_cfg.green_width = 8;
++ g_pp_cfg.blue_width = 8;
++ g_pp_cfg.red_offset = 24;
++ g_pp_cfg.green_offset = 16;
++ g_pp_cfg.blue_offset = 8;
++ g_pp_cfg.rgb_resolution = 32;
++ break;
++ case V4L2_PIX_FMT_YUYV:
++ g_pp_cfg.red_width = 0;
++ g_pp_cfg.green_width = 0;
++ g_pp_cfg.blue_width = 0;
++ g_pp_cfg.red_offset = 0;
++ g_pp_cfg.green_offset = 0;
++ g_pp_cfg.blue_offset = PP_PIX_YUYV;
++ g_pp_cfg.rgb_resolution = 16;
++ break;
++ case V4L2_PIX_FMT_UYVY:
++ g_pp_cfg.red_width = 0;
++ g_pp_cfg.green_width = 0;
++ g_pp_cfg.blue_width = 0;
++ g_pp_cfg.red_offset = 0;
++ g_pp_cfg.green_offset = 0;
++ g_pp_cfg.blue_offset = PP_PIX_UYVY;
++ g_pp_cfg.rgb_resolution = 16;
++ break;
++ case V4L2_PIX_FMT_RGB565:
++ default:
++ g_pp_cfg.red_width = 5;
++ g_pp_cfg.green_width = 6;
++ g_pp_cfg.blue_width = 5;
++ g_pp_cfg.red_offset = 11;
++ g_pp_cfg.green_offset = 5;
++ g_pp_cfg.blue_offset = 0;
++ g_pp_cfg.rgb_resolution = 16;
++ break;
++ }
++ spin_lock_irqsave(&vout->irq_lock, flags);
++ if (vout->active != NULL) {
++ g_pp_cfg.ptr.y = vout->active->dma_desc.dma_addr;
++ } else {
++ g_pp_cfg.ptr.y = 0;
++ }
++ spin_unlock_irqrestore(&vout->irq_lock, flags);
++ g_pp_cfg.ptr.u = g_pp_cfg.ptr.v = g_pp_cfg.ptr.qp = 0;
++ if ((vout->crop_rect.width != 0) && (vout->crop_rect.height != 0)) {
++ g_pp_cfg.dim.in.width = vout->crop_rect.width;
++ g_pp_cfg.dim.in.height = vout->crop_rect.height;
++ } else {
++ g_pp_cfg.dim.in.width = vout->v2f.fmt.pix.width;
++ g_pp_cfg.dim.in.height = vout->v2f.fmt.pix.height;
++ }
++
++ g_pp_cfg.dim.out.width = vout->crop_current.width;
++ g_pp_cfg.dim.out.height = vout->crop_current.height;
++ g_pp_cfg.dim.num.width = 0;
++ g_pp_cfg.dim.num.height = 0;
++ g_pp_cfg.dim.den.width = 0;
++ g_pp_cfg.dim.den.height = 0;
++
++ pr_debug("input dimensions: %dx%d output dimensions: %dx%d\n",
++ g_pp_cfg.dim.in.width, g_pp_cfg.dim.in.height,
++ g_pp_cfg.dim.out.width, g_pp_cfg.dim.out.height);
++
++ if (scale_2d(&g_pp_cfg.dim)) {
++ printk(KERN_ERR "unsupported resize ratio\n");
++ return -EINVAL;
++ }
++
++ g_pp_cfg.dim.out.width = vout->crop_current.width;
++ g_pp_cfg.dim.out.height = vout->crop_current.height;
++
++ g_pp_cfg.in_y_stride = vout->v2f.fmt.pix.width;
++ g_pp_cfg.in_height = vout->v2f.fmt.pix.height;
++ if (set_output_addr(&g_pp_cfg, vout)) {
++ printk(KERN_ERR "failed to set pp output address\n");
++ return -EINVAL;
++ }
++
++ return pphw_cfg(&g_pp_cfg);
++}
++
++irqreturn_t mxc_v4l2out_pp_in_irq_handler(int irq, void *dev_id);
++
++/*!
++ * @brief PP IRQ handler.
++ */
++static irqreturn_t pp_isr(int irq, void *dev_id)
++{
++ int status;
++ vout_data *vout = dev_id;
++
++interrupts++;
++ status = pphw_isr();
++ if ((status & 0x1) == 0) { /* Not frame complete interrupt */
++ pr_debug("not pp frame complete interrupt\n");
++ return IRQ_HANDLED;
++ }
++
++ if (vout->v4l2_fb.flags == V4L2_FBUF_FLAG_OVERLAY) {
++ g_disp_num = g_disp_num ? 0 : 1;
++ g_pp_cfg.outptr = (unsigned int)vout->display_bufs[g_disp_num];
++ pphw_outptr(&g_pp_cfg);
++ }
++
++ return mxc_v4l2out_pp_in_irq_handler(irq, dev_id);
++}
++
++/*!
++ * @brief Set PP output address.
++ * @param cfg Pointer to emma_pp_cfg structure
++ * @param vout Pointer to _vout_data structure
++ * @return Zero on success, others on failure
++ */
++static int set_output_addr(emma_pp_cfg *cfg, vout_data *vout)
++{
++ if (vout->v4l2_fb.flags == V4L2_FBUF_FLAG_OVERLAY) {
++ g_disp_num = 0;
++ cfg->outptr = (unsigned int)vout->display_bufs[g_disp_num];
++ cfg->out_stride = vout->crop_current.width;
++ return 0;
++ } else {
++ struct fb_info *fb;
++
++ fb = registered_fb[vout->output_fb_num[vout->cur_disp_output]];
++ if (!fb)
++ return -ENODEV;
++
++ cfg->outptr = fb->fix.smem_start;
++ cfg->outptr += vout->crop_current.top * fb->var.xres_virtual *
++ (fb->var.bits_per_pixel >> 3) +
++ vout->crop_current.left * (fb->var.bits_per_pixel >> 3);
++ cfg->out_stride = fb->var.xres_virtual;
++
++ return 0;
++ }
++}
++
++/*!
++ * @brief Get maximum common divisor.
++ * @param x First input value
++ * @param y Second input value
++ * @return Maximum common divisor of x and y
++ */
++static int gcd(int x, int y)
++{
++ int k;
++
++ if (x < y) {
++ k = x;
++ x = y;
++ y = k;
++ }
++
++ while ((k = x % y)) {
++ x = y;
++ y = k;
++ }
++
++ return y;
++}
++
++/*!
++ * @brief Get ratio.
++ * @param x First input value
++ * @param y Second input value
++ * @param den Denominator of the ratio (corresponding to y)
++ * @return Numerator of the ratio (corresponding to x)
++ */
++static int ratio(int x, int y, int *den)
++{
++ int g;
++
++ if (!x || !y)
++ return 0;
++
++ g = gcd(x, y);
++ *den = y / g;
++
++ return x / g;
++}
++
++/*!
++ * @brief Build PP coefficient entry
++ * Build one or more coefficient entries for PP coefficient table based
++ * on given coefficient.
++ *
++ * @param k The index of the coefficient in coefficient table
++ * @param coeff The weighting coefficient
++ * @param base The base of the coefficient
++ * @param nxt Number of pixels to be read
++ *
++ * @return The index of the next coefficient entry on success
++ * -1 on failure
++ */
++static int scale_0d(int k, int coeff, int base, int nxt)
++{
++ if (k >= PP_TBL_MAX) {
++ /* no more space in table */
++ pr_debug("no space in scale table, k = %d\n", k);
++ return -ENOSPC;
++ }
++
++ coeff = ((coeff << BC_COEF) + (base >> 1)) / base;
++
++ /*
++ * Valid values for weighting coefficient are 0, 2 to 30, and 31.
++ * A value of 31 is treated as 32 and therefore 31 is an
++ * invalid co-efficient.
++ */
++ if (coeff >= SZ_COEF - 1)
++ coeff--;
++ else if (coeff == 1)
++ coeff++;
++ coeff = coeff << BC_NXT;
++
++ if (nxt < SZ_NXT) {
++ coeff |= nxt;
++ coeff <<= 1;
++ coeff |= 1;
++ } else {
++ /*
++ * src inc field is 2 bit wide, for 4+, use special
++ * code 0:0:1 to prevent dest inc
++ */
++ coeff |= PP_SKIP;
++ coeff <<= 1;
++ coeff |= 1;
++ nxt -= PP_SKIP;
++ do {
++ pr_debug("tbl = %03X\n", coeff);
++ scale_tbl[k++] = coeff;
++ coeff = (nxt > PP_SKIP) ? PP_SKIP : nxt;
++ coeff <<= 1;
++ } while ((nxt -= PP_SKIP) > 0);
++ }
++ pr_debug("tbl = %03X\n", coeff);
++ scale_tbl[k++] = coeff;
++
++ return k;
++}
++
++/*!
++ * @brief Get approximate ratio
++ *
++ * @param pscale The pointer to scale_t structure which holdes
++ * coefficient tables
++ * @param mt Scale ratio numerator
++ * @param nt Scale ratio denominator
++ * @param *n denominator of approximate ratio
++ * @return numerator of approximate ratio
++ */
++static int approx_ratio(int mt, int nt, int *n)
++{
++ int index = sizeof(pp_coeftab) / sizeof(pp_coeftab[0]) / 2;
++ int left = 0;
++ int right = index - 1;
++ int nom = 0, den = 0;
++ while (index > 0) {
++ nom = pp_coeftab[(((right + left) >> 1) << 1)];
++ den = pp_coeftab[(((right + left) >> 1) << 1) + 1];
++ if ((nom * nt - mt * den) > 0) {
++ left = (right + left) >> 1;
++ } else {
++ right = (right + left) >> 1;
++ }
++ index = index >> 1;
++ }
++ *n = pp_coeftab[right * 2 + 1];
++ nom = pp_coeftab[right * 2];
++ return nom;
++}
++
++/*
++ * @brief Build PP coefficient table
++ * Build PP coefficient table for one dimension (width or height)
++ * based on given input and output resolution
++ *
++ * @param inv input resolution
++ * @param outv output resolution
++ * @param k index of free table entry
++ *
++ * @return The index of the next free coefficient entry on success
++ * -1 on failure
++ */
++static int scale_1d(int inv, int outv, int k)
++{
++ int v; /* overflow counter */
++ int coeff, nxt; /* table output */
++
++ if (inv == outv)
++ return scale_0d(k, 1, 1, 1); /* force scaling */
++
++ v = 0;
++ if (inv < outv) {
++ /* upscale: mix <= 2 input pixels per output pixel */
++ do {
++ coeff = outv - v;
++ v += inv;
++ if (v >= outv) {
++ v -= outv;
++ nxt = 1;
++ } else
++ nxt = 0;
++ pr_debug("upscale: coeff = %d/%d nxt = %d\n", coeff,
++ outv, nxt);
++ k = scale_0d(k, coeff, outv, nxt);
++ if (k < 0)
++ return -1;
++ } while (v);
++ } else if (inv >= 2 * outv) {
++ /* PP doesn't support resize ratio > 2:1 except 4:1. */
++ if ((inv != 2 * outv) && (inv != 4 * outv))
++ return -1;
++ /* downscale: >=2:1 bilinear approximation */
++ coeff = inv - 2 * outv;
++ v = 0;
++ nxt = 0;
++ do {
++ v += coeff;
++ nxt = 2;
++ while (v >= outv) {
++ v -= outv;
++ nxt++;
++ }
++ pr_debug("downscale: coeff = 1/2 nxt = %d\n", nxt);
++ k = scale_0d(k, 1, 2, nxt);
++ if (k < 0)
++ return -1;
++ } while (v);
++ } else {
++ /* downscale: bilinear */
++ int in_pos_inc = 2 * outv;
++ int out_pos = inv;
++ int out_pos_inc = 2 * inv;
++ int init_carry = inv - outv;
++ int carry = init_carry;
++
++ v = outv + in_pos_inc;
++ do {
++ coeff = v - out_pos;
++ out_pos += out_pos_inc;
++ carry += out_pos_inc;
++ for (nxt = 0; v < out_pos; nxt++) {
++ v += in_pos_inc;
++ carry -= in_pos_inc;
++ }
++ pr_debug("downscale: coeff = %d/%d nxt = %d\n", coeff,
++ in_pos_inc, nxt);
++ k = scale_0d(k, coeff, in_pos_inc, nxt);
++ if (k < 0)
++ return -1;
++ } while (carry != init_carry);
++ }
++ return k;
++}
++
++/*
++ * @brief Build PP coefficient table
++ * Build PP coefficient table for one dimension (width or height)
++ * based on given input and output resolution. The given input
++ * and output resolution might be not supported due to hardware
++ * limits. In this case this functin rounds the input and output
++ * to closest possible values and return them to caller.
++ *
++ * @param inv input resolution, might be modified after the call
++ * @param outv output resolution, might be modified after the call
++ * @param k index of free table entry
++ *
++ * @return The index of the next free coefficient entry on success
++ * -1 on failure
++ */
++static int scale_1d_smart(int *inv, int *outv, int index)
++{
++ int len, num, den, approx_num, approx_den;
++ static int num1, den1;
++
++ if (!inv || !outv)
++ return -1;
++
++ /* Both should be non-zero */
++ if (!(*inv) || !(*outv))
++ return -1;
++
++ if ((*outv > 4 * (*inv)) ||
++ ((*inv > 2 * (*outv)) && (*inv != 4 * (*outv)))) {
++ pr_debug("unsupported pp resize ration: inv/outv = %d/%d\n",
++ *inv, *outv);
++ return -1;
++ }
++
++ num = ratio(*inv, *outv, &den);
++ pr_debug("%s: num=%d den=%d\n", __FUNCTION__, num, den);
++
++ if (index == 0) {
++ if ((num > 20) || (den > 20)) {
++ approx_num = approx_ratio(num, den, &approx_den);
++ num = approx_num;
++ den = approx_den;
++ }
++ } else {
++ if ((num > (40 - index)) || (den > (40 - index))) {
++ approx_num = approx_ratio(num, den, &approx_den);
++ num = approx_num;
++ den = approx_den;
++ }
++ if ((num == num1) && (den == den1)) {
++ return index;
++ }
++ }
++
++ if ((len = scale_1d(num, den, index)) < 0) {
++ return -1;
++ } else {
++ if (index == 0) {
++ num1 = num;
++ den1 = den;
++ }
++ return len;
++ }
++
++}
++
++/*
++ * @brief Build PP coefficient table for both width and height
++ * Build PP coefficient table for both width and height based on
++ * given resizing ratios.
++ *
++ * @param sz Structure contains resizing ratio informations
++ *
++ * @return 0 on success, others on failure
++ */
++static int scale_2d(emma_pp_scale *sz)
++{
++ int inv, outv;
++
++ /* horizontal resizing. parameter check - must provide in size */
++ if (!sz->in.width)
++ return -1;
++
++ /* Resizing based on num:den */
++ inv = sz->num.width;
++ outv = sz->den.width;
++
++ if ((g_hlen = scale_1d_smart(&inv, &outv, 0)) > 0) {
++ /* Resizing succeeded */
++ sz->den.width = outv;
++ sz->out.width = (sz->in.width * outv) / inv;
++ } else {
++ /* Resizing based on in:out */
++ inv = sz->in.width;
++ outv = sz->out.width;
++
++ if ((g_hlen = scale_1d_smart(&inv, &outv, 0)) > 0) {
++ /* Resizing succeeded */
++ sz->out.width = outv;
++ sz->num.width = ratio(sz->in.width, sz->out.width,
++ &sz->den.width);
++ } else
++ return -1;
++ }
++
++ sz->out.width &= ~1;
++
++ /* vertical resizing. parameter check - must provide in size */
++ if (!sz->in.height)
++ return -1;
++
++ /* Resizing based on num:den */
++ inv = sz->num.height;
++ outv = sz->den.height;
++
++ if ((g_vlen = scale_1d_smart(&inv, &outv, g_hlen)) > 0) {
++ /* Resizing succeeded */
++ sz->den.height = outv;
++ sz->out.height = (sz->in.height * outv) / inv;
++ } else {
++ /* Resizing based on in:out */
++ inv = sz->in.height;
++ outv = sz->out.height;
++
++ if ((g_vlen = scale_1d_smart(&inv, &outv, g_hlen)) > 0) {
++ /* Resizing succeeded */
++ sz->out.height = outv;
++ sz->num.height = ratio(sz->in.height, sz->out.height,
++ &sz->den.height);
++ } else
++ return -1;
++ }
++
++ return 0;
++}
++
++/*!
++ * @brief Set PP resizing registers.
++ * @param sz Pointer to pp scaling structure
++ * @return Zero on success, others on failure
++ */
++static int pphw_scale(emma_pp_scale * sz)
++{
++ __raw_writel((sz->out.width << 16) | sz->out.height,
++ PP_DEST_IMAGE_SIZE);
++ __raw_writel(((g_hlen - 1) << 16) | (g_vlen ==
++ g_hlen ? 0 : (g_hlen << 8)) |
++ (g_vlen - 1), PP_RESIZE_INDEX);
++ for (g_hlen = 0; g_hlen < g_vlen; g_hlen++)
++ __raw_writel(scale_tbl[g_hlen],
++ PP_RESIZE_COEF_TBL + g_hlen * 4);
++
++ return 0;
++}
++
++/*!
++ * @brief Reset PP.
++ * @return Zero on success, others on failure
++ */
++static int pphw_reset(void)
++{
++ int i;
++ u32 pp_cntl;
++
++ __raw_writel(0x100, PP_CNTL);
++
++ /* timeout */
++ for (i = 0; i < 1000; i++) {
++ if (!((pp_cntl = __raw_readl(PP_CNTL)) & 0x100)) {
++ pr_debug("pp reset over\n");
++ break;
++ }
++ }
++
++ /* check reset value */
++ if (pp_cntl != 0x876) {
++ pr_debug("pp reset value err = 0x%08X\n", pp_cntl);
++ return -EIO;
++ }
++
++ return 0;
++}
++
++/*!
++ * @brief Enable or disable PP.
++ * @param flag Zero to disable PP, others to enable PP
++ * @return Zero on success, others on failure
++ */
++static int pphw_enable(int flag)
++{
++ int ret = 0;
++
++ if (flag)
++ __raw_writel(__raw_readl(PP_CNTL) | 1, PP_CNTL);
++ else
++ ret = pphw_reset();
++
++ return ret;
++}
++
++/*!
++ * @brief Set PP input address.
++ * @param cfg The pointer to PP configuration parameter
++ * @return Zero on success, others on failure
++ */
++static int pphw_ptr(emma_pp_cfg *cfg)
++{
++ __raw_writel(cfg->ptr.y, PP_SOURCE_Y_PTR);
++ __raw_writel(cfg->ptr.u, PP_SOURCE_CB_PTR);
++ __raw_writel(cfg->ptr.v, PP_SOURCE_CR_PTR);
++ __raw_writel(cfg->ptr.qp, PP_QUANTIZER_PTR);
++
++ return 0;
++}
++
++/*!
++ * @brief Set PP output address.
++ * @param cfg The pointer to PP configuration parameter
++ * @return Zero on success, others on failure
++ */
++static int pphw_outptr(emma_pp_cfg *cfg)
++{
++ __raw_writel(cfg->outptr, PP_DEST_RGB_PTR);
++ return 0;
++}
++
++/*!
++ * @brief Configuration PP.
++ * @param cfg The pointer to PP configuration parameter
++ * @return Zero on success, others on failure
++ */
++static int pphw_cfg(emma_pp_cfg *cfg)
++{
++ int rt;
++ register int r;
++
++ pphw_scale(&cfg->dim);
++
++ if (!cfg->in_y_stride)
++ cfg->in_y_stride = cfg->dim.in.width;
++
++ if (!cfg->out_stride)
++ cfg->out_stride = cfg->dim.out.width;
++
++ r = __raw_readl(PP_CNTL) & ~EN_MASK;
++
++ /* config parms */
++ r |= cfg->operation & EN_MASK;
++ if (cfg->operation & EN_MACROBLOCK) {
++ /* Macroblock Mode */
++ r |= 0x0200;
++ __raw_writel(0x06, PP_INTRCNTL);
++ } else {
++ /* Frame mode */
++ __raw_writel(0x05, PP_INTRCNTL);
++ }
++
++ if (cfg->red_width | cfg->green_width | cfg->blue_width) {
++ /* color conversion to be performed */
++ r |= EN_CSC;
++ if (!(cfg->red_offset | cfg->green_offset)) {
++ /* auto offset B:G:R LSb to Msb */
++ cfg->green_offset = cfg->blue_offset + cfg->blue_width;
++ cfg->red_offset = cfg->green_offset + cfg->green_width;
++ }
++ if (!cfg->rgb_resolution) {
++ /* derive minimum resolution required */
++ int w, w2;
++
++ w = cfg->red_offset + cfg->red_width;
++ w2 = cfg->blue_offset + cfg->blue_width;
++ if (w < w2)
++ w = w2;
++ w2 = cfg->green_offset + cfg->green_width;
++ if (w < w2)
++ w = w2;
++ if (w > 16)
++ w = 24;
++ else if (w > 8)
++ w = 16;
++ else
++ w = 8;
++ cfg->rgb_resolution = w;
++ }
++ /* 00,11 - 32 bpp, 10 - 16 bpp, 01 - 8 bpp */
++ r &= ~0xC00;
++ if (cfg->rgb_resolution < 32)
++ r |= (cfg->rgb_resolution << 7);
++ __raw_writel((cfg->red_offset << 26) |
++ (cfg->green_offset << 21) |
++ (cfg->blue_offset << 16) |
++ (cfg->red_width << 8) |
++ (cfg->green_width << 4) |
++ cfg->blue_width, PP_DEST_FRAME_FMT_CNTL);
++ } else {
++ /* add YUV422 formatting */
++ static const unsigned int _422[] = {
++ 0x62000888,
++ 0x60100888,
++ 0x43080888,
++ 0x41180888
++ };
++
++ __raw_writel(_422[(cfg->blue_offset >> 3) & 3],
++ PP_DEST_FRAME_FMT_CNTL);
++ cfg->rgb_resolution = 16;
++ r &= ~0xC00;
++ r |= (cfg->rgb_resolution << 7);
++ }
++
++ /* add csc formatting */
++ if (!cfg->csc_table[1]) {
++ static const unsigned short _csc[][6] = {
++ {0x80, 0xb4, 0x2c, 0x5b, 0x0e4, 0},
++ {0x95, 0xcc, 0x32, 0x68, 0x104, 1},
++ {0x80, 0xca, 0x18, 0x3c, 0x0ec, 0},
++ {0x95, 0xe5, 0x1b, 0x44, 0x10e, 1},
++ };
++ memcpy(cfg->csc_table, _csc[cfg->csc_table[0]],
++ sizeof(_csc[0]));
++ }
++ __raw_writel((cfg->csc_table[0] << 24) |
++ (cfg->csc_table[1] << 16) |
++ (cfg->csc_table[2] << 8) |
++ cfg->csc_table[3], PP_CSC_COEF_0123);
++ __raw_writel((cfg->csc_table[5] ? (1 << 9) : 0) | cfg->csc_table[4],
++ PP_CSC_COEF_4);
++
++ __raw_writel(r, PP_CNTL);
++
++ pphw_ptr(cfg);
++ pphw_outptr(cfg);
++
++ /*
++ * #MB in a row = input_width / 16pix
++ * 1 byte per QP per MB
++ * QP must be formatted to be 4-byte aligned
++ * YUV lines are to be 4-byte aligned as well
++ * So Y is 8 byte aligned, as U = V = Y/2 for 420
++ * MPEG MBs are 16x16 anyway
++ */
++ __raw_writel((cfg->dim.in.width << 16) | cfg->dim.in.height,
++ PP_PROCESS_FRAME_PARA);
++ __raw_writel(cfg->in_y_stride | (PP_CALC_QP_WIDTH(cfg) << 16),
++ PP_SOURCE_FRAME_WIDTH);
++
++ /* in bytes */
++ rt = cfg->rgb_resolution >> 3;
++ if (rt == 3)
++ rt = 4;
++ __raw_writel(cfg->out_stride * rt, PP_DEST_DISPLAY_WIDTH);
++
++ pp_dump();
++ return 0;
++}
++
++/*!
++ * @brief Check PP interrupt status.
++ * @return PP interrupt status
++ */
++static int pphw_isr(void)
++{
++ unsigned long status;
++
++ status = __raw_readl(PP_INTRSTATUS) & 7;
++ if (!status) {
++ pr_debug("pp: not my isr err\n");
++ return status;
++ }
++
++ if (status & 4)
++ pr_debug("pp: isr state error\n");
++
++ /* clear interrupt status */
++ __raw_writel(status, PP_INTRSTATUS);
++
++ return status;
++}
++
++static struct clk *emma_clk;
++
++/*!
++ * @brief PP module clock enable
++ */
++static int pphw_init(void)
++{
++ emma_clk = clk_get(NULL, "emma_clk");
++ if (IS_ERR(emma_clk)) {
++ printk(KERN_ERR "Could not get emma_clk: %ld\n", PTR_ERR(emma_clk));
++ return PTR_ERR(emma_clk);
++ }
++ clk_enable(emma_clk);
++ return 0;
++}
++
++/*!
++ * @brief PP module clock disable
++ */
++static void pphw_exit(void)
++{
++ clk_disable(emma_clk);
++ clk_put(emma_clk);
++}
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/output/mx27_pp.h linux-2.6.28-karo/drivers/media/video/mxc/output/mx27_pp.h
+--- linux-2.6.28/drivers/media/video/mxc/output/mx27_pp.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/output/mx27_pp.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,181 @@
++/*
++ * Copyright 2005-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @file mx27_pp.h
++ *
++ * @brief Header file for MX27 V4L2 Video Output Driver
++ *
++ * @ingroup MXC_V4L2_OUTPUT
++ */
++#ifndef __MX27_PP_H__
++#define __MX27_PP_H__
++
++#include "mxc_v4l2_output.h"
++
++/* PP register definitions */
++#define PP_REG(ofs) (IO_ADDRESS(EMMA_PP_BASE_ADDR) + ofs)
++
++/* Register offsets */
++#define PP_CNTL PP_REG(0x00)
++#define PP_INTRCNTL PP_REG(0x04)
++#define PP_INTRSTATUS PP_REG(0x08)
++#define PP_SOURCE_Y_PTR PP_REG(0x0C)
++#define PP_SOURCE_CB_PTR PP_REG(0x10)
++#define PP_SOURCE_CR_PTR PP_REG(0x14)
++#define PP_DEST_RGB_PTR PP_REG(0x18)
++#define PP_QUANTIZER_PTR PP_REG(0x1C)
++#define PP_PROCESS_FRAME_PARA PP_REG(0x20)
++#define PP_SOURCE_FRAME_WIDTH PP_REG(0x24)
++#define PP_DEST_DISPLAY_WIDTH PP_REG(0x28)
++#define PP_DEST_IMAGE_SIZE PP_REG(0x2C)
++#define PP_DEST_FRAME_FMT_CNTL PP_REG(0x30)
++#define PP_RESIZE_INDEX PP_REG(0x34)
++#define PP_CSC_COEF_0123 PP_REG(0x38)
++#define PP_CSC_COEF_4 PP_REG(0x3C)
++#define PP_RESIZE_COEF_TBL PP_REG(0x100)
++
++/* resize table dimensions
++ dest pixel index left/32 right/32 #src pixels to read
++ 0 [BC_COEF] [BC_COEF] [BC_NXT]
++ :
++ pp_tbl_max-1
++*/
++#define BC_NXT 2
++#define BC_COEF 5
++#define SZ_COEF (1 << BC_COEF)
++#define SZ_NXT (1 << BC_NXT)
++
++/* PP operations */
++#define EN_DEBLOCK 0x02
++#define EN_DERING 0x04
++#define EN_CSC 0x10
++#define EN_MACROBLOCK 0x20
++#define EN_DEF 0x16
++#define EN_MASK 0x36
++#define EN_BIGDATA 0x1000
++#define EN_BIGQP 0x2000
++
++/* PP CSC tables */
++#define CSC_TBL_NONE 0x80
++#define CSC_TBL_REUSE 0x81
++#define CSC_TBL_A1 0x00
++#define CSC_TBL_A0 0x20
++#define CSC_TBL_B1 0x40
++#define CSC_TBL_B0 0x60
++/* converts from 4 decimal fixed point to hw setting & vice versa */
++#define PP_CSC_FP4_2_HW(coeff) ((((coeff) << 7) + 5000) / 10000)
++#define PP_CSC_HW_2_FP4(coeff) ((((coeff) * 10000) + 64) >> 7)
++
++#define PP_PIX_YUYV 0
++#define PP_PIX_YVYU 8
++#define PP_PIX_UYVY 16
++#define PP_PIX_VYUY 24
++
++/* PP size & width calculation macros */
++#define PP_CALC_QP_WIDTH(cfg) \
++ (!((cfg)->operation & (EN_DEBLOCK | EN_DERING)) ? 0 : \
++ (((((cfg)->in_y_stride + 15) >> 4) + 3) & ~3))
++#define PP_CALC_Y_SIZE(cfg) \
++ ((cfg)->in_y_stride * (cfg)->in_height)
++#define PP_CALC_CH_SIZE(cfg) (PP_CALC_Y_SIZE(cfg) >> 2)
++#define PP_CALC_BPP(cfg) \
++ ((cfg)->rgb_resolution > 16 ? 4 : ((cfg)->rgb_resolution >> 3))
++#define PP_CALC_YUV_SIZE(cfg) \
++ ((PP_CALC_Y_SIZE(cfg) * 3) >> 1)
++#define PP_CALC_QP_SIZE(cfg) \
++ (PP_CALC_QP_WIDTH(cfg) * (((cfg)->in_height + 15) >> 4))
++#define PP_CALC_DEST_WIDTH(cfg) \
++ (((cfg)->out_stride & ~1) * PP_CALC_BPP(cfg))
++#define PP_CALC_DEST_SIZE(cfg) \
++ ((cfg)->dim.out.height * PP_CALC_DEST_WIDTH(cfg))
++
++/*
++ * physical addresses for bus mastering
++ * v=0 -> yuv packed
++ * v=0 & qp=0 -> yuv packed with qp appended
++ */
++typedef struct _emma_pp_ptr {
++ unsigned int y; /* Y data (line align8) */
++ unsigned int u; /* U data (line align4) */
++ unsigned int v; /* V data (line align4) */
++ unsigned int qp; /* Quantization (line align4) */
++} emma_pp_ptr;
++
++typedef struct _emma_pp_size {
++ int width;
++ int height;
++} emma_pp_size;
++
++/*
++ * if num.width != 0
++ * resize ratio = num.width : den.width
++ * else
++ * resize ratio = in.width : out.width
++ * same for height
++ */
++typedef struct _emma_pp_scale {
++ emma_pp_size num;
++ emma_pp_size den;
++ emma_pp_size in; /* clip */
++ emma_pp_size out; /* 0 -> same as in */
++} emma_pp_scale;
++
++typedef struct _emma_pp_cfg {
++ unsigned char operation; /* OR of EN_xx defines */
++
++ /*
++ * input color coeff
++ * fixed pt 8 bits, steps of 1/128
++ * csc[5] is 1 or 0 to indicate Y + 16
++ * csc[0] is matrix id 0-3 while csc[1-5]=0
++ */
++ unsigned short csc_table[6];
++
++ /*
++ * Output color (shade width, shade offset, pixel resolution)
++ * Eg. 16bpp RGB565 resolution, the values could be:
++ * red_width = 5, green_width = 6, blue_width = 6
++ * red_offset = 11, green_offset = 5, blue_offset = 0 (defaults)
++ * rgb_resolution = 16 (default)
++ * For YUV422: xxx_width=0, blue_offset=PP_PIX_xxx
++ */
++ unsigned short red_width;
++ unsigned short green_width;
++ unsigned short blue_width;
++ /* if offsets are 0, the offsets are by width LSb to MSb B:G:R */
++ unsigned short red_offset;
++ unsigned short blue_offset;
++ unsigned short green_offset;
++ /* if resolution is 0, the minimum for the sum of widths is chosen */
++ short rgb_resolution; /* 8,16,24 bpp only */
++
++ emma_pp_ptr ptr; /* dma buffer pointers */
++ unsigned int outptr; /* RGB/YUV output */
++ emma_pp_scale dim; /* in/out dimensions */
++
++ /* pixels between two adjacent input Y rows */
++ unsigned short in_y_stride; /* 0 = in_width */
++ unsigned short in_height; /* Input image height */
++ /* PIXELS between two adjacent output rows */
++ unsigned short out_stride; /* 0 = out_width */
++} emma_pp_cfg;
++
++int pp_ptr(unsigned long ptr, struct v4l2_mxc_offset offset);
++int pp_enable(int flag);
++int pp_cfg(vout_data * vout);
++int pp_init(vout_data * vout);
++int pp_num_last(void);
++void pp_exit(vout_data * vout);
++
++#endif /* __MX27_PP_H__ */
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/output/mx27_v4l2_output.c linux-2.6.28-karo/drivers/media/video/mxc/output/mx27_v4l2_output.c
+--- linux-2.6.28/drivers/media/video/mxc/output/mx27_v4l2_output.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/output/mx27_v4l2_output.c 2009-03-11 13:48:58.000000000 +0100
+@@ -0,0 +1,1897 @@
++/*
++ * Copyright 2005-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @file mx27_v4l2_output.c
++ *
++ * @brief MX27 V4L2 Video Output Driver
++ *
++ * Video4Linux2 Output Device using MX27 eMMA Post-processing functionality.
++ *
++ * @ingroup MXC_V4L2_OUTPUT
++ */
++#include <linux/module.h>
++#include <linux/init.h>
++#include <linux/fs.h>
++#include <linux/irq.h>
++#include <linux/delay.h>
++#include <linux/fb.h>
++#include <linux/pci.h>
++#include <linux/platform_device.h>
++#include <media/v4l2-dev.h>
++#include <media/v4l2-common.h>
++#include <media/v4l2-ioctl.h>
++#include <asm/poll.h>
++#include <asm/io.h>
++//#include <asm/semaphore.h>
++#include <mach/imxfb.h>
++
++#include "mxc_v4l2_output.h"
++#include "mx27_pp.h"
++
++//#define SDC_FG_FB_FORMAT V4L2_PIX_FMT_RGB565
++#define SDC_FG_FB_FORMAT V4L2_PIX_FMT_RGB32
++
++struct v4l2_output mxc_outputs[1] = {
++ {
++ .index = 0,
++ .name = "DISP%d Video Out",
++ .type = V4L2_OUTPUT_TYPE_ANALOG, /* not really correct,
++ but no other choice */
++ .audioset = 0,
++ .modulator = 0,
++ .std = V4L2_STD_UNKNOWN,
++ },
++};
++
++#ifdef DEBUG
++int debug = 1;
++#define dbg_lvl(n) ((n) < debug)
++module_param(debug, int, S_IRUGO | S_IWUSR);
++
++#define DBG(lvl, fmt...) do { if (dbg_lvl(lvl)) printk(KERN_DEBUG fmt); } while (0)
++#else
++int debug;
++#define dbg_lvl(n) 0
++#define DBG(lvl, fmt...) do { } while (0)
++module_param(debug, int, 0);
++#endif
++#define pr_warn(fmt,arg...) printk(KERN_WARNING fmt,##arg)
++
++
++static int video_nr = 16;
++
++/* debug counters */
++static uint32_t g_irq_cnt;
++static uint32_t g_buf_output_cnt;
++static uint32_t g_buf_q_cnt;
++static uint32_t g_buf_dq_cnt;
++static uint32_t g_paused_cnt;
++static uint32_t g_busy_cnt;
++static uint32_t g_late_cnt;
++
++static int dq_intr_cnt;
++
++static int using_default_fps = -1;
++
++static inline unsigned long diff_usec(struct timeval *t1, struct timeval *t0)
++{
++ unsigned long diff = (t1->tv_sec - t0->tv_sec) * 1000000;
++
++ if (t1->tv_usec >= t0->tv_usec) {
++ diff += t1->tv_usec - t0->tv_usec;
++ } else {
++ diff += 1000000 - (t0->tv_usec - t1->tv_usec);
++ }
++ return diff;
++}
++
++#ifdef CONFIG_VIDEO_MXC_OUTPUT_FBSYNC
++
++static int fb_event_notify(struct notifier_block *self,
++ unsigned long action, void *data)
++{
++ vout_data *vout = container_of(self, vout_data, fb_event_notifier);
++ struct fb_event *event = data;
++ struct fb_info *info = event->info;
++ unsigned long flags;
++ int blank, i;
++
++ for (i = 0; i < num_registered_fb; i++)
++ if (registered_fb[i] == info)
++ break;
++
++ /*
++ * Check if the event is sent by the framebuffer in which
++ * the video is displayed.
++ */
++ if (i != vout->output_fb)
++ return 0;
++
++ switch (action) {
++ case FB_EVENT_BLANK:
++ DBG(0, "%s: BLANK\n", __FUNCTION__);
++ blank = *(int *)event->data;
++ down(&vout->user_lock);
++ vout->fb_enabled = !blank;
++ if (blank && vout->pp_ready) {
++ if (pp_enable(1))
++ pr_warn("unable to enable PP\n");
++ vout->pp_ready = 0;
++ }
++ up(&vout->user_lock);
++ break;
++ case FB_EVENT_MXC_EOF:
++ down(&vout->user_lock);
++ vout->fb_enabled = 1;
++ if (vout->pp_ready) {
++ DBG(2, "%s: enable PP\n", __FUNCTION__);
++ if (pp_enable(1))
++ pr_warn("unable to enable PP\n");
++ vout->pp_ready = 0;
++ }
++ up(&vout->user_lock);
++ break;
++ }
++
++ return 0;
++}
++#endif
++
++static inline struct v4l_queue *find_buffer(struct list_head *list, int index)
++{
++ struct v4l_queue *q;
++
++ list_for_each_entry(q, list, head) {
++ if (q->buf.index == index) {
++ return q;
++ }
++ }
++ return NULL;
++}
++
++static __inline unsigned long get_jiffies(struct timeval *t)
++{
++ struct timeval cur;
++
++ if (WARN_ON(t->tv_usec >= 1000000)) {
++ t->tv_sec += t->tv_usec / 1000000;
++ t->tv_usec = t->tv_usec % 1000000;
++ }
++
++ do_gettimeofday(&cur);
++ if ((t->tv_sec < cur.tv_sec) ||
++ ((t->tv_sec == cur.tv_sec) && (t->tv_usec < cur.tv_usec))) {
++ DBG(0, "%s: timestamp is %luµs in the past\n", __FUNCTION__,
++ (cur.tv_sec - t->tv_sec) * 1000000 + cur.tv_usec - t->tv_usec);
++ g_late_cnt++;
++ return jiffies;
++ }
++
++ if (t->tv_usec < cur.tv_usec) {
++ cur.tv_sec = t->tv_sec - cur.tv_sec - 1;
++ cur.tv_usec = t->tv_usec + 1000000 - cur.tv_usec;
++ } else {
++ cur.tv_sec = t->tv_sec - cur.tv_sec;
++ cur.tv_usec = t->tv_usec - cur.tv_usec;
++ }
++
++ return jiffies + timeval_to_jiffies(&cur);
++}
++
++/* must be called with irq_lock held */
++static void mxc_v4l2out_start_pp(vout_data *vout)
++{
++ BUG_ON(vout->active == NULL);
++ DBG(0, "%s: Updating HW PTR to %08x\n", __FUNCTION__,
++ vout->active->dma_desc.dma_addr);
++ do_gettimeofday(&vout->frame_start);
++ if (pp_ptr(vout->active->dma_desc.dma_addr, vout->offset)) {
++ pr_warn("%s: unable to update buffer\n", __FUNCTION__);
++ return;
++ }
++#ifdef CONFIG_VIDEO_MXC_OUTPUT_FBSYNC
++ if (vout->tear_protection == TEARING_PROTECTION_ACTIVE) {
++ if (vout->fb_enabled && (vout->v4l2_fb.flags != V4L2_FBUF_FLAG_OVERLAY)) {
++ vout->pp_ready = 1;
++ } else {
++ pp_enable(1);
++ }
++ } else {
++ pp_enable(1);
++ }
++#else
++ pp_enable(1);
++#endif
++ vout->busy = 1;
++}
++
++static __inline void mxc_v4l2out_schedule_frame(vout_data *vout, struct timeval *tv, int init)
++{
++ unsigned long timeout;
++
++ DBG(1, "%s: ts %6lu.%06lu\n", __FUNCTION__, tv->tv_sec, tv->tv_usec);
++ if (0 || ((tv->tv_sec == 0) && (tv->tv_usec == 0))) {
++ if (using_default_fps != 1) {
++ DBG(-1, "%s: using default fps\n", __FUNCTION__);
++ using_default_fps = 1;
++ }
++ mxc_v4l2out_start_pp(vout);
++ return;
++ } else {
++ timeout = get_jiffies(tv);
++ if (using_default_fps != 0) {
++ struct timeval now;
++ do_gettimeofday(&now);
++
++ DBG(-1, "%s: using timestamp %lu.%06lu (now: %lu.%06lu)\n", __FUNCTION__,
++ tv->tv_sec, tv->tv_usec, now.tv_sec, now.tv_usec);
++ using_default_fps = 0;
++ }
++ }
++#ifdef DEBUG
++ {
++ unsigned long cur = jiffies;
++ static unsigned long last_tv;
++ static unsigned int last_fps;
++ static unsigned int last_avg;
++ static unsigned int fps_avg;
++ static unsigned int avg_count;
++
++ if (vout->frame_count == 0) {
++ avg_count = 0;
++ }
++ if (avg_count++ == 0) {
++ last_tv = timeout;
++ fps_avg = 0;
++ last_fps = 0;
++ } else {
++ unsigned long t = timeout;
++ unsigned int fps;
++
++ DBG(3, "%s: t=%lu last=%lu diff=%ld\n", __FUNCTION__,
++ t, last_tv, t - last_tv);
++ if (t - last_tv) {
++ fps = HZ / (t - last_tv);
++ fps_avg = ((fps_avg * (avg_count - 1)) + fps) / avg_count;
++ last_tv = t;
++ if (last_fps != fps || last_avg != fps_avg) {
++ DBG(1, "%s: FPS: %u AVG %u\n", __FUNCTION__, fps, fps_avg);
++ last_fps = fps;
++ last_avg = fps_avg;
++ }
++ } else {
++ avg_count--;
++ }
++ }
++
++ DBG(1, "%s: frame %u start %lu cur %lu timeout %lu\n", __FUNCTION__,
++ vout->frame_count, vout->start_jiffies, cur, timeout);
++ }
++#endif
++ if (!time_before(jiffies, timeout)) {
++ DBG(0, "%s: timeout already expired: %lu >= %lu\n", __FUNCTION__,
++ jiffies, timeout);
++ mxc_v4l2out_start_pp(vout);
++ return;
++ }
++ if (init) {
++ vout->output_timer.expires = timeout;
++ add_timer(&vout->output_timer);
++ } else {
++ WARN_ON(mod_timer(&vout->output_timer, timeout));
++ }
++}
++
++/*!
++ * Private function to free buffers
++ *
++ * @param bufs_paddr Array of physical address of buffers to be freed
++ *
++ * @param bufs_vaddr Array of virtual address of buffers to be freed
++ *
++ * @param num_buf Number of buffers to be freed
++ *
++ * @param size Size for each buffer to be free
++ *
++ * @return status 0 success.
++ */
++static void mxc_free_buffers(vout_data *vout, struct list_head *list)
++{
++ struct v4l_queue *q;
++ struct v4l_queue *tmp;
++ struct device *dev = vout->video_dev->parent;
++ unsigned long flags;
++
++ list_for_each_entry(q, list, head) {
++ DBG(-1, "%s: dma_addr: %08x cpu_addr: %p size %08x\n", __FUNCTION__,
++ q->dma_desc.dma_addr, q->dma_desc.cpu_addr, q->dma_desc.size);
++ }
++ list_for_each_entry_safe(q, tmp, list, head) {
++ struct dma_buf_desc *dma = &q->dma_desc;
++ DBG(-1, "%s: dma_addr: %08x cpu_addr: %p size %08x\n", __FUNCTION__,
++ dma->dma_addr, dma->cpu_addr, dma->size);
++ if (dma->cpu_addr != NULL) {
++ dma_free_coherent(dev,
++ dma->size, dma->cpu_addr,
++ dma->dma_addr);
++ DBG(2, "freed @ paddr=0x%08x\n", dma->dma_addr);
++ }
++ spin_lock_irqsave(&vout->irq_lock, flags);
++ list_del(&q->head);
++ spin_unlock_irqrestore(&vout->irq_lock, flags);
++ kfree(q);
++ }
++}
++
++/*!
++ * Private function to allocate buffers
++ *
++ * @param bufs_paddr Output array of physical address of buffers allocated
++ *
++ * @param bufs_vaddr Output array of virtual address of buffers allocated
++ *
++ * @param num_buf Input number of buffers to allocate
++ *
++ * @param size Input size for each buffer to allocate
++ *
++ * @return status -0 Successfully allocated a buffer, -ENOBUFS failed.
++ */
++static int mxc_allocate_buffers(vout_data *vout, struct list_head *list, int num_buf, int size)
++{
++ int i;
++ unsigned long flags;
++ struct device *dev = vout->video_dev->parent;
++
++ DBG(2, "Trying to allocate %u buffers of %08x byte each\n",
++ num_buf, size);
++
++ for (i = 0; i < num_buf; i++) {
++ struct v4l_queue *q = kzalloc(sizeof(*q), GFP_KERNEL);
++ struct dma_buf_desc *dma;
++
++ if (q == NULL) {
++ pr_err("failed to allocate memory for queue descriptor\n");
++ return i;
++ }
++ DBG(2, "%s: queue header allocated %p\n", __FUNCTION__, q);
++ q->buf.index = i;
++ dma = &q->dma_desc;
++ if (size != 0) {
++ DBG(2, "Trying to allocate %08x byte DMA memory for queue %p\n",
++ size, list);
++ dma->size = size;
++ dma->cpu_addr = dma_alloc_coherent(dev,
++ size,
++ &dma->dma_addr,
++ GFP_KERNEL);
++ if (dma->cpu_addr == NULL) {
++ pr_err("dma_alloc_coherent failed for buffer %u out of %u\n",
++ i, num_buf);
++ kfree(q);
++ return i;
++ }
++ DBG(2, "allocated @ paddr=0x%08x, size=%d\n",
++ dma->dma_addr, size);
++ }
++ spin_lock_irqsave(&vout->irq_lock, flags);
++ list_add_tail(&q->head, list);
++ spin_unlock_irqrestore(&vout->irq_lock, flags);
++ }
++ return i;
++}
++
++static void mxc_v4l2out_timer_handler(unsigned long arg)
++{
++ unsigned long flags;
++ vout_data *vout = (vout_data *)arg;
++ struct timeval now;
++
++ do_gettimeofday(&now);
++
++ DBG(1, "timer handler: %lu\n", jiffies);
++
++ spin_lock_irqsave(&vout->irq_lock, flags);
++
++ if (vout->state == STATE_STREAM_OFF && vout->active == NULL) {
++ pr_warn("stream has stopped\n");
++ spin_unlock_irqrestore(&vout->irq_lock, flags);
++ return;
++ }
++
++ BUG_ON(vout->active == NULL);
++
++ if (vout->busy) {
++ pr_warn("Buffer %d is still busy in frame %u\n",
++ vout->active->buf.index, vout->frame_count);
++ g_busy_cnt++;
++ }
++ vout->frame_count++;
++ mxc_v4l2out_start_pp(vout);
++
++ spin_unlock_irqrestore(&vout->irq_lock, flags);
++}
++
++irqreturn_t mxc_v4l2out_pp_in_irq_handler(int irq, void *dev_id)
++{
++ unsigned long flags;
++ vout_data *vout = dev_id;
++ struct timeval now;
++
++ spin_lock_irqsave(&vout->irq_lock, flags);
++
++ do_gettimeofday(&now);
++ DBG(2, "%s: PP IRQ%d #%d\n", __FUNCTION__, irq, g_irq_cnt);
++
++ g_irq_cnt++;
++
++ if (WARN_ON(vout->state == STATE_STREAM_OFF && vout->active == NULL)) {
++ spin_unlock_irqrestore(&vout->irq_lock, flags);
++ return IRQ_HANDLED;
++ }
++#if 0
++ if (vout->v4l2_fb.flags == V4L2_FBUF_FLAG_OVERLAY) {
++ struct fb_gwinfo gwinfo;
++
++ gwinfo.enabled = 1;
++ gwinfo.alpha_value = 255;
++ gwinfo.ck_enabled = 0;
++ gwinfo.xpos = vout->crop_current.left;
++ gwinfo.ypos = vout->crop_current.top;
++ gwinfo.base = vout->display_bufs[pp_num_last()];
++ gwinfo.xres = vout->crop_current.width;
++ gwinfo.yres = vout->crop_current.height;
++ gwinfo.xres_virtual = vout->crop_current.width;
++ gwinfo.vs_reversed = 0;
++
++ mx2_gw_set(&gwinfo);
++ }
++#endif
++ /* Process previous buffer */
++ BUG_ON(vout->active == NULL);
++ BUG_ON(vout->active->buf.index >= vout->buffer_cnt);
++ DBG(3, "%s: queue entry %p g_irq_cnt %d\n", __FUNCTION__,
++ vout->active, g_irq_cnt);
++ DBG(0, "%s: frame %u finished after %lums\n", __FUNCTION__,
++ g_buf_output_cnt, diff_usec(&now, &vout->frame_start) / 1000);
++ g_buf_output_cnt++;
++ vout->active->buf.flags |= V4L2_BUF_FLAG_DONE;
++ list_add_tail(&vout->active->head, &vout->done_q);
++ DBG(2, "%s: buffer[%d] %p moved to done queue %p\n", __FUNCTION__,
++ vout->active->buf.index, vout->active, &vout->done_q);
++ vout->active = NULL;
++ vout->busy = 0;
++ wake_up_interruptible(&vout->v4l_bufq);
++
++ BUG_ON(vout->active != NULL);
++
++ if ((vout->state == STATE_STREAM_ON ||
++ vout->state == STATE_STREAM_STOPPING ||
++ vout->state == STATE_STREAM_PAUSED) &&
++ !list_empty(&vout->ready_q)) {
++ vout->active = list_first_entry(&vout->ready_q, struct v4l_queue, head);
++ list_del_init(&vout->active->head);
++ vout->queued--;
++ g_buf_dq_cnt++;
++ DBG(3, "next buffer[%d] %p\n", vout->active->buf.index, vout->active);
++ if (vout->state == STATE_STREAM_PAUSED) {
++ vout->state = STATE_STREAM_ON;
++ }
++ mxc_v4l2out_schedule_frame(vout, &vout->active->buf.timestamp, 0);
++ }
++ if (list_empty(&vout->ready_q)) {
++ if (vout->state == STATE_STREAM_STOPPING) {
++ DBG(-1, "Stream idle\n");
++ //vout->state = STATE_STREAM_OFF;
++ } else if (vout->state == STATE_STREAM_ON) {
++ DBG(0, "No more buffers ready; pausing stream; queued: %d\n", vout->queued);
++ g_paused_cnt++;
++ vout->state = STATE_STREAM_PAUSED;
++ }
++ }
++ spin_unlock_irqrestore(&vout->irq_lock, flags);
++
++ return IRQ_HANDLED;
++}
++
++/*!
++ * Start the output stream
++ *
++ * @param vout structure vout_data *
++ *
++ * @return status 0 Success
++ *
++ * called with vout->user_lock held
++ */
++static int mxc_v4l2out_streamon(vout_data *vout)
++{
++ unsigned long flags;
++ int ret;
++
++ if (vout->state != STATE_STREAM_OFF) {
++ return -EBUSY;
++ }
++ if (vout->v4l2_fb.flags == V4L2_FBUF_FLAG_OVERLAY) {
++ int index = 0;
++ struct v4l_queue *q;
++
++ /* Free previously allocated buffer */
++ mxc_free_buffers(vout, &vout->display_q);
++ /* Allocate buffers for foreground */
++ ret = mxc_allocate_buffers(vout, &vout->display_q,
++ 2, vout->display_buf_size);
++ if (ret != 2) {
++ pr_err("unable to allocate SDC FG buffers\n");
++ return ret;
++ }
++ list_for_each_entry(q, &vout->display_q, head) {
++ DBG(2, "%s: Display buffer %d @ %08x\n", __FUNCTION__,
++ index, q->dma_desc.dma_addr);
++ vout->display_bufs[index++] = q->dma_desc.dma_addr;
++ }
++ }
++
++ /* Configure PP */
++ ret = pp_cfg(vout);
++ if (ret != 0) {
++ /* Free previously allocated buffer */
++ mxc_free_buffers(vout, &vout->display_q);
++ pr_err("failed to config PP: %d\n", ret);
++ return ret;
++ }
++
++#ifdef CONFIG_VIDEO_MXC_OUTPUT_FBSYNC
++ if (vout->tear_protection == TEARING_PROTECTION_ACTIVE) {
++ vout->output_fb = vout->output_fb_num[vout->cur_disp_output];
++ vout->fb_enabled = 0;
++ vout->pp_ready = 0;
++ vout->fb_event_notifier.notifier_call = fb_event_notify;
++ fb_register_client(&vout->fb_event_notifier);
++ mx2fb_register_client(&vout->fb_event_notifier);
++ }
++#endif
++ spin_lock_irqsave(&vout->irq_lock, flags);
++ if (list_empty(&vout->ready_q)) {
++ pr_warn("no buffers queued yet!\n");
++ spin_unlock_irqrestore(&vout->irq_lock, flags);
++ return EINVAL;
++ }
++ BUG_ON(vout->active);
++ vout->active = list_first_entry(&vout->ready_q, struct v4l_queue, head);
++ list_del_init(&vout->active->head);
++ vout->queued--;
++ g_buf_dq_cnt++;
++ DBG(2, "%s: processing buffer[%d] %p from ready queue %p\n", __FUNCTION__,
++ vout->active->buf.index, vout->active, &vout->ready_q);
++
++ vout->frame_count = 0;
++ vout->state = STATE_STREAM_ON;
++ vout->start_jiffies = jiffies;
++ mxc_v4l2out_schedule_frame(vout, &vout->active->buf.timestamp, 1);
++ spin_unlock_irqrestore(&vout->irq_lock, flags);
++
++ return 0;
++}
++
++/*!
++ * Shut down the voutera
++ *
++ * @param vout structure vout_data *
++ *
++ * @return status 0 Success
++ *
++ * called with vout->user_lock held
++ */
++static int mxc_v4l2out_streamoff(vout_data *vout)
++{
++ struct v4l_queue *q;
++ struct v4l_queue *tmp;
++ unsigned long flags;
++
++ if (vout->state == STATE_STREAM_OFF) {
++ return -EINVAL;
++ }
++
++ del_timer_sync(&vout->output_timer);
++
++ pp_enable(0); /* Disable PP */
++
++ spin_lock_irqsave(&vout->irq_lock, flags);
++
++ if (vout->active != NULL) {
++#ifdef DEBUG
++ list_for_each_entry(q, &vout->free_q, head) {
++ DBG(-1, "%s: dma_addr: %08x cpu_addr: %p size %08x\n", __FUNCTION__,
++ q->dma_desc.dma_addr, q->dma_desc.cpu_addr, q->dma_desc.size);
++ }
++#endif
++ DBG(-1, "%s: Moving active buffer %p (%08x:%p) to free list\n", __FUNCTION__,
++ vout->active, vout->active->dma_desc.dma_addr,
++ vout->active->dma_desc.cpu_addr);
++ list_add_tail(&vout->active->head, &vout->free_q);
++#ifdef DEBUG
++ list_for_each_entry(q, &vout->free_q, head) {
++ DBG(-1, "%s: dma_addr: %08x cpu_addr: %p size %08x\n", __FUNCTION__,
++ q->dma_desc.dma_addr, q->dma_desc.cpu_addr, q->dma_desc.size);
++ }
++#endif
++ vout->active = NULL;
++ }
++ list_for_each_entry_safe(q, tmp, &vout->ready_q, head) {
++ DBG(-1, "%s: Moving buffer %p (%08x:%p) from ready to free list\n", __FUNCTION__,
++ q, q->dma_desc.dma_addr, q->dma_desc.cpu_addr);
++ list_move_tail(&q->head, &vout->free_q);
++ vout->queued--;
++ }
++ list_for_each_entry_safe(q, tmp, &vout->done_q, head) {
++ DBG(-1, "%s: Moving buffer %p (%08x:%p) from done to free list\n", __FUNCTION__,
++ q, q->dma_desc.dma_addr, q->dma_desc.cpu_addr);
++ list_move_tail(&q->head, &vout->free_q);
++ }
++
++ BUG_ON(vout->queued < 0);
++ if (WARN_ON(vout->queued)) {
++ DBG(-1, "queued=%d\n", vout->queued);
++ }
++ list_for_each_entry(q, &vout->free_q, head) {
++ DBG(-1, "%s: Reinitializing buffer %p (%08x:%p)\n", __FUNCTION__,
++ q, q->dma_desc.dma_addr, q->dma_desc.cpu_addr);
++ q->buf.flags = 0;
++ q->buf.timestamp.tv_sec = 0;
++ q->buf.timestamp.tv_usec = 0;
++ }
++
++ vout->state = STATE_STREAM_OFF;
++#if 0
++ if (vout->v4l2_fb.flags == V4L2_FBUF_FLAG_OVERLAY) {
++ struct fb_gwinfo gwinfo;
++
++ /* Disable graphic window */
++ gwinfo.enabled = 0;
++ mx2_gw_set(&gwinfo);
++ }
++#endif
++#ifdef CONFIG_VIDEO_MXC_OUTPUT_FBSYNC
++ if (vout->tear_protection == TEARING_PROTECTION_ACTIVE) {
++ vout->output_fb = -1;
++ vout->fb_enabled = 0;
++ vout->pp_ready = 0;
++ fb_unregister_client(&vout->fb_event_notifier);
++ mx2fb_unregister_client(&vout->fb_event_notifier);
++ }
++#endif
++ spin_unlock_irqrestore(&vout->irq_lock, flags);
++ mxc_free_buffers(vout, &vout->display_q);
++ up(&vout->user_lock);
++
++ return 0;
++}
++
++/*
++ * Valid whether the palette is supported
++ *
++ * @param palette V4L2_PIX_FMT_RGB565, V4L2_PIX_FMT_BGR24 or V4L2_PIX_FMT_BGR32
++ *
++ * @return 1 if supported, 0 if failed
++ */
++static inline int valid_mode(u32 palette)
++{
++ return palette == V4L2_PIX_FMT_YUV420 ||
++ palette == V4L2_PIX_FMT_YUV422P;
++}
++
++/*
++ * Returns bits per pixel for given pixel format
++ *
++ * @param pixelformat V4L2_PIX_FMT_RGB565, V4L2_PIX_FMT_BGR24 or V4L2_PIX_FMT_BGR32
++ *
++ * @return bits per pixel of pixelformat
++ */
++static u32 fmt_to_bpp(u32 pixelformat)
++{
++ u32 bpp;
++
++ switch (pixelformat) {
++ case V4L2_PIX_FMT_RGB565:
++ bpp = 16;
++ break;
++ case V4L2_PIX_FMT_BGR24:
++ case V4L2_PIX_FMT_RGB24:
++ bpp = 24;
++ break;
++ case V4L2_PIX_FMT_BGR32:
++ case V4L2_PIX_FMT_RGB32:
++ bpp = 32;
++ break;
++ default:
++ bpp = 8;
++ break;
++ }
++ return bpp;
++}
++
++/*
++ * V4L2 - Handles VIDIOC_G_FMT Ioctl
++ *
++ * @param vout structure vout_data *
++ *
++ * @param v4l2_format structure v4l2_format *
++ *
++ * @return status 0 success, EINVAL failed
++ */
++static int mxc_v4l2out_g_fmt(vout_data *vout, struct v4l2_format *f)
++{
++ if (f->type != V4L2_BUF_TYPE_VIDEO_OUTPUT) {
++ return -EINVAL;
++ }
++ *f = vout->v2f;
++ DBG(0, "%s: fmt=%dx%d %08x bpl=%d size=%d colorspace=%d\n", __FUNCTION__,
++ f->fmt.pix.width, f->fmt.pix.height, f->fmt.pix.pixelformat,
++ f->fmt.pix.bytesperline, f->fmt.pix.sizeimage, f->fmt.pix.colorspace);
++ return 0;
++}
++
++/*
++ * V4L2 - Handles VIDIOC_S_FMT Ioctl
++ *
++ * @param vout structure vout_data *
++ *
++ * @param v4l2_format structure v4l2_format *
++ *
++ * @return status 0 success, EINVAL failed
++ */
++static int mxc_v4l2out_s_fmt(vout_data *vout, struct v4l2_format *f)
++{
++ int retval = 0;
++ u32 size = 0;
++ u32 bytesperline;
++
++ DBG(0, "%s: fmt=%dx%d %08x bpl=%d size=%d colorspace=%d\n", __FUNCTION__,
++ f->fmt.pix.width, f->fmt.pix.height, f->fmt.pix.pixelformat,
++ f->fmt.pix.bytesperline, f->fmt.pix.sizeimage, f->fmt.pix.colorspace);
++
++ if (f->type != V4L2_BUF_TYPE_VIDEO_OUTPUT) {
++ pr_err("buffer type %d not supported\n", f->type);
++ retval = -EINVAL;
++ goto err0;
++ }
++ if (!valid_mode(f->fmt.pix.pixelformat)) {
++ pr_err("pixel format %08x not supported\n", f->fmt.pix.pixelformat);
++ retval = -EINVAL;
++ goto err0;
++ }
++
++ bytesperline = (f->fmt.pix.width * fmt_to_bpp(f->fmt.pix.pixelformat)) / 8;
++ if (f->fmt.pix.bytesperline < bytesperline) {
++ f->fmt.pix.bytesperline = bytesperline;
++ } else {
++ bytesperline = f->fmt.pix.bytesperline;
++ }
++ if (bytesperline == 0) {
++ return -EINVAL;
++ }
++
++ vout->v2f.fmt.pix.width = f->fmt.pix.width;
++ vout->v2f.fmt.pix.height = f->fmt.pix.height;
++ vout->v2f.fmt.pix.priv = f->fmt.pix.priv;
++ if (vout->v2f.fmt.pix.priv) {
++ if (copy_from_user(&vout->crop_rect,
++ (void __user *)vout->v2f.fmt.pix.priv,
++ sizeof(struct v4l2_rect))) {
++ retval = -EFAULT;
++ goto err0;
++ }
++ }
++ DBG(0, "%s: format: %dx%d-%dbpp %u bytes per line\n", __FUNCTION__,
++ f->fmt.pix.width, f->fmt.pix.height, fmt_to_bpp(f->fmt.pix.pixelformat),
++ bytesperline);
++
++ if ((vout->crop_rect.top > f->fmt.pix.height) ||
++ (vout->crop_rect.left > f->fmt.pix.width)) {
++ retval = -EINVAL;
++ goto err0;
++ }
++
++ if ((vout->crop_rect.top + vout->crop_rect.height) > f->fmt.pix.height)
++ vout->crop_rect.height =
++ f->fmt.pix.height - vout->crop_rect.top;
++ if ((vout->crop_rect.left + vout->crop_rect.width) > f->fmt.pix.width)
++ vout->crop_rect.width = f->fmt.pix.width - vout->crop_rect.left;
++
++ vout->crop_rect.width -= vout->crop_rect.width % 8;
++ vout->crop_rect.height -= vout->crop_rect.height % 8;
++ vout->crop_rect.top -= vout->crop_rect.top % 2;
++ vout->crop_rect.left += vout->crop_rect.left % 2;
++
++ switch (f->fmt.pix.pixelformat) {
++ case V4L2_PIX_FMT_YUV422P:
++ /* bytesperline for YUV planar formats is for
++ Y plane only */
++ size = bytesperline * f->fmt.pix.height * 2;
++ if ((vout->crop_rect.width != 0)
++ && (vout->crop_rect.height != 0)) {
++ vout->offset.y_offset =
++ (vout->v2f.fmt.pix.width * vout->crop_rect.top) +
++ (vout->crop_rect.left);
++ vout->offset.u_offset =
++ (vout->v2f.fmt.pix.width *
++ vout->v2f.fmt.pix.height) +
++ ((vout->v2f.fmt.pix.width *
++ vout->crop_rect.top) >> 1) +
++ (vout->crop_rect.left >> 1);
++ vout->offset.v_offset =
++ (vout->v2f.fmt.pix.width *
++ vout->v2f.fmt.pix.height) +
++ ((vout->v2f.fmt.pix.width *
++ vout->v2f.fmt.pix.height) >> 1) +
++ ((vout->v2f.fmt.pix.width *
++ vout->crop_rect.top) >> 1) +
++ (vout->crop_rect.left >> 1);
++ vout->offset.qp_offset =
++ ((vout->v2f.fmt.pix.width *
++ vout->v2f.fmt.pix.height) << 1) +
++ (((vout->v2f.fmt.pix.width * vout->crop_rect.top) +
++ vout->crop_rect.left) >> 10);
++ } else {
++ vout->offset.y_offset = 0;
++ vout->offset.u_offset =
++ (vout->v2f.fmt.pix.width *
++ vout->v2f.fmt.pix.height);
++ vout->offset.v_offset =
++ vout->offset.u_offset +
++ ((vout->v2f.fmt.pix.width *
++ vout->v2f.fmt.pix.height) >> 1);
++ vout->offset.qp_offset =
++ vout->offset.v_offset +
++ ((vout->v2f.fmt.pix.width *
++ vout->v2f.fmt.pix.height) >> 1);
++ }
++ break;
++ case V4L2_PIX_FMT_YUV420:
++ size = (bytesperline * f->fmt.pix.height * 3) / 2;
++ if ((vout->crop_rect.width != 0)
++ && (vout->crop_rect.height != 0)) {
++ vout->offset.y_offset =
++ (vout->v2f.fmt.pix.width * vout->crop_rect.top) +
++ (vout->crop_rect.left);
++ vout->offset.u_offset =
++ (vout->v2f.fmt.pix.width *
++ vout->v2f.fmt.pix.height) +
++ ((vout->v2f.fmt.pix.width *
++ vout->crop_rect.top) >> 2) +
++ (vout->crop_rect.left >> 1);
++ vout->offset.v_offset =
++ (vout->v2f.fmt.pix.width *
++ vout->v2f.fmt.pix.height) +
++ ((vout->v2f.fmt.pix.width *
++ vout->v2f.fmt.pix.height) >> 2) +
++ ((vout->v2f.fmt.pix.width *
++ vout->crop_rect.top) >> 2) +
++ (vout->crop_rect.left >> 1);
++ vout->offset.qp_offset =
++ (vout->v2f.fmt.pix.width *
++ vout->v2f.fmt.pix.height) +
++ ((vout->v2f.fmt.pix.width *
++ vout->v2f.fmt.pix.height) >> 1) +
++ (((vout->v2f.fmt.pix.width * vout->crop_rect.top) +
++ vout->crop_rect.left) >> 10);
++ } else {
++ vout->offset.y_offset = 0;
++ vout->offset.u_offset =
++ (vout->v2f.fmt.pix.width *
++ vout->v2f.fmt.pix.height);
++ vout->offset.v_offset =
++ vout->offset.u_offset +
++ ((vout->v2f.fmt.pix.width *
++ vout->v2f.fmt.pix.height) >> 2);
++ vout->offset.qp_offset =
++ vout->offset.v_offset +
++ ((vout->v2f.fmt.pix.width *
++ vout->v2f.fmt.pix.height) >> 2);
++ }
++ break;
++ default:
++ size = bytesperline * f->fmt.pix.height;
++ vout->offset.y_offset = 0;
++ vout->offset.u_offset =
++ (vout->v2f.fmt.pix.width * vout->v2f.fmt.pix.height);
++ vout->offset.v_offset =
++ vout->offset.u_offset +
++ ((vout->v2f.fmt.pix.width * vout->v2f.fmt.pix.height) >> 2);
++ vout->offset.qp_offset =
++ vout->offset.v_offset +
++ ((vout->v2f.fmt.pix.width * vout->v2f.fmt.pix.height) >> 2);
++ break;
++ }
++
++ /* Return the actual size of the image to the app */
++ f->fmt.pix.sizeimage = size;
++ vout->v2f.fmt.pix.sizeimage = size;
++ vout->v2f.fmt.pix.pixelformat = f->fmt.pix.pixelformat;
++ vout->v2f.fmt.pix.bytesperline = f->fmt.pix.bytesperline;
++
++ DBG(0, "%s: Y: %08x..%08x\n", __FUNCTION__,
++ vout->offset.y_offset, vout->offset.y_offset + f->fmt.pix.height * bytesperline);
++ DBG(0, "%s: U: %08x..%08x\n", __FUNCTION__,
++ vout->offset.u_offset, vout->offset.u_offset + f->fmt.pix.height * bytesperline / 2);
++ DBG(0, "%s: V: %08x..%08x\n", __FUNCTION__,
++ vout->offset.v_offset, vout->offset.v_offset + f->fmt.pix.height * bytesperline / 2);
++
++ retval = 0;
++ err0:
++ return retval;
++}
++
++/*
++ * V4L2 - Handles VIDIOC_G_CTRL Ioctl
++ *
++ * @param vout structure vout_data *
++ *
++ * @param c structure v4l2_control *
++ *
++ * @return status 0 success, EINVAL failed
++ */
++static int mxc_get_v4l2out_control(vout_data *vout, struct v4l2_control *c)
++{
++ switch (c->id) {
++ case V4L2_CID_HFLIP:
++ c->value = (vout->rotate & IPU_ROTATE_HORIZ_FLIP) ? 1 : 0;
++ break;
++ case V4L2_CID_VFLIP:
++ c->value = (vout->rotate & IPU_ROTATE_VERT_FLIP) ? 1 : 0;
++ break;
++ case (V4L2_CID_PRIVATE_BASE + 1):
++ c->value = vout->rotate;
++ break;
++ case V4L2_CID_MXC_TEAR_PROTECT:
++ c->value = vout->tear_protection;
++ break;
++ default:
++ return -EINVAL;
++ }
++ return 0;
++}
++
++/*
++ * V4L2 - Handles VIDIOC_S_CTRL Ioctl
++ *
++ * @param vout structure vout_data *
++ *
++ * @param c structure v4l2_control *
++ *
++ * @return status 0 success, EINVAL failed
++ */
++static int mxc_set_v4l2out_control(vout_data *vout, struct v4l2_control *c)
++{
++ switch (c->id) {
++ case V4L2_CID_HFLIP:
++ case V4L2_CID_VFLIP:
++ case V4L2_CID_MXC_ROT:
++ if (c->value < 0 || c->value > 7) {
++ return -EINVAL;
++ }
++ vout->rotate = c->value;
++ break;
++ case V4L2_CID_MXC_TEAR_PROTECT:
++#ifdef CONFIG_VIDEO_MXC_OUTPUT_FBSYNC
++ if (c->value == TEARING_PROTECTION_ACTIVE)
++ vout->tear_protection = TEARING_PROTECTION_ACTIVE;
++ else
++ vout->tear_protection = TEARING_PROTECTION_INACTIVE;;
++#else
++ vout->tear_protection = TEARING_PROTECTION_UNSUPPORTED;
++#endif
++ break;
++ default:
++ return -EINVAL;
++ }
++ return 0;
++}
++
++/*!
++ * V4L2 interface - open function
++ *
++ * @param inode structure inode *
++ *
++ * @param file structure file *
++ *
++ * @return status 0 success, ENODEV invalid device instance,
++ * ENODEV timeout, ERESTARTSYS interrupted by user
++ */
++static int mxc_v4l2out_open(struct inode *inode, struct file *file)
++{
++ struct video_device *dev = video_devdata(file);
++ vout_data *vout = video_get_drvdata(dev);
++ int err;
++
++ dq_intr_cnt = 0;
++ if (!vout) {
++ pr_info("Internal error, vout_data not found!\n");
++ return -ENODEV;
++ }
++
++ down(&vout->user_lock);
++ if (vout->open_count == 0) {
++ err = pp_init(vout);
++ if (err) {
++ goto out;
++ }
++ init_timer(&vout->output_timer);
++ vout->output_timer.function = mxc_v4l2out_timer_handler;
++ vout->output_timer.data = (unsigned long)vout;
++
++ vout->state = STATE_STREAM_OFF;
++ vout->queued = 0;
++ g_irq_cnt = g_buf_output_cnt = g_buf_q_cnt = g_buf_dq_cnt = 0;
++ g_busy_cnt = 0;
++ g_paused_cnt = 0;
++ g_late_cnt = 0;
++ using_default_fps = -1;
++ vout->open_count++;
++ }
++#ifdef CONFIG_VIDEO_MXC_OUTPUT_FBSYNC
++ vout->tear_protection = TEARING_PROTECTION_ACTIVE;
++#else
++ vout->tear_protection = TEARING_PROTECTION_UNSUPPORTED;
++#endif
++
++ file->private_data = dev;
++ out:
++ up(&vout->user_lock);
++ return err;
++}
++
++/*!
++ * V4L2 interface - close function
++ *
++ * @param inode struct inode *
++ *
++ * @param file struct file *
++ *
++ * @return 0 success
++ */
++static int mxc_v4l2out_close(struct inode *inode, struct file *file)
++{
++ struct video_device *dev = file->private_data;
++ vout_data *vout = video_get_drvdata(dev);
++
++ down(&vout->user_lock);
++ if (vout->open_count) {
++ DBG(-1, "%s: %u interrupts handled frames output: %u queued: %u dequeued: %u paused: %u busy: %u late: %u\n",
++ __FUNCTION__, g_irq_cnt, g_buf_output_cnt, g_buf_q_cnt, g_buf_dq_cnt,
++ g_paused_cnt, g_busy_cnt, g_late_cnt);
++
++ pp_exit(vout);
++ mxc_v4l2out_streamoff(vout);
++ file->private_data = NULL;
++ WARN_ON(!list_empty(&vout->done_q));
++ if (WARN_ON(vout->active)) {
++ DBG(-1, "%s: Moving active buffer %p (%08x:%p) to done list\n",
++ __FUNCTION__, vout->active, vout->active->dma_desc.dma_addr,
++ vout->active->dma_desc.cpu_addr);
++ list_add_tail(&vout->active->head, &vout->done_q);
++ vout->active = NULL;
++ }
++ WARN_ON(!list_empty(&vout->ready_q));
++ DBG(-1, "%s: Releasing buffers from ready queue\n", __FUNCTION__);
++ mxc_free_buffers(vout, &vout->ready_q);
++ DBG(-1, "%s: Releasing buffers from done queue\n", __FUNCTION__);
++ mxc_free_buffers(vout, &vout->done_q);
++ DBG(-1, "%s: Releasing buffers from free queue\n", __FUNCTION__);
++ mxc_free_buffers(vout, &vout->free_q);
++ vout->buffer_cnt = 0;
++ DBG(-1, "%s: Releasing display buffers\n", __FUNCTION__);
++ mxc_free_buffers(vout, &vout->display_q);
++
++ /* capture off */
++ wake_up_interruptible(&vout->v4l_bufq);
++ WARN_ON(--vout->open_count);
++ }
++ up(&vout->user_lock);
++
++ return 0;
++}
++
++static int do_dequeue(vout_data *vout, struct v4l2_buffer *user_buf)
++{
++ int retval;
++ unsigned long flags;
++
++ if (vout->active == NULL &&
++ list_empty(&vout->done_q) &&
++ list_empty(&vout->ready_q)) {
++ return -EINVAL;
++ }
++ up(&vout->user_lock);
++ retval = wait_event_interruptible_timeout(vout->v4l_bufq,
++ !list_empty(&vout->done_q),
++ 10 * HZ);
++ down(&vout->user_lock);
++ spin_lock_irqsave(&vout->irq_lock, flags);
++ if (retval >= 0 && !list_empty(&vout->done_q)) {
++ struct v4l_queue *q;
++ struct v4l2_buffer *buf;
++
++ q = list_first_entry(&vout->done_q, struct v4l_queue, head);
++ DBG(2, "%s: processing buffer[%d] %p from done queue %p\n",
++ __FUNCTION__, q->buf.index, q, &vout->done_q);
++ WARN_ON(!(q->buf.flags & V4L2_BUF_FLAG_DONE));
++ q->buf.flags &= ~(V4L2_BUF_FLAG_DONE | V4L2_BUF_FLAG_QUEUED);
++ list_move_tail(&q->head, &vout->free_q);
++ buf = &q->buf;
++ DBG(2, "%s: buffer[%d] %p moved to free queue %p\n", __FUNCTION__,
++ buf->index, q, &vout->free_q);
++ BUG_ON(buf->index >= vout->buffer_cnt);
++ if (user_buf != NULL) {
++ memcpy(user_buf, buf, sizeof(*user_buf));
++ }
++ DBG(1, "VIDIOC_DQBUF: buffer %d: %d\n", buf->index, retval);
++ retval = 0;
++ } else if (retval == 0) {
++ pr_warn("VIDIOC_DQBUF: timeout\n");
++ retval = -ETIME;
++ DBG(3, "VIDIOC_DQBUF: ret: %d\n", retval);
++ }
++ spin_unlock_irqrestore(&vout->irq_lock, flags);
++ return retval;
++}
++/*!
++ * V4L2 interface - ioctl function
++ *
++ * @param inode struct inode *
++ *
++ * @param file struct file *
++ *
++ * @param ioctlnr unsigned int
++ *
++ * @param arg void *
++ *
++ * @return 0 success, ENODEV for invalid device instance,
++ * -1 for other errors.
++ */
++static int
++mxc_v4l2out_do_ioctl(struct inode *inode, struct file *file,
++ unsigned int ioctlnr, void *arg)
++{
++ struct video_device *vdev = file->private_data;
++ vout_data *vout = video_get_drvdata(vdev);
++ int retval = 0;
++ int i = 0;
++
++ if (dbg_lvl(3)) {
++ v4l_printk_ioctl(ioctlnr);
++ printk(" arg=%p\n", arg);
++ }
++
++ if (!vout)
++ return -EBADF;
++
++ down(&vout->user_lock);
++ switch (ioctlnr) {
++ case VIDIOC_QUERYCAP:
++ {
++ struct v4l2_capability *cap = arg;
++ strcpy(cap->driver, "mxc_v4l2_output");
++ cap->version = 0;
++ cap->capabilities = V4L2_CAP_VIDEO_OUTPUT |
++ V4L2_CAP_STREAMING;
++ strlcpy(cap->card, "MX27 eMMA", sizeof(cap->card));
++ cap->bus_info[0] = '\0';
++ retval = 0;
++ }
++ break;
++ case VIDIOC_G_FMT:
++ {
++ struct v4l2_format *gf = arg;
++ retval = mxc_v4l2out_g_fmt(vout, gf);
++ }
++ break;
++ case VIDIOC_S_FMT:
++ {
++ struct v4l2_format *sf = arg;
++ if (vout->state != STATE_STREAM_OFF) {
++ retval = -EBUSY;
++ break;
++ }
++ retval = mxc_v4l2out_s_fmt(vout, sf);
++ }
++ break;
++ case VIDIOC_REQBUFS:
++ {
++ struct v4l2_requestbuffers *req = arg;
++ struct v4l_queue *q;
++ size_t bufsize = PAGE_ALIGN(vout->v2f.fmt.pix.sizeimage);
++ unsigned long flags;
++
++ if ((req->type != V4L2_BUF_TYPE_VIDEO_OUTPUT) ||
++ (req->memory != V4L2_MEMORY_MMAP)) {
++ pr_warn("VIDIOC_REQBUFS: incorrect buffer type: %d\n", req->type);
++ retval = -EINVAL;
++ break;
++ }
++
++ DBG(0, "%s: VIDIOC_REQBUFS: %d buffers of %u byte requested\n",
++ __FUNCTION__, req->count, bufsize);
++
++ if (req->count == 0) {
++ spin_lock_irqsave(&vout->irq_lock, flags);
++ if (vout->state == STATE_STREAM_ON) {
++ vout->state = STATE_STREAM_STOPPING;
++ }
++ spin_unlock_irqrestore(&vout->irq_lock, flags);
++ while (vout->active || !list_empty(&vout->ready_q)) {
++ DBG(-1, "%s: Waiting for stream to drain queued: %d\n",
++ __FUNCTION__, vout->queued);
++ retval = do_dequeue(vout, NULL);
++ if (retval != 0) {
++ DBG(-1, "do_dequeue() returned %d\n", retval);
++ }
++ if (retval == -ERESTARTSYS) {
++ break;
++ }
++ }
++#if 1
++ retval = mxc_v4l2out_streamoff(vout);
++#else
++ pp_enable(0); /* Disable PP */
++ WARN_ON(!list_empty(&vout->ready_q));
++ WARN_ON(vout->active);
++ mxc_free_buffers(vout, &vout->done_q);
++ mxc_free_buffers(vout, &vout->free_q);
++#endif
++ break;
++ }
++ if (vout->state != STATE_STREAM_OFF) {
++ pr_warn("VIDIOC_REQBUFS: streaming active\n");
++ retval = -EBUSY;
++ break;
++ }
++#if 0
++ if (!list_empty(&vout->done_q) || !list_empty(&vout->ready_q)) {
++ pr_warn("VIDIOC_REQBUFS: busy buffers exist\n");
++ retval = -EBUSY;
++ break;
++ }
++#endif
++ /* This needs no interrupt locking, since we have the busy_lock
++ * and are in STATE_STREAM_OFF where there cannot be any interrupts
++ */
++ if (!list_empty(&vout->free_q)) {
++ q = list_first_entry(&vout->free_q, struct v4l_queue, head);
++ if (q->buf.length != bufsize ||
++ q->buf.type != req->type ||
++ q->buf.memory != req->memory) {
++ pr_err("VIDIOC_REQBUFS: Mismatch between old and new buffer parameters\n");
++ retval = -EINVAL;
++ break;
++ }
++ }
++
++ if (req->count > MAX_FRAME_NUM) {
++ req->count = MAX_FRAME_NUM;
++ }
++ if (req->count == vout->buffer_cnt) {
++ break;
++ } else if (req->count < vout->buffer_cnt) {
++ struct list_head release_q;
++ struct list_head *tmp;
++ struct list_head *x;
++
++ list_for_each_prev_safe(x, tmp, &vout->free_q) {
++ if (vout->buffer_cnt-- > req->count) {
++ q = container_of(x, struct v4l_queue, head);
++ DBG(-1, "%s: Moving buffer[%d:%d] %p (%08x:%p) from free to release list\n",
++ __FUNCTION__, vout->buffer_cnt,
++ q->buf.index, q, q->dma_desc.dma_addr,
++ q->dma_desc.cpu_addr);
++ list_move(x, &release_q);
++ }
++ }
++ BUG_ON(vout->buffer_cnt != req->count);
++ mxc_free_buffers(vout, &release_q);
++ break;
++ }
++ retval = mxc_allocate_buffers(vout, &vout->free_q,
++ req->count - vout->buffer_cnt,
++ bufsize);
++ if (retval + vout->buffer_cnt < req->count) {
++ pr_warn("Could only allocate %u of %u buffers\n",
++ retval + vout->buffer_cnt, req->count);
++ req->count = retval + vout->buffer_cnt;
++ }
++ vout->buffer_cnt = req->count;
++ i = 0;
++ list_for_each_entry(q, &vout->free_q, head) {
++ q->buf.memory = V4L2_MEMORY_MMAP;
++ q->buf.index = i;
++ q->buf.type = V4L2_BUF_TYPE_VIDEO_OUTPUT;
++ q->buf.length = bufsize;
++ q->buf.m.offset = q->dma_desc.dma_addr;
++ i++;
++ }
++ BUG_ON(i != vout->buffer_cnt);
++ }
++ break;
++ case VIDIOC_QUERYBUF:
++ {
++ struct v4l2_buffer *buf = arg;
++ struct v4l_queue *q;
++ unsigned long flags;
++
++ if (buf->type != V4L2_BUF_TYPE_VIDEO_OUTPUT) {
++ pr_warn("VIDIOC_QUERYBUF: incorrect buffer type: %d expected %d\n",
++ buf->type, V4L2_BUF_TYPE_VIDEO_OUTPUT);
++ retval = -EINVAL;
++ break;
++ }
++ if (buf->index >= vout->buffer_cnt) {
++ pr_warn("VIDIOC_QUERYBUF: buffer index out of range: %d [0..%d]\n",
++ buf->index, vout->buffer_cnt);
++ retval = -EINVAL;
++ break;
++ }
++ spin_lock_irqsave(&vout->irq_lock, flags);
++ q = find_buffer(&vout->free_q, buf->index);
++ if (q == NULL) {
++ q = find_buffer(&vout->ready_q, buf->index);
++ if (q == NULL) {
++ q = find_buffer(&vout->done_q, buf->index);
++ if (q != NULL) {
++ DBG(2, "VIDIOC_QUERYBUF: buffer[%d] %p found in done list\n",
++ q->buf.index, q);
++ }
++ } else {
++ DBG(2, "VIDIOC_QUERYBUF: buffer[%d] %p found in ready list\n",
++ q->buf.index, q);
++ }
++ } else {
++ DBG(2, "VIDIOC_QUERYBUF: buffer[%d] %p found in free list\n",
++ q->buf.index, q);
++ }
++ spin_unlock_irqrestore(&vout->irq_lock, flags);
++ if (q == NULL) {
++ pr_warn("VIDIOC_QUERYBUF: buffer %d not found in any list\n",
++ buf->index);
++ retval = -ENOENT;
++ break;
++ }
++ memcpy(buf, &q->buf, sizeof(q->buf));
++ }
++ break;
++ case VIDIOC_QBUF:
++ {
++ struct v4l2_buffer *buf = arg;
++ struct v4l_queue *q;
++ unsigned long flags;
++
++ if (buf->type != V4L2_BUF_TYPE_VIDEO_OUTPUT) {
++ pr_err("VIDIOC_QBUF: invalid buffer type: %u\n",
++ buf->type);
++ retval = -EINVAL;
++ break;
++ }
++
++ if (buf->index >= vout->buffer_cnt) {
++ pr_err("VIDIOC_QBUF: invalid buffer index: %u/%u\n",
++ buf->index, vout->buffer_cnt);
++ retval = -EINVAL;
++ break;
++ }
++
++ if (vout->state == STATE_STREAM_STOPPING) {
++ pr_err("VIDIOC_QBUF: stream is draining\n");
++ retval = -EINVAL;
++ break;
++ }
++ DBG(1, "VIDIOC_QBUF: %d/%d ts %6lu.%06lu\n", buf->index,
++ vout->queued, buf->timestamp.tv_sec, buf->timestamp.tv_usec);
++
++ spin_lock_irqsave(&vout->irq_lock, flags);
++ q = find_buffer(&vout->free_q, buf->index);
++ if (q == NULL || (q->buf.flags & V4L2_BUF_FLAG_QUEUED)) {
++ spin_unlock_irqrestore(&vout->irq_lock, flags);
++ if (q == NULL) {
++ printk(KERN_ERR "buffer %d not on free queue %p\n",
++ buf->index, &vout->free_q);
++ retval = -ENOENT;
++ } else {
++ retval = -EBUSY;
++ }
++ break;
++ }
++ q->buf.timestamp = buf->timestamp;
++ buf = &q->buf;
++ WARN_ON(buf->flags & V4L2_BUF_FLAG_DONE);
++ buf->flags |= V4L2_BUF_FLAG_QUEUED;
++
++ BUG_ON(buf->index >= vout->buffer_cnt);
++
++ if (vout->state == STATE_STREAM_PAUSED &&
++ vout->active == NULL) {
++ DBG(0, "%s: Restarting stream\n", __FUNCTION__);
++ vout->active = q;
++ list_del_init(&vout->active->head);
++ vout->state = STATE_STREAM_ON;
++
++ mxc_v4l2out_schedule_frame(vout, &buf->timestamp, 0);
++ } else {
++ list_move_tail(&q->head, &vout->ready_q);
++ vout->queued++;
++ g_buf_q_cnt++;
++ DBG(2, "%s: buffer[%d] %p moved to ready queue %p\n", __FUNCTION__,
++ buf->index, q, &vout->ready_q);
++ }
++ if (vout->state == STATE_STREAM_PAUSED) {
++ vout->state = STATE_STREAM_ON;
++ }
++ spin_unlock_irqrestore(&vout->irq_lock, flags);
++ }
++ break;
++ case VIDIOC_DQBUF:
++ {
++ struct v4l2_buffer *buf = arg;
++
++ DBG(1, "VIDIOC_DQBUF: queued: %d\n", vout->queued);
++
++ if (list_empty(&vout->done_q) &&
++ (file->f_flags & O_NONBLOCK)) {
++ retval = -EAGAIN;
++ DBG(3, "VIDIOC_DQBUF: ret: %d\n", retval);
++ break;
++ }
++
++ retval = do_dequeue(vout, buf);
++ if (retval == -ETIME) {
++ pr_warn("VIDIOC_DQBUF: timeout\n");
++ DBG(3, "VIDIOC_DQBUF: ret: %d\n", retval);
++ } else if (retval == -ERESTARTSYS) {
++ if (dq_intr_cnt == 0)
++ DBG(0, "VIDIOC_DQBUF: interrupt received\n");
++ dq_intr_cnt++;
++ DBG(0, "VIDIOC_DQBUF: ret: %d\n", retval);
++ } else {
++ DBG(0, "VIDIOC_DQBUF: ret: %d\n", retval);
++ }
++ }
++ break;
++ case VIDIOC_STREAMON:
++ retval = mxc_v4l2out_streamon(vout);
++ break;
++ case VIDIOC_STREAMOFF:
++ retval = mxc_v4l2out_streamoff(vout);
++ break;
++ case VIDIOC_G_CTRL:
++ retval = mxc_get_v4l2out_control(vout, arg);
++ break;
++ case VIDIOC_S_CTRL:
++ retval = mxc_set_v4l2out_control(vout, arg);
++ break;
++ case VIDIOC_CROPCAP:
++ {
++ struct v4l2_cropcap *cap = arg;
++
++ if (cap->type != V4L2_BUF_TYPE_VIDEO_OUTPUT) {
++ retval = -EINVAL;
++ break;
++ }
++ cap->bounds = vout->crop_bounds[vout->cur_disp_output];
++ cap->defrect = vout->crop_bounds[vout->cur_disp_output];
++ retval = 0;
++ }
++ break;
++ case VIDIOC_G_CROP:
++ {
++ struct v4l2_crop *crop = arg;
++
++ if (crop->type != V4L2_BUF_TYPE_VIDEO_OUTPUT) {
++ retval = -EINVAL;
++ break;
++ }
++ crop->c = vout->crop_current;
++ }
++ break;
++ case VIDIOC_S_CROP:
++ {
++ struct v4l2_crop *crop = arg;
++ struct v4l2_rect *b = &vout->crop_bounds[vout->cur_disp_output];
++
++ if (crop->type != V4L2_BUF_TYPE_VIDEO_OUTPUT) {
++ retval = -EINVAL;
++ break;
++ }
++ if (crop->c.height < 0) {
++ retval = -EINVAL;
++ break;
++ }
++ if (crop->c.width < 0) {
++ retval = -EINVAL;
++ break;
++ }
++
++ if (crop->c.top < b->top)
++ crop->c.top = b->top;
++ if (crop->c.top > b->top + b->height)
++ crop->c.top = b->top + b->height;
++ if (crop->c.height > b->top - crop->c.top + b->height)
++ crop->c.height =
++ b->top - crop->c.top + b->height;
++
++ if (crop->c.left < b->left)
++ crop->c.top = b->left;
++ if (crop->c.left > b->left + b->width)
++ crop->c.top = b->left + b->width;
++ if (crop->c.width > b->left - crop->c.left + b->width)
++ crop->c.width =
++ b->left - crop->c.left + b->width;
++
++ /* stride line limitation */
++ crop->c.height -= crop->c.height % 8;
++ crop->c.width -= crop->c.width % 8;
++
++ vout->crop_current = crop->c;
++ DBG(0, "%s: crop rect: %dx%d%+d%+d\n", __FUNCTION__,
++ crop->c.width, crop->c.height,
++ crop->c.left, crop->c.top);
++
++ vout->display_buf_size = vout->crop_current.width *
++ vout->crop_current.height;
++ vout->display_buf_size *=
++ fmt_to_bpp(SDC_FG_FB_FORMAT) / 8;
++ }
++ break;
++ case VIDIOC_ENUMOUTPUT:
++ {
++ struct v4l2_output *output = arg;
++
++ if ((output->index >= 2) || !vout->output_enabled[output->index]) {
++ retval = -EINVAL;
++ break;
++ }
++ memcpy(output, &mxc_outputs[0], sizeof(*output));
++ snprintf(output->name, sizeof(output->name), mxc_outputs[0].name,
++ output->index);
++ }
++ break;
++ case VIDIOC_G_OUTPUT:
++ {
++ u32 *index = arg;
++
++ *index = vout->cur_disp_output;
++ }
++ break;
++ case VIDIOC_S_OUTPUT:
++ {
++ u32 *index = arg;
++
++ if ((*index >= 2) || !vout->output_enabled[*index]) {
++ retval = -EINVAL;
++ break;
++ }
++
++ if (vout->state != STATE_STREAM_OFF) {
++ retval = -EBUSY;
++ break;
++ }
++
++ vout->cur_disp_output = *index;
++ }
++ break;
++ case VIDIOC_G_FBUF:
++ {
++ struct v4l2_framebuffer *fb = arg;
++
++ memcpy(fb, &vout->v4l2_fb, sizeof(*fb));
++ }
++ break;
++ case VIDIOC_S_FBUF:
++ {
++ struct v4l2_framebuffer *fb = arg;
++
++ memcpy(&vout->v4l2_fb, fb, sizeof(vout->v4l2_fb));
++ vout->v4l2_fb.capability = V4L2_FBUF_CAP_EXTERNOVERLAY;
++ }
++ break;
++ case VIDIOC_ENUM_FMT:
++ case VIDIOC_TRY_FMT:
++ case VIDIOC_QUERYCTRL:
++ case VIDIOC_G_PARM:
++ case VIDIOC_ENUMSTD:
++ case VIDIOC_G_STD:
++ case VIDIOC_S_STD:
++ case VIDIOC_G_TUNER:
++ case VIDIOC_S_TUNER:
++ case VIDIOC_G_FREQUENCY:
++ case VIDIOC_S_FREQUENCY:
++ default:
++ retval = -EINVAL;
++ }
++
++ up(&vout->user_lock);
++ return retval;
++}
++
++/*
++ * V4L2 interface - ioctl function
++ *
++ * @return None
++ */
++static int
++mxc_v4l2out_ioctl(struct inode *inode, struct file *file,
++ unsigned int cmd, unsigned long arg)
++{
++ return video_usercopy(inode, file, cmd, arg, mxc_v4l2out_do_ioctl);
++}
++
++/*!
++ * V4L2 interface - mmap function
++ *
++ * @param file structure file *
++ *
++ * @param vma structure vm_area_struct *
++ *
++ * @return status 0 Success, EINTR busy lock error,
++ * ENOBUFS remap_page error
++ */
++static int mxc_v4l2out_mmap(struct file *file, struct vm_area_struct *vma)
++{
++ struct video_device *vdev = file->private_data;
++ unsigned long start = vma->vm_start;
++ unsigned long size = vma->vm_end - vma->vm_start;
++ int res = 0;
++ vout_data *vout = video_get_drvdata(vdev);
++
++ down(&vout->user_lock);
++
++ /* make buffers write-thru cacheable */
++ vma->vm_page_prot = __pgprot(pgprot_val(vma->vm_page_prot) &
++ ~L_PTE_BUFFERABLE);
++
++ if (remap_pfn_range(vma, start, vma->vm_pgoff, size, vma->vm_page_prot)) {
++ pr_err("%s: - remap_pfn_range failed\n", __FUNCTION__);
++ res = -ENOBUFS;
++ }
++
++ up(&vout->user_lock);
++ return res;
++}
++
++/*!
++ * V4L2 interface - poll function
++ *
++ * @param file structure file *
++ *
++ * @param wait structure poll_table *
++ *
++ * @return status POLLIN | POLLRDNORM
++ */
++static unsigned int mxc_v4l2out_poll(struct file *file, poll_table *wait)
++{
++ struct video_device *vdev = file->private_data;
++ vout_data *vout = video_get_drvdata(vdev);
++
++ wait_queue_head_t *queue = NULL;
++ int res = POLLIN | POLLRDNORM;
++
++ down(&vout->user_lock);
++
++ queue = &vout->v4l_bufq;
++ poll_wait(file, queue, wait);
++
++ up(&vout->user_lock);
++ return res;
++}
++
++static struct file_operations mxc_v4l2out_fops = {
++ .owner = THIS_MODULE,
++ .open = mxc_v4l2out_open,
++ .release = mxc_v4l2out_close,
++ .ioctl = mxc_v4l2out_ioctl,
++ .mmap = mxc_v4l2out_mmap,
++ .poll = mxc_v4l2out_poll,
++};
++
++static struct video_device mxc_v4l2out_template = {
++ //.owner = THIS_MODULE,
++ .name = "MXC Video Output",
++ //.type = 0,
++ //.type2 = V4L2_CAP_VIDEO_OUTPUT | V4L2_CAP_STREAMING,
++ //.hardware = 39,
++ .fops = &mxc_v4l2out_fops,
++ .release = video_device_release,
++};
++
++/*!
++ * Probe routine for the framebuffer driver. It is called during the
++ * driver binding process. The following functions are performed in
++ * this routine: Framebuffer initialization, Memory allocation and
++ * mapping, Framebuffer registration, IPU initialization.
++ *
++ * @return Appropriate error code to the kernel common code
++ */
++static int mxc_v4l2out_probe(struct platform_device *pdev)
++{
++ int ret;
++ int i;
++ vout_data *vout;
++
++ /*
++ * Allocate sufficient memory for the fb structure
++ */
++ vout = kzalloc(sizeof(vout_data), GFP_KERNEL);
++ if (!vout)
++ return -ENOMEM;
++
++ vout->video_dev = video_device_alloc();
++ if (vout->video_dev == NULL) {
++ kfree(vout);
++ return -ENOMEM;
++ }
++ *vout->video_dev = mxc_v4l2out_template;
++
++ vout->video_dev->parent = &pdev->dev;
++ vout->video_dev->minor = -1;
++
++ init_waitqueue_head(&vout->v4l_bufq);
++
++ INIT_LIST_HEAD(&vout->free_q);
++ INIT_LIST_HEAD(&vout->ready_q);
++ INIT_LIST_HEAD(&vout->done_q);
++
++ INIT_LIST_HEAD(&vout->display_q);
++
++ spin_lock_init(&vout->irq_lock);
++
++ init_MUTEX(&vout->user_lock);
++ //mutex_init(&vout->user_lock);
++
++ /* register v4l device */
++ ret = video_register_device(vout->video_dev,
++ VFL_TYPE_GRABBER, video_nr);
++ if (ret != 0) {
++ pr_err("video_register_device failed: %d\n", ret);
++ kfree(vout->video_dev);
++ kfree(vout);
++ return ret;
++ }
++ DBG(0, "mxc_v4l2out: registered device video%d\n",
++ vout->video_dev->minor & 0x1f);
++
++ video_set_drvdata(vout->video_dev, vout);
++ platform_set_drvdata(pdev, vout);
++
++ /* setup outputs and cropping */
++ vout->cur_disp_output = -1;
++#ifdef CONFIG_VIDEO_MXC_OUTPUT_FBSYNC
++ vout->output_fb = -1;
++#endif
++ for (i = 0; i < num_registered_fb && i < MXC_V4L2_OUT_NUM_OUTPUTS; i++) {
++ char *idstr = registered_fb[i]->fix.id;
++ DBG(0, "Checking FB '%s'\n", idstr);
++ if (strncmp(idstr, "IMX", 3) == 0) {
++ int disp_num = i;
++ vout->crop_bounds[disp_num].left = 0;
++ vout->crop_bounds[disp_num].top = 0;
++ vout->crop_bounds[disp_num].width =
++ registered_fb[i]->var.xres;
++ vout->crop_bounds[disp_num].height =
++ registered_fb[i]->var.yres;
++ vout->output_enabled[disp_num] = true;
++ vout->output_fb_num[disp_num] = i;
++ DBG(0, "crop bounds: %dx%d%+d%+d\n",
++ vout->crop_bounds[disp_num].left,
++ vout->crop_bounds[disp_num].top,
++ vout->crop_bounds[disp_num].width,
++ vout->crop_bounds[disp_num].height);
++ if (vout->cur_disp_output == -1)
++ vout->cur_disp_output = disp_num;
++ }
++
++ }
++ if (vout->cur_disp_output >= 0) {
++ vout->crop_current = vout->crop_bounds[vout->cur_disp_output];
++ }
++
++ /* Setup framebuffer parameters */
++ vout->v4l2_fb.capability = V4L2_FBUF_CAP_EXTERNOVERLAY;
++ vout->v4l2_fb.flags = V4L2_FBUF_FLAG_PRIMARY;
++ // FIXME: Use the pixelformat of the FB driver!
++ vout->v4l2_fb.fmt.pixelformat = V4L2_PIX_FMT_RGB565;
++
++ return 0;
++}
++
++static int mxc_v4l2out_remove(struct platform_device *pdev)
++{
++ vout_data *vout = platform_get_drvdata(pdev);
++
++ video_unregister_device(vout->video_dev);
++ kfree(vout);
++ return 0;
++}
++
++#define DRV_NAME "MXC Video Output"
++/*!
++ * This structure contains pointers to the power management callback functions.
++ */
++static struct platform_driver mxc_v4l2out_driver = {
++ .driver = {
++ .name = DRV_NAME,
++ .owner = THIS_MODULE,
++ },
++ .probe = mxc_v4l2out_probe,
++ .remove = mxc_v4l2out_remove,
++};
++
++#ifdef REGISTER_PDEV
++#if 1
++static struct platform_device *mxc_v4l2out_device;
++#else
++static struct platform_device mxc_v4l2out_device = {
++ .name = "MXC Video Output",
++ .id = 0,
++};
++#endif
++#endif
++
++/*!
++ * mxc v4l2 init function
++ *
++ */
++static int mxc_v4l2out_init(void)
++{
++ int err;
++
++ err = platform_driver_register(&mxc_v4l2out_driver);
++#ifdef REGISTER_PDEV
++ if (err == 0) {
++#if 1
++ mxc_v4l2out_device = platform_device_register_simple(DRV_NAME, 0, NULL, 0);
++ if (IS_ERR(mxc_v4l2out_device)) {
++ platform_driver_unregister(&mxc_v4l2out_driver);
++ return PTR_ERR(mxc_v4l2out_device);
++ }
++#else
++ platform_device_register(&mxc_v4l2out_device);
++#endif
++ }
++#endif
++ return err;
++}
++
++/*!
++ * mxc v4l2 cleanup function
++ *
++ */
++static void mxc_v4l2out_clean(void)
++{
++ DBG(0, "unregistering video\n");
++
++ platform_driver_unregister(&mxc_v4l2out_driver);
++#ifdef REGISTER_PDEV
++#if 1
++ platform_device_unregister(mxc_v4l2out_device);
++#else
++ platform_device_unregister(&mxc_v4l2out_device);
++#endif
++#endif
++}
++
++module_init(mxc_v4l2out_init);
++module_exit(mxc_v4l2out_clean);
++
++module_param(video_nr, int, S_IRUGO);
++
++MODULE_AUTHOR("Freescale Semiconductor, Inc.");
++MODULE_DESCRIPTION("V4L2-driver for MXC video output");
++MODULE_LICENSE("GPL");
++MODULE_SUPPORTED_DEVICE("video");
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/output/mxc_v4l2_output.c linux-2.6.28-karo/drivers/media/video/mxc/output/mxc_v4l2_output.c
+--- linux-2.6.28/drivers/media/video/mxc/output/mxc_v4l2_output.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/output/mxc_v4l2_output.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,1720 @@
++/*
++ * Copyright 2005-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @file drivers/media/video/mxc/output/mxc_v4l2_output.c
++ *
++ * @brief MXC V4L2 Video Output Driver
++ *
++ * Video4Linux2 Output Device using MXC IPU Post-processing functionality.
++ *
++ * @ingroup MXC_V4L2_OUTPUT
++ */
++#include <linux/module.h>
++#include <linux/init.h>
++#include <linux/console.h>
++#include <linux/fs.h>
++#include <linux/delay.h>
++#include <linux/platform_device.h>
++#include <asm/cacheflush.h>
++#include <asm/io.h>
++#include <asm/semaphore.h>
++#include <linux/dma-mapping.h>
++
++#include <asm/arch/mxcfb.h>
++#include <mach/ipu.h>
++
++#include "mxc_v4l2_output.h"
++
++vout_data *g_vout;
++#define SDC_FG_FB_FORMAT IPU_PIX_FMT_RGB565
++
++struct v4l2_output mxc_outputs[2] = {
++ {
++ .index = MXC_V4L2_OUT_2_SDC,
++ .name = "DISP3 Video Out",
++ .type = V4L2_OUTPUT_TYPE_ANALOG, /* not really correct,
++ but no other choice */
++ .audioset = 0,
++ .modulator = 0,
++ .std = V4L2_STD_UNKNOWN},
++ {
++ .index = MXC_V4L2_OUT_2_ADC,
++ .name = "DISPx Video Out",
++ .type = V4L2_OUTPUT_TYPE_ANALOG, /* not really correct,
++ but no other choice */
++ .audioset = 0,
++ .modulator = 0,
++ .std = V4L2_STD_UNKNOWN}
++};
++
++static int video_nr = 16;
++static spinlock_t g_lock = SPIN_LOCK_UNLOCKED;
++
++/* debug counters */
++uint32_t g_irq_cnt;
++uint32_t g_buf_output_cnt;
++uint32_t g_buf_q_cnt;
++uint32_t g_buf_dq_cnt;
++
++static int dq_intr_cnt=0;
++static int dq_timeout_cnt=0;
++
++#define QUEUE_SIZE (MAX_FRAME_NUM + 1)
++static __inline int queue_size(v4l_queue * q)
++{
++ if (q->tail >= q->head)
++ return (q->tail - q->head);
++ else
++ return ((q->tail + QUEUE_SIZE) - q->head);
++}
++
++static __inline int queue_buf(v4l_queue * q, int idx)
++{
++ if (((q->tail + 1) % QUEUE_SIZE) == q->head)
++ return -1; /* queue full */
++ q->list[q->tail] = idx;
++ q->tail = (q->tail + 1) % QUEUE_SIZE;
++ return 0;
++}
++
++static __inline int dequeue_buf(v4l_queue * q)
++{
++ int ret;
++ if (q->tail == q->head)
++ return -1; /* queue empty */
++ ret = q->list[q->head];
++ q->head = (q->head + 1) % QUEUE_SIZE;
++ return ret;
++}
++
++static __inline int peek_next_buf(v4l_queue * q)
++{
++ if (q->tail == q->head)
++ return -1; /* queue empty */
++ return q->list[q->head];
++}
++
++static __inline unsigned long get_jiffies(struct timeval *t)
++{
++ struct timeval cur;
++
++ if (t->tv_usec >= 1000000) {
++ t->tv_sec += t->tv_usec / 1000000;
++ t->tv_usec = t->tv_usec % 1000000;
++ }
++
++ do_gettimeofday(&cur);
++ if ((t->tv_sec < cur.tv_sec)
++ || ((t->tv_sec == cur.tv_sec) && (t->tv_usec < cur.tv_usec)))
++ return jiffies;
++
++ if (t->tv_usec < cur.tv_usec) {
++ cur.tv_sec = t->tv_sec - cur.tv_sec - 1;
++ cur.tv_usec = t->tv_usec + 1000000 - cur.tv_usec;
++ } else {
++ cur.tv_sec = t->tv_sec - cur.tv_sec;
++ cur.tv_usec = t->tv_usec - cur.tv_usec;
++ }
++
++ return jiffies + timeval_to_jiffies(&cur);
++}
++
++/*!
++ * Private function to free buffers
++ *
++ * @param bufs_paddr Array of physical address of buffers to be freed
++ *
++ * @param bufs_vaddr Array of virtual address of buffers to be freed
++ *
++ * @param num_buf Number of buffers to be freed
++ *
++ * @param size Size for each buffer to be free
++ *
++ * @return status 0 success.
++ */
++static int mxc_free_buffers(dma_addr_t bufs_paddr[], void *bufs_vaddr[],
++ int num_buf, int size)
++{
++ int i;
++
++ for (i = 0; i < num_buf; i++) {
++ if (bufs_vaddr[i] != 0) {
++ dma_free_coherent(0, size, bufs_vaddr[i],
++ bufs_paddr[i]);
++ pr_debug("freed @ paddr=0x%08X\n", (u32) bufs_paddr[i]);
++ bufs_paddr[i] = 0;
++ bufs_vaddr[i] = NULL;
++ }
++ }
++ return 0;
++}
++
++/*!
++ * Private function to allocate buffers
++ *
++ * @param bufs_paddr Output array of physical address of buffers allocated
++ *
++ * @param bufs_vaddr Output array of virtual address of buffers allocated
++ *
++ * @param num_buf Input number of buffers to allocate
++ *
++ * @param size Input size for each buffer to allocate
++ *
++ * @return status -0 Successfully allocated a buffer, -ENOBUFS failed.
++ */
++static int mxc_allocate_buffers(dma_addr_t bufs_paddr[], void *bufs_vaddr[],
++ int num_buf, int size)
++{
++ int i;
++
++ for (i = 0; i < num_buf; i++) {
++ bufs_vaddr[i] = dma_alloc_coherent(0, size,
++ &bufs_paddr[i],
++ GFP_DMA | GFP_KERNEL);
++
++ if (bufs_vaddr[i] == 0) {
++ mxc_free_buffers(bufs_paddr, bufs_vaddr, i, size);
++ printk(KERN_ERR "dma_alloc_coherent failed.\n");
++ return -ENOBUFS;
++ }
++ pr_debug("allocated @ paddr=0x%08X, size=%d.\n",
++ (u32) bufs_paddr[i], size);
++ }
++
++ return 0;
++}
++
++/*
++ * Returns bits per pixel for given pixel format
++ *
++ * @param pixelformat V4L2_PIX_FMT_RGB565, V4L2_PIX_FMT_BGR24 or V4L2_PIX_FMT_BGR32
++ *
++ * @return bits per pixel of pixelformat
++ */
++static u32 fmt_to_bpp(u32 pixelformat)
++{
++ u32 bpp;
++
++ switch (pixelformat) {
++ case V4L2_PIX_FMT_RGB565:
++ bpp = 16;
++ break;
++ case V4L2_PIX_FMT_BGR24:
++ case V4L2_PIX_FMT_RGB24:
++ bpp = 24;
++ break;
++ case V4L2_PIX_FMT_BGR32:
++ case V4L2_PIX_FMT_RGB32:
++ bpp = 32;
++ break;
++ default:
++ bpp = 8;
++ break;
++ }
++ return bpp;
++}
++
++static void mxc_v4l2out_timer_handler(unsigned long arg)
++{
++ int index;
++ unsigned long timeout;
++ unsigned long lock_flags = 0;
++ vout_data *vout = (vout_data *) arg;
++
++ dev_dbg(vout->video_dev->dev, "timer handler: %lu\n", jiffies);
++
++ spin_lock_irqsave(&g_lock, lock_flags);
++
++ /*
++ * If timer occurs before IPU h/w is ready, then set the state to
++ * paused and the timer will be set again when next buffer is queued
++ * or PP comletes
++ */
++ if (vout->ipu_buf[vout->next_rdy_ipu_buf] != -1) {
++ dev_dbg(vout->video_dev->dev, "IPU buffer busy\n");
++ vout->state = STATE_STREAM_PAUSED;
++ goto exit0;
++ }
++
++ /* Dequeue buffer and pass to IPU */
++ index = dequeue_buf(&vout->ready_q);
++ if (index == -1) { /* no buffers ready, should never occur */
++ dev_err(vout->video_dev->dev,
++ "mxc_v4l2out: timer - no queued buffers ready\n");
++ goto exit0;
++ }
++
++ g_buf_dq_cnt++;
++ vout->frame_count++;
++ vout->ipu_buf[vout->next_rdy_ipu_buf] = index;
++ if (ipu_update_channel_buffer(vout->post_proc_ch, IPU_INPUT_BUFFER,
++ vout->next_rdy_ipu_buf,
++ vout->v4l2_bufs[index].m.offset) < 0) {
++ dev_err(vout->video_dev->dev,
++ "unable to update buffer %d address\n",
++ vout->next_rdy_ipu_buf);
++ goto exit0;
++ }
++ if (ipu_select_buffer(vout->post_proc_ch, IPU_INPUT_BUFFER,
++ vout->next_rdy_ipu_buf) < 0) {
++ dev_err(vout->video_dev->dev,
++ "unable to set IPU buffer ready\n");
++ }
++ vout->next_rdy_ipu_buf = !vout->next_rdy_ipu_buf;
++
++ /* Setup timer for next buffer */
++ index = peek_next_buf(&vout->ready_q);
++ if (index != -1) {
++ /* if timestamp is 0, then default to 30fps */
++ if ((vout->v4l2_bufs[index].timestamp.tv_sec == 0)
++ && (vout->v4l2_bufs[index].timestamp.tv_usec == 0))
++ timeout =
++ vout->start_jiffies + vout->frame_count * HZ / 30;
++ else
++ timeout =
++ get_jiffies(&vout->v4l2_bufs[index].timestamp);
++
++ if (jiffies >= timeout) {
++ dev_dbg(vout->video_dev->dev,
++ "warning: timer timeout already expired.\n");
++ }
++ if (mod_timer(&vout->output_timer, timeout))
++ dev_dbg(vout->video_dev->dev,
++ "warning: timer was already set\n");
++
++ dev_dbg(vout->video_dev->dev,
++ "timer handler next schedule: %lu\n", timeout);
++ } else {
++ vout->state = STATE_STREAM_PAUSED;
++ }
++
++ exit0:
++ spin_unlock_irqrestore(&g_lock, lock_flags);
++}
++
++static irqreturn_t mxc_v4l2out_pp_in_irq_handler(int irq, void *dev_id)
++{
++ int last_buf;
++ int index;
++ unsigned long timeout;
++ unsigned long lock_flags = 0;
++ vout_data *vout = dev_id;
++
++ spin_lock_irqsave(&g_lock, lock_flags);
++
++ g_irq_cnt++;
++
++ /* Process previous buffer */
++ last_buf = vout->ipu_buf[vout->next_done_ipu_buf];
++ if (last_buf != -1) {
++ g_buf_output_cnt++;
++ vout->v4l2_bufs[last_buf].flags = V4L2_BUF_FLAG_DONE;
++ queue_buf(&vout->done_q, last_buf);
++ vout->ipu_buf[vout->next_done_ipu_buf] = -1;
++ wake_up_interruptible(&vout->v4l_bufq);
++ /* printk("pp_irq: buf %d done\n", vout->next_done_ipu_buf); */
++ vout->next_done_ipu_buf = !vout->next_done_ipu_buf;
++ }
++
++ if (vout->state == STATE_STREAM_STOPPING) {
++ if ((vout->ipu_buf[0] == -1) && (vout->ipu_buf[1] == -1)) {
++ vout->state = STATE_STREAM_OFF;
++ }
++ } else if ((vout->state == STATE_STREAM_PAUSED)
++ && ((index = peek_next_buf(&vout->ready_q)) != -1)) {
++ /* Setup timer for next buffer, when stream has been paused */
++ pr_debug("next index %d\n", index);
++
++ /* if timestamp is 0, then default to 30fps */
++ if ((vout->v4l2_bufs[index].timestamp.tv_sec == 0)
++ && (vout->v4l2_bufs[index].timestamp.tv_usec == 0))
++ timeout =
++ vout->start_jiffies + vout->frame_count * HZ / 30;
++ else
++ timeout =
++ get_jiffies(&vout->v4l2_bufs[index].timestamp);
++
++ if (jiffies >= timeout) {
++ pr_debug("warning: timer timeout already expired.\n");
++ }
++
++ vout->state = STATE_STREAM_ON;
++
++ if (mod_timer(&vout->output_timer, timeout))
++ pr_debug("warning: timer was already set\n");
++
++ pr_debug("timer handler next schedule: %lu\n", timeout);
++ }
++
++ spin_unlock_irqrestore(&g_lock, lock_flags);
++
++ return IRQ_HANDLED;
++}
++
++/*!
++ * Start the output stream
++ *
++ * @param vout structure vout_data *
++ *
++ * @return status 0 Success
++ */
++static int mxc_v4l2out_streamon(vout_data * vout)
++{
++ struct device *dev = vout->video_dev->dev;
++ ipu_channel_params_t params;
++ struct mxcfb_pos fb_pos;
++ struct fb_var_screeninfo fbvar;
++ struct fb_info *fbi =
++ registered_fb[vout->output_fb_num[vout->cur_disp_output]];
++ int pp_in_buf[2];
++ u16 out_width;
++ u16 out_height;
++ ipu_channel_t display_input_ch = MEM_PP_MEM;
++ bool use_direct_adc = false;
++
++ if (!vout)
++ return -EINVAL;
++
++ if (vout->state != STATE_STREAM_OFF)
++ return -EBUSY;
++
++ if (queue_size(&vout->ready_q) < 2) {
++ dev_err(dev, "2 buffers not been queued yet!\n");
++ return -EINVAL;
++ }
++
++ out_width = vout->crop_current.width;
++ out_height = vout->crop_current.height;
++
++ vout->next_done_ipu_buf = vout->next_rdy_ipu_buf = 0;
++ vout->ipu_buf[0] = pp_in_buf[0] = dequeue_buf(&vout->ready_q);
++ vout->ipu_buf[1] = pp_in_buf[1] = dequeue_buf(&vout->ready_q);
++ vout->frame_count = 2;
++
++ ipu_enable_irq(IPU_IRQ_PP_IN_EOF);
++
++ /* Init Display Channel */
++#ifdef CONFIG_FB_MXC_ASYNC_PANEL
++ if (vout->cur_disp_output < DISP3) {
++ mxcfb_set_refresh_mode(fbi, MXCFB_REFRESH_OFF, 0);
++ if (vout->rotate < IPU_ROTATE_90_RIGHT) {
++ dev_dbg(dev, "Using PP direct to ADC channel\n");
++ use_direct_adc = true;
++ vout->display_ch = MEM_PP_ADC;
++ vout->post_proc_ch = MEM_PP_ADC;
++
++ memset(&params, 0, sizeof(params));
++ params.mem_pp_adc.in_width = vout->v2f.fmt.pix.width;
++ params.mem_pp_adc.in_height = vout->v2f.fmt.pix.height;
++ params.mem_pp_adc.in_pixel_fmt =
++ vout->v2f.fmt.pix.pixelformat;
++ params.mem_pp_adc.out_width = out_width;
++ params.mem_pp_adc.out_height = out_height;
++ params.mem_pp_adc.out_pixel_fmt = SDC_FG_FB_FORMAT;
++#ifdef CONFIG_FB_MXC_EPSON_PANEL
++ params.mem_pp_adc.out_left =
++ 2 + vout->crop_current.left;
++#else
++ params.mem_pp_adc.out_left =
++ 12 + vout->crop_current.left;
++#endif
++ params.mem_pp_adc.out_top = vout->crop_current.top;
++ if (ipu_init_channel(vout->post_proc_ch, &params) != 0) {
++ dev_err(dev, "Error initializing PP chan\n");
++ return -EINVAL;
++ }
++
++ if (ipu_init_channel_buffer(vout->post_proc_ch,
++ IPU_INPUT_BUFFER,
++ params.mem_pp_adc.
++ in_pixel_fmt,
++ params.mem_pp_adc.in_width,
++ params.mem_pp_adc.in_height,
++ vout->v2f.fmt.pix.
++ bytesperline /
++ bytes_per_pixel(params.
++ mem_pp_adc.
++ in_pixel_fmt),
++ vout->rotate,
++ vout->
++ v4l2_bufs[pp_in_buf[0]].m.
++ offset,
++ vout->
++ v4l2_bufs[pp_in_buf[1]].m.
++ offset,
++ vout->offset.u_offset,
++ vout->offset.v_offset) !=
++ 0) {
++ dev_err(dev, "Error initializing PP in buf\n");
++ return -EINVAL;
++ }
++
++ if (ipu_init_channel_buffer(vout->post_proc_ch,
++ IPU_OUTPUT_BUFFER,
++ params.mem_pp_adc.
++ out_pixel_fmt, out_width,
++ out_height, out_width,
++ vout->rotate, 0, 0, 0,
++ 0) != 0) {
++ dev_err(dev,
++ "Error initializing PP output buffer\n");
++ return -EINVAL;
++ }
++
++ } else {
++ dev_dbg(dev, "Using ADC SYS2 channel\n");
++ vout->display_ch = ADC_SYS2;
++ vout->post_proc_ch = MEM_PP_MEM;
++
++ if (vout->display_bufs[0]) {
++ mxc_free_buffers(vout->display_bufs,
++ vout->display_bufs_vaddr,
++ 2, vout->display_buf_size);
++ }
++
++ vout->display_buf_size = vout->crop_current.width *
++ vout->crop_current.height *
++ fmt_to_bpp(SDC_FG_FB_FORMAT) / 8;
++ mxc_allocate_buffers(vout->display_bufs,
++ vout->display_bufs_vaddr,
++ 2, vout->display_buf_size);
++
++ memset(&params, 0, sizeof(params));
++ params.adc_sys2.disp = vout->cur_disp_output;
++ params.adc_sys2.ch_mode = WriteTemplateNonSeq;
++#ifdef CONFIG_FB_MXC_EPSON_PANEL
++ params.adc_sys2.out_left = 2 + vout->crop_current.left;
++#else
++ params.adc_sys2.out_left = 12 + vout->crop_current.left;
++#endif
++ params.adc_sys2.out_top = vout->crop_current.top;
++ if (ipu_init_channel(ADC_SYS2, &params) < 0)
++ return -EINVAL;
++
++ if (ipu_init_channel_buffer(vout->display_ch,
++ IPU_INPUT_BUFFER,
++ SDC_FG_FB_FORMAT,
++ out_width, out_height,
++ out_width, IPU_ROTATE_NONE,
++ vout->display_bufs[0],
++ vout->display_bufs[1], 0,
++ 0) != 0) {
++ dev_err(dev,
++ "Error initializing SDC FG buffer\n");
++ return -EINVAL;
++ }
++ }
++ } else
++#endif
++ { /* Use SDC */
++ dev_dbg(dev, "Using SDC channel\n");
++
++ fbvar = fbi->var;
++ fbvar.xres = fbvar.xres_virtual = out_width;
++ fbvar.yres = out_height;
++ fbvar.yres_virtual = out_height * 2;
++ fbvar.bits_per_pixel = 16;
++ fb_set_var(fbi, &fbvar);
++
++ fb_pos.x = vout->crop_current.left;
++ fb_pos.y = vout->crop_current.top;
++ if (fbi->fbops->fb_ioctl)
++ fbi->fbops->fb_ioctl(fbi, MXCFB_SET_OVERLAY_POS,
++ (unsigned long)&fb_pos);
++
++ vout->display_bufs[0] = fbi->fix.smem_start;
++ vout->display_bufs[1] = fbi->fix.smem_start +
++ (fbi->fix.line_length * fbi->var.yres);
++ vout->display_buf_size = vout->crop_current.width *
++ vout->crop_current.height *
++ fmt_to_bpp(SDC_FG_FB_FORMAT) / 8;
++
++ if (vout->cur_disp_output == 3)
++ vout->display_ch = MEM_SDC_FG;
++ else
++ vout->display_ch = MEM_SDC_BG;
++
++ vout->post_proc_ch = MEM_PP_MEM;
++ }
++
++ /* Init PP */
++ if (use_direct_adc == false) {
++ if (vout->rotate >= IPU_ROTATE_90_RIGHT) {
++ out_width = vout->crop_current.height;
++ out_height = vout->crop_current.width;
++ }
++ memset(&params, 0, sizeof(params));
++ params.mem_pp_mem.in_width = vout->v2f.fmt.pix.width;
++ params.mem_pp_mem.in_height = vout->v2f.fmt.pix.height;
++ params.mem_pp_mem.in_pixel_fmt = vout->v2f.fmt.pix.pixelformat;
++ params.mem_pp_mem.out_width = out_width;
++ params.mem_pp_mem.out_height = out_height;
++ params.mem_pp_mem.out_pixel_fmt = SDC_FG_FB_FORMAT;
++ if (ipu_init_channel(vout->post_proc_ch, &params) != 0) {
++ dev_err(dev, "Error initializing PP channel\n");
++ return -EINVAL;
++ }
++
++ if (ipu_init_channel_buffer(vout->post_proc_ch,
++ IPU_INPUT_BUFFER,
++ params.mem_pp_mem.in_pixel_fmt,
++ params.mem_pp_mem.in_width,
++ params.mem_pp_mem.in_height,
++ vout->v2f.fmt.pix.bytesperline /
++ bytes_per_pixel(params.mem_pp_mem.
++ in_pixel_fmt),
++ IPU_ROTATE_NONE,
++ vout->v4l2_bufs[pp_in_buf[0]].m.
++ offset,
++ vout->v4l2_bufs[pp_in_buf[1]].m.
++ offset, vout->offset.u_offset,
++ vout->offset.v_offset) != 0) {
++ dev_err(dev, "Error initializing PP input buffer\n");
++ return -EINVAL;
++ }
++
++ if (vout->rotate >= IPU_ROTATE_90_RIGHT) {
++ if (vout->rot_pp_bufs[0]) {
++ mxc_free_buffers(vout->rot_pp_bufs,
++ vout->rot_pp_bufs_vaddr, 2,
++ vout->display_buf_size);
++ }
++ if (mxc_allocate_buffers
++ (vout->rot_pp_bufs, vout->rot_pp_bufs_vaddr, 2,
++ vout->display_buf_size) < 0) {
++ return -ENOBUFS;
++ }
++
++ if (ipu_init_channel_buffer(vout->post_proc_ch,
++ IPU_OUTPUT_BUFFER,
++ params.mem_pp_mem.
++ out_pixel_fmt, out_width,
++ out_height, out_width,
++ IPU_ROTATE_NONE,
++ vout->rot_pp_bufs[0],
++ vout->rot_pp_bufs[1], 0,
++ 0) != 0) {
++ dev_err(dev,
++ "Error initializing PP output buffer\n");
++ return -EINVAL;
++ }
++
++ if (ipu_init_channel(MEM_ROT_PP_MEM, NULL) != 0) {
++ dev_err(dev,
++ "Error initializing PP ROT channel\n");
++ return -EINVAL;
++ }
++
++ if (ipu_init_channel_buffer(MEM_ROT_PP_MEM,
++ IPU_INPUT_BUFFER,
++ params.mem_pp_mem.
++ out_pixel_fmt, out_width,
++ out_height, out_width,
++ vout->rotate,
++ vout->rot_pp_bufs[0],
++ vout->rot_pp_bufs[1], 0,
++ 0) != 0) {
++ dev_err(dev,
++ "Error initializing PP ROT input buffer\n");
++ return -EINVAL;
++ }
++
++ /* swap width and height */
++ out_width = vout->crop_current.width;
++ out_height = vout->crop_current.height;
++
++ if (ipu_init_channel_buffer(MEM_ROT_PP_MEM,
++ IPU_OUTPUT_BUFFER,
++ params.mem_pp_mem.
++ out_pixel_fmt, out_width,
++ out_height, out_width,
++ IPU_ROTATE_NONE,
++ vout->display_bufs[0],
++ vout->display_bufs[1], 0,
++ 0) != 0) {
++ dev_err(dev,
++ "Error initializing PP output buffer\n");
++ return -EINVAL;
++ }
++
++ if (ipu_link_channels(vout->post_proc_ch,
++ MEM_ROT_PP_MEM) < 0) {
++ return -EINVAL;
++ }
++ ipu_select_buffer(MEM_ROT_PP_MEM, IPU_OUTPUT_BUFFER, 0);
++ ipu_select_buffer(MEM_ROT_PP_MEM, IPU_OUTPUT_BUFFER, 1);
++
++ ipu_enable_channel(MEM_ROT_PP_MEM);
++
++ display_input_ch = MEM_ROT_PP_MEM;
++ } else {
++ if (ipu_init_channel_buffer(vout->post_proc_ch,
++ IPU_OUTPUT_BUFFER,
++ params.mem_pp_mem.
++ out_pixel_fmt, out_width,
++ out_height, out_width,
++ vout->rotate,
++ vout->display_bufs[0],
++ vout->display_bufs[1], 0,
++ 0) != 0) {
++ dev_err(dev,
++ "Error initializing PP output buffer\n");
++ return -EINVAL;
++ }
++ }
++ if (ipu_link_channels(display_input_ch, vout->display_ch) < 0) {
++ dev_err(dev, "Error linking ipu channels\n");
++ return -EINVAL;
++ }
++ }
++
++ vout->state = STATE_STREAM_PAUSED;
++
++ ipu_select_buffer(vout->post_proc_ch, IPU_INPUT_BUFFER, 0);
++ ipu_select_buffer(vout->post_proc_ch, IPU_INPUT_BUFFER, 1);
++
++ if (use_direct_adc == false) {
++ ipu_select_buffer(vout->post_proc_ch, IPU_OUTPUT_BUFFER, 0);
++ ipu_select_buffer(vout->post_proc_ch, IPU_OUTPUT_BUFFER, 1);
++
++ ipu_enable_channel(vout->post_proc_ch);
++ if ((vout->display_ch == MEM_SDC_FG) ||
++ (vout->display_ch == MEM_SDC_BG)) {
++ acquire_console_sem();
++ fb_blank(fbi, FB_BLANK_UNBLANK);
++ release_console_sem();
++ } else {
++ ipu_enable_channel(vout->display_ch);
++ }
++ } else {
++ ipu_enable_channel(vout->post_proc_ch);
++ }
++
++ vout->start_jiffies = jiffies;
++ dev_dbg(dev,
++ "streamon: start time = %lu jiffies\n", vout->start_jiffies);
++
++ return 0;
++}
++
++/*!
++ * Shut down the voutera
++ *
++ * @param vout structure vout_data *
++ *
++ * @return status 0 Success
++ */
++static int mxc_v4l2out_streamoff(vout_data * vout)
++{
++ struct fb_info *fbi =
++ registered_fb[vout->output_fb_num[vout->cur_disp_output]];
++ int i, retval = 0;
++ unsigned long lockflag = 0;
++
++ if (!vout)
++ return -EINVAL;
++
++ if (vout->state == STATE_STREAM_OFF) {
++ return 0;
++ }
++
++ spin_lock_irqsave(&g_lock, lockflag);
++
++ del_timer(&vout->output_timer);
++
++ if (vout->state == STATE_STREAM_ON) {
++ vout->state = STATE_STREAM_STOPPING;
++ }
++
++ ipu_disable_irq(IPU_IRQ_PP_IN_EOF);
++
++ spin_unlock_irqrestore(&g_lock, lockflag);
++
++ if (vout->post_proc_ch == MEM_PP_MEM) { /* SDC or ADC with Rotation */
++ if (vout->rotate >= IPU_ROTATE_90_RIGHT) {
++ ipu_unlink_channels(MEM_PP_MEM, MEM_ROT_PP_MEM);
++ ipu_unlink_channels(MEM_ROT_PP_MEM, vout->display_ch);
++ ipu_disable_channel(MEM_ROT_PP_MEM, true);
++ } else {
++ ipu_unlink_channels(MEM_PP_MEM, vout->display_ch);
++ }
++ ipu_disable_channel(MEM_PP_MEM, true);
++ if ((vout->display_ch != MEM_SDC_FG) &&
++ (vout->display_ch != MEM_SDC_BG)) {
++ ipu_disable_channel(vout->display_ch, true);
++ ipu_uninit_channel(vout->display_ch);
++ } else {
++ fbi->var.activate |= FB_ACTIVATE_FORCE;
++ fb_set_var(fbi, &fbi->var);
++ }
++
++ ipu_uninit_channel(MEM_PP_MEM);
++ if (vout->rotate >= IPU_ROTATE_90_RIGHT)
++ ipu_uninit_channel(MEM_ROT_PP_MEM);
++ } else { /* ADC Direct */
++ ipu_disable_channel(MEM_PP_ADC, true);
++ ipu_uninit_channel(MEM_PP_ADC);
++ }
++ vout->ready_q.head = vout->ready_q.tail = 0;
++ vout->done_q.head = vout->done_q.tail = 0;
++ for (i = 0; i < vout->buffer_cnt; i++) {
++ vout->v4l2_bufs[i].flags = 0;
++ vout->v4l2_bufs[i].timestamp.tv_sec = 0;
++ vout->v4l2_bufs[i].timestamp.tv_usec = 0;
++ }
++
++ vout->state = STATE_STREAM_OFF;
++
++ if (vout->display_bufs[0] != 0) {
++ mxc_free_buffers(vout->display_bufs,
++ vout->display_bufs_vaddr, 2,
++ vout->display_buf_size);
++ }
++#ifdef CONFIG_FB_MXC_ASYNC_PANEL
++ if (vout->cur_disp_output < DISP3) {
++ mxcfb_set_refresh_mode(registered_fb
++ [vout->
++ output_fb_num[vout->cur_disp_output]],
++ MXCFB_REFRESH_PARTIAL, 0);
++ }
++#endif
++
++ return retval;
++}
++
++/*
++ * Valid whether the palette is supported
++ *
++ * @param palette V4L2_PIX_FMT_RGB565, V4L2_PIX_FMT_BGR24 or V4L2_PIX_FMT_BGR32
++ *
++ * @return 1 if supported, 0 if failed
++ */
++static inline int valid_mode(u32 palette)
++{
++ return ((palette == V4L2_PIX_FMT_RGB565) ||
++ (palette == V4L2_PIX_FMT_BGR24) ||
++ (palette == V4L2_PIX_FMT_RGB24) ||
++ (palette == V4L2_PIX_FMT_BGR32) ||
++ (palette == V4L2_PIX_FMT_RGB32) ||
++ (palette == V4L2_PIX_FMT_YUV422P) ||
++ (palette == V4L2_PIX_FMT_YUV420));
++}
++
++/*
++ * V4L2 - Handles VIDIOC_G_FMT Ioctl
++ *
++ * @param vout structure vout_data *
++ *
++ * @param v4l2_format structure v4l2_format *
++ *
++ * @return status 0 success, EINVAL failed
++ */
++static int mxc_v4l2out_g_fmt(vout_data * vout, struct v4l2_format *f)
++{
++ if (f->type != V4L2_BUF_TYPE_VIDEO_OUTPUT) {
++ return -EINVAL;
++ }
++ *f = vout->v2f;
++ return 0;
++}
++
++/*
++ * V4L2 - Handles VIDIOC_S_FMT Ioctl
++ *
++ * @param vout structure vout_data *
++ *
++ * @param v4l2_format structure v4l2_format *
++ *
++ * @return status 0 success, EINVAL failed
++ */
++static int mxc_v4l2out_s_fmt(vout_data * vout, struct v4l2_format *f)
++{
++ int retval = 0;
++ u32 size = 0;
++ u32 bytesperline;
++
++ if (f->type != V4L2_BUF_TYPE_VIDEO_OUTPUT) {
++ retval = -EINVAL;
++ goto err0;
++ }
++ if (!valid_mode(f->fmt.pix.pixelformat)) {
++ dev_err(vout->video_dev->dev, "pixel format not supported\n");
++ retval = -EINVAL;
++ goto err0;
++ }
++
++ bytesperline = (f->fmt.pix.width * fmt_to_bpp(f->fmt.pix.pixelformat)) /
++ 8;
++ if (f->fmt.pix.bytesperline < bytesperline) {
++ f->fmt.pix.bytesperline = bytesperline;
++ } else {
++ bytesperline = f->fmt.pix.bytesperline;
++ }
++
++ switch (f->fmt.pix.pixelformat) {
++ case V4L2_PIX_FMT_YUV422P:
++ /* byteperline for YUV planar formats is for
++ Y plane only */
++ size = bytesperline * f->fmt.pix.height * 2;
++ break;
++ case V4L2_PIX_FMT_YUV420:
++ size = (bytesperline * f->fmt.pix.height * 3) / 2;
++ break;
++ default:
++ size = bytesperline * f->fmt.pix.height;
++ break;
++ }
++
++ /* Return the actual size of the image to the app */
++ if (f->fmt.pix.sizeimage < size) {
++ f->fmt.pix.sizeimage = size;
++ } else {
++ size = f->fmt.pix.sizeimage;
++ }
++
++ vout->v2f.fmt.pix = f->fmt.pix;
++ if (vout->v2f.fmt.pix.priv != 0) {
++ if (copy_from_user(&vout->offset,
++ (void *)vout->v2f.fmt.pix.priv,
++ sizeof(vout->offset))) {
++ retval = -EFAULT;
++ goto err0;
++ }
++ }
++
++ retval = 0;
++ err0:
++ return retval;
++}
++
++/*
++ * V4L2 - Handles VIDIOC_G_CTRL Ioctl
++ *
++ * @param vout structure vout_data *
++ *
++ * @param c structure v4l2_control *
++ *
++ * @return status 0 success, EINVAL failed
++ */
++static int mxc_get_v42lout_control(vout_data * vout, struct v4l2_control *c)
++{
++ switch (c->id) {
++ case V4L2_CID_HFLIP:
++ return (vout->rotate & IPU_ROTATE_HORIZ_FLIP) ? 1 : 0;
++ case V4L2_CID_VFLIP:
++ return (vout->rotate & IPU_ROTATE_VERT_FLIP) ? 1 : 0;
++ case (V4L2_CID_PRIVATE_BASE + 1):
++ return vout->rotate;
++ default:
++ return -EINVAL;
++ }
++}
++
++/*
++ * V4L2 - Handles VIDIOC_S_CTRL Ioctl
++ *
++ * @param vout structure vout_data *
++ *
++ * @param c structure v4l2_control *
++ *
++ * @return status 0 success, EINVAL failed
++ */
++static int mxc_set_v42lout_control(vout_data * vout, struct v4l2_control *c)
++{
++ switch (c->id) {
++ case V4L2_CID_HFLIP:
++ vout->rotate |= c->value ? IPU_ROTATE_HORIZ_FLIP :
++ IPU_ROTATE_NONE;
++ break;
++ case V4L2_CID_VFLIP:
++ vout->rotate |= c->value ? IPU_ROTATE_VERT_FLIP :
++ IPU_ROTATE_NONE;
++ break;
++ case V4L2_CID_MXC_ROT:
++ vout->rotate = c->value;
++ break;
++ default:
++ return -EINVAL;
++ }
++ return 0;
++}
++
++/*!
++ * V4L2 interface - open function
++ *
++ * @param inode structure inode *
++ *
++ * @param file structure file *
++ *
++ * @return status 0 success, ENODEV invalid device instance,
++ * ENODEV timeout, ERESTARTSYS interrupted by user
++ */
++static int mxc_v4l2out_open(struct inode *inode, struct file *file)
++{
++ struct video_device *dev = video_devdata(file);
++ vout_data *vout = video_get_drvdata(dev);
++ int err;
++
++ dq_intr_cnt = 0;
++ dq_timeout_cnt = 0;
++ if (!vout) {
++ return -ENODEV;
++ }
++
++ down(&vout->busy_lock);
++
++ err = -EINTR;
++ if (signal_pending(current))
++ goto oops;
++
++ if (vout->open_count++ == 0) {
++ ipu_request_irq(IPU_IRQ_PP_IN_EOF,
++ mxc_v4l2out_pp_in_irq_handler,
++ 0, dev->name, vout);
++
++ init_waitqueue_head(&vout->v4l_bufq);
++
++ init_timer(&vout->output_timer);
++ vout->output_timer.function = mxc_v4l2out_timer_handler;
++ vout->output_timer.data = (unsigned long)vout;
++
++ vout->state = STATE_STREAM_OFF;
++ g_irq_cnt = g_buf_output_cnt = g_buf_q_cnt = g_buf_dq_cnt = 0;
++
++ }
++
++ file->private_data = dev;
++
++ up(&vout->busy_lock);
++
++ return 0;
++
++ oops:
++ up(&vout->busy_lock);
++ return err;
++}
++
++/*!
++ * V4L2 interface - close function
++ *
++ * @param inode struct inode *
++ *
++ * @param file struct file *
++ *
++ * @return 0 success
++ */
++static int mxc_v4l2out_close(struct inode *inode, struct file *file)
++{
++ struct video_device *dev = video_devdata(file);
++ vout_data *vout = video_get_drvdata(dev);
++
++ if (--vout->open_count == 0) {
++ if (vout->state != STATE_STREAM_OFF)
++ mxc_v4l2out_streamoff(vout);
++
++ ipu_free_irq(IPU_IRQ_PP_IN_EOF, vout);
++
++ file->private_data = NULL;
++
++ mxc_free_buffers(vout->queue_buf_paddr, vout->queue_buf_vaddr,
++ vout->buffer_cnt, vout->queue_buf_size);
++ vout->buffer_cnt = 0;
++ mxc_free_buffers(vout->rot_pp_bufs, vout->rot_pp_bufs_vaddr, 2,
++ vout->display_buf_size);
++
++ /* capture off */
++ wake_up_interruptible(&vout->v4l_bufq);
++ }
++
++ return 0;
++}
++
++/*!
++ * V4L2 interface - ioctl function
++ *
++ * @param inode struct inode *
++ *
++ * @param file struct file *
++ *
++ * @param ioctlnr unsigned int
++ *
++ * @param arg void *
++ *
++ * @return 0 success, ENODEV for invalid device instance,
++ * -1 for other errors.
++ */
++static int
++mxc_v4l2out_do_ioctl(struct inode *inode, struct file *file,
++ unsigned int ioctlnr, void *arg)
++{
++ struct video_device *vdev = file->private_data;
++ vout_data *vout = video_get_drvdata(vdev);
++ int retval = 0;
++ int i = 0;
++
++ if (!vout)
++ return -EBADF;
++
++ /* make this _really_ smp-safe */
++ if (down_interruptible(&vout->busy_lock))
++ return -EBUSY;
++
++ switch (ioctlnr) {
++ case VIDIOC_QUERYCAP:
++ {
++ struct v4l2_capability *cap = arg;
++ strcpy(cap->driver, "mxc_v4l2_output");
++ cap->version = 0;
++ cap->capabilities =
++ V4L2_CAP_VIDEO_OUTPUT | V4L2_CAP_STREAMING;
++ cap->card[0] = '\0';
++ cap->bus_info[0] = '\0';
++ retval = 0;
++ break;
++ }
++ case VIDIOC_G_FMT:
++ {
++ struct v4l2_format *gf = arg;
++ retval = mxc_v4l2out_g_fmt(vout, gf);
++ break;
++ }
++ case VIDIOC_S_FMT:
++ {
++ struct v4l2_format *sf = arg;
++ if (vout->state != STATE_STREAM_OFF) {
++ retval = -EBUSY;
++ break;
++ }
++ retval = mxc_v4l2out_s_fmt(vout, sf);
++ break;
++ }
++ case VIDIOC_REQBUFS:
++ {
++ struct v4l2_requestbuffers *req = arg;
++ if ((req->type != V4L2_BUF_TYPE_VIDEO_OUTPUT) ||
++ (req->memory != V4L2_MEMORY_MMAP)) {
++ dev_dbg(vdev->dev,
++ "VIDIOC_REQBUFS: incorrect buffer type\n");
++ retval = -EINVAL;
++ break;
++ }
++
++ if (req->count == 0)
++ mxc_v4l2out_streamoff(vout);
++
++ if (vout->state == STATE_STREAM_OFF) {
++ if (vout->queue_buf_paddr[0] != 0) {
++ mxc_free_buffers(vout->queue_buf_paddr,
++ vout->queue_buf_vaddr,
++ vout->buffer_cnt,
++ vout->queue_buf_size);
++ dev_dbg(vdev->dev,
++ "VIDIOC_REQBUFS: freed buffers\n");
++ }
++ vout->buffer_cnt = 0;
++ } else {
++ dev_dbg(vdev->dev,
++ "VIDIOC_REQBUFS: Buffer is in use\n");
++ retval = -EBUSY;
++ break;
++ }
++
++ if (req->count == 0)
++ break;
++
++ if (req->count < MIN_FRAME_NUM) {
++ req->count = MIN_FRAME_NUM;
++ } else if (req->count > MAX_FRAME_NUM) {
++ req->count = MAX_FRAME_NUM;
++ }
++ vout->buffer_cnt = req->count;
++ vout->queue_buf_size =
++ PAGE_ALIGN(vout->v2f.fmt.pix.sizeimage);
++
++ retval = mxc_allocate_buffers(vout->queue_buf_paddr,
++ vout->queue_buf_vaddr,
++ vout->buffer_cnt,
++ vout->queue_buf_size);
++ if (retval < 0)
++ break;
++
++ /* Init buffer queues */
++ vout->done_q.head = 0;
++ vout->done_q.tail = 0;
++ vout->ready_q.head = 0;
++ vout->ready_q.tail = 0;
++
++ for (i = 0; i < vout->buffer_cnt; i++) {
++ memset(&(vout->v4l2_bufs[i]), 0,
++ sizeof(vout->v4l2_bufs[i]));
++ vout->v4l2_bufs[i].flags = 0;
++ vout->v4l2_bufs[i].memory = V4L2_MEMORY_MMAP;
++ vout->v4l2_bufs[i].index = i;
++ vout->v4l2_bufs[i].type =
++ V4L2_BUF_TYPE_VIDEO_OUTPUT;
++ vout->v4l2_bufs[i].length =
++ PAGE_ALIGN(vout->v2f.fmt.pix.sizeimage);
++ vout->v4l2_bufs[i].m.offset =
++ (unsigned long)vout->queue_buf_paddr[i];
++ vout->v4l2_bufs[i].timestamp.tv_sec = 0;
++ vout->v4l2_bufs[i].timestamp.tv_usec = 0;
++ }
++ break;
++ }
++ case VIDIOC_QUERYBUF:
++ {
++ struct v4l2_buffer *buf = arg;
++ u32 type = buf->type;
++ int index = buf->index;
++
++ if ((type != V4L2_BUF_TYPE_VIDEO_OUTPUT) ||
++ (index >= vout->buffer_cnt)) {
++ dev_dbg(vdev->dev,
++ "VIDIOC_QUERYBUFS: incorrect buffer type\n");
++ retval = -EINVAL;
++ break;
++ }
++ down(&vout->param_lock);
++ memcpy(buf, &(vout->v4l2_bufs[index]), sizeof(*buf));
++ up(&vout->param_lock);
++ break;
++ }
++ case VIDIOC_QBUF:
++ {
++ struct v4l2_buffer *buf = arg;
++ int index = buf->index;
++ unsigned long lock_flags;
++
++ if ((buf->type != V4L2_BUF_TYPE_VIDEO_OUTPUT) ||
++ (index >= vout->buffer_cnt)) {
++ retval = -EINVAL;
++ break;
++ }
++
++ dev_dbg(vdev->dev, "VIDIOC_QBUF: %d\n", buf->index);
++
++ /* mmapped buffers are L1 WB cached,
++ * so we need to clean them */
++ if (buf->flags & V4L2_BUF_FLAG_MAPPED) {
++ flush_cache_all();
++ }
++
++ spin_lock_irqsave(&g_lock, lock_flags);
++
++ memcpy(&(vout->v4l2_bufs[index]), buf, sizeof(*buf));
++ vout->v4l2_bufs[index].flags |= V4L2_BUF_FLAG_QUEUED;
++
++ g_buf_q_cnt++;
++ queue_buf(&vout->ready_q, index);
++ if (vout->state == STATE_STREAM_PAUSED) {
++ unsigned long timeout;
++
++ index = peek_next_buf(&vout->ready_q);
++
++ /* if timestamp is 0, then default to 30fps */
++ if ((vout->v4l2_bufs[index].timestamp.tv_sec ==
++ 0)
++ && (vout->v4l2_bufs[index].timestamp.
++ tv_usec == 0))
++ timeout =
++ vout->start_jiffies +
++ vout->frame_count * HZ / 30;
++ else
++ timeout =
++ get_jiffies(&vout->v4l2_bufs[index].
++ timestamp);
++
++ if (jiffies >= timeout) {
++ dev_dbg(vout->video_dev->dev,
++ "warning: timer timeout already expired.\n");
++ }
++ vout->output_timer.expires = timeout;
++ dev_dbg(vdev->dev,
++ "QBUF: frame #%u timeout @ %lu jiffies, current = %lu\n",
++ vout->frame_count, timeout, jiffies);
++ add_timer(&vout->output_timer);
++ vout->state = STATE_STREAM_ON;
++ }
++
++ spin_unlock_irqrestore(&g_lock, lock_flags);
++ break;
++ }
++ case VIDIOC_DQBUF:
++ {
++ struct v4l2_buffer *buf = arg;
++ int idx;
++
++ if ((queue_size(&vout->done_q) == 0) &&
++ (file->f_flags & O_NONBLOCK)) {
++ retval = -EAGAIN;
++ break;
++ }
++
++ if (!wait_event_interruptible_timeout(vout->v4l_bufq,
++ queue_size(&vout->
++ done_q)
++ != 0, 10 * HZ)) {
++ if(dq_timeout_cnt == 0){
++ dev_dbg(vdev->dev, "VIDIOC_DQBUF: timeout\n");
++ }
++ dq_timeout_cnt++;
++ retval = -ETIME;
++ break;
++ } else if (signal_pending(current)) {
++ if (dq_intr_cnt == 0) {
++ dev_dbg(vdev->dev,
++ "VIDIOC_DQBUF: interrupt received\n");
++ }
++ dq_intr_cnt++;
++ retval = -ERESTARTSYS;
++ break;
++ }
++ idx = dequeue_buf(&vout->done_q);
++ if (idx == -1) { /* No frame free */
++ dev_dbg(vdev->dev,
++ "VIDIOC_DQBUF: no free buffers, returning\n");
++ retval = -EAGAIN;
++ break;
++ }
++ if ((vout->v4l2_bufs[idx].flags & V4L2_BUF_FLAG_DONE) ==
++ 0)
++ dev_dbg(vdev->dev,
++ "VIDIOC_DQBUF: buffer in done q, but not "
++ "flagged as done\n");
++
++ vout->v4l2_bufs[idx].flags = 0;
++ memcpy(buf, &(vout->v4l2_bufs[idx]), sizeof(*buf));
++ dev_dbg(vdev->dev, "VIDIOC_DQBUF: %d\n", buf->index);
++ break;
++ }
++ case VIDIOC_STREAMON:
++ {
++ retval = mxc_v4l2out_streamon(vout);
++ break;
++ }
++ case VIDIOC_STREAMOFF:
++ {
++ retval = mxc_v4l2out_streamoff(vout);
++ break;
++ }
++ case VIDIOC_G_CTRL:
++ {
++ retval = mxc_get_v42lout_control(vout, arg);
++ break;
++ }
++ case VIDIOC_S_CTRL:
++ {
++ retval = mxc_set_v42lout_control(vout, arg);
++ break;
++ }
++ case VIDIOC_CROPCAP:
++ {
++ struct v4l2_cropcap *cap = arg;
++
++ if (cap->type != V4L2_BUF_TYPE_VIDEO_OUTPUT) {
++ retval = -EINVAL;
++ break;
++ }
++ cap->bounds = vout->crop_bounds[vout->cur_disp_output];
++ cap->defrect = vout->crop_bounds[vout->cur_disp_output];
++ retval = 0;
++ break;
++ }
++ case VIDIOC_G_CROP:
++ {
++ struct v4l2_crop *crop = arg;
++
++ if (crop->type != V4L2_BUF_TYPE_VIDEO_OUTPUT) {
++ retval = -EINVAL;
++ break;
++ }
++ crop->c = vout->crop_current;
++ break;
++ }
++ case VIDIOC_S_CROP:
++ {
++ struct v4l2_crop *crop = arg;
++ struct v4l2_rect *b =
++ &(vout->crop_bounds[vout->cur_disp_output]);
++
++ if (crop->type != V4L2_BUF_TYPE_VIDEO_OUTPUT) {
++ retval = -EINVAL;
++ break;
++ }
++ if (crop->c.height < 0) {
++ retval = -EINVAL;
++ break;
++ }
++ if (crop->c.width < 0) {
++ retval = -EINVAL;
++ break;
++ }
++
++ /* only full screen supported for SDC BG */
++ if (vout->cur_disp_output == 4) {
++ crop->c = vout->crop_current;
++ break;
++ }
++
++ if (crop->c.top < b->top)
++ crop->c.top = b->top;
++ if (crop->c.top >= b->top + b->height)
++ crop->c.top = b->top + b->height - 1;
++ if (crop->c.height > b->top - crop->c.top + b->height)
++ crop->c.height =
++ b->top - crop->c.top + b->height;
++
++ if (crop->c.left < b->left)
++ crop->c.left = b->left;
++ if (crop->c.left >= b->left + b->width)
++ crop->c.left = b->left + b->width - 1;
++ if (crop->c.width > b->left - crop->c.left + b->width)
++ crop->c.width =
++ b->left - crop->c.left + b->width;
++
++ /* stride line limitation */
++ crop->c.height -= crop->c.height % 8;
++ crop->c.width -= crop->c.width % 8;
++
++ vout->crop_current = crop->c;
++ break;
++ }
++ case VIDIOC_ENUMOUTPUT:
++ {
++ struct v4l2_output *output = arg;
++
++ if ((output->index >= 5) ||
++ (vout->output_enabled[output->index] == false)) {
++ retval = -EINVAL;
++ break;
++ }
++
++ if (output->index < 3) {
++ *output = mxc_outputs[MXC_V4L2_OUT_2_ADC];
++ output->name[4] = '0' + output->index;
++ } else {
++ *output = mxc_outputs[MXC_V4L2_OUT_2_SDC];
++ }
++ break;
++ }
++ case VIDIOC_G_OUTPUT:
++ {
++ int *p_output_num = arg;
++
++ *p_output_num = vout->cur_disp_output;
++ break;
++ }
++ case VIDIOC_S_OUTPUT:
++ {
++ int *p_output_num = arg;
++
++ if ((*p_output_num >= 5) ||
++ (vout->output_enabled[*p_output_num] == false)) {
++ retval = -EINVAL;
++ break;
++ }
++
++ if (vout->state != STATE_STREAM_OFF) {
++ retval = -EBUSY;
++ break;
++ }
++
++ vout->cur_disp_output = *p_output_num;
++ vout->crop_current =
++ vout->crop_bounds[vout->cur_disp_output];
++ break;
++ }
++ case VIDIOC_ENUM_FMT:
++ case VIDIOC_TRY_FMT:
++ case VIDIOC_QUERYCTRL:
++ case VIDIOC_G_PARM:
++ case VIDIOC_ENUMSTD:
++ case VIDIOC_G_STD:
++ case VIDIOC_S_STD:
++ case VIDIOC_G_TUNER:
++ case VIDIOC_S_TUNER:
++ case VIDIOC_G_FREQUENCY:
++ case VIDIOC_S_FREQUENCY:
++ default:
++ retval = -EINVAL;
++ break;
++ }
++
++ up(&vout->busy_lock);
++ return retval;
++}
++
++/*
++ * V4L2 interface - ioctl function
++ *
++ * @return None
++ */
++static int
++mxc_v4l2out_ioctl(struct inode *inode, struct file *file,
++ unsigned int cmd, unsigned long arg)
++{
++ return video_usercopy(inode, file, cmd, arg, mxc_v4l2out_do_ioctl);
++}
++
++/*!
++ * V4L2 interface - mmap function
++ *
++ * @param file structure file *
++ *
++ * @param vma structure vm_area_struct *
++ *
++ * @return status 0 Success, EINTR busy lock error,
++ * ENOBUFS remap_page error
++ */
++static int mxc_v4l2out_mmap(struct file *file, struct vm_area_struct *vma)
++{
++ struct video_device *vdev = video_devdata(file);
++ unsigned long size = vma->vm_end - vma->vm_start;
++ int res = 0;
++ int i;
++ vout_data *vout = video_get_drvdata(vdev);
++
++ dev_dbg(vdev->dev, "pgoff=0x%lx, start=0x%lx, end=0x%lx\n",
++ vma->vm_pgoff, vma->vm_start, vma->vm_end);
++
++ /* make this _really_ smp-safe */
++ if (down_interruptible(&vout->busy_lock))
++ return -EINTR;
++
++ for (i = 0; i < vout->buffer_cnt; i++) {
++ if ((vout->v4l2_bufs[i].m.offset ==
++ (vma->vm_pgoff << PAGE_SHIFT)) &&
++ (vout->v4l2_bufs[i].length >= size)) {
++ vout->v4l2_bufs[i].flags |= V4L2_BUF_FLAG_MAPPED;
++ break;
++ }
++ }
++ if (i == vout->buffer_cnt) {
++ res = -ENOBUFS;
++ goto mxc_mmap_exit;
++ }
++
++ /* make buffers inner write-back, outer write-thru cacheable */
++ vma->vm_page_prot = pgprot_outer_wrthru(vma->vm_page_prot);
++
++ if (remap_pfn_range(vma, vma->vm_start,
++ vma->vm_pgoff, size, vma->vm_page_prot)) {
++ dev_dbg(vdev->dev, "mmap remap_pfn_range failed\n");
++ res = -ENOBUFS;
++ goto mxc_mmap_exit;
++ }
++
++ vma->vm_flags &= ~VM_IO; /* using shared anonymous pages */
++
++ mxc_mmap_exit:
++ up(&vout->busy_lock);
++ return res;
++}
++
++/*!
++ * V4L2 interface - poll function
++ *
++ * @param file structure file *
++ *
++ * @param wait structure poll_table *
++ *
++ * @return status POLLIN | POLLRDNORM
++ */
++static unsigned int mxc_v4l2out_poll(struct file *file, poll_table * wait)
++{
++ struct video_device *dev = video_devdata(file);
++ vout_data *vout = video_get_drvdata(dev);
++
++ wait_queue_head_t *queue = NULL;
++ int res = POLLIN | POLLRDNORM;
++
++ if (down_interruptible(&vout->busy_lock))
++ return -EINTR;
++
++ queue = &vout->v4l_bufq;
++ poll_wait(file, queue, wait);
++
++ up(&vout->busy_lock);
++ return res;
++}
++
++static struct
++file_operations mxc_v4l2out_fops = {
++ .owner = THIS_MODULE,
++ .open = mxc_v4l2out_open,
++ .release = mxc_v4l2out_close,
++ .ioctl = mxc_v4l2out_ioctl,
++ .mmap = mxc_v4l2out_mmap,
++ .poll = mxc_v4l2out_poll,
++};
++
++static struct video_device mxc_v4l2out_template = {
++ .owner = THIS_MODULE,
++ .name = "MXC Video Output",
++ .type = 0,
++ .type2 = V4L2_CAP_VIDEO_OUTPUT | V4L2_CAP_STREAMING,
++ .hardware = 0,
++ .fops = &mxc_v4l2out_fops,
++ .release = video_device_release,
++};
++
++/*!
++ * Probe routine for the framebuffer driver. It is called during the
++ * driver binding process. The following functions are performed in
++ * this routine: Framebuffer initialization, Memory allocation and
++ * mapping, Framebuffer registration, IPU initialization.
++ *
++ * @return Appropriate error code to the kernel common code
++ */
++static int mxc_v4l2out_probe(struct platform_device *pdev)
++{
++ int i;
++ vout_data *vout;
++
++ /*
++ * Allocate sufficient memory for the fb structure
++ */
++ g_vout = vout = kmalloc(sizeof(vout_data), GFP_KERNEL);
++
++ if (!vout)
++ return 0;
++
++ memset(vout, 0, sizeof(vout_data));
++
++ vout->video_dev = video_device_alloc();
++ if (vout->video_dev == NULL)
++ return -1;
++ vout->video_dev->dev = &pdev->dev;
++ vout->video_dev->minor = -1;
++
++ *(vout->video_dev) = mxc_v4l2out_template;
++
++ /* register v4l device */
++ if (video_register_device(vout->video_dev,
++ VFL_TYPE_GRABBER, video_nr) == -1) {
++ dev_dbg(&pdev->dev, "video_register_device failed\n");
++ return 0;
++ }
++ dev_info(&pdev->dev, "Registered device video%d\n",
++ vout->video_dev->minor & 0x1f);
++ vout->video_dev->dev = &pdev->dev;
++
++ video_set_drvdata(vout->video_dev, vout);
++
++ init_MUTEX(&vout->param_lock);
++ init_MUTEX(&vout->busy_lock);
++
++ /* setup outputs and cropping */
++ vout->cur_disp_output = -1;
++ for (i = 0; i < num_registered_fb; i++) {
++ char *idstr = registered_fb[i]->fix.id;
++ if (strncmp(idstr, "DISP", 4) == 0) {
++ int disp_num = idstr[4] - '0';
++ if ((disp_num == 3) &&
++ (strncmp(idstr, "DISP3 BG", 8) == 0)) {
++ disp_num = 4;
++ }
++ vout->crop_bounds[disp_num].left = 0;
++ vout->crop_bounds[disp_num].top = 0;
++ vout->crop_bounds[disp_num].width =
++ registered_fb[i]->var.xres;
++ vout->crop_bounds[disp_num].height =
++ registered_fb[i]->var.yres;
++ vout->output_enabled[disp_num] = true;
++ vout->output_fb_num[disp_num] = i;
++ if (vout->cur_disp_output == -1) {
++ vout->cur_disp_output = disp_num;
++ }
++ }
++
++ }
++ vout->crop_current = vout->crop_bounds[vout->cur_disp_output];
++
++ platform_set_drvdata(pdev, vout);
++
++ return 0;
++}
++
++static int mxc_v4l2out_remove(struct platform_device *pdev)
++{
++ vout_data *vout = platform_get_drvdata(pdev);
++
++ if (vout->video_dev) {
++ if (-1 != vout->video_dev->minor)
++ video_unregister_device(vout->video_dev);
++ else
++ video_device_release(vout->video_dev);
++ vout->video_dev = NULL;
++ }
++
++ platform_set_drvdata(pdev, NULL);
++
++ kfree(vout);
++
++ return 0;
++}
++
++/*!
++ * This structure contains pointers to the power management callback functions.
++ */
++static struct platform_driver mxc_v4l2out_driver = {
++ .driver = {
++ .name = "MXC Video Output",
++ },
++ .probe = mxc_v4l2out_probe,
++ .remove = mxc_v4l2out_remove,
++};
++
++static struct platform_device mxc_v4l2out_device = {
++ .name = "MXC Video Output",
++ .id = 0,
++};
++
++/*!
++ * mxc v4l2 init function
++ *
++ */
++static int mxc_v4l2out_init(void)
++{
++ u8 err = 0;
++
++ err = platform_driver_register(&mxc_v4l2out_driver);
++ if (err == 0) {
++ platform_device_register(&mxc_v4l2out_device);
++ }
++ return err;
++}
++
++/*!
++ * mxc v4l2 cleanup function
++ *
++ */
++static void mxc_v4l2out_clean(void)
++{
++ video_unregister_device(g_vout->video_dev);
++
++ platform_driver_unregister(&mxc_v4l2out_driver);
++ platform_device_unregister(&mxc_v4l2out_device);
++ kfree(g_vout);
++ g_vout = NULL;
++}
++
++module_init(mxc_v4l2out_init);
++module_exit(mxc_v4l2out_clean);
++
++module_param(video_nr, int, 0444);
++MODULE_AUTHOR("Freescale Semiconductor, Inc.");
++MODULE_DESCRIPTION("V4L2-driver for MXC video output");
++MODULE_LICENSE("GPL");
++MODULE_SUPPORTED_DEVICE("video");
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/media/video/mxc/output/mxc_v4l2_output.h linux-2.6.28-karo/drivers/media/video/mxc/output/mxc_v4l2_output.h
+--- linux-2.6.28/drivers/media/video/mxc/output/mxc_v4l2_output.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/media/video/mxc/output/mxc_v4l2_output.h 2009-03-11 13:47:25.000000000 +0100
+@@ -0,0 +1,169 @@
++/*
++ * Copyright 2005-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @defgroup MXC_V4L2_OUTPUT MXC V4L2 Video Output Driver
++ */
++/*!
++ * @file mxc_v4l2_output.h
++ *
++ * @brief MXC V4L2 Video Output Driver Header file
++ *
++ * Video4Linux2 Output Device using MXC IPU Post-processing functionality.
++ *
++ * @ingroup MXC_V4L2_OUTPUT
++ */
++#ifndef __MXC_V4L2_OUTPUT_H__
++#define __MXC_V4L2_OUTPUT_H__
++
++#include <linux/list.h>
++#include <linux/spinlock.h>
++#include <media/v4l2-dev.h>
++
++#ifdef __KERNEL__
++
++#include <mach/ipu.h>
++#include <mach/mxc_v4l2.h>
++
++#define MIN_FRAME_NUM 2
++#define MAX_FRAME_NUM 30
++
++#define MXC_V4L2_OUT_NUM_OUTPUTS 5
++#define MXC_V4L2_OUT_2_SDC 0
++#define MXC_V4L2_OUT_2_ADC 1
++
++struct dma_buf_desc {
++ dma_addr_t dma_addr;
++ void *cpu_addr;
++ size_t size;
++};
++
++typedef enum {
++ BUF_IDLE,
++ BUF_DONE,
++ BUF_CANCELED,
++ BUF_ERROR,
++ BUF_BUSY,
++} v4l_buf_state_t;
++
++typedef struct v4l_queue {
++ struct v4l2_buffer buf;
++ struct list_head head;
++ v4l_buf_state_t state;
++ struct dma_buf_desc dma_desc;
++} v4l_queue_t;
++
++/*!
++ * States for the video stream
++ */
++typedef enum {
++ STATE_STREAM_OFF,
++ STATE_STREAM_ON,
++ STATE_STREAM_PAUSED,
++ STATE_STREAM_STOPPING,
++} v4lout_state;
++
++/*!
++ * States for tearing protection
++ */
++typedef enum {
++ TEARING_PROTECTION_INACTIVE,
++ TEARING_PROTECTION_ACTIVE,
++ TEARING_PROTECTION_UNSUPPORTED
++} v4l_tear_protect;
++
++/*!
++ * common v4l2 driver structure.
++ */
++typedef struct _vout_data {
++ struct video_device *video_dev;
++ /*!
++ * semaphore guard against SMP multithreading
++ */
++#if 1
++ struct semaphore user_lock;
++#else
++ struct mutex user_lock;
++#endif
++ spinlock_t irq_lock;
++
++ struct notifier_block fb_event_notifier;
++
++ /*!
++ * number of process that have device open
++ */
++ int open_count;
++
++ v4l_tear_protect tear_protection;
++
++ /*!
++ * params lock for this camera
++ */
++ struct semaphore param_lock;
++
++ struct timer_list output_timer;
++ unsigned long start_jiffies;
++ u32 frame_count;
++ struct list_head ready_q;
++ struct list_head done_q;
++ struct list_head free_q;
++ struct v4l_queue *active;
++
++ s8 next_rdy_ipu_buf;
++ s8 next_done_ipu_buf;
++ s8 ipu_buf[2];
++ v4lout_state state;
++ int busy;
++
++ int cur_disp_output;
++ int output_fb_num[MXC_V4L2_OUT_NUM_OUTPUTS];
++ int output_enabled[MXC_V4L2_OUT_NUM_OUTPUTS];
++ struct v4l2_framebuffer v4l2_fb;
++#ifdef CONFIG_VIDEO_MXC_IPU_OUTPUT
++ ipu_channel_t display_ch;
++ ipu_channel_t post_proc_ch;
++#endif
++ int buffer_cnt;
++ struct list_head display_q;
++ u32 display_buf_size;
++ dma_addr_t display_bufs[2];
++
++ /*!
++ * Poll wait queue
++ */
++ wait_queue_head_t v4l_bufq;
++
++ /*!
++ * v4l2 format
++ */
++ struct v4l2_format v2f;
++ struct v4l2_mxc_offset offset;
++ struct v4l2_rect crop_rect;
++ enum ipu_rotate_mode rotate;
++
++ /* crop */
++ struct v4l2_rect crop_bounds[MXC_V4L2_OUT_NUM_OUTPUTS];
++ struct v4l2_rect crop_current;
++#ifdef CONFIG_VIDEO_MXC_OUTPUT_FBSYNC
++ int output_fb;
++ int fb_enabled;
++ int pp_ready;
++#endif
++#if 1
++ int queued;
++ struct timeval frame_start;
++#endif
++} vout_data;
++
++#endif
++#endif /* __MXC_V4L2_OUTPUT_H__ */
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/mmc/host/Kconfig linux-2.6.28-karo/drivers/mmc/host/Kconfig
+--- linux-2.6.28/drivers/mmc/host/Kconfig 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/mmc/host/Kconfig 2009-03-11 13:16:24.000000000 +0100
+@@ -135,6 +135,16 @@ config MMC_IMX
+
+ If unsure, say N.
+
++config MMC_MXC
++ tristate "Freescale i.MX2/3 Multimedia Card Interface support"
++ depends on ARCH_MXC
++ help
++ This selects the Freescale i.MX2/3 Multimedia card Interface.
++ If you have a i.MX platform with a Multimedia Card slot,
++ say Y or M here.
++
++ If unsure, say N.
++
+ config MMC_TIFM_SD
+ tristate "TI Flash Media MMC/SD Interface support (EXPERIMENTAL)"
+ depends on EXPERIMENTAL && PCI
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/mmc/host/Makefile linux-2.6.28-karo/drivers/mmc/host/Makefile
+--- linux-2.6.28/drivers/mmc/host/Makefile 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/mmc/host/Makefile 2009-03-11 13:16:24.000000000 +0100
+@@ -9,6 +9,7 @@ endif
+ obj-$(CONFIG_MMC_ARMMMCI) += mmci.o
+ obj-$(CONFIG_MMC_PXA) += pxamci.o
+ obj-$(CONFIG_MMC_IMX) += imxmmc.o
++obj-$(CONFIG_MMC_MXC) += mxcmmc.o
+ obj-$(CONFIG_MMC_SDHCI) += sdhci.o
+ obj-$(CONFIG_MMC_SDHCI_PCI) += sdhci-pci.o
+ obj-$(CONFIG_MMC_RICOH_MMC) += ricoh_mmc.o
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/mmc/host/mxcmmc.c linux-2.6.28-karo/drivers/mmc/host/mxcmmc.c
+--- linux-2.6.28/drivers/mmc/host/mxcmmc.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/mmc/host/mxcmmc.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,893 @@
++/*
++ * linux/drivers/mmc/host/mxcmmc.c - Freescale i.MX MMCI driver
++ *
++ * This is a driver for the SDHC controller found in Freescale MX2/MX3
++ * SoCs. It is basically the same hardware as found on MX1 (imxmmc.c).
++ * Unlike the hardware found on MX1, this hardware just works and does
++ * not need all the quirks found in imxmmc.c, hence the seperate driver.
++ *
++ * Copyright (C) 2008 Sascha Hauer, Pengutronix <s.hauer@pengutronix.de>
++ * Copyright (C) 2006 Pavel Pisa, PiKRON <ppisa@pikron.com>
++ *
++ * derived from pxamci.c by Russell King
++ *
++ * 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/ioport.h>
++#include <linux/platform_device.h>
++#include <linux/interrupt.h>
++#include <linux/irq.h>
++#include <linux/blkdev.h>
++#include <linux/dma-mapping.h>
++#include <linux/mmc/host.h>
++#include <linux/mmc/card.h>
++#include <linux/delay.h>
++#include <linux/clk.h>
++#include <linux/io.h>
++#include <linux/gpio.h>
++
++#include <asm/dma.h>
++#include <asm/irq.h>
++#include <asm/sizes.h>
++#include <mach/mmc.h>
++
++#ifdef CONFIG_ARCH_MX2
++#include <mach/dma-mx1-mx2.h>
++#define HAS_DMA
++#endif
++
++#define DRIVER_NAME "imx-mmc"
++
++#define MMC_REG_STR_STP_CLK 0x00
++#define MMC_REG_STATUS 0x04
++#define MMC_REG_CLK_RATE 0x08
++#define MMC_REG_CMD_DAT_CONT 0x0C
++#define MMC_REG_RES_TO 0x10
++#define MMC_REG_READ_TO 0x14
++#define MMC_REG_BLK_LEN 0x18
++#define MMC_REG_NOB 0x1C
++#define MMC_REG_REV_NO 0x20
++#define MMC_REG_INT_CNTR 0x24
++#define MMC_REG_CMD 0x28
++#define MMC_REG_ARG 0x2C
++#define MMC_REG_RES_FIFO 0x34
++#define MMC_REG_BUFFER_ACCESS 0x38
++
++#define STR_STP_CLK_RESET (1 << 3)
++#define STR_STP_CLK_START_CLK (1 << 1)
++#define STR_STP_CLK_STOP_CLK (1 << 0)
++
++#define STATUS_CARD_INSERTION (1 << 31)
++#define STATUS_CARD_REMOVAL (1 << 30)
++#define STATUS_YBUF_EMPTY (1 << 29)
++#define STATUS_XBUF_EMPTY (1 << 28)
++#define STATUS_YBUF_FULL (1 << 27)
++#define STATUS_XBUF_FULL (1 << 26)
++#define STATUS_BUF_UND_RUN (1 << 25)
++#define STATUS_BUF_OVFL (1 << 24)
++#define STATUS_SDIO_INT_ACTIVE (1 << 14)
++#define STATUS_END_CMD_RESP (1 << 13)
++#define STATUS_WRITE_OP_DONE (1 << 12)
++#define STATUS_DATA_TRANS_DONE (1 << 11)
++#define STATUS_READ_OP_DONE (1 << 11)
++#define STATUS_WR_CRC_ERROR_CODE_MASK (3 << 10)
++#define STATUS_CARD_BUS_CLK_RUN (1 << 8)
++#define STATUS_BUF_READ_RDY (1 << 7)
++#define STATUS_BUF_WRITE_RDY (1 << 6)
++#define STATUS_RESP_CRC_ERR (1 << 5)
++#define STATUS_CRC_READ_ERR (1 << 3)
++#define STATUS_CRC_WRITE_ERR (1 << 2)
++#define STATUS_TIME_OUT_RESP (1 << 1)
++#define STATUS_TIME_OUT_READ (1 << 0)
++#define STATUS_ERR_MASK 0x2f
++
++#define CMD_DAT_CONT_CMD_RESP_LONG_OFF (1 << 12)
++#define CMD_DAT_CONT_STOP_READWAIT (1 << 11)
++#define CMD_DAT_CONT_START_READWAIT (1 << 10)
++#define CMD_DAT_CONT_BUS_WIDTH_4 (2 << 8)
++#define CMD_DAT_CONT_INIT (1 << 7)
++#define CMD_DAT_CONT_WRITE (1 << 4)
++#define CMD_DAT_CONT_DATA_ENABLE (1 << 3)
++#define CMD_DAT_CONT_RESPONSE_48BIT_CRC (1 << 0)
++#define CMD_DAT_CONT_RESPONSE_136BIT (2 << 0)
++#define CMD_DAT_CONT_RESPONSE_48BIT (3 << 0)
++
++#define INT_SDIO_INT_WKP_EN (1 << 18)
++#define INT_CARD_INSERTION_WKP_EN (1 << 17)
++#define INT_CARD_REMOVAL_WKP_EN (1 << 16)
++#define INT_CARD_INSERTION_EN (1 << 15)
++#define INT_CARD_REMOVAL_EN (1 << 14)
++#define INT_SDIO_IRQ_EN (1 << 13)
++#define INT_DAT0_EN (1 << 12)
++#define INT_BUF_READ_EN (1 << 4)
++#define INT_BUF_WRITE_EN (1 << 3)
++#define INT_END_CMD_RES_EN (1 << 2)
++#define INT_WRITE_OP_DONE_EN (1 << 1)
++#define INT_READ_OP_EN (1 << 0)
++
++struct mxcmci_host {
++ struct mmc_host *mmc;
++ struct resource *res;
++ void __iomem *base;
++ int irq;
++ int detect_irq;
++ int dma;
++ int do_dma;
++ unsigned int power_mode;
++ struct imxmmc_platform_data *pdata;
++
++ struct mmc_request *req;
++ struct mmc_command *cmd;
++ struct mmc_data *data;
++
++ unsigned int dma_nents;
++ unsigned int datasize;
++ unsigned int dma_dir;
++
++ u16 rev_no;
++ unsigned int cmdat;
++
++ struct clk *clk;
++
++ int clock;
++
++ struct work_struct datawork;
++};
++
++#ifdef HAS_DMA
++static inline int mxcmci_use_dma(struct mxcmci_host *host)
++{
++ return host->do_dma;
++}
++#else
++static inline int mxcmci_use_dma(struct mxcmci_host *host)
++{
++ return 0;
++}
++#endif
++
++static void mxcmci_softreset(struct mxcmci_host *host)
++{
++ int i;
++
++ /* reset sequence */
++ writew(STR_STP_CLK_RESET, host->base + MMC_REG_STR_STP_CLK);
++ writew(STR_STP_CLK_RESET | STR_STP_CLK_START_CLK,
++ host->base + MMC_REG_STR_STP_CLK);
++
++ for (i = 0; i < 8; i++)
++ writew(STR_STP_CLK_START_CLK, host->base + MMC_REG_STR_STP_CLK);
++
++ writew(0xff, host->base + MMC_REG_RES_TO);
++}
++
++static void mxcmci_setup_data(struct mxcmci_host *host, struct mmc_data *data)
++{
++ unsigned int nob = data->blocks;
++ unsigned int blksz = data->blksz;
++ unsigned int datasize = nob * blksz;
++ struct scatterlist *sg;
++ int i;
++
++ if (data->flags & MMC_DATA_STREAM)
++ nob = 0xffff;
++
++ host->data = data;
++ data->bytes_xfered = 0;
++
++ writew(nob, host->base + MMC_REG_NOB);
++ writew(blksz, host->base + MMC_REG_BLK_LEN);
++ host->datasize = datasize;
++
++#ifdef HAS_DMA
++ for_each_sg(data->sg, sg, data->sg_len, i) {
++ if (sg->offset & 3 || sg->length & 3) {
++ host->do_dma = 0;
++ return;
++ }
++ }
++
++ if (data->flags & MMC_DATA_READ) {
++ host->dma_dir = DMA_FROM_DEVICE;
++ host->dma_nents = dma_map_sg(mmc_dev(host->mmc), data->sg,
++ data->sg_len, host->dma_dir);
++
++ imx_dma_setup_sg(host->dma, data->sg, host->dma_nents, datasize,
++ host->res->start + MMC_REG_BUFFER_ACCESS,
++ DMA_MODE_READ);
++ } else {
++ host->dma_dir = DMA_TO_DEVICE;
++ host->dma_nents = dma_map_sg(mmc_dev(host->mmc), data->sg,
++ data->sg_len, host->dma_dir);
++
++ imx_dma_setup_sg(host->dma, data->sg, host->dma_nents, datasize,
++ host->res->start + MMC_REG_BUFFER_ACCESS,
++ DMA_MODE_WRITE);
++ }
++
++ wmb();
++
++ imx_dma_enable(host->dma);
++#endif /* HAS_DMA */
++}
++
++static int mxcmci_start_cmd(struct mxcmci_host *host, struct mmc_command *cmd,
++ unsigned int cmdat)
++{
++ WARN_ON(host->cmd != NULL);
++ host->cmd = cmd;
++
++ switch (mmc_resp_type(cmd)) {
++ case MMC_RSP_R1: /* short CRC, OPCODE */
++ case MMC_RSP_R1B:/* short CRC, OPCODE, BUSY */
++ cmdat |= CMD_DAT_CONT_RESPONSE_48BIT_CRC;
++ break;
++ case MMC_RSP_R2: /* long 136 bit + CRC */
++ cmdat |= CMD_DAT_CONT_RESPONSE_136BIT;
++ break;
++ case MMC_RSP_R3: /* short */
++ cmdat |= CMD_DAT_CONT_RESPONSE_48BIT;
++ break;
++ case MMC_RSP_NONE:
++ break;
++ default:
++ dev_err(mmc_dev(host->mmc), "unhandled response type 0x%x\n",
++ mmc_resp_type(cmd));
++ cmd->error = -EINVAL;
++ return -EINVAL;
++ }
++
++ if (mxcmci_use_dma(host))
++ writel(INT_READ_OP_EN | INT_WRITE_OP_DONE_EN |
++ INT_END_CMD_RES_EN,
++ host->base + MMC_REG_INT_CNTR);
++ else
++ writel(INT_END_CMD_RES_EN, host->base + MMC_REG_INT_CNTR);
++
++ writew(cmd->opcode, host->base + MMC_REG_CMD);
++ writel(cmd->arg, host->base + MMC_REG_ARG);
++ writew(cmdat, host->base + MMC_REG_CMD_DAT_CONT);
++
++ return 0;
++}
++
++static void mxcmci_finish_request(struct mxcmci_host *host,
++ struct mmc_request *req)
++{
++ writel(0, host->base + MMC_REG_INT_CNTR);
++
++ host->req = NULL;
++ host->cmd = NULL;
++ host->data = NULL;
++
++ mmc_request_done(host->mmc, req);
++}
++
++static int mxcmci_finish_data(struct mxcmci_host *host, unsigned int stat)
++{
++ struct mmc_data *data = host->data;
++ int data_error;
++
++#ifdef HAS_DMA
++ if (mxcmci_use_dma(host)) {
++ imx_dma_disable(host->dma);
++ dma_unmap_sg(mmc_dev(host->mmc), data->sg, host->dma_nents,
++ host->dma_dir);
++ }
++#endif
++
++ if (stat & STATUS_ERR_MASK) {
++ dev_dbg(mmc_dev(host->mmc), "request failed. status: 0x%08x\n",
++ stat);
++ if (stat & STATUS_CRC_READ_ERR) {
++ data->error = -EILSEQ;
++ } else if (stat & STATUS_CRC_WRITE_ERR) {
++ u32 err_code = (stat >> 9) & 0x3;
++ if (err_code == 2) /* No CRC response */
++ data->error = -ETIMEDOUT;
++ else
++ data->error = -EILSEQ;
++ } else if (stat & STATUS_TIME_OUT_READ) {
++ data->error = -ETIMEDOUT;
++ } else {
++ data->error = -EIO;
++ }
++ } else {
++ data->bytes_xfered = host->datasize;
++ }
++
++ data_error = data->error;
++
++ host->data = NULL;
++
++ return data_error;
++}
++
++static void imxmmc_read_response(struct mxcmci_host *host, unsigned int stat)
++{
++ struct mmc_command *cmd = host->cmd;
++ int i;
++ u32 a, b, c;
++
++ if (!cmd)
++ return;
++
++ if (stat & STATUS_TIME_OUT_RESP) {
++ dev_dbg(mmc_dev(host->mmc), "CMD TIMEOUT\n");
++ cmd->error = -ETIMEDOUT;
++ } else if (stat & STATUS_RESP_CRC_ERR && cmd->flags & MMC_RSP_CRC) {
++ dev_dbg(mmc_dev(host->mmc), "cmd crc error\n");
++ cmd->error = -EILSEQ;
++ }
++
++ if (cmd->flags & MMC_RSP_PRESENT) {
++ if (cmd->flags & MMC_RSP_136) {
++ for (i = 0; i < 4; i++) {
++ a = readw(host->base + MMC_REG_RES_FIFO);
++ b = readw(host->base + MMC_REG_RES_FIFO);
++ cmd->resp[i] = a << 16 | b;
++ }
++ } else {
++ a = readw(host->base + MMC_REG_RES_FIFO);
++ b = readw(host->base + MMC_REG_RES_FIFO);
++ c = readw(host->base + MMC_REG_RES_FIFO);
++ cmd->resp[0] = a << 24 | b << 8 | c >> 8;
++ }
++ }
++}
++
++static int mxcmci_poll_status(struct mxcmci_host *host, u32 mask)
++{
++ u32 stat;
++
++ do {
++ stat = readl(host->base + MMC_REG_STATUS);
++ if (stat & STATUS_ERR_MASK)
++ return stat;
++ } while (!(stat & mask));
++
++ return 0;
++}
++
++static int mxcmci_pull(struct mxcmci_host *host, void *_buf, int bytes)
++{
++ unsigned int stat;
++ u32 *buf = _buf;
++
++ while (bytes > 3) {
++ stat = mxcmci_poll_status(host, STATUS_BUF_READ_RDY | STATUS_READ_OP_DONE);
++ if (stat)
++ return stat;
++ *buf++ = readl(host->base + MMC_REG_BUFFER_ACCESS);
++ bytes -= 4;
++ }
++
++ if (bytes) {
++ u8 *b = (u8 *)buf;
++ u32 tmp;
++
++ stat = mxcmci_poll_status(host, STATUS_BUF_READ_RDY | STATUS_READ_OP_DONE);
++ if (stat)
++ return stat;
++ tmp = readl(host->base + MMC_REG_BUFFER_ACCESS);
++ memcpy(b, &tmp, bytes);
++ }
++
++ return 0;
++}
++
++static int mxcmci_push(struct mxcmci_host *host, void *_buf, int bytes)
++{
++ unsigned int stat;
++ u32 *buf = _buf;
++
++ while (bytes > 3) {
++ stat = mxcmci_poll_status(host, STATUS_BUF_WRITE_RDY);
++ if (stat)
++ return stat;
++ writel(*buf++, host->base + MMC_REG_BUFFER_ACCESS);
++ bytes -= 4;
++ }
++
++ if (bytes) {
++ u8 *b = (u8 *)buf;
++ u32 tmp;
++
++ stat = mxcmci_poll_status(host, STATUS_BUF_WRITE_RDY);
++ if (stat)
++ return stat;
++
++ memcpy(&tmp, b, bytes);
++ writel(tmp, host->base + MMC_REG_BUFFER_ACCESS);
++ }
++
++ stat = mxcmci_poll_status(host, STATUS_BUF_WRITE_RDY);
++ if (stat)
++ return stat;
++
++ return 0;
++}
++
++static int mxcmci_transfer_data(struct mxcmci_host *host)
++{
++ struct mmc_data *data = host->req->data;
++ struct scatterlist *sg;
++ unsigned int nob = data->blocks;
++ int stat, i;
++
++ host->datasize = 0;
++
++ if (data->flags & MMC_DATA_STREAM)
++ nob = 0xffff;
++
++ host->data = data;
++ host->datasize = 0;
++
++ if (data->flags & MMC_DATA_READ) {
++ for_each_sg(data->sg, sg, data->sg_len, i) {
++ stat = mxcmci_pull(host, sg_virt(sg), sg->length);
++ if (stat)
++ return stat;
++ host->datasize += sg->length;
++ }
++ } else {
++ for_each_sg(data->sg, sg, data->sg_len, i) {
++ stat = mxcmci_push(host, sg_virt(sg), sg->length);
++ if (stat)
++ return stat;
++ host->datasize += sg->length;
++ }
++ stat = mxcmci_poll_status(host, STATUS_WRITE_OP_DONE);
++ if (stat)
++ return stat;
++ }
++ return 0;
++}
++
++static void mxcmci_datawork(struct work_struct *work)
++{
++ struct mxcmci_host *host = container_of(work, struct mxcmci_host,
++ datawork);
++ int datastat = mxcmci_transfer_data(host);
++ mxcmci_finish_data(host, datastat);
++
++ if (host->req->stop) {
++ if (mxcmci_start_cmd(host, host->req->stop, 0)) {
++ mxcmci_finish_request(host, host->req);
++ return;
++ }
++ } else {
++ mxcmci_finish_request(host, host->req);
++ }
++}
++
++#ifdef HAS_DMA
++static void mxcmci_data_done(struct mxcmci_host *host, unsigned int stat)
++{
++ struct mmc_data *data = host->data;
++ int data_error;
++
++ if (!data)
++ return;
++
++ data_error = mxcmci_finish_data(host, stat);
++
++ imxmmc_read_response(host, stat);
++ host->cmd = NULL;
++
++ if (host->req->stop) {
++ if (mxcmci_start_cmd(host, host->req->stop, 0)) {
++ mxcmci_finish_request(host, host->req);
++ return;
++ }
++ } else {
++ mxcmci_finish_request(host, host->req);
++ }
++}
++#endif /* HAS_DMA */
++
++static void mxcmci_cmd_done(struct mxcmci_host *host, unsigned int stat)
++{
++ imxmmc_read_response(host, stat);
++ host->cmd = NULL;
++
++ if (!host->data && host->req) {
++ mxcmci_finish_request(host, host->req);
++ return;
++ }
++
++ /* For the DMA case the DMA engine handles the data transfer
++ * automatically. For non DMA we have to to it ourselves.
++ * Don't do it in interrupt context though.
++ */
++ if (!mxcmci_use_dma(host) && host->data)
++ schedule_work(&host->datawork);
++
++}
++
++static irqreturn_t mxcmci_irq(int irq, void *devid)
++{
++ struct mxcmci_host *host = devid;
++ u32 stat;
++
++ stat = readl(host->base + MMC_REG_STATUS);
++ writel(stat, host->base + MMC_REG_STATUS);
++
++ dev_dbg(mmc_dev(host->mmc), "%s: 0x%08x\n", __func__, stat);
++
++ if (stat & STATUS_END_CMD_RESP)
++ mxcmci_cmd_done(host, stat);
++#ifdef HAS_DMA
++ if (mxcmci_use_dma(host) &&
++ (stat & (STATUS_DATA_TRANS_DONE | STATUS_WRITE_OP_DONE)))
++ mxcmci_data_done(host, stat);
++#endif
++ return IRQ_HANDLED;
++}
++
++static void mxcmci_request(struct mmc_host *mmc, struct mmc_request *req)
++{
++ struct mxcmci_host *host = mmc_priv(mmc);
++ unsigned int cmdat = host->cmdat;
++
++ WARN_ON(host->req != NULL);
++
++ host->req = req;
++ host->cmdat &= ~CMD_DAT_CONT_INIT;
++#ifdef HAS_DMA
++ host->do_dma = 1;
++#endif
++ if (req->data) {
++ mxcmci_setup_data(host, req->data);
++
++ cmdat |= CMD_DAT_CONT_DATA_ENABLE;
++
++ if (req->data->flags & MMC_DATA_WRITE)
++ cmdat |= CMD_DAT_CONT_WRITE;
++ }
++
++ if (mxcmci_start_cmd(host, req->cmd, cmdat))
++ mxcmci_finish_request(host, req);
++}
++
++static void mxcmci_set_clk_rate(struct mxcmci_host *host, unsigned int clk_ios)
++{
++ unsigned int divider;
++ int prescaler = 0;
++ unsigned int clk_in = clk_get_rate(host->clk);
++
++ while (prescaler <= 0x800) {
++ for (divider = 1; divider <= 0xF; divider++) {
++ int x;
++
++ x = (clk_in / (divider + 1));
++
++ if (prescaler)
++ x /= (prescaler * 2);
++
++ if (x <= clk_ios)
++ break;
++ }
++ if (divider < 0x10)
++ break;
++
++ if (prescaler == 0)
++ prescaler = 1;
++ else
++ prescaler <<= 1;
++ }
++
++ writew((prescaler << 4) | divider, host->base + MMC_REG_CLK_RATE);
++
++ dev_dbg(mmc_dev(host->mmc), "scaler: %d divider: %d in: %d out: %d\n",
++ prescaler, divider, clk_in, clk_ios);
++}
++
++static void mxcmci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios)
++{
++ struct mxcmci_host *host = mmc_priv(mmc);
++#ifdef HAS_DMA
++ unsigned int blen;
++ /*
++ * use burstlen of 64 in 4 bit mode (--> reg value 0)
++ * use burstlen of 16 in 1 bit mode (--> reg value 16)
++ */
++ if (ios->bus_width == MMC_BUS_WIDTH_4)
++ blen = 0;
++ else
++ blen = 16;
++
++ imx_dma_config_burstlen(host->dma, blen);
++#endif
++ if (ios->bus_width == MMC_BUS_WIDTH_4)
++ host->cmdat |= CMD_DAT_CONT_BUS_WIDTH_4;
++ else
++ host->cmdat &= ~CMD_DAT_CONT_BUS_WIDTH_4;
++
++ if (host->power_mode != ios->power_mode) {
++ if (host->pdata->setpower)
++ host->pdata->setpower(mmc_dev(mmc), ios->vdd);
++ host->power_mode = ios->power_mode;
++ if (ios->power_mode == MMC_POWER_ON)
++ host->cmdat |= CMD_DAT_CONT_INIT;
++ }
++
++ if (ios->clock) {
++ mxcmci_set_clk_rate(host, ios->clock);
++ writew(STR_STP_CLK_START_CLK, host->base + MMC_REG_STR_STP_CLK);
++ } else {
++ writew(STR_STP_CLK_STOP_CLK, host->base + MMC_REG_STR_STP_CLK);
++ }
++
++ host->clock = ios->clock;
++}
++
++static irqreturn_t mxcmci_detect_irq(int irq, void *data)
++{
++ struct mmc_host *mmc = data;
++
++ dev_dbg(mmc_dev(mmc), "%s\n", __func__);
++
++ mmc_detect_change(mmc, msecs_to_jiffies(250));
++ return IRQ_HANDLED;
++}
++
++static int mxcmci_get_ro(struct mmc_host *mmc)
++{
++ struct mxcmci_host *host = mmc_priv(mmc);
++
++ if (host->pdata->get_ro)
++ return !!host->pdata->get_ro(mmc_dev(mmc));
++ /*
++ * Board doesn't support read only detection; let the mmc core
++ * decide what to do.
++ */
++ return -ENOSYS;
++}
++
++
++static const struct mmc_host_ops mxcmci_ops = {
++ .request = mxcmci_request,
++ .set_ios = mxcmci_set_ios,
++ .get_ro = mxcmci_get_ro,
++};
++
++static int mxcmci_probe(struct platform_device *pdev)
++{
++ struct mmc_host *mmc;
++ struct mxcmci_host *host = NULL;
++ struct resource *r;
++ int ret = 0, irq;
++
++ printk(KERN_INFO "i.MX SDHC driver\n");
++
++ r = platform_get_resource(pdev, IORESOURCE_MEM, 0);
++ irq = platform_get_irq(pdev, 0);
++ if (!r || irq < 0)
++ return -ENODEV;
++
++ mmc = mmc_alloc_host(sizeof(struct mxcmci_host), &pdev->dev);
++ if (!mmc) {
++ return -ENOMEM;
++ }
++
++ r = request_mem_region(r->start, resource_size(r), pdev->name);
++ if (!r) {
++ ret = -EBUSY;
++ goto out_free;
++ }
++
++ mmc->ops = &mxcmci_ops;
++ mmc->ocr_avail = MMC_VDD_32_33 | MMC_VDD_33_34;
++ mmc->caps = MMC_CAP_4_BIT_DATA;
++
++ /* MMC core transfer sizes tunable parameters */
++ mmc->max_hw_segs = 64;
++ mmc->max_phys_segs = 64;
++ mmc->max_seg_size = 64 * 512; /* default PAGE_CACHE_SIZE */
++ mmc->max_req_size = 64 * 512; /* default PAGE_CACHE_SIZE */
++ mmc->max_blk_size = 2048;
++ mmc->max_blk_count = 65535;
++
++ host = mmc_priv(mmc);
++ host->base = ioremap(r->start, resource_size(r));
++ if (!host->base) {
++ ret = -ENOMEM;
++ goto out_release_mem;
++ }
++
++ host->mmc = mmc;
++ host->pdata = pdev->dev.platform_data;
++ if (!host->pdata) {
++ ret = -ENODEV;
++ dev_err(&pdev->dev, "No platform data provided!\n");
++ goto out_iounmap;
++ }
++
++ host->res = r;
++ host->irq = irq;
++
++ host->clk = clk_get(&pdev->dev, "sdhc_clk");
++ if (IS_ERR(host->clk)) {
++ ret = PTR_ERR(host->clk);
++ goto out_iounmap;
++ }
++ clk_enable(host->clk);
++
++ mxcmci_softreset(host);
++
++ host->rev_no = readw(host->base + MMC_REG_REV_NO);
++ if (host->rev_no != 0x400) {
++ ret = -ENODEV;
++ dev_err(mmc_dev(host->mmc), "wrong rev.no. 0x%08x. aborting.\n",
++ host->rev_no);
++ goto out_clk_put;
++ }
++
++ mmc->f_min = clk_get_rate(host->clk) >> 7;
++ mmc->f_max = clk_get_rate(host->clk) >> 1;
++
++ /* recommended in data sheet */
++ writew(0x2db4, host->base + MMC_REG_READ_TO);
++
++ writel(0, host->base + MMC_REG_INT_CNTR);
++
++#ifdef HAS_DMA
++ host->dma = imx_dma_request_by_prio(DRIVER_NAME, DMA_PRIO_LOW);
++ if (host->dma < 0) {
++ dev_err(mmc_dev(host->mmc), "imx_dma_request_by_prio failed\n");
++ ret = -EBUSY;
++ goto out_clk_put;
++ }
++
++ r = platform_get_resource(pdev, IORESOURCE_DMA, 0);
++ if (!r) {
++ ret = -ENODEV;
++ goto out_free_dma;
++ }
++
++ ret = imx_dma_config_channel(host->dma,
++ IMX_DMA_MEMSIZE_32 | IMX_DMA_TYPE_FIFO,
++ IMX_DMA_MEMSIZE_32 | IMX_DMA_TYPE_LINEAR,
++ r->start, 0);
++ if (ret) {
++ dev_err(mmc_dev(host->mmc), "failed to configure DMA channel\n");
++ goto out_free_dma;
++ }
++#endif
++ INIT_WORK(&host->datawork, mxcmci_datawork);
++
++ ret = request_irq(host->irq, mxcmci_irq, 0, DRIVER_NAME, host);
++ if (ret)
++ goto out_free_dma;
++
++ platform_set_drvdata(pdev, mmc);
++ device_set_wakeup_enable(&pdev->dev, 1);
++
++ if (host->pdata->init) {
++ ret = host->pdata->init(&pdev->dev, mxcmci_detect_irq,
++ host->mmc);
++ if (ret)
++ goto out_free_irq;
++ }
++
++ mmc_add_host(mmc);
++
++ return 0;
++
++out_free_irq:
++ free_irq(host->irq, host);
++out_free_dma:
++#ifdef HAS_DMA
++ imx_dma_free(host->dma);
++#endif
++out_clk_put:
++ clk_disable(host->clk);
++ clk_put(host->clk);
++out_iounmap:
++ iounmap(host->base);
++out_release_mem:
++ release_mem_region(host->res->start, resource_size(host->res));
++out_free:
++ mmc_free_host(mmc);
++ return ret;
++}
++
++static int mxcmci_remove(struct platform_device *pdev)
++{
++ struct mmc_host *mmc = platform_get_drvdata(pdev);
++ struct mxcmci_host *host = mmc_priv(mmc);
++
++ platform_set_drvdata(pdev, NULL);
++
++ mmc_remove_host(mmc);
++
++ if (host->pdata->exit)
++ host->pdata->exit(&pdev->dev, mmc);
++
++ free_irq(host->irq, host);
++ iounmap(host->base);
++#ifdef HAS_DMA
++ imx_dma_free(host->dma);
++#endif
++ clk_disable(host->clk);
++ clk_put(host->clk);
++
++ release_resource(host->res);
++
++ mmc_free_host(mmc);
++
++ return 0;
++}
++
++#ifdef CONFIG_PM
++static int mxcmci_suspend(struct platform_device *pdev, pm_message_t state)
++{
++ struct mmc_host *mmc = platform_get_drvdata(pdev);
++ struct mxcmci_host *host = mmc_priv(mmc);
++ int ret = 0;
++
++ if (host->pdata->suspend)
++ ret = host->pdata->suspend(&pdev->dev, state);
++ if (ret == 0) {
++ ret = mmc_suspend_host(mmc, state);
++ if (ret)
++ if (host->pdata->resume)
++ host->pdata->resume(&pdev->dev);
++ }
++
++ return ret;
++}
++
++static int mxcmci_resume(struct platform_device *pdev)
++{
++ struct mmc_host *mmc = platform_get_drvdata(pdev);
++ struct mxcmci_host *host = mmc_priv(mmc);
++ int ret;
++
++ ret = mmc_resume_host(mmc);
++ if (ret == 0)
++ if (host->pdata->resume)
++ ret = host->pdata->resume(&pdev->dev);
++
++ return ret;
++}
++#else
++#define mxcmci_suspend NULL
++#define mxcmci_resume NULL
++#endif /* CONFIG_PM */
++
++static struct platform_driver mxcmci_driver = {
++ .probe = mxcmci_probe,
++ .remove = mxcmci_remove,
++ .suspend = mxcmci_suspend,
++ .resume = mxcmci_resume,
++ .driver = {
++ .name = DRIVER_NAME,
++ .owner = THIS_MODULE,
++ }
++};
++
++static int __init mxcmci_init(void)
++{
++ return platform_driver_register(&mxcmci_driver);
++}
++
++static void __exit mxcmci_exit(void)
++{
++ platform_driver_unregister(&mxcmci_driver);
++}
++
++module_init(mxcmci_init);
++module_exit(mxcmci_exit);
++
++MODULE_DESCRIPTION("i.MX Multimedia Card Interface Driver");
++MODULE_AUTHOR("Sascha Hauer, Pengutronix");
++MODULE_LICENSE("GPL");
++MODULE_ALIAS("platform:imx-mmc");
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/mtd/mtdblock.c linux-2.6.28-karo/drivers/mtd/mtdblock.c
+--- linux-2.6.28/drivers/mtd/mtdblock.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/mtd/mtdblock.c 2009-03-11 13:16:24.000000000 +0100
+@@ -102,9 +102,9 @@ static int write_cached_data (struct mtd
+ if (mtdblk->cache_state != STATE_DIRTY)
+ return 0;
+
+- DEBUG(MTD_DEBUG_LEVEL2, "mtdblock: writing cached data for \"%s\" "
+- "at 0x%lx, size 0x%x\n", mtd->name,
+- mtdblk->cache_offset, mtdblk->cache_size);
++ MTD_DEBUG(MTD_DEBUG_LEVEL2, "mtdblock: writing cached data for \"%s\" "
++ "at 0x%lx, size 0x%x\n", mtd->name,
++ mtdblk->cache_offset, mtdblk->cache_size);
+
+ ret = erase_write (mtd, mtdblk->cache_offset,
+ mtdblk->cache_size, mtdblk->cache_data);
+@@ -131,8 +131,8 @@ static int do_cached_write (struct mtdbl
+ size_t retlen;
+ int ret;
+
+- DEBUG(MTD_DEBUG_LEVEL2, "mtdblock: write on \"%s\" at 0x%lx, size 0x%x\n",
+- mtd->name, pos, len);
++ MTD_DEBUG(MTD_DEBUG_LEVEL2, "mtdblock: write on \"%s\" at 0x%lx, size 0x%x\n",
++ mtd->name, pos, len);
+
+ if (!sect_size)
+ return mtd->write(mtd, pos, len, &retlen, buf);
+@@ -201,8 +201,8 @@ static int do_cached_read (struct mtdblk
+ size_t retlen;
+ int ret;
+
+- DEBUG(MTD_DEBUG_LEVEL2, "mtdblock: read on \"%s\" at 0x%lx, size 0x%x\n",
+- mtd->name, pos, len);
++ MTD_DEBUG(MTD_DEBUG_LEVEL2, "mtdblock: read on \"%s\" at 0x%lx, size 0x%x\n",
++ mtd->name, pos, len);
+
+ if (!sect_size)
+ return mtd->read(mtd, pos, len, &retlen, buf);
+@@ -268,7 +268,7 @@ static int mtdblock_open(struct mtd_blkt
+ struct mtd_info *mtd = mbd->mtd;
+ int dev = mbd->devnum;
+
+- DEBUG(MTD_DEBUG_LEVEL1,"mtdblock_open\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL1,"mtdblock_open\n");
+
+ if (mtdblks[dev]) {
+ mtdblks[dev]->count++;
+@@ -292,7 +292,7 @@ static int mtdblock_open(struct mtd_blkt
+
+ mtdblks[dev] = mtdblk;
+
+- DEBUG(MTD_DEBUG_LEVEL1, "ok\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL1, "ok\n");
+
+ return 0;
+ }
+@@ -302,7 +302,7 @@ static int mtdblock_release(struct mtd_b
+ int dev = mbd->devnum;
+ struct mtdblk_dev *mtdblk = mtdblks[dev];
+
+- DEBUG(MTD_DEBUG_LEVEL1, "mtdblock_release\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL1, "mtdblock_release\n");
+
+ mutex_lock(&mtdblk->cache_mutex);
+ write_cached_data(mtdblk);
+@@ -316,7 +316,7 @@ static int mtdblock_release(struct mtd_b
+ vfree(mtdblk->cache_data);
+ kfree(mtdblk);
+ }
+- DEBUG(MTD_DEBUG_LEVEL1, "ok\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL1, "ok\n");
+
+ return 0;
+ }
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/mtd/mtdchar.c linux-2.6.28-karo/drivers/mtd/mtdchar.c
+--- linux-2.6.28/drivers/mtd/mtdchar.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/mtd/mtdchar.c 2009-03-11 13:16:24.000000000 +0100
+@@ -90,7 +90,7 @@ static int mtd_open(struct inode *inode,
+ struct mtd_info *mtd;
+ struct mtd_file_info *mfi;
+
+- DEBUG(MTD_DEBUG_LEVEL0, "MTD_open\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL0, "MTD_open\n");
+
+ if (devnum >= MAX_MTD_DEVICES)
+ return -ENODEV;
+@@ -141,7 +141,7 @@ static int mtd_close(struct inode *inode
+ struct mtd_file_info *mfi = file->private_data;
+ struct mtd_info *mtd = mfi->mtd;
+
+- DEBUG(MTD_DEBUG_LEVEL0, "MTD_close\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL0, "MTD_close\n");
+
+ /* Only sync if opened RW */
+ if ((file->f_mode & FMODE_WRITE) && mtd->sync)
+@@ -169,7 +169,7 @@ static ssize_t mtd_read(struct file *fil
+ int len;
+ char *kbuf;
+
+- DEBUG(MTD_DEBUG_LEVEL0,"MTD_read\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL0,"MTD_read\n");
+
+ if (*ppos + count > mtd->size)
+ count = mtd->size - *ppos;
+@@ -262,7 +262,7 @@ static ssize_t mtd_write(struct file *fi
+ int ret=0;
+ int len;
+
+- DEBUG(MTD_DEBUG_LEVEL0,"MTD_write\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL0,"MTD_write\n");
+
+ if (*ppos == mtd->size)
+ return -ENOSPC;
+@@ -388,7 +388,7 @@ static int mtd_ioctl(struct inode *inode
+ u_long size;
+ struct mtd_info_user info;
+
+- DEBUG(MTD_DEBUG_LEVEL0, "MTD_ioctl\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL0, "MTD_ioctl\n");
+
+ size = (cmd & IOCSIZE_MASK) >> IOCSIZE_SHIFT;
+ if (cmd & IOC_IN) {
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/mtd/mtdcore.c linux-2.6.28-karo/drivers/mtd/mtdcore.c
+--- linux-2.6.28/drivers/mtd/mtdcore.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/mtd/mtdcore.c 2009-03-11 13:16:24.000000000 +0100
+@@ -67,7 +67,7 @@ int add_mtd_device(struct mtd_info *mtd)
+ mtd->name);
+ }
+
+- DEBUG(0, "mtd: Giving out device %d to %s\n",i, mtd->name);
++ MTD_DEBUG(0, "mtd: Giving out device %d to %s\n",i, mtd->name);
+ /* No need to get a refcount on the module containing
+ the notifier, since we hold the mtd_table_mutex */
+ list_for_each_entry(not, &mtd_notifiers, list)
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/mtd/mtdsuper.c linux-2.6.28-karo/drivers/mtd/mtdsuper.c
+--- linux-2.6.28/drivers/mtd/mtdsuper.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/mtd/mtdsuper.c 2009-03-11 13:16:24.000000000 +0100
+@@ -23,13 +23,13 @@ static int get_sb_mtd_compare(struct sup
+ struct mtd_info *mtd = _mtd;
+
+ if (sb->s_mtd == mtd) {
+- DEBUG(2, "MTDSB: Match on device %d (\"%s\")\n",
+- mtd->index, mtd->name);
++ MTD_DEBUG(2, "MTDSB: Match on device %d (\"%s\")\n",
++ mtd->index, mtd->name);
+ return 1;
+ }
+
+- DEBUG(2, "MTDSB: No match, device %d (\"%s\"), device %d (\"%s\")\n",
+- sb->s_mtd->index, sb->s_mtd->name, mtd->index, mtd->name);
++ MTD_DEBUG(2, "MTDSB: No match, device %d (\"%s\"), device %d (\"%s\")\n",
++ sb->s_mtd->index, sb->s_mtd->name, mtd->index, mtd->name);
+ return 0;
+ }
+
+@@ -67,8 +67,8 @@ static int get_sb_mtd_aux(struct file_sy
+ goto already_mounted;
+
+ /* fresh new superblock */
+- DEBUG(1, "MTDSB: New superblock for device %d (\"%s\")\n",
+- mtd->index, mtd->name);
++ MTD_DEBUG(1, "MTDSB: New superblock for device %d (\"%s\")\n",
++ mtd->index, mtd->name);
+
+ sb->s_flags = flags;
+
+@@ -85,8 +85,8 @@ static int get_sb_mtd_aux(struct file_sy
+
+ /* new mountpoint for an already mounted superblock */
+ already_mounted:
+- DEBUG(1, "MTDSB: Device %d (\"%s\") is already mounted\n",
+- mtd->index, mtd->name);
++ MTD_DEBUG(1, "MTDSB: Device %d (\"%s\") is already mounted\n",
++ mtd->index, mtd->name);
+ ret = simple_set_mnt(mnt, sb);
+ goto out_put;
+
+@@ -109,7 +109,7 @@ static int get_sb_mtd_nr(struct file_sys
+
+ mtd = get_mtd_device(NULL, mtdnr);
+ if (IS_ERR(mtd)) {
+- DEBUG(0, "MTDSB: Device #%u doesn't appear to exist\n", mtdnr);
++ MTD_DEBUG(0, "MTDSB: Device #%u doesn't appear to exist\n", mtdnr);
+ return PTR_ERR(mtd);
+ }
+
+@@ -134,7 +134,7 @@ int get_sb_mtd(struct file_system_type *
+ if (!dev_name)
+ return -EINVAL;
+
+- DEBUG(2, "MTDSB: dev_name \"%s\"\n", dev_name);
++ MTD_DEBUG(2, "MTDSB: dev_name \"%s\"\n", dev_name);
+
+ /* the preferred way of mounting in future; especially when
+ * CONFIG_BLOCK=n - we specify the underlying MTD device by number or
+@@ -145,8 +145,8 @@ int get_sb_mtd(struct file_system_type *
+ struct mtd_info *mtd;
+
+ /* mount by MTD device name */
+- DEBUG(1, "MTDSB: mtd:%%s, name \"%s\"\n",
+- dev_name + 4);
++ MTD_DEBUG(1, "MTDSB: mtd:%%s, name \"%s\"\n",
++ dev_name + 4);
+
+ for (mtdnr = 0; mtdnr < MAX_MTD_DEVICES; mtdnr++) {
+ mtd = get_mtd_device(NULL, mtdnr);
+@@ -172,8 +172,8 @@ int get_sb_mtd(struct file_system_type *
+ mtdnr = simple_strtoul(dev_name + 3, &endptr, 0);
+ if (!*endptr) {
+ /* It was a valid number */
+- DEBUG(1, "MTDSB: mtd%%d, mtdnr %d\n",
+- mtdnr);
++ MTD_DEBUG(1, "MTDSB: mtd%%d, mtdnr %d\n",
++ mtdnr);
+ return get_sb_mtd_nr(fs_type, flags,
+ dev_name, data,
+ mtdnr, fill_super, mnt);
+@@ -188,10 +188,10 @@ int get_sb_mtd(struct file_system_type *
+ bdev = lookup_bdev(dev_name);
+ if (IS_ERR(bdev)) {
+ ret = PTR_ERR(bdev);
+- DEBUG(1, "MTDSB: lookup_bdev() returned %d\n", ret);
++ MTD_DEBUG(1, "MTDSB: lookup_bdev() returned %d\n", ret);
+ return ret;
+ }
+- DEBUG(1, "MTDSB: lookup_bdev() returned 0\n");
++ MTD_DEBUG(1, "MTDSB: lookup_bdev() returned 0\n");
+
+ ret = -EINVAL;
+
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/mtd/nand/Kconfig linux-2.6.28-karo/drivers/mtd/nand/Kconfig
+--- linux-2.6.28/drivers/mtd/nand/Kconfig 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/mtd/nand/Kconfig 2009-03-12 15:04:48.000000000 +0100
+@@ -358,6 +358,10 @@ config MTD_NAND_TMIO
+ Support for NAND flash connected to a Toshiba Mobile IO
+ Controller in some PDAs, including the Sharp SL6000x.
+
++config MTD_NAND_MXC_FLASH_BBT
++ bool "Support a flash based bad block table"
++ depends on MTD_NAND_MXC
++
+ config MTD_NAND_NANDSIM
+ tristate "Support for NAND Flash Simulator"
+ depends on MTD_PARTITIONS
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/mtd/nand/mxc_nand.c linux-2.6.28-karo/drivers/mtd/nand/mxc_nand.c
+--- linux-2.6.28/drivers/mtd/nand/mxc_nand.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/mtd/nand/mxc_nand.c 2009-03-16 11:04:21.000000000 +0100
+@@ -141,6 +141,31 @@ static struct nand_ecclayout nand_hw_ecc
+ .oobfree = {{0, 6}, {12, 4}, }
+ };
+
++#ifdef CONFIG_MTD_NAND_MXC_FLASH_BBT
++static u8 bbt_pattern[] = {'B', 'b', 't', '0' };
++static u8 mirror_pattern[] = {'1', 't', 'b', 'B' };
++
++static struct nand_bbt_descr bbt_main_descr = {
++ .options = (NAND_BBT_LASTBLOCK | NAND_BBT_WRITE |
++ NAND_BBT_2BIT | NAND_BBT_VERSION | NAND_BBT_PERCHIP),
++ .offs = 12,
++ .len = 4,
++ .veroffs = 11,
++ .maxblocks = 4,
++ .pattern = bbt_pattern,
++};
++
++static struct nand_bbt_descr bbt_mirror_descr = {
++ .options = (NAND_BBT_LASTBLOCK | NAND_BBT_WRITE |
++ NAND_BBT_2BIT | NAND_BBT_VERSION | NAND_BBT_PERCHIP),
++ .offs = 12,
++ .len = 4,
++ .veroffs = 11,
++ .maxblocks = 4,
++ .pattern = mirror_pattern,
++};
++#endif
++
+ #ifdef CONFIG_MTD_PARTITIONS
+ static const char *part_probes[] = { "RedBoot", "cmdlinepart", NULL };
+ #endif
+@@ -193,8 +218,8 @@ static void wait_op_done(struct mxc_nand
+ udelay(1);
+ }
+ if (max_retries <= 0)
+- DEBUG(MTD_DEBUG_LEVEL0, "%s(%d): INT not set\n",
+- __func__, param);
++ MTD_DEBUG(MTD_DEBUG_LEVEL0, "%s(%d): INT not set\n",
++ __func__, param);
+ }
+ }
+
+@@ -202,7 +227,7 @@ static void wait_op_done(struct mxc_nand
+ * waits for completion. */
+ static void send_cmd(struct mxc_nand_host *host, uint16_t cmd, int useirq)
+ {
+- DEBUG(MTD_DEBUG_LEVEL3, "send_cmd(host, 0x%x, %d)\n", cmd, useirq);
++ MTD_DEBUG(MTD_DEBUG_LEVEL3, "send_cmd(host, 0x%x, %d)\n", cmd, useirq);
+
+ writew(cmd, host->regs + NFC_FLASH_CMD);
+ writew(NFC_CMD, host->regs + NFC_CONFIG2);
+@@ -216,7 +241,7 @@ static void send_cmd(struct mxc_nand_hos
+ * a NAND command. */
+ static void send_addr(struct mxc_nand_host *host, uint16_t addr, int islast)
+ {
+- DEBUG(MTD_DEBUG_LEVEL3, "send_addr(host, 0x%x %d)\n", addr, islast);
++ MTD_DEBUG(MTD_DEBUG_LEVEL3, "send_addr(host, 0x%x %d)\n", addr, islast);
+
+ writew(addr, host->regs + NFC_FLASH_ADDR);
+ writew(NFC_ADDR, host->regs + NFC_CONFIG2);
+@@ -230,7 +255,7 @@ static void send_addr(struct mxc_nand_ho
+ static void send_prog_page(struct mxc_nand_host *host, uint8_t buf_id,
+ int spare_only)
+ {
+- DEBUG(MTD_DEBUG_LEVEL3, "send_prog_page (%d)\n", spare_only);
++ MTD_DEBUG(MTD_DEBUG_LEVEL3, "send_prog_page (%d)\n", spare_only);
+
+ /* NANDFC buffer 0 is used for page read/write */
+ writew(buf_id, host->regs + NFC_BUF_ADDR);
+@@ -256,7 +281,7 @@ static void send_prog_page(struct mxc_na
+ static void send_read_page(struct mxc_nand_host *host, uint8_t buf_id,
+ int spare_only)
+ {
+- DEBUG(MTD_DEBUG_LEVEL3, "send_read_page (%d)\n", spare_only);
++ MTD_DEBUG(MTD_DEBUG_LEVEL3, "send_read_page (%d)\n", spare_only);
+
+ /* NANDFC buffer 0 is used for page read/write */
+ writew(buf_id, host->regs + NFC_BUF_ADDR);
+@@ -372,8 +397,8 @@ static int mxc_nand_correct_data(struct
+ uint16_t ecc_status = readw(host->regs + NFC_ECC_STATUS_RESULT);
+
+ if (((ecc_status & 0x3) == 2) || ((ecc_status >> 2) == 2)) {
+- DEBUG(MTD_DEBUG_LEVEL0,
+- "MXC_NAND: HWECC uncorrectable 2-bit ECC error\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL0,
++ "MXC_NAND: HWECC uncorrectable 2-bit ECC error\n");
+ return -1;
+ }
+
+@@ -427,8 +452,8 @@ static uint16_t mxc_nand_read_word(struc
+ uint16_t col, rd_word, ret;
+ uint16_t __iomem *p;
+
+- DEBUG(MTD_DEBUG_LEVEL3,
+- "mxc_nand_read_word(col = %d)\n", host->col_addr);
++ MTD_DEBUG(MTD_DEBUG_LEVEL3,
++ "mxc_nand_read_word(col = %d)\n", host->col_addr);
+
+ col = host->col_addr;
+ /* Adjust saved column address */
+@@ -465,9 +490,9 @@ static void mxc_nand_write_buf(struct mt
+ struct mxc_nand_host *host = nand_chip->priv;
+ int n, col, i = 0;
+
+- DEBUG(MTD_DEBUG_LEVEL3,
+- "mxc_nand_write_buf(col = %d, len = %d)\n", host->col_addr,
+- len);
++ MTD_DEBUG(MTD_DEBUG_LEVEL3,
++ "mxc_nand_write_buf(col = %d, len = %d)\n",
++ host->col_addr, len);
+
+ col = host->col_addr;
+
+@@ -478,8 +503,8 @@ static void mxc_nand_write_buf(struct mt
+ n = mtd->writesize + mtd->oobsize - col;
+ n = min(len, n);
+
+- DEBUG(MTD_DEBUG_LEVEL3,
+- "%s:%d: col = %d, n = %d\n", __func__, __LINE__, col, n);
++ MTD_DEBUG(MTD_DEBUG_LEVEL3,
++ "%s:%d: col = %d, n = %d\n", __func__, __LINE__, col, n);
+
+ while (n) {
+ void __iomem *p;
+@@ -490,8 +515,8 @@ static void mxc_nand_write_buf(struct mt
+ p = host->regs + SPARE_AREA0 -
+ mtd->writesize + (col & ~3);
+
+- DEBUG(MTD_DEBUG_LEVEL3, "%s:%d: p = %p\n", __func__,
+- __LINE__, p);
++ MTD_DEBUG(MTD_DEBUG_LEVEL3, "%s:%d: p = %p\n", __func__,
++ __LINE__, p);
+
+ if (((col | (int)&buf[i]) & 3) || n < 16) {
+ uint32_t data = 0;
+@@ -539,9 +564,9 @@ static void mxc_nand_write_buf(struct mt
+
+ m = min(n, m) & ~3;
+
+- DEBUG(MTD_DEBUG_LEVEL3,
+- "%s:%d: n = %d, m = %d, i = %d, col = %d\n",
+- __func__, __LINE__, n, m, i, col);
++ MTD_DEBUG(MTD_DEBUG_LEVEL3,
++ "%s:%d: n = %d, m = %d, i = %d, col = %d\n",
++ __func__, __LINE__, n, m, i, col);
+
+ memcpy(p, &buf[i], m);
+ col += m;
+@@ -563,8 +588,9 @@ static void mxc_nand_read_buf(struct mtd
+ struct mxc_nand_host *host = nand_chip->priv;
+ int n, col, i = 0;
+
+- DEBUG(MTD_DEBUG_LEVEL3,
+- "mxc_nand_read_buf(col = %d, len = %d)\n", host->col_addr, len);
++ MTD_DEBUG(MTD_DEBUG_LEVEL3,
++ "mxc_nand_read_buf(col = %d, len = %d)\n",
++ host->col_addr, len);
+
+ col = host->col_addr;
+
+@@ -649,8 +675,9 @@ static void mxc_nand_select_chip(struct
+
+ #ifdef CONFIG_MTD_NAND_MXC_FORCE_CE
+ if (chip > 0) {
+- DEBUG(MTD_DEBUG_LEVEL0,
+- "ERROR: Illegal chip select (chip = %d)\n", chip);
++ MTD_DEBUG(MTD_DEBUG_LEVEL0,
++ "ERROR: Illegal chip select (chip = %d)\n",
++ chip);
+ return;
+ }
+
+@@ -694,9 +721,9 @@ static void mxc_nand_command(struct mtd_
+ struct mxc_nand_host *host = nand_chip->priv;
+ int useirq = true;
+
+- DEBUG(MTD_DEBUG_LEVEL3,
+- "mxc_nand_command (cmd = 0x%x, col = 0x%x, page = 0x%x)\n",
+- command, column, page_addr);
++ MTD_DEBUG(MTD_DEBUG_LEVEL3,
++ "mxc_nand_command (cmd = 0x%x, col = 0x%x, page = 0x%x)\n",
++ command, column, page_addr);
+
+ /* Reset command state information */
+ host->status_request = false;
+@@ -831,6 +858,7 @@ static void mxc_nand_command(struct mtd_
+ break;
+
+ case NAND_CMD_READID:
++ host->col_addr = 0;
+ send_read_id(host);
+ break;
+
+@@ -879,10 +907,17 @@ static int __init mxcnd_probe(struct pla
+ this->write_buf = mxc_nand_write_buf;
+ this->read_buf = mxc_nand_read_buf;
+ this->verify_buf = mxc_nand_verify_buf;
++#ifdef CONFIG_MTD_NAND_MXC_FLASH_BBT
++ this->bbt_td = &bbt_main_descr;
++ this->bbt_md = &bbt_mirror_descr;
++ this->options |= NAND_USE_FLASH_BBT;
++#endif
+
+ host->clk = clk_get(&pdev->dev, "nfc_clk");
+- if (IS_ERR(host->clk))
++ if (IS_ERR(host->clk)) {
++ err = PTR_ERR(host->clk);
+ goto eclk;
++ }
+
+ clk_enable(host->clk);
+ host->clk_act = 1;
+@@ -895,7 +930,7 @@ static int __init mxcnd_probe(struct pla
+
+ host->regs = ioremap(res->start, res->end - res->start + 1);
+ if (!host->regs) {
+- err = -EIO;
++ err = -ENOMEM;
+ goto eres;
+ }
+
+@@ -952,13 +987,17 @@ static int __init mxcnd_probe(struct pla
+ this->ecc.layout = &nand_hw_eccoob_16;
+ }
+
+- host->pagesize_2k = 0;
+-
+ /* Scan to find existence of the device */
+- if (nand_scan(mtd, 1)) {
+- DEBUG(MTD_DEBUG_LEVEL0,
+- "MXC_ND: Unable to find any NAND device.\n");
+- err = -ENXIO;
++ err = nand_scan_ident(mtd, 1);
++ if (err) {
++ MTD_DEBUG(MTD_DEBUG_LEVEL0,
++ "MXC_ND: Unable to find any NAND device.\n");
++ goto escan;
++ }
++ /* this is required before completing the scan */
++ host->pagesize_2k = mtd->writesize == 2048;
++ err = nand_scan_tail(mtd);
++ if (err) {
+ goto escan;
+ }
+
+@@ -1010,43 +1049,51 @@ static int __devexit mxcnd_remove(struct
+ #ifdef CONFIG_PM
+ static int mxcnd_suspend(struct platform_device *pdev, pm_message_t state)
+ {
+- struct mtd_info *info = platform_get_drvdata(pdev);
++ struct mtd_info *mtd = platform_get_drvdata(pdev);
++ struct nand_chip *nand_chip = mtd->priv;
++ struct mxc_nand_host *host = nand_chip->priv;
+ int ret = 0;
+
+- DEBUG(MTD_DEBUG_LEVEL0, "MXC_ND : NAND suspend\n");
+- if (info)
+- ret = info->suspend(info);
++ MTD_DEBUG(MTD_DEBUG_LEVEL0, "MXC_ND : NAND suspend\n");
++ if (mtd)
++ ret = mtd->suspend(mtd);
+
++ if (host->clk_act) {
+ /* Disable the NFC clock */
+- clk_disable(nfc_clk); /* FIXME */
++ clk_disable(host->clk);
++ }
+
+ return ret;
+ }
+
+ static int mxcnd_resume(struct platform_device *pdev)
+ {
+- struct mtd_info *info = platform_get_drvdata(pdev);
++ struct mtd_info *mtd = platform_get_drvdata(pdev);
++ struct nand_chip *nand_chip = mtd->priv;
++ struct mxc_nand_host *host = nand_chip->priv;
+ int ret = 0;
+
+- DEBUG(MTD_DEBUG_LEVEL0, "MXC_ND : NAND resume\n");
+- /* Enable the NFC clock */
+- clk_enable(nfc_clk); /* FIXME */
++ MTD_DEBUG(MTD_DEBUG_LEVEL0, "MXC_ND : NAND resume\n");
+
+- if (info)
+- info->resume(info);
++ if (host->clk_act) {
++ /* Enable the NFC clock */
++ clk_enable(host->clk);
++ }
++ if (mtd)
++ mtd->resume(mtd);
+
+ return ret;
+ }
+
+ #else
+-# define mxcnd_suspend NULL
+-# define mxcnd_resume NULL
++#define mxcnd_suspend NULL
++#define mxcnd_resume NULL
+ #endif /* CONFIG_PM */
+
+ static struct platform_driver mxcnd_driver = {
+ .driver = {
+ .name = DRIVER_NAME,
+- },
++ },
+ .remove = __exit_p(mxcnd_remove),
+ .suspend = mxcnd_suspend,
+ .resume = mxcnd_resume,
+@@ -1054,13 +1101,15 @@ static struct platform_driver mxcnd_driv
+
+ static int __init mxc_nd_init(void)
+ {
++ int ret;
++
+ /* Register the device driver structure. */
+ pr_info("MXC MTD nand Driver\n");
+- if (platform_driver_probe(&mxcnd_driver, mxcnd_probe) != 0) {
++ ret = platform_driver_probe(&mxcnd_driver, mxcnd_probe);
++ if (ret != 0) {
+ printk(KERN_ERR "Driver register failed for mxcnd_driver\n");
+- return -ENODEV;
+ }
+- return 0;
++ return ret;
+ }
+
+ static void __exit mxc_nd_cleanup(void)
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/mtd/nand/nand_base.c linux-2.6.28-karo/drivers/mtd/nand/nand_base.c
+--- linux-2.6.28/drivers/mtd/nand/nand_base.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/mtd/nand/nand_base.c 2009-03-11 13:16:24.000000000 +0100
+@@ -1356,8 +1356,8 @@ static int nand_do_read_oob(struct mtd_i
+ int len;
+ uint8_t *buf = ops->oobbuf;
+
+- DEBUG(MTD_DEBUG_LEVEL3, "nand_read_oob: from = 0x%08Lx, len = %i\n",
+- (unsigned long long)from, readlen);
++ MTD_DEBUG(MTD_DEBUG_LEVEL3, "nand_read_oob: from = 0x%08Lx, len = %i\n",
++ (unsigned long long)from, readlen);
+
+ if (ops->mode == MTD_OOB_AUTO)
+ len = chip->ecc.layout->oobavail;
+@@ -1365,8 +1365,8 @@ static int nand_do_read_oob(struct mtd_i
+ len = mtd->oobsize;
+
+ if (unlikely(ops->ooboffs >= len)) {
+- DEBUG(MTD_DEBUG_LEVEL0, "nand_read_oob: "
+- "Attempt to start read outside oob\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL0,
++ "nand_read_oob: Attempt to start read outside oob\n");
+ return -EINVAL;
+ }
+
+@@ -1374,8 +1374,8 @@ static int nand_do_read_oob(struct mtd_i
+ if (unlikely(from >= mtd->size ||
+ ops->ooboffs + readlen > ((mtd->size >> chip->page_shift) -
+ (from >> chip->page_shift)) * len)) {
+- DEBUG(MTD_DEBUG_LEVEL0, "nand_read_oob: "
+- "Attempt read beyond end of device\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL0,
++ "nand_read_oob: Attempt read beyond end of device\n");
+ return -EINVAL;
+ }
+
+@@ -1449,8 +1449,8 @@ static int nand_read_oob(struct mtd_info
+
+ /* Do not allow reads past end of device */
+ if (ops->datbuf && (from + ops->len) > mtd->size) {
+- DEBUG(MTD_DEBUG_LEVEL0, "nand_read_oob: "
+- "Attempt read beyond end of device\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL0,
++ "nand_read_oob: Attempt read beyond end of device\n");
+ return -EINVAL;
+ }
+
+@@ -1847,8 +1847,8 @@ static int nand_do_write_oob(struct mtd_
+ int chipnr, page, status, len;
+ struct nand_chip *chip = mtd->priv;
+
+- DEBUG(MTD_DEBUG_LEVEL3, "nand_write_oob: to = 0x%08x, len = %i\n",
+- (unsigned int)to, (int)ops->ooblen);
++ MTD_DEBUG(MTD_DEBUG_LEVEL3, "nand_write_oob: to = 0x%08x, len = %i\n",
++ (unsigned int)to, (int)ops->ooblen);
+
+ if (ops->mode == MTD_OOB_AUTO)
+ len = chip->ecc.layout->oobavail;
+@@ -1857,14 +1857,14 @@ static int nand_do_write_oob(struct mtd_
+
+ /* Do not allow write past end of page */
+ if ((ops->ooboffs + ops->ooblen) > len) {
+- DEBUG(MTD_DEBUG_LEVEL0, "nand_write_oob: "
+- "Attempt to write past end of page\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL0,
++ "nand_write_oob: Attempt to write past end of page\n");
+ return -EINVAL;
+ }
+
+ if (unlikely(ops->ooboffs >= len)) {
+- DEBUG(MTD_DEBUG_LEVEL0, "nand_read_oob: "
+- "Attempt to start write outside oob\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL0,
++ "nand_read_oob: Attempt to start write outside oob\n");
+ return -EINVAL;
+ }
+
+@@ -1873,8 +1873,8 @@ static int nand_do_write_oob(struct mtd_
+ ops->ooboffs + ops->ooblen >
+ ((mtd->size >> chip->page_shift) -
+ (to >> chip->page_shift)) * len)) {
+- DEBUG(MTD_DEBUG_LEVEL0, "nand_read_oob: "
+- "Attempt write beyond end of device\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL0,
++ "nand_read_oob: Attempt write beyond end of device\n");
+ return -EINVAL;
+ }
+
+@@ -1929,8 +1929,8 @@ static int nand_write_oob(struct mtd_inf
+
+ /* Do not allow writes past end of device */
+ if (ops->datbuf && (to + ops->len) > mtd->size) {
+- DEBUG(MTD_DEBUG_LEVEL0, "nand_read_oob: "
+- "Attempt read beyond end of device\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL0,
++ "nand_read_oob: Attempt read beyond end of device\n");
+ return -EINVAL;
+ }
+
+@@ -2019,26 +2019,24 @@ int nand_erase_nand(struct mtd_info *mtd
+ int rewrite_bbt[NAND_MAX_CHIPS]={0};
+ unsigned int bbt_masked_page = 0xffffffff;
+
+- DEBUG(MTD_DEBUG_LEVEL3, "nand_erase: start = 0x%08x, len = %i\n",
+- (unsigned int)instr->addr, (unsigned int)instr->len);
++ MTD_DEBUG(MTD_DEBUG_LEVEL3, "nand_erase: start = 0x%08x, len = %i\n",
++ (unsigned int)instr->addr, (unsigned int)instr->len);
+
+ /* Start address must align on block boundary */
+ if (instr->addr & ((1 << chip->phys_erase_shift) - 1)) {
+- DEBUG(MTD_DEBUG_LEVEL0, "nand_erase: Unaligned address\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL0, "nand_erase: Unaligned address\n");
+ return -EINVAL;
+ }
+
+ /* Length must align on block boundary */
+ if (instr->len & ((1 << chip->phys_erase_shift) - 1)) {
+- DEBUG(MTD_DEBUG_LEVEL0, "nand_erase: "
+- "Length not block aligned\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL0, "nand_erase: Length not block aligned\n");
+ return -EINVAL;
+ }
+
+ /* Do not allow erase past end of device */
+ if ((instr->len + instr->addr) > mtd->size) {
+- DEBUG(MTD_DEBUG_LEVEL0, "nand_erase: "
+- "Erase past end of device\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL0, "nand_erase: Erase past end of device\n");
+ return -EINVAL;
+ }
+
+@@ -2059,8 +2057,7 @@ int nand_erase_nand(struct mtd_info *mtd
+
+ /* Check, if it is write protected */
+ if (nand_check_wp(mtd)) {
+- DEBUG(MTD_DEBUG_LEVEL0, "nand_erase: "
+- "Device is write protected!!!\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL0, "nand_erase: Device is write protected!!!\n");
+ instr->state = MTD_ERASE_FAILED;
+ goto erase_exit;
+ }
+@@ -2113,8 +2110,9 @@ int nand_erase_nand(struct mtd_info *mtd
+
+ /* See if block erase succeeded */
+ if (status & NAND_STATUS_FAIL) {
+- DEBUG(MTD_DEBUG_LEVEL0, "nand_erase: "
+- "Failed erase, page 0x%08x\n", page);
++ MTD_DEBUG(MTD_DEBUG_LEVEL0,
++ "nand_erase: Failed erase, page 0x%08x\n",
++ page);
+ instr->state = MTD_ERASE_FAILED;
+ instr->fail_addr = (page << chip->page_shift);
+ goto erase_exit;
+@@ -2172,9 +2170,10 @@ int nand_erase_nand(struct mtd_info *mtd
+ if (!rewrite_bbt[chipnr])
+ continue;
+ /* update the BBT for chip */
+- DEBUG(MTD_DEBUG_LEVEL0, "nand_erase_nand: nand_update_bbt "
+- "(%d:0x%0x 0x%0x)\n", chipnr, rewrite_bbt[chipnr],
+- chip->bbt_td->pages[chipnr]);
++ MTD_DEBUG(MTD_DEBUG_LEVEL0,
++ "nand_erase_nand: nand_update_bbt (%d:0x%0x 0x%0x)\n",
++ chipnr, rewrite_bbt[chipnr],
++ chip->bbt_td->pages[chipnr]);
+ nand_update_bbt(mtd, rewrite_bbt[chipnr]);
+ }
+
+@@ -2192,7 +2191,7 @@ static void nand_sync(struct mtd_info *m
+ {
+ struct nand_chip *chip = mtd->priv;
+
+- DEBUG(MTD_DEBUG_LEVEL3, "nand_sync: called\n");
++ MTD_DEBUG(MTD_DEBUG_LEVEL3, "nand_sync: called\n");
+
+ /* Grab the lock and see if the device is available */
+ nand_get_device(chip, mtd, FL_SYNCING);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/mtd/nand/nand_bbt.c linux-2.6.28-karo/drivers/mtd/nand/nand_bbt.c
+--- linux-2.6.28/drivers/mtd/nand/nand_bbt.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/mtd/nand/nand_bbt.c 2009-03-11 13:16:24.000000000 +0100
+@@ -1201,8 +1201,9 @@ int nand_isbad_bbt(struct mtd_info *mtd,
+ block = (int)(offs >> (this->bbt_erase_shift - 1));
+ res = (this->bbt[block >> 3] >> (block & 0x06)) & 0x03;
+
+- DEBUG(MTD_DEBUG_LEVEL2, "nand_isbad_bbt(): bbt info for offs 0x%08x: (block %d) 0x%02x\n",
+- (unsigned int)offs, block >> 1, res);
++ MTD_DEBUG(MTD_DEBUG_LEVEL2,
++ "nand_isbad_bbt(): bbt info for offs 0x%08x: (block %d) 0x%02x\n",
++ (unsigned int)offs, block >> 1, res);
+
+ switch ((int)res) {
+ case 0x00:
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/mtd/nand/nand_ecc.c linux-2.6.28-karo/drivers/mtd/nand/nand_ecc.c
+--- linux-2.6.28/drivers/mtd/nand/nand_ecc.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/mtd/nand/nand_ecc.c 2009-03-11 13:16:24.000000000 +0100
+@@ -492,7 +492,6 @@ int nand_correct_data(struct mtd_info *m
+ if ((bitsperbyte[b0] + bitsperbyte[b1] + bitsperbyte[b2]) == 1)
+ return 1; /* error in ecc data; no action needed */
+
+- printk(KERN_ERR "uncorrectable error : ");
+ return -1;
+ }
+ EXPORT_SYMBOL(nand_correct_data);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/mxc/Kconfig linux-2.6.28-karo/drivers/mxc/Kconfig
+--- linux-2.6.28/drivers/mxc/Kconfig 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/mxc/Kconfig 2009-03-11 13:39:19.000000000 +0100
+@@ -0,0 +1,27 @@
++# drivers/video/mxc/Kconfig
++
++#if ARCH_MXC
++
++menuconfig DRIVERS_MXC
++ bool "MXC support drivers"
++ depends on ARCH_MXC
++
++menu "MXC VPU(Video Processing Unit) support"
++
++config MXC_VPU
++ tristate "Support for MXC VPU(Video Processing Unit)"
++ depends on DRIVERS_MXC
++ depends on MACH_MX27
++ help
++ The VPU codec device provides codec function for H.264/MPEG4/H.263
++
++config MXC_VPU_DEBUG
++ bool "MXC VPU debugging"
++ depends on MXC_VPU
++ help
++ This is an option for the developers; most people should
++ say N here. This enables MXC VPU driver debugging.
++
++endmenu
++
++#endif
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/mxc/Makefile linux-2.6.28-karo/drivers/mxc/Makefile
+--- linux-2.6.28/drivers/mxc/Makefile 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/mxc/Makefile 2009-03-11 13:40:14.000000000 +0100
+@@ -0,0 +1,11 @@
++#
++# Makefile for the VPU drivers.
++#
++
++ifeq ($(CONFIG_MXC_VPU_DEBUG),y)
++ EXTRA_CFLAGS += -DDEBUG
++endif
++
++vpu-objs := mxc_vpu.o
++obj-$(CONFIG_MXC_VPU) += vpu.o
++
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/mxc/mxc_vpu.c linux-2.6.28-karo/drivers/mxc/mxc_vpu.c
+--- linux-2.6.28/drivers/mxc/mxc_vpu.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/mxc/mxc_vpu.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,540 @@
++/*
++ * Copyright 2006-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++
++/*!
++ * @file mxc_vpu.c
++ *
++ * @brief VPU system initialization and file operation implementation
++ *
++ * @ingroup VPU
++ */
++
++#include <linux/kernel.h>
++#include <linux/fs.h>
++#include <linux/mm.h>
++#include <linux/interrupt.h>
++#include <linux/autoconf.h>
++#include <linux/ioport.h>
++#include <linux/stat.h>
++#include <linux/platform_device.h>
++#include <linux/kdev_t.h>
++#include <linux/dma-mapping.h>
++#include <linux/wait.h>
++#include <linux/list.h>
++#include <linux/clk.h>
++#include <linux/io.h>
++
++#include <asm/uaccess.h>
++#include <asm/sizes.h>
++#include <asm/dma-mapping.h>
++#include <asm/atomic.h>
++
++#include <mach/hardware.h>
++
++#include "mxc_vpu.h"
++
++
++#define BIT_INT_CLEAR 0x00C
++#define BIT_INT_STATUS 0x010
++#define BIT_INT_ENABLE 0x170
++
++typedef struct vpu_t {
++ struct fasync_struct *async_queue;
++ struct device *dev;
++} vpu_t;
++
++/* To track the allocated memory buffer */
++typedef struct memalloc_record {
++ struct list_head list;
++ vpu_mem_desc mem;
++} memalloc_record;
++
++static DEFINE_MUTEX(vpu_list_lock);
++static DEFINE_MUTEX(vpu_mutex);
++static LIST_HEAD(head);
++
++static int vpu_major = 0;
++static struct class *vpu_class;
++static struct vpu_t vpu_data;
++static int open_count = 0;
++static struct clk *vpu_clk;
++
++/* implement the blocking ioctl */
++static int codec_done = 0;
++static wait_queue_head_t vpu_queue;
++static int wait_intr_cnt = 0;
++
++/*!
++ * Private function to free buffers
++ * @return status 0 success.
++ */
++static int vpu_free_buffers(vpu_t *vpu_data)
++{
++ struct memalloc_record *rec, *n;
++ vpu_mem_desc mem;
++
++ mutex_lock(&vpu_list_lock);
++ list_for_each_entry_safe(rec, n, &head, list) {
++ mem = rec->mem;
++ if (mem.cpu_addr != 0) {
++ dma_free_coherent(vpu_data->dev, PAGE_ALIGN(mem.size),
++ mem.cpu_addr, mem.phy_addr);
++ pr_debug("[FREE] freed paddr=0x%08X\n", mem.phy_addr);
++
++ /* delete from list */
++ list_del(&rec->list);
++ kfree(rec);
++ }
++ }
++ mutex_unlock(&vpu_list_lock);
++
++ return 0;
++}
++
++/*!
++ * @brief vpu interrupt handler
++ */
++static irqreturn_t vpu_irq_handler(int irq, void *dev_id)
++{
++ struct vpu_t *dev = dev_id;
++
++ //(void)__raw_readl(IO_ADDRESS(VPU_BASE_ADDR + BIT_INT_STATUS));
++ __raw_writel(0x1, IO_ADDRESS(VPU_BASE_ADDR + BIT_INT_CLEAR));
++ if (dev->async_queue)
++ kill_fasync(&dev->async_queue, SIGIO, POLL_IN);
++
++ BUG_ON(codec_done < 0);
++ codec_done++;
++ WARN_ON(codec_done > 1);
++ wake_up_interruptible(&vpu_queue);
++
++ return IRQ_HANDLED;
++}
++
++/*!
++ * @brief vpu hardware enable function
++ *
++ * @return 0 on success or negative error code on error
++ */
++static int vpu_hardware_enable(void)
++{
++ u32 tmp;
++
++ clk_enable(vpu_clk);
++
++ /* Let userspace access the codec engine through the
++ * peripheral access register.
++ */
++#if 0
++ tmp = __raw_readl(IO_ADDRESS(AIPI_BASE_ADDR + 0x00020008));
++ __raw_writel(tmp & ~(1 << 3), IO_ADDRESS(AIPI_BASE_ADDR + 0x00020008));
++#endif
++ codec_done = 0;
++
++ return 0;
++}
++
++/*!
++ * @brief vpu hardware disable function
++ *
++ * @return 0 on success or negative error code on error
++ */
++static int vpu_hardware_disable(void)
++{
++ clk_disable(vpu_clk);
++ return 0;
++
++}
++
++/*!
++ * @brief open function for vpu file operation
++ *
++ * @return 0 on success or negative error code on error
++ */
++static int vpu_open(struct inode *inode, struct file *filp)
++{
++ int ret = 0;
++
++ mutex_lock(&vpu_mutex);
++ if (open_count == 0) {
++ open_count++;
++ filp->private_data = &vpu_data;
++ vpu_hardware_enable();
++ } else {
++ printk(KERN_ERR "VPU has already been opened.\n");
++ ret = -EBUSY;
++ }
++ mutex_unlock(&vpu_mutex);
++
++ return ret;
++}
++
++/*!
++ * @brief IO ctrl function for vpu file operation
++ * @param cmd IO ctrl command
++ * @return 0 on success or negative error code on error
++ */
++static int vpu_ioctl(struct inode *inode, struct file *filp, u_int cmd,
++ u_long arg)
++{
++ int ret = 0;
++ vpu_t *vpu_data = filp->private_data;
++
++ switch (cmd) {
++ case VPU_IOC_PHYMEM_ALLOC:
++ {
++ struct memalloc_record *rec;
++
++ rec = kzalloc(sizeof(*rec), GFP_KERNEL);
++ if (!rec)
++ return -ENOMEM;
++
++ ret = copy_from_user(&rec->mem, (const void __user *)arg,
++ sizeof(vpu_mem_desc));
++ if (ret) {
++ kfree(rec);
++ return -EFAULT;
++ }
++
++ pr_debug("[ALLOC] mem alloc size = 0x%x\n",
++ rec->mem.size);
++ rec->mem.cpu_addr = dma_alloc_coherent(vpu_data->dev,
++ PAGE_ALIGN(rec->mem.size),
++ &rec->mem.phy_addr,
++ GFP_DMA | GFP_KERNEL);
++ pr_debug("[ALLOC] mem alloc cpu_addr = 0x%p\n",
++ rec->mem.cpu_addr);
++ if (rec->mem.cpu_addr == NULL) {
++ kfree(rec);
++ printk(KERN_ERR
++ "Physical memory allocation error!\n");
++ ret = -ENOMEM;
++ break;
++ }
++ ret = copy_to_user((void __user *)arg, &rec->mem,
++ sizeof(vpu_mem_desc));
++ if (ret) {
++ kfree(rec);
++ ret = -EFAULT;
++ break;
++ }
++
++ mutex_lock(&vpu_list_lock);
++ list_add(&rec->list, &head);
++ mutex_unlock(&vpu_list_lock);
++
++ break;
++ }
++ case VPU_IOC_PHYMEM_FREE:
++ {
++ struct memalloc_record *rec, *n;
++ vpu_mem_desc vpu_mem;
++ int found = 0;
++
++ ret = copy_from_user(&vpu_mem, (const void __user *)arg,
++ sizeof(vpu_mem_desc));
++ if (ret)
++ return -EFAULT;
++
++ pr_debug("[FREE] mem freed cpu_addr = 0x%p\n",
++ vpu_mem.cpu_addr);
++ mutex_lock(&vpu_list_lock);
++ list_for_each_entry_safe(rec, n, &head, list) {
++ if (rec->mem.cpu_addr == vpu_mem.cpu_addr) {
++ /* delete from list */
++ list_del(&rec->list);
++ kfree(rec);
++ found = 1;
++ break;
++ }
++ }
++
++ if (vpu_mem.cpu_addr != NULL) {
++ dma_free_coherent(vpu_data->dev,
++ PAGE_ALIGN(vpu_mem.size),
++ vpu_mem.cpu_addr,
++ vpu_mem.phy_addr);
++ }
++ BUG_ON(!found);
++ mutex_unlock(&vpu_list_lock);
++
++ break;
++ }
++ case VPU_IOC_WAIT4INT:
++ {
++ unsigned long flags;
++ u_long timeout = arg;
++
++ ret = wait_event_interruptible_timeout(vpu_queue,
++ codec_done,
++ msecs_to_jiffies(timeout));
++ local_irq_save(flags);
++ if (codec_done) {
++ codec_done--;
++ BUG_ON(codec_done < 0);
++ if (codec_done > 0) {
++ printk(KERN_DEBUG "%s: WAIT4INT: codec_done=%d\n",
++ __FUNCTION__, codec_done);
++ }
++ ret = 0;
++ } else if (signal_pending(current)) {
++ if (wait_intr_cnt == 0) {
++ printk(KERN_WARNING "wait for VPU interrupt canceled\n");
++ }
++ wait_intr_cnt++;
++ } else if (ret == 0) {
++ if (timeout != 0) {
++ printk(KERN_WARNING "VPU blocking: timeout.\n");
++ }
++ ret = (filp->f_flags & O_NONBLOCK) ? -EAGAIN : -ETIME;
++ }
++ local_irq_restore(flags);
++ break;
++ }
++/* RMW: this is not portable across platforms */
++#ifdef CONFIG_MACH_MX27
++ /* set/clear LHD (Latency Hiding Disable) bit in ESDCFG0 reg.
++ Tends to fix MPEG4 issue on MX27 TO2 */
++ case VPU_IOC_LHD:
++ {
++ u_int disable = (u_int)arg;
++ u_int reg;
++ void __iomem *reg_addr;
++
++ reg_addr = IO_ADDRESS(SDRAMC_BASE_ADDR + 0x10);
++ reg = __raw_readl(reg_addr);
++ pr_debug("ESDCFG0: [ 0x%08x ]\n", reg);
++
++ if (disable == 0) {
++ __raw_writel(reg & ~0x00000020, reg_addr);
++ pr_debug("Latency Hiding Disable\n");
++ } else {
++ __raw_writel(reg | 0x00000020, reg_addr);
++ pr_debug("Latency Hiding Enable\n");
++ }
++
++ pr_debug("ESDCFG0: [ 0x%08x ]\n",
++ __raw_readl(reg_addr));
++
++ break;
++ }
++#endif
++ case VPU_IOC_VL2CC_FLUSH:
++ break;
++ case VPU_IOC_REG_DUMP:
++ break;
++ case VPU_IOC_PHYMEM_DUMP:
++ break;
++ default:
++ printk(KERN_ERR "No such IOCTL, cmd is %d\n", cmd);
++ ret = -EINVAL;
++ }
++ return ret;
++}
++
++/*!
++ * @brief Release function for vpu file operation
++ * @return 0 on success or negative error code on error
++ */
++static int vpu_release(struct inode *inode, struct file *filp)
++{
++ int ret = 0;
++ struct vpu_t *vpu_data = filp->private_data;
++
++ mutex_lock(&vpu_mutex);
++ if (open_count == 1) {
++ open_count--;
++ vpu_free_buffers(vpu_data);
++ vpu_hardware_disable();
++ } else {
++ ret = -EINVAL;
++ }
++ mutex_unlock(&vpu_mutex);
++ return ret;
++}
++
++/*!
++ * @brief fasync function for vpu file operation
++ * @return 0 on success or negative error code on error
++ */
++static int vpu_fasync(int fd, struct file *filp, int mode)
++{
++ struct vpu_t *vpu_data = filp->private_data;
++ return fasync_helper(fd, filp, mode, &vpu_data->async_queue);
++}
++
++/*!
++ * @brief memory map function of harware registers for vpu file operation
++ * @return 0 on success or negative error code on error
++ */
++static int vpu_map_hwregs(struct file *fp, struct vm_area_struct *vm)
++{
++ unsigned long pfn;
++
++ vm->vm_flags |= VM_IO | VM_RESERVED;
++ vm->vm_page_prot = pgprot_noncached(vm->vm_page_prot);
++ pfn = VPU_BASE_ADDR >> PAGE_SHIFT;
++ pr_debug("size=0x%08lx, page no.=0x%08lx\n",
++ vm->vm_end - vm->vm_start, pfn);
++ return remap_pfn_range(vm, vm->vm_start, pfn, vm->vm_end - vm->vm_start,
++ vm->vm_page_prot) ? -EAGAIN : 0;
++}
++
++/*!
++ * @brief memory map function of memory for vpu file operation
++ * @return 0 on success or negative error code on error
++ */
++static int vpu_map_mem(struct file *fp, struct vm_area_struct *vm)
++{
++ int request_size = vm->vm_end - vm->vm_start;
++
++ pr_debug(" start=0x%08lx, pgoff=0x%08lx, size=0x%08x\n",
++ vm->vm_start, vm->vm_pgoff,
++ request_size);
++
++ vm->vm_flags |= VM_IO | VM_RESERVED;
++ vm->vm_page_prot = pgprot_noncached(vm->vm_page_prot);
++
++ return remap_pfn_range(vm, vm->vm_start, vm->vm_pgoff,
++ request_size, vm->vm_page_prot) ? -EAGAIN : 0;
++
++}
++
++/*!
++ * @brief memory map interface for vpu file operation
++ * @return 0 on success or negative error code on error
++ */
++static int vpu_mmap(struct file *fp, struct vm_area_struct *vm)
++{
++ if (vm->vm_pgoff)
++ return vpu_map_mem(fp, vm);
++ else
++ return vpu_map_hwregs(fp, vm);
++}
++
++struct file_operations vpu_fops = {
++ .owner = THIS_MODULE,
++ .open = vpu_open,
++ .ioctl = vpu_ioctl,
++ .release = vpu_release,
++ .fasync = vpu_fasync,
++ .mmap = vpu_mmap,
++};
++
++/*!
++ * This function is called by the driver framework to initialize the vpu device.
++ * @param dev The device structure for the vpu passed in by the framework.
++ * @return 0 on success or negative error code on error
++ */
++static int vpu_dev_probe(struct platform_device *pdev)
++{
++ int err = 0;
++ struct device *temp_device;
++
++ init_waitqueue_head(&vpu_queue);
++
++ vpu_major = register_chrdev(vpu_major, "mxc_vpu", &vpu_fops);
++ if (vpu_major < 0) {
++ printk(KERN_ERR "vpu: unable to get a major for VPU\n");
++ err = vpu_major;
++ goto error;
++ }
++
++ vpu_class = class_create(THIS_MODULE, "mxc_vpu");
++ if (IS_ERR(vpu_class)) {
++ err = PTR_ERR(vpu_class);
++ goto err_out_chrdev;
++ }
++
++ temp_device = device_create(vpu_class, NULL,
++ MKDEV(vpu_major, 0), NULL, "mxc_vpu");
++ if (IS_ERR(temp_device)) {
++ err = PTR_ERR(temp_device);
++ goto err_out_class;
++ }
++
++ vpu_data.dev = &pdev->dev;
++ platform_set_drvdata(pdev, &vpu_data);
++
++ vpu_clk = clk_get(&pdev->dev, "vpu_clk");
++ if (IS_ERR(vpu_clk)) {
++ err = PTR_ERR(vpu_clk);
++ goto err_out_class;
++ }
++
++ err = request_irq(MXC_INT_VPU, vpu_irq_handler, 0, "VPU_CODEC_IRQ",
++ &vpu_data);
++ if (err)
++ goto err_out_class;
++
++ printk(KERN_INFO "VPU initialized\n");
++ goto out;
++
++err_out_class:
++ device_destroy(vpu_class, MKDEV(vpu_major, 0));
++ class_destroy(vpu_class);
++err_out_chrdev:
++ unregister_chrdev(vpu_major, "mxc_vpu");
++error:
++out:
++ return err;
++}
++
++static int __devexit vpu_dev_remove(struct platform_device *pdev)
++{
++ vpu_t *vpu_data = platform_get_drvdata(pdev);
++
++ free_irq(MXC_INT_VPU, vpu_data);
++ if (vpu_major > 0) {
++ device_destroy(vpu_class, MKDEV(vpu_major, 0));
++ class_destroy(vpu_class);
++ unregister_chrdev(vpu_major, "mxc_vpu");
++ vpu_major = 0;
++ }
++
++ clk_put(vpu_clk);
++ return 0;
++}
++
++/*! Driver definition
++ *
++ */
++static struct platform_driver mxcvpu_driver = {
++ .driver = {
++ .name = "mxc_vpu",
++ },
++ .probe = vpu_dev_probe,
++ .remove = __devexit_p(vpu_dev_remove),
++};
++
++static int __init vpu_init(void)
++{
++ int ret = platform_driver_register(&mxcvpu_driver);
++
++ printk(KERN_DEBUG "%s: platform_driver_register returned %d\n", __FUNCTION__, ret);
++
++ return ret;
++}
++
++static void __exit vpu_exit(void)
++{
++ platform_driver_unregister(&mxcvpu_driver);
++ return;
++}
++
++MODULE_AUTHOR("Freescale Semiconductor, Inc.");
++MODULE_DESCRIPTION("Linux VPU driver for Freescale i.MX27");
++MODULE_LICENSE("GPL");
++
++module_init(vpu_init);
++module_exit(vpu_exit);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/mxc/mxc_vpu.h linux-2.6.28-karo/drivers/mxc/mxc_vpu.h
+--- linux-2.6.28/drivers/mxc/mxc_vpu.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/mxc/mxc_vpu.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,22 @@
++typedef struct vpu_mem_desc {
++ size_t size;
++ dma_addr_t phy_addr;
++ void *cpu_addr; /* cpu address to free the dma mem */
++ void *virt_uaddr; /* virtual user space address */
++} vpu_mem_desc;
++
++#define VPU_IOC_MAGIC 'V'
++
++#define VPU_IOC_PHYMEM_ALLOC _IO(VPU_IOC_MAGIC, 0)
++#define VPU_IOC_PHYMEM_FREE _IO(VPU_IOC_MAGIC, 1)
++#define VPU_IOC_WAIT4INT _IO(VPU_IOC_MAGIC, 2)
++#define VPU_IOC_PHYMEM_DUMP _IO(VPU_IOC_MAGIC, 3)
++#define VPU_IOC_REG_DUMP _IO(VPU_IOC_MAGIC, 4)
++#define VPU_IOC_LHD _IO(VPU_IOC_MAGIC, 5)
++#define VPU_IOC_VL2CC_FLUSH _IO(VPU_IOC_MAGIC, 6)
++
++int vl2cc_init(u32 vl2cc_hw_base);
++void vl2cc_enable(void);
++void vl2cc_flush(void);
++void vl2cc_disable(void);
++void vl2cc_cleanup(void);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/net/Kconfig linux-2.6.28-karo/drivers/net/Kconfig
+--- linux-2.6.28/drivers/net/Kconfig 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/net/Kconfig 2009-03-11 13:16:24.000000000 +0100
+@@ -1810,11 +1810,11 @@ config 68360_ENET
+ the Motorola 68360 processor.
+
+ config FEC
+- bool "FEC ethernet controller (of ColdFire CPUs)"
+- depends on M523x || M527x || M5272 || M528x || M520x
++ tristate "FEC ethernet controller"
++ depends on M523x || M527x || M5272 || M528x || M520x || MACH_MX27
+ help
+ Say Y here if you want to use the built-in 10/100 Fast ethernet
+- controller on some Motorola ColdFire processors.
++ controller on some Motorola/Freescale processors.
+
+ config FEC2
+ bool "Second FEC ethernet controller (on some ColdFire CPUs)"
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/net/fec.c linux-2.6.28-karo/drivers/net/fec.c
+--- linux-2.6.28/drivers/net/fec.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/net/fec.c 2009-03-11 13:16:24.000000000 +0100
+@@ -2,6 +2,12 @@
+ * Fast Ethernet Controller (FEC) driver for Motorola MPC8xx.
+ * Copyright (c) 1997 Dan Malek (dmalek@jlc.net)
+ *
++ * This version of the driver is specific to the FADS implementation,
++ * since the board contains control registers external to the processor
++ * for the control of the LevelOne LXT970 transceiver. The MPC860T manual
++ * describes connections using the internal parallel port I/O, which
++ * is basically all of Port D.
++ *
+ * Right now, I am very wasteful with the buffers. I allocate memory
+ * pages and then divide them into 2K frame buffers. This way I know I
+ * have buffers large enough to hold one frame within one buffer descriptor.
+@@ -18,48 +24,74 @@
+ * Bug fixes and cleanup by Philippe De Muyter (phdm@macqel.be)
+ * Copyright (c) 2004-2006 Macq Electronique SA.
+ */
++/*
++ * Copyright 2006-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
+
+ #include <linux/module.h>
+ #include <linux/kernel.h>
+-#include <linux/string.h>
+-#include <linux/ptrace.h>
++#include <linux/resource.h>
+ #include <linux/errno.h>
+ #include <linux/ioport.h>
+ #include <linux/slab.h>
+ #include <linux/interrupt.h>
+-#include <linux/pci.h>
+-#include <linux/init.h>
++#include <linux/wait.h>
+ #include <linux/delay.h>
++#include <linux/platform_device.h>
+ #include <linux/netdevice.h>
+ #include <linux/etherdevice.h>
+ #include <linux/skbuff.h>
+ #include <linux/spinlock.h>
+-#include <linux/workqueue.h>
+-#include <linux/bitops.h>
++#include <linux/dma-mapping.h>
++#include <linux/clk.h>
++#include <linux/fec_enet.h>
++#include <linux/phy.h>
+
+ #include <asm/irq.h>
+-#include <asm/uaccess.h>
+ #include <asm/io.h>
+-#include <asm/pgtable.h>
+-#include <asm/cacheflush.h>
+
++#define DRV_NAME "fec_enet"
++
++#ifdef DEBUG
++static int debug = 1;
++#define dbg_lvl(n) ((n) < debug)
++module_param(debug, int, S_IRUGO | S_IWUSR);
++
++#define DBG(lvl, fmt...) do { if (dbg_lvl(lvl)) printk(KERN_DEBUG fmt); } while (0)
++#else
++static int debug;
++#define dbg_lvl(n) 0
++module_param(debug, int, 0);
++
++#define DBG(lvl, fmt...) do { } while (0)
++#endif
++
++#if defined(CONFIG_M523x) || defined(CONFIG_M527x) || \
++ defined(CONFIG_M5272) || defined(CONFIG_M528x) || \
++ defined(CONFIG_M520x) || defined(CONFIG_M532x)
+ #include <asm/coldfire.h>
+ #include <asm/mcfsim.h>
+ #include "fec.h"
+-
+-#if defined(CONFIG_FEC2)
+-#define FEC_MAX_PORTS 2
++#define FEC_ALIGNMENT (0x03) /*FEC needs 4bytes alignment*/
++#elif defined(CONFIG_ARCH_MXC)
++#include <mach/hardware.h>
++#include <mach/iim.h>
++#include "fec.h"
++#define FEC_ALIGNMENT (0x0F) /*FEC needs 128bits(32bytes) alignment*/
+ #else
+-#define FEC_MAX_PORTS 1
++#include <asm/8xx_immap.h>
++#include <asm/mpc8xx.h>
++#include "commproc.h"
++#define FEC_ALIGNMENT (0x03) /*FEC needs 4bytes alignment */
+ #endif
+
+-#if defined(CONFIG_M5272)
+-#define HAVE_mii_link_interrupt
+-#endif
++#define FEC_ADDR_ALIGNMENT(x) ((unsigned char *)(((unsigned long)(x) + (FEC_ALIGNMENT)) & (~FEC_ALIGNMENT)))
+
++#if 0
+ /*
+ * Define the fixed address of the FEC hardware.
+ */
++/* USE resources provided by platform_device! */
+ static unsigned int fec_hw[] = {
+ #if defined(CONFIG_M5272)
+ (MCF_MBAR + 0x840),
+@@ -72,23 +104,26 @@ static unsigned int fec_hw[] = {
+ (MCF_MBAR+0x30000),
+ #elif defined(CONFIG_M532x)
+ (MCF_MBAR+0xfc030000),
++#elif defined(CONFIG_ARCH_MXC)
++ (IO_ADDRESS(FEC_BASE_ADDR)),
+ #else
+ &(((immap_t *)IMAP_ADDR)->im_cpm.cp_fec),
+ #endif
+ };
++#endif
+
+-static unsigned char fec_mac_default[] = {
+- 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+-};
+-
++#if 0
+ /*
+ * Some hardware gets it MAC address out of local flash memory.
+ * if this is non-zero then assume it is the address to get MAC from.
+ */
++/* implemented using platform_data! */
+ #if defined(CONFIG_NETtel)
+ #define FEC_FLASHMAC 0xf0006006
+ #elif defined(CONFIG_GILBARCONAP) || defined(CONFIG_SCALES)
+ #define FEC_FLASHMAC 0xf0006000
++#elif defined (CONFIG_MTD_KeyTechnology)
++#define FEC_FLASHMAC 0xffe04000
+ #elif defined(CONFIG_CANCam)
+ #define FEC_FLASHMAC 0xf0020000
+ #elif defined (CONFIG_M5272C3)
+@@ -98,10 +133,13 @@ static unsigned char fec_mac_default[] =
+ #else
+ #define FEC_FLASHMAC 0
+ #endif
++#endif
++
++#define platform_func(p, args...) ((p) ? (p)(args) : 0)
+
+ /* Forward declarations of some structures to support different PHYs
+ */
+-
++#ifndef CONFIG_PHYLIB
+ typedef struct {
+ uint mii_data;
+ void (*funct)(uint mii_reg, struct net_device *dev);
+@@ -116,6 +154,7 @@ typedef struct {
+ const phy_cmd_t *ack_int;
+ const phy_cmd_t *shutdown;
+ } phy_info_t;
++#endif
+
+ /* The number of Tx and Rx buffers. These are allocated from the page
+ * pool. The code may assume these are power of two, so it it best
+@@ -129,12 +168,13 @@ typedef struct {
+ #define RX_RING_SIZE (FEC_ENET_RX_FRPPG * FEC_ENET_RX_PAGES)
+ #define FEC_ENET_TX_FRSIZE 2048
+ #define FEC_ENET_TX_FRPPG (PAGE_SIZE / FEC_ENET_TX_FRSIZE)
+-#define TX_RING_SIZE 16 /* Must be power of two */
+-#define TX_RING_MOD_MASK 15 /* for this to work */
++#define TX_RING_SIZE 16 /* Must be power of two */
++#define TX_RING_MOD_MASK (TX_RING_SIZE - 1) /* for this to work */
+
+ #if (((RX_RING_SIZE + TX_RING_SIZE) * 8) > PAGE_SIZE)
+ #error "FEC: descriptor ring size constants too large"
+ #endif
++#define CBD_BUF_SIZE ((RX_RING_SIZE + TX_RING_SIZE) * sizeof(cbd_t))
+
+ /* Interrupt events/masks.
+ */
+@@ -149,6 +189,12 @@ typedef struct {
+ #define FEC_ENET_MII ((uint)0x00800000) /* MII interrupt */
+ #define FEC_ENET_EBERR ((uint)0x00400000) /* SDMA bus error */
+
++#ifndef CONFIG_ARCH_MXC
++#define FEC_ENET_MASK ((uint)0xffc00000)
++#else
++#define FEC_ENET_MASK ((uint)0xfff80000)
++#endif
++
+ /* The FEC stores dest/src/type, data, and checksum for receive packets.
+ */
+ #define PKT_MAXBUF_SIZE 1518
+@@ -162,8 +208,8 @@ typedef struct {
+ * account when setting it.
+ */
+ #if defined(CONFIG_M523x) || defined(CONFIG_M527x) || defined(CONFIG_M528x) || \
+- defined(CONFIG_M520x) || defined(CONFIG_M532x)
+-#define OPT_FRAME_SIZE (PKT_MAXBUF_SIZE << 16)
++ defined(CONFIG_M520x) || defined(CONFIG_M532x) || defined(CONFIG_ARCH_MXC)
++#define OPT_FRAME_SIZE (RCR_MAX_FL_set(PKT_MAXBUF_SIZE))
+ #else
+ #define OPT_FRAME_SIZE 0
+ #endif
+@@ -178,28 +224,44 @@ typedef struct {
+ */
+ struct fec_enet_private {
+ /* Hardware registers of the FEC device */
+- volatile fec_t *hwp;
+-
+- struct net_device *netdev;
++ volatile void __iomem *reg_base;
++ struct resource *res_mem1;
++ struct resource *res_mem2;
++ int etn_irq;
++ int mii_irq;
++#ifndef CONFIG_PHYLIB
++ struct timer_list *phy_timer;
++#else
++ struct mii_bus *mii;
++ int mii_complete;
++#endif
++ u32 msg_enable;
+
+ /* The saved address of a sent-in-place packet/buffer, for skfree(). */
+- unsigned char *tx_bounce[TX_RING_SIZE];
++ void *tx_bounce[TX_RING_SIZE];
+ struct sk_buff* tx_skbuff[TX_RING_SIZE];
++ struct sk_buff* rx_skbuff[RX_RING_SIZE];
+ ushort skb_cur;
+ ushort skb_dirty;
+
+ /* CPM dual port RAM relative addresses.
+ */
++ struct device *dma_dev; /* pointer to (platform_)device for dma_sync*() functions */
++ void *cbd_mem_base; /* save the virtual base address of rx&tx buffer descriptor */
++ dma_addr_t cbd_phys_base; /* physical address of buffer descriptor memory for access by FEC HW */
++
+ cbd_t *rx_bd_base; /* Address of Rx and Tx buffers. */
+ cbd_t *tx_bd_base;
+- cbd_t *cur_rx, *cur_tx; /* The next free ring entry */
+- cbd_t *dirty_tx; /* The ring entries to be free()ed. */
++ cbd_t *cur_rx, *cur_tx; /* The next free ring entry */
++ cbd_t *dirty_tx; /* The ring entries to be free()ed. */
++ struct net_device_stats stats;
+ uint tx_full;
+- /* hold while accessing the HW like ringbuffer for tx/rx but not MAC */
+- spinlock_t hw_lock;
+- /* hold while accessing the mii_list_t() elements */
+- spinlock_t mii_lock;
++ spinlock_t lock;
+
++#ifdef CONFIG_PHYLIB
++ struct phy_device *phy;
++ uint phy_speed;
++#else
+ uint phy_id;
+ uint phy_id_done;
+ uint phy_status;
+@@ -209,28 +271,41 @@ struct fec_enet_private {
+
+ uint sequence_done;
+ uint mii_phy_task_queued;
+-
++#endif
+ uint phy_addr;
+
+- int index;
+- int opened;
+- int link;
+- int old_link;
+- int full_duplex;
++ unsigned int opened:1;
++ unsigned int phy_int_enabled:1;
++ unsigned int linkstatus:1;
++#ifndef CONFIG_PHYLIB
++ unsigned int old_linkstatus:1;
++#endif
++ unsigned int full_duplex:1;
++
++ struct clk *clk;
+ };
+
+-static int fec_enet_open(struct net_device *dev);
+-static int fec_enet_start_xmit(struct sk_buff *skb, struct net_device *dev);
+-static void fec_enet_mii(struct net_device *dev);
+-static irqreturn_t fec_enet_interrupt(int irq, void * dev_id);
++#ifdef CONFIG_PHYLIB
++static int fec_connect_phy(struct net_device *dev, struct fec_enet_private *fep);
++#else
++static irqreturn_t mii_link_interrupt(int irq, void *dev_id);
++#endif
++static void fec_restart(struct net_device *dev, int duplex);
+ static void fec_enet_tx(struct net_device *dev);
+ static void fec_enet_rx(struct net_device *dev);
+-static int fec_enet_close(struct net_device *dev);
+-static void set_multicast_list(struct net_device *dev);
+-static void fec_restart(struct net_device *dev, int duplex);
++static void fec_enet_mii(struct net_device *dev);
+ static void fec_stop(struct net_device *dev);
+-static void fec_set_mac_address(struct net_device *dev);
++static void _fec_set_mac_address(struct net_device *dev);
+
++/*
++ * fec_copy_threshold controls the copy when receiving ethernet frame.
++ * If ethernet header is aligned on a 4byte boundary, the ip header and
++ * higher level header will not be aligned.
++ * The reason is, that an ethernet header is 14bytes long.
++ * And the max size of tcp & ip header is 128bytes. Normally it is 40bytes.
++ * So I set the default value between 128 to 256.
++ */
++static int fec_copy_threshold = 192;
+
+ /* MII processing. We keep this as simple as possible. Requests are
+ * placed on the list (if there is room). When the request is finished
+@@ -242,14 +317,16 @@ typedef struct mii_list {
+ struct mii_list *mii_next;
+ } mii_list_t;
+
++#ifndef CONFIG_PHYLIB
+ #define NMII 20
+ static mii_list_t mii_cmds[NMII];
+ static mii_list_t *mii_free;
+ static mii_list_t *mii_head;
+ static mii_list_t *mii_tail;
+
+-static int mii_queue(struct net_device *dev, int request,
+- void (*func)(uint, struct net_device *));
++static int mii_queue(struct net_device *dev, int request,
++ void (*func)(uint, struct net_device *));
++#endif
+
+ /* Make MII read/write commands for the FEC.
+ */
+@@ -294,47 +371,147 @@ static int mii_queue(struct net_device *
+ #define PHY_STAT_100HDX 0x4000 /* 100 Mbit half duplex selected */
+ #define PHY_STAT_100FDX 0x8000 /* 100 Mbit full duplex selected */
+
++#ifndef DEBUG
++static inline unsigned long fec_reg_read(struct fec_enet_private *fep, unsigned int reg)
++{
++ return readl(fep->reg_base + reg);
++}
++
++static inline void fec_reg_write(struct fec_enet_private *fep, unsigned int reg, unsigned long val)
++{
++ writel(val, fep->reg_base + reg);
++}
++#else
++#define fec_reg_read(fep, reg) __fec_reg_read(fep, reg, __FUNCTION__, #reg)
++#define fec_reg_write(fep, reg, val) __fec_reg_write(fep, reg, val, __FUNCTION__, #reg)
++
++static inline unsigned long __fec_reg_read(struct fec_enet_private *fep, unsigned int reg,
++ const char *func, const char *reg_name)
++{
++ unsigned long val = readl(fep->reg_base + reg);
++ DBG(3, "%s: Read %08lx from %s(%03x)\n", func, val, reg_name, reg);
++ return val;
++}
++
++static inline void __fec_reg_write(struct fec_enet_private *fep, unsigned int reg,
++ unsigned long val, const char *func, const char *reg_name)
++{
++ DBG(3, "%s: Writing %08lx to %s(%03x)\n", func, val, reg_name, reg);
++ writel(val, fep->reg_base + reg);
++}
++#endif
++
++static inline void fec_enet_cbd_get(struct fec_enet_private *fep)
++{
++ DBG(2, "%s: Requesting cbd area: %08lx\n", __FUNCTION__, (ulong)fep->cbd_phys_base);
++ dma_sync_single_for_cpu(fep->dma_dev, fep->cbd_phys_base, CBD_BUF_SIZE, DMA_BIDIRECTIONAL);
++}
++
++static inline void fec_enet_cbd_put(struct fec_enet_private *fep)
++{
++ DBG(2, "%s: Flushing changes to cbd area\n", __FUNCTION__);
++ dma_sync_single_for_device(fep->dma_dev, fep->cbd_phys_base,
++ CBD_BUF_SIZE, DMA_BIDIRECTIONAL);
++}
++
++static inline void fec_enet_rxbuf_get(struct fec_enet_private *fep, cbd_t *bdp, ushort len)
++{
++ DBG(2, "%s: Requesting RX buffer %08lx(%u)\n", __FUNCTION__, (ulong)bdp->cbd_bufaddr, len);
++ dma_sync_single_for_cpu(fep->dma_dev, bdp->cbd_bufaddr, len, DMA_FROM_DEVICE);
++}
++
++static inline void fec_enet_rxbuf_put(struct fec_enet_private *fep, cbd_t *bdp, ushort len)
++{
++ DBG(2, "%s: Releasing RX buffer %08lx(%u)\n", __FUNCTION__, (ulong)bdp->cbd_bufaddr, len);
++ dma_sync_single_for_device(fep->dma_dev, bdp->cbd_bufaddr, len, DMA_FROM_DEVICE);
++}
++
++static inline void fec_enet_rxbuf_map(struct fec_enet_private *fep, cbd_t *bdp,
++ void *buf, ushort len)
++{
++ BUG_ON(!dma_mapping_error(fep->dma_dev, bdp->cbd_bufaddr));
++ bdp->cbd_bufaddr = dma_map_single(fep->dma_dev, buf, len, DMA_FROM_DEVICE);
++ DBG(2, "%s: RX buffer %p(%u) mapped to %08lx\n", __FUNCTION__, buf, len,
++ (ulong)bdp->cbd_bufaddr);
++}
++
++static inline void fec_enet_rxbuf_unmap(struct fec_enet_private *fep, cbd_t *bdp, ushort len)
++{
++ DBG(2, "%s: Unmapping RX buffer %08lx(%u)\n", __FUNCTION__, (ulong)bdp->cbd_bufaddr, len);
++ BUG_ON(dma_mapping_error(fep->dma_dev, bdp->cbd_bufaddr));
++ dma_unmap_single(fep->dma_dev, bdp->cbd_bufaddr, len, DMA_FROM_DEVICE);
++ bdp->cbd_bufaddr = ~0;
++}
++
++static inline void fec_enet_txbuf_map(struct fec_enet_private *fep, cbd_t *bdp,
++ void *buf, ushort len)
++{
++ BUG_ON(!dma_mapping_error(fep->dma_dev, bdp->cbd_bufaddr));
++ bdp->cbd_bufaddr = dma_map_single(fep->dma_dev, buf, len, DMA_TO_DEVICE);
++ DBG(2, "%s: TX buffer %p(%u) mapped to %08lx\n", __FUNCTION__, buf, len,
++ (ulong)bdp->cbd_bufaddr);
++}
++
++static inline void fec_enet_txbuf_unmap(struct fec_enet_private *fep, cbd_t *bdp, ushort len)
++{
++ DBG(2, "%s: Unmapping TX buffer %08lx(%u)\n", __FUNCTION__, (ulong)bdp->cbd_bufaddr, len);
++ BUG_ON(dma_mapping_error(fep->dma_dev, bdp->cbd_bufaddr));
++ dma_unmap_single(fep->dma_dev, bdp->cbd_bufaddr, len, DMA_TO_DEVICE);
++ bdp->cbd_bufaddr = ~0;
++}
++
++static inline void fec_enet_txbuf_get(struct fec_enet_private *fep, cbd_t *bdp, ushort len)
++{
++ DBG(2, "%s: Requesting TX buffer %08lx(%u)\n", __FUNCTION__, (ulong)bdp->cbd_bufaddr, len);
++ dma_sync_single_for_cpu(fep->dma_dev, bdp->cbd_bufaddr, len, DMA_TO_DEVICE);
++}
++
++static inline void fec_enet_txbuf_put(struct fec_enet_private *fep, cbd_t *bdp, ushort len)
++{
++ DBG(2, "%s: Releasing TX buffer %08lx(%u)\n", __FUNCTION__, (ulong)bdp->cbd_bufaddr, len);
++ dma_sync_single_for_device(fep->dma_dev, bdp->cbd_bufaddr, len, DMA_TO_DEVICE);
++}
+
+ static int
+ fec_enet_start_xmit(struct sk_buff *skb, struct net_device *dev)
+ {
+- struct fec_enet_private *fep;
+- volatile fec_t *fecp;
+- volatile cbd_t *bdp;
+- unsigned short status;
++ struct fec_enet_private *fep = netdev_priv(dev);
++ cbd_t *bdp;
++ unsigned short status;
+ unsigned long flags;
+
+- fep = netdev_priv(dev);
+- fecp = (volatile fec_t*)dev->base_addr;
+-
+- if (!fep->link) {
++ if (!fep->linkstatus) {
++ DBG(0, "%s: Cannot send packet; link is down\n", __FUNCTION__);
+ /* Link is down or autonegotiation is in progress. */
+ return 1;
+ }
+
+- spin_lock_irqsave(&fep->hw_lock, flags);
++ spin_lock_irqsave(&fep->lock, flags);
++
++ //WARN_ON(fec_reg_read(fep, FEC_TDAR) & TDAR_BUSY);
++ fec_enet_cbd_get(fep);
++
+ /* Fill in a Tx ring entry */
+ bdp = fep->cur_tx;
+
+ status = bdp->cbd_sc;
+-#ifndef final_version
++#ifdef DEBUG
+ if (status & BD_ENET_TX_READY) {
+ /* Ooops. All transmit buffers are full. Bail out.
+ * This should not happen, since dev->tbusy should be set.
+ */
+ printk("%s: tx queue full!.\n", dev->name);
+- spin_unlock_irqrestore(&fep->hw_lock, flags);
++ fec_enet_cbd_put(fep);
++ spin_unlock_irqrestore(&fep->lock, flags);
+ return 1;
+ }
+ #endif
+-
+ /* Clear all of the status flags.
+ */
+ status &= ~BD_ENET_TX_STATS;
+
+ /* Set buffer length and buffer pointer.
+ */
+- bdp->cbd_bufaddr = __pa(skb->data);
+ bdp->cbd_datlen = skb->len;
+
+ /*
+@@ -342,30 +519,25 @@ fec_enet_start_xmit(struct sk_buff *skb,
+ * 4-byte boundaries. Use bounce buffers to copy data
+ * and get it aligned. Ugh.
+ */
+- if (bdp->cbd_bufaddr & 0x3) {
++ if (unlikely((bdp->cbd_bufaddr) & FEC_ALIGNMENT)) {
+ unsigned int index;
+ index = bdp - fep->tx_bd_base;
+- memcpy(fep->tx_bounce[index], (void *) bdp->cbd_bufaddr, bdp->cbd_datlen);
+- bdp->cbd_bufaddr = __pa(fep->tx_bounce[index]);
++ memcpy(fep->tx_bounce[index], skb->data, skb->len);
++ fec_enet_txbuf_map(fep, bdp, fep->tx_bounce[index], skb->len);
++ } else {
++ fec_enet_txbuf_map(fep, bdp, skb->data, skb->len);
+ }
+
+ /* Save skb pointer.
+ */
+ fep->tx_skbuff[fep->skb_cur] = skb;
+
+- dev->stats.tx_bytes += skb->len;
+- fep->skb_cur = (fep->skb_cur+1) & TX_RING_MOD_MASK;
+-
+- /* Push the data cache so the CPM does not get stale memory
+- * data.
+- */
+- flush_dcache_range((unsigned long)skb->data,
+- (unsigned long)skb->data + skb->len);
++ fep->stats.tx_bytes += skb->len;
++ fep->skb_cur = (fep->skb_cur + 1) & TX_RING_MOD_MASK;
+
+ /* Send it on its way. Tell FEC it's ready, interrupt when done,
+ * it's the last BD of the frame, and to put the CRC on the end.
+ */
+-
+ status |= (BD_ENET_TX_READY | BD_ENET_TX_INTR
+ | BD_ENET_TX_LAST | BD_ENET_TX_TC);
+ bdp->cbd_sc = status;
+@@ -373,7 +545,7 @@ fec_enet_start_xmit(struct sk_buff *skb,
+ dev->trans_start = jiffies;
+
+ /* Trigger transmission start */
+- fecp->fec_x_des_active = 0;
++ fec_reg_write(fep, FEC_TDAR, DONT_CARE);
+
+ /* If this was the last BD in the ring, start at the beginning again.
+ */
+@@ -385,12 +557,14 @@ fec_enet_start_xmit(struct sk_buff *skb,
+
+ if (bdp == fep->dirty_tx) {
+ fep->tx_full = 1;
++ DBG(0, "TX ring full, stopping netif queue\n");
+ netif_stop_queue(dev);
+ }
+
+- fep->cur_tx = (cbd_t *)bdp;
++ fep->cur_tx = bdp;
++ fec_enet_cbd_put(fep);
+
+- spin_unlock_irqrestore(&fep->hw_lock, flags);
++ spin_unlock_irqrestore(&fep->lock, flags);
+
+ return 0;
+ }
+@@ -400,68 +574,73 @@ fec_timeout(struct net_device *dev)
+ {
+ struct fec_enet_private *fep = netdev_priv(dev);
+
+- printk("%s: transmit timed out.\n", dev->name);
+- dev->stats.tx_errors++;
+-#ifndef final_version
++ printk(KERN_WARNING "%s: transmit timed out.\n", dev->name);
++ fep->stats.tx_errors++;
++#ifdef DEBUG
+ {
+- int i;
+- cbd_t *bdp;
++ int i;
++ cbd_t *bdp;
+
+- printk("Ring data dump: cur_tx %lx%s, dirty_tx %lx cur_rx: %lx\n",
+- (unsigned long)fep->cur_tx, fep->tx_full ? " (full)" : "",
+- (unsigned long)fep->dirty_tx,
+- (unsigned long)fep->cur_rx);
++ fec_enet_cbd_get(fep);
+
+- bdp = fep->tx_bd_base;
+- printk(" tx: %u buffers\n", TX_RING_SIZE);
+- for (i = 0 ; i < TX_RING_SIZE; i++) {
+- printk(" %08x: %04x %04x %08x\n",
+- (uint) bdp,
+- bdp->cbd_sc,
+- bdp->cbd_datlen,
+- (int) bdp->cbd_bufaddr);
+- bdp++;
+- }
++ printk(KERN_DEBUG "%s: Ring data dump: cur_tx %p%s, dirty_tx %p cur_rx: %p\n",
++ __FUNCTION__,
++ fep->cur_tx, fep->tx_full ? " (full)" : "",
++ fep->dirty_tx,
++ fep->cur_rx);
+
+- bdp = fep->rx_bd_base;
+- printk(" rx: %lu buffers\n", (unsigned long) RX_RING_SIZE);
+- for (i = 0 ; i < RX_RING_SIZE; i++) {
+- printk(" %08x: %04x %04x %08x\n",
+- (uint) bdp,
+- bdp->cbd_sc,
+- bdp->cbd_datlen,
+- (int) bdp->cbd_bufaddr);
+- bdp++;
+- }
++ bdp = fep->tx_bd_base;
++ printk(" tx: %u buffers\n", TX_RING_SIZE);
++ for (i = 0 ; i < TX_RING_SIZE; i++) {
++ printk(" %p: %04x %04x %08x\n",
++ bdp,
++ bdp->cbd_sc,
++ bdp->cbd_datlen,
++ bdp->cbd_bufaddr);
++ bdp++;
++ }
++
++ bdp = fep->rx_bd_base;
++ printk(" rx: %lu buffers\n", RX_RING_SIZE);
++ for (i = 0 ; i < RX_RING_SIZE; i++) {
++ printk(" %p: %04x %04x %08x\n",
++ bdp,
++ bdp->cbd_sc,
++ bdp->cbd_datlen,
++ bdp->cbd_bufaddr);
++ bdp++;
++ }
++ fec_enet_cbd_put(fep);
+ }
+ #endif
+ fec_restart(dev, fep->full_duplex);
+- netif_wake_queue(dev);
++ DBG(0, "%s: Scheduling netif queue\n", __FUNCTION__);
++ //netif_schedule(dev);
+ }
+
+ /* The interrupt handler.
+ * This is called from the MPC core interrupt.
+ */
+ static irqreturn_t
+-fec_enet_interrupt(int irq, void * dev_id)
++fec_enet_interrupt(int irq, void *dev_id)
+ {
+- struct net_device *dev = dev_id;
+- volatile fec_t *fecp;
+- uint int_events;
+- irqreturn_t ret = IRQ_NONE;
+-
+- fecp = (volatile fec_t*)dev->base_addr;
++ struct net_device *dev = dev_id;
++ struct fec_enet_private *fep = netdev_priv(dev);
++ uint int_events;
++ int handled = 0;
+
++ DBG(2, "%s: %08lx:%08lx\n", __FUNCTION__,
++ fec_reg_read(fep, FEC_EIR), fec_reg_read(fep, FEC_EIMR));
+ /* Get the interrupt events that caused us to be here.
+ */
+- do {
+- int_events = fecp->fec_ievent;
+- fecp->fec_ievent = int_events;
++ while ((int_events = fec_reg_read(fep, FEC_EIR)) != 0) {
++ fec_reg_write(fep, FEC_EIR, int_events);
+
+ /* Handle receive event in its own function.
+ */
+- if (int_events & FEC_ENET_RXF) {
+- ret = IRQ_HANDLED;
++ if (int_events & (FEC_ENET_RXF | FEC_ENET_RXB)) {
++ DBG(2, "%s: Handling RX Interrupt\n", __FUNCTION__);
++ handled = 1;
+ fec_enet_rx(dev);
+ }
+
+@@ -469,32 +648,43 @@ fec_enet_interrupt(int irq, void * dev_i
+ descriptors. FEC handles all errors, we just discover
+ them as part of the transmit process.
+ */
+- if (int_events & FEC_ENET_TXF) {
+- ret = IRQ_HANDLED;
++ if (int_events & (FEC_ENET_TXF | FEC_ENET_TXB)) {
++ DBG(2, "%s: Handling TX Interrupt\n", __FUNCTION__);
++ handled = 1;
+ fec_enet_tx(dev);
+ }
+
+- if (int_events & FEC_ENET_MII) {
+- ret = IRQ_HANDLED;
++ if (int_events & (FEC_ENET_MII | FEC_ENET_HBERR)) {
++ DBG(2, "%s: Handling MII Interrupt\n", __FUNCTION__);
++ handled = 1;
+ fec_enet_mii(dev);
+ }
+-
+- } while (int_events);
+-
+- return ret;
++ }
++ return IRQ_RETVAL(handled);
+ }
+
++static void fec_free_skb(struct fec_enet_private *fep, cbd_t *bdp, struct sk_buff **pskb)
++{
++ struct sk_buff *skb = *pskb;
++ if (!dma_mapping_error(fep->dma_dev, bdp->cbd_bufaddr)) {
++ fec_enet_txbuf_unmap(fep, bdp, skb->len);
++ }
++ dev_kfree_skb_any(skb);
++ *pskb = NULL;
++}
+
+ static void
+ fec_enet_tx(struct net_device *dev)
+ {
+- struct fec_enet_private *fep;
+- volatile cbd_t *bdp;
++ struct fec_enet_private *fep = netdev_priv(dev);
++ cbd_t *bdp;
+ unsigned short status;
+- struct sk_buff *skb;
++ struct sk_buff *skb;
+
+- fep = netdev_priv(dev);
+- spin_lock_irq(&fep->hw_lock);
++ spin_lock(&fep->lock);
++
++ //WARN_ON(fec_reg_read(fep, FEC_TDAR) & TDAR_BUSY);
++ fec_enet_cbd_get(fep);
+ bdp = fep->dirty_tx;
+
+ while (((status = bdp->cbd_sc) & BD_ENET_TX_READY) == 0) {
+@@ -505,22 +695,22 @@ fec_enet_tx(struct net_device *dev)
+ if (status & (BD_ENET_TX_HB | BD_ENET_TX_LC |
+ BD_ENET_TX_RL | BD_ENET_TX_UN |
+ BD_ENET_TX_CSL)) {
+- dev->stats.tx_errors++;
++ fep->stats.tx_errors++;
+ if (status & BD_ENET_TX_HB) /* No heartbeat */
+- dev->stats.tx_heartbeat_errors++;
++ fep->stats.tx_heartbeat_errors++;
+ if (status & BD_ENET_TX_LC) /* Late collision */
+- dev->stats.tx_window_errors++;
++ fep->stats.tx_window_errors++;
+ if (status & BD_ENET_TX_RL) /* Retrans limit */
+- dev->stats.tx_aborted_errors++;
++ fep->stats.tx_aborted_errors++;
+ if (status & BD_ENET_TX_UN) /* Underrun */
+- dev->stats.tx_fifo_errors++;
++ fep->stats.tx_fifo_errors++;
+ if (status & BD_ENET_TX_CSL) /* Carrier lost */
+- dev->stats.tx_carrier_errors++;
++ fep->stats.tx_carrier_errors++;
+ } else {
+- dev->stats.tx_packets++;
++ fep->stats.tx_packets++;
+ }
+
+-#ifndef final_version
++#ifdef DEBUG
+ if (status & BD_ENET_TX_READY)
+ printk("HEY! Enet xmit interrupt and TX_READY.\n");
+ #endif
+@@ -528,12 +718,11 @@ fec_enet_tx(struct net_device *dev)
+ * but we eventually sent the packet OK.
+ */
+ if (status & BD_ENET_TX_DEF)
+- dev->stats.collisions++;
++ fep->stats.collisions++;
+
+ /* Free the sk buffer associated with this last transmit.
+ */
+- dev_kfree_skb_any(skb);
+- fep->tx_skbuff[fep->skb_dirty] = NULL;
++ fec_free_skb(fep, bdp, &fep->tx_skbuff[fep->skb_dirty]);
+ fep->skb_dirty = (fep->skb_dirty + 1) & TX_RING_MOD_MASK;
+
+ /* Update pointer to next buffer descriptor to be transmitted.
+@@ -548,12 +737,15 @@ fec_enet_tx(struct net_device *dev)
+ */
+ if (fep->tx_full) {
+ fep->tx_full = 0;
+- if (netif_queue_stopped(dev))
++ if (netif_queue_stopped(dev)) {
++ DBG(0, "%s: Waking up netif queue\n", __FUNCTION__);
+ netif_wake_queue(dev);
++ }
+ }
+ }
+- fep->dirty_tx = (cbd_t *)bdp;
+- spin_unlock_irq(&fep->hw_lock);
++ fec_enet_cbd_put(fep);
++ fep->dirty_tx = bdp;
++ spin_unlock(&fep->lock);
+ }
+
+
+@@ -565,22 +757,22 @@ fec_enet_tx(struct net_device *dev)
+ static void
+ fec_enet_rx(struct net_device *dev)
+ {
+- struct fec_enet_private *fep;
+- volatile fec_t *fecp;
+- volatile cbd_t *bdp;
++ struct fec_enet_private *fep = netdev_priv(dev);
++ cbd_t *bdp;
+ unsigned short status;
+- struct sk_buff *skb;
+- ushort pkt_len;
+- __u8 *data;
++ struct sk_buff *skb;
++ ushort pkt_len;
++ int rx_index;
+
+ #ifdef CONFIG_M532x
++ /* This is probably nonsense
++ Proper use of dma-mapping functions should make this obsolete
++ */
+ flush_cache_all();
+ #endif
+-
+- fep = netdev_priv(dev);
+- fecp = (volatile fec_t*)dev->base_addr;
+-
+- spin_lock_irq(&fep->hw_lock);
++ /* reserve the dual port memory area for our use */
++ //WARN_ON(fec_reg_read(fep, FEC_RDAR) & RDAR_BUSY);
++ fec_enet_cbd_get(fep);
+
+ /* First, grab all of the stats for the incoming packet.
+ * These get messed up if we get called due to a busy condition.
+@@ -588,8 +780,8 @@ fec_enet_rx(struct net_device *dev)
+ bdp = fep->cur_rx;
+
+ while (!((status = bdp->cbd_sc) & BD_ENET_RX_EMPTY)) {
+-
+-#ifndef final_version
++ rx_index = bdp - fep->rx_bd_base;
++#ifdef DEBUG
+ /* Since we have allocated space to hold a complete frame,
+ * the last indicator should be set.
+ */
+@@ -597,23 +789,24 @@ while (!((status = bdp->cbd_sc) & BD_ENE
+ printk("FEC ENET: rcv is not +last\n");
+ #endif
+
+- if (!fep->opened)
++ if (!fep->opened) {
++ DBG(0, "%s: Driver not opened; ignoring packet\n", __FUNCTION__);
+ goto rx_processing_done;
+-
++ }
+ /* Check for errors. */
+ if (status & (BD_ENET_RX_LG | BD_ENET_RX_SH | BD_ENET_RX_NO |
+ BD_ENET_RX_CR | BD_ENET_RX_OV)) {
+- dev->stats.rx_errors++;
++ fep->stats.rx_errors++;
+ if (status & (BD_ENET_RX_LG | BD_ENET_RX_SH)) {
+ /* Frame too long or too short. */
+- dev->stats.rx_length_errors++;
++ fep->stats.rx_length_errors++;
+ }
+ if (status & BD_ENET_RX_NO) /* Frame alignment */
+- dev->stats.rx_frame_errors++;
++ fep->stats.rx_frame_errors++;
+ if (status & BD_ENET_RX_CR) /* CRC Error */
+- dev->stats.rx_crc_errors++;
++ fep->stats.rx_crc_errors++;
+ if (status & BD_ENET_RX_OV) /* FIFO overrun */
+- dev->stats.rx_fifo_errors++;
++ fep->stats.rx_fifo_errors++;
+ }
+
+ /* Report late collisions as a frame error.
+@@ -621,36 +814,63 @@ while (!((status = bdp->cbd_sc) & BD_ENE
+ * have in the buffer. So, just drop this frame on the floor.
+ */
+ if (status & BD_ENET_RX_CL) {
+- dev->stats.rx_errors++;
+- dev->stats.rx_frame_errors++;
++ fep->stats.rx_errors++;
++ fep->stats.rx_frame_errors++;
++ DBG(0, "%s: Collision detected; dropping packet\n", __FUNCTION__);
+ goto rx_processing_done;
+ }
+
+ /* Process the incoming frame.
+ */
+- dev->stats.rx_packets++;
++ fep->stats.rx_packets++;
+ pkt_len = bdp->cbd_datlen;
+- dev->stats.rx_bytes += pkt_len;
+- data = (__u8*)__va(bdp->cbd_bufaddr);
++ fep->stats.rx_bytes += pkt_len;
+
+ /* This does 16 byte alignment, exactly what we need.
+ * The packet length includes FCS, but we don't want to
+ * include that when passing upstream as it messes up
+ * bridging applications.
+ */
+- skb = dev_alloc_skb(pkt_len-4);
++ if ((pkt_len - 4) < fec_copy_threshold) {
++ skb = dev_alloc_skb(pkt_len);
++ } else {
++ skb = dev_alloc_skb(FEC_ENET_RX_FRSIZE);
++ }
+
+ if (skb == NULL) {
+ printk("%s: Memory squeeze, dropping packet.\n", dev->name);
+- dev->stats.rx_dropped++;
++ fep->stats.rx_dropped++;
+ } else {
+- skb_put(skb,pkt_len-4); /* Make room */
+- skb_copy_to_linear_data(skb, data, pkt_len-4);
+- skb->protocol=eth_type_trans(skb,dev);
++ if ((pkt_len - 4) < fec_copy_threshold) {
++ /* skip 2 bytes, so IP header is on a 4 bytes boundary */
++ skb_reserve(skb, 2);
++ skb_put(skb, pkt_len - 4); /* Make room */
++ fec_enet_rxbuf_get(fep, bdp, pkt_len - 4);
++ skb_copy_to_linear_data(skb,
++ fep->rx_skbuff[rx_index]->data,
++ pkt_len - 4);
++ fec_enet_rxbuf_put(fep, bdp, pkt_len - 4);
++ } else {
++ struct sk_buff *pskb = fep->rx_skbuff[rx_index];
++
++ /* unmap the skb we are going to hand down to the network layer */
++ fec_enet_rxbuf_unmap(fep, bdp, FEC_ENET_RX_FRSIZE);
++
++ /* init the newly allocated skb */
++ fep->rx_skbuff[rx_index] = skb;
++ skb->data = FEC_ADDR_ALIGNMENT(skb->data);
++ /* map the newly allocated skb's data buffer for DMA */
++ fec_enet_rxbuf_map(fep, bdp, skb->data, FEC_ENET_RX_FRSIZE);
++
++ skb_put(pskb, pkt_len - 4); /* Make room */
++ skb = pskb;
++
++ }
++ skb->dev = dev;
++ skb->protocol = eth_type_trans(skb, dev);
+ netif_rx(skb);
+ }
+ rx_processing_done:
+-
+ /* Clear the status flags for this buffer.
+ */
+ status &= ~BD_ENET_RX_STATS;
+@@ -660,6 +880,9 @@ while (!((status = bdp->cbd_sc) & BD_ENE
+ status |= BD_ENET_RX_EMPTY;
+ bdp->cbd_sc = status;
+
++ /* release the dual port memory area for use by the FEC hardware */
++ fec_enet_cbd_put(fep);
++
+ /* Update BD pointer to next entry.
+ */
+ if (status & BD_ENET_RX_WRAP)
+@@ -672,10 +895,10 @@ while (!((status = bdp->cbd_sc) & BD_ENE
+ * incoming frames. On a heavily loaded network, we should be
+ * able to keep up at the expense of system resources.
+ */
+- fecp->fec_r_des_active = 0;
++ fec_reg_write(fep, FEC_RDAR, DONT_CARE);
+ #endif
+ } /* while (!((status = bdp->cbd_sc) & BD_ENET_RX_EMPTY)) */
+- fep->cur_rx = (cbd_t *)bdp;
++ fep->cur_rx = bdp;
+
+ #if 0
+ /* Doing this here will allow us to process all frames in the
+@@ -685,27 +908,28 @@ while (!((status = bdp->cbd_sc) & BD_ENE
+ * our way back to the interrupt return only to come right back
+ * here.
+ */
+- fecp->fec_r_des_active = 0;
++ fec_reg_write(fep, FEC_RDAR, DONT_CARE);
+ #endif
+-
+- spin_unlock_irq(&fep->hw_lock);
+ }
+
+-
++#ifdef CONFIG_PHYLIB
+ /* called from interrupt context */
++static void fec_enet_mii(struct net_device *dev)
++{
++ struct fec_enet_private *fep = netdev_priv(dev);
++ fep->mii_complete = 1;
++}
++#else
+ static void
+ fec_enet_mii(struct net_device *dev)
+ {
+- struct fec_enet_private *fep;
+- volatile fec_t *ep;
++ struct fec_enet_private *fep = netdev_priv(dev);
+ mii_list_t *mip;
+ uint mii_reg;
+
+- fep = netdev_priv(dev);
+- spin_lock_irq(&fep->mii_lock);
++ mii_reg = fec_reg_read(fep, FEC_MMFR);
+
+- ep = fep->hwp;
+- mii_reg = ep->fec_mii_data;
++ spin_lock(&fep->lock);
+
+ if ((mip = mii_head) == NULL) {
+ printk("MII and no head!\n");
+@@ -720,27 +944,27 @@ fec_enet_mii(struct net_device *dev)
+ mii_free = mip;
+
+ if ((mip = mii_head) != NULL)
+- ep->fec_mii_data = mip->mii_regval;
++ fec_reg_write(fep, FEC_MMFR, mip->mii_regval);
+
+ unlock:
+- spin_unlock_irq(&fep->mii_lock);
++ spin_unlock(&fep->lock);
+ }
+
+ static int
+ mii_queue(struct net_device *dev, int regval, void (*func)(uint, struct net_device *))
+ {
+- struct fec_enet_private *fep;
++ struct fec_enet_private *fep = netdev_priv(dev);
+ unsigned long flags;
+ mii_list_t *mip;
+ int retval;
+
++ retval = 0;
++
++ spin_lock_irqsave(&fep->lock,flags);
++
+ /* Add PHY address to register command.
+ */
+- fep = netdev_priv(dev);
+- spin_lock_irqsave(&fep->mii_lock, flags);
+-
+ regval |= fep->phy_addr << 23;
+- retval = 0;
+
+ if ((mip = mii_free) != NULL) {
+ mii_free = mip->mii_next;
+@@ -752,32 +976,32 @@ mii_queue(struct net_device *dev, int re
+ mii_tail = mip;
+ } else {
+ mii_head = mii_tail = mip;
+- fep->hwp->fec_mii_data = regval;
++ fec_reg_write(fep, FEC_MMFR, regval);
+ }
+ } else {
+ retval = 1;
+ }
+
+- spin_unlock_irqrestore(&fep->mii_lock, flags);
+- return retval;
++ spin_unlock_irqrestore(&fep->lock,flags);
++
++ return(retval);
+ }
+
+ static void mii_do_cmd(struct net_device *dev, const phy_cmd_t *c)
+ {
+- if(!c)
+- return;
++ int k;
+
+- for (; c->mii_data != mk_mii_end; c++)
+- mii_queue(dev, c->mii_data, c->funct);
++ for (k = 0; c != NULL && c[k].mii_data != mk_mii_end; k++) {
++ mii_queue(dev, c[k].mii_data, c[k].funct);
++ }
+ }
+
+ static void mii_parse_sr(uint mii_reg, struct net_device *dev)
+ {
+ struct fec_enet_private *fep = netdev_priv(dev);
+- volatile uint *s = &(fep->phy_status);
+ uint status;
+
+- status = *s & ~(PHY_STAT_LINK | PHY_STAT_FAULT | PHY_STAT_ANC);
++ status = fep->phy_status & ~(PHY_STAT_LINK | PHY_STAT_FAULT | PHY_STAT_ANC);
+
+ if (mii_reg & 0x0004)
+ status |= PHY_STAT_LINK;
+@@ -785,31 +1009,30 @@ static void mii_parse_sr(uint mii_reg, s
+ status |= PHY_STAT_FAULT;
+ if (mii_reg & 0x0020)
+ status |= PHY_STAT_ANC;
+- *s = status;
++
++ fep->phy_status = status;
+ }
+
+ static void mii_parse_cr(uint mii_reg, struct net_device *dev)
+ {
+ struct fec_enet_private *fep = netdev_priv(dev);
+- volatile uint *s = &(fep->phy_status);
+ uint status;
+
+- status = *s & ~(PHY_CONF_ANE | PHY_CONF_LOOP);
++ status = fep->phy_status & ~(PHY_CONF_ANE | PHY_CONF_LOOP);
+
+ if (mii_reg & 0x1000)
+ status |= PHY_CONF_ANE;
+ if (mii_reg & 0x4000)
+ status |= PHY_CONF_LOOP;
+- *s = status;
++ fep->phy_status = status;
+ }
+
+ static void mii_parse_anar(uint mii_reg, struct net_device *dev)
+ {
+ struct fec_enet_private *fep = netdev_priv(dev);
+- volatile uint *s = &(fep->phy_status);
+ uint status;
+
+- status = *s & ~(PHY_CONF_SPMASK);
++ status = fep->phy_status & ~(PHY_CONF_SPMASK);
+
+ if (mii_reg & 0x0020)
+ status |= PHY_CONF_10HDX;
+@@ -819,7 +1042,7 @@ static void mii_parse_anar(uint mii_reg,
+ status |= PHY_CONF_100HDX;
+ if (mii_reg & 0x00100)
+ status |= PHY_CONF_100FDX;
+- *s = status;
++ fep->phy_status = status;
+ }
+
+ /* ------------------------------------------------------------------------- */
+@@ -834,10 +1057,9 @@ static void mii_parse_anar(uint mii_reg,
+ static void mii_parse_lxt970_csr(uint mii_reg, struct net_device *dev)
+ {
+ struct fec_enet_private *fep = netdev_priv(dev);
+- volatile uint *s = &(fep->phy_status);
+ uint status;
+
+- status = *s & ~(PHY_STAT_SPMASK);
++ status = fep->phy_status & ~(PHY_STAT_SPMASK);
+ if (mii_reg & 0x0800) {
+ if (mii_reg & 0x1000)
+ status |= PHY_STAT_100FDX;
+@@ -849,7 +1071,7 @@ static void mii_parse_lxt970_csr(uint mi
+ else
+ status |= PHY_STAT_10HDX;
+ }
+- *s = status;
++ fep->phy_status = status;
+ }
+
+ static phy_cmd_t const phy_cmd_lxt970_config[] = {
+@@ -905,16 +1127,15 @@ static phy_info_t const phy_info_lxt970
+ static void mii_parse_lxt971_sr2(uint mii_reg, struct net_device *dev)
+ {
+ struct fec_enet_private *fep = netdev_priv(dev);
+- volatile uint *s = &(fep->phy_status);
+ uint status;
+
+- status = *s & ~(PHY_STAT_SPMASK | PHY_STAT_LINK | PHY_STAT_ANC);
++ status = fep->phy_status & ~(PHY_STAT_SPMASK | PHY_STAT_LINK | PHY_STAT_ANC);
+
+ if (mii_reg & 0x0400) {
+- fep->link = 1;
++ fep->linkstatus = 1;
+ status |= PHY_STAT_LINK;
+ } else {
+- fep->link = 0;
++ fep->linkstatus = 0;
+ }
+ if (mii_reg & 0x0080)
+ status |= PHY_STAT_ANC;
+@@ -932,7 +1153,7 @@ static void mii_parse_lxt971_sr2(uint mi
+ if (mii_reg & 0x0008)
+ status |= PHY_STAT_FAULT;
+
+- *s = status;
++ fep->phy_status = status;
+ }
+
+ static phy_cmd_t const phy_cmd_lxt971_config[] = {
+@@ -989,10 +1210,9 @@ static phy_info_t const phy_info_lxt971
+ static void mii_parse_qs6612_pcr(uint mii_reg, struct net_device *dev)
+ {
+ struct fec_enet_private *fep = netdev_priv(dev);
+- volatile uint *s = &(fep->phy_status);
+ uint status;
+
+- status = *s & ~(PHY_STAT_SPMASK);
++ status = fep->phy_status & ~(PHY_STAT_SPMASK);
+
+ switch((mii_reg >> 2) & 7) {
+ case 1: status |= PHY_STAT_10HDX; break;
+@@ -1001,7 +1221,7 @@ static void mii_parse_qs6612_pcr(uint mi
+ case 6: status |= PHY_STAT_100FDX; break;
+ }
+
+- *s = status;
++ fep->phy_status = status;
+ }
+
+ static phy_cmd_t const phy_cmd_qs6612_config[] = {
+@@ -1059,10 +1279,9 @@ static phy_info_t const phy_info_qs6612
+ static void mii_parse_am79c874_dr(uint mii_reg, struct net_device *dev)
+ {
+ struct fec_enet_private *fep = netdev_priv(dev);
+- volatile uint *s = &(fep->phy_status);
+ uint status;
+
+- status = *s & ~(PHY_STAT_SPMASK | PHY_STAT_ANC);
++ status = fep->phy_status & ~(PHY_STAT_SPMASK | PHY_STAT_ANC);
+
+ if (mii_reg & 0x0080)
+ status |= PHY_STAT_ANC;
+@@ -1071,7 +1290,7 @@ static void mii_parse_am79c874_dr(uint m
+ else
+ status |= ((mii_reg & 0x0800) ? PHY_STAT_10FDX : PHY_STAT_10HDX);
+
+- *s = status;
++ fep->phy_status = status;
+ }
+
+ static phy_cmd_t const phy_cmd_am79c874_config[] = {
+@@ -1155,33 +1374,32 @@ static phy_info_t const phy_info_ks8721b
+
+ static void mii_parse_dp8384x_sr2(uint mii_reg, struct net_device *dev)
+ {
+- struct fec_enet_private *fep = dev->priv;
+- volatile uint *s = &(fep->phy_status);
++ struct fec_enet_private *fep = netdev_priv(dev);
+
+- *s &= ~(PHY_STAT_SPMASK | PHY_STAT_LINK | PHY_STAT_ANC);
++ fep->phy_status &= ~(PHY_STAT_SPMASK | PHY_STAT_LINK | PHY_STAT_ANC);
+
+ /* Link up */
+ if (mii_reg & 0x0001) {
+- fep->link = 1;
+- *s |= PHY_STAT_LINK;
++ fep->linkstatus = 1;
++ fep->phy_status |= PHY_STAT_LINK;
+ } else
+- fep->link = 0;
++ fep->linkstatus = 0;
+ /* Status of link */
+ if (mii_reg & 0x0010) /* Autonegotioation complete */
+- *s |= PHY_STAT_ANC;
++ fep->phy_status |= PHY_STAT_ANC;
+ if (mii_reg & 0x0002) { /* 10MBps? */
+ if (mii_reg & 0x0004) /* Full Duplex? */
+- *s |= PHY_STAT_10FDX;
++ fep->phy_status |= PHY_STAT_10FDX;
+ else
+- *s |= PHY_STAT_10HDX;
++ fep->phy_status |= PHY_STAT_10HDX;
+ } else { /* 100 Mbps? */
+ if (mii_reg & 0x0004) /* Full Duplex? */
+- *s |= PHY_STAT_100FDX;
++ fep->phy_status |= PHY_STAT_100FDX;
+ else
+- *s |= PHY_STAT_100HDX;
++ fep->phy_status |= PHY_STAT_100HDX;
+ }
+ if (mii_reg & 0x0008)
+- *s |= PHY_STAT_FAULT;
++ fep->phy_status |= PHY_STAT_FAULT;
+ }
+
+ static phy_info_t phy_info_dp83848= {
+@@ -1218,660 +1436,249 @@ static phy_info_t const * const phy_info
+ &phy_info_dp83848,
+ NULL
+ };
+-
+-/* ------------------------------------------------------------------------- */
+-#ifdef HAVE_mii_link_interrupt
+-static irqreturn_t
+-mii_link_interrupt(int irq, void * dev_id);
+ #endif
+
+-#if defined(CONFIG_M5272)
+ /*
+- * Code specific to Coldfire 5272 setup.
++ * do some initializtion based architecture of this chip
++MOVED to platform_data hooks!
+ */
+-static void __inline__ fec_request_intrs(struct net_device *dev)
+-{
+- volatile unsigned long *icrp;
+- static const struct idesc {
+- char *name;
+- unsigned short irq;
+- irq_handler_t handler;
+- } *idp, id[] = {
+- { "fec(RX)", 86, fec_enet_interrupt },
+- { "fec(TX)", 87, fec_enet_interrupt },
+- { "fec(OTHER)", 88, fec_enet_interrupt },
+- { "fec(MII)", 66, mii_link_interrupt },
+- { NULL },
+- };
+-
+- /* Setup interrupt handlers. */
+- for (idp = id; idp->name; idp++) {
+- if (request_irq(idp->irq, idp->handler, IRQF_DISABLED, idp->name, dev) != 0)
+- printk("FEC: Could not allocate %s IRQ(%d)!\n", idp->name, idp->irq);
+- }
+
+- /* Unmask interrupt at ColdFire 5272 SIM */
+- icrp = (volatile unsigned long *) (MCF_MBAR + MCFSIM_ICR3);
+- *icrp = 0x00000ddd;
+- icrp = (volatile unsigned long *) (MCF_MBAR + MCFSIM_ICR1);
+- *icrp = 0x0d000000;
+-}
++#define PHY_POLL_LINK_ON (1 * HZ)
++#define PHY_POLL_LINK_OFF (HZ / 5)
+
+-static void __inline__ fec_set_mii(struct net_device *dev, struct fec_enet_private *fep)
++#ifdef CONFIG_PHYLIB
++static void fec_link_change(struct net_device *dev)
+ {
+- volatile fec_t *fecp;
+-
+- fecp = fep->hwp;
+- fecp->fec_r_cntrl = OPT_FRAME_SIZE | 0x04;
+- fecp->fec_x_cntrl = 0x00;
+-
+- /*
+- * Set MII speed to 2.5 MHz
+- * See 5272 manual section 11.5.8: MSCR
+- */
+- fep->phy_speed = ((((MCF_CLK / 4) / (2500000 / 10)) + 5) / 10) * 2;
+- fecp->fec_mii_speed = fep->phy_speed;
++ struct fec_enet_private *fep = netdev_priv(dev);
++ struct phy_device *phydev = fep->phy;
+
+- fec_restart(dev, 0);
++ if (phydev->link != fep->linkstatus ||
++ phydev->duplex != fep->full_duplex) {
++ DBG(0, "%s: link status changed from %d to %d %s -> %s duplex\n", __FUNCTION__,
++ fep->linkstatus, phydev->link, fep->full_duplex ? "full" : "half",
++ phydev->duplex ? "full" : "half");
++ if (phydev->link) {
++ fec_restart(dev, phydev->duplex);
++ } else {
++ fec_stop(dev);
++ }
++ if (fep->linkstatus != phydev->link && netif_msg_link(fep)) {
++ phy_print_status(phydev);
++ }
++ fep->linkstatus = phydev->link;
++ }
+ }
+-
+-static void __inline__ fec_get_mac(struct net_device *dev)
++#else
++static void fec_link_change(struct net_device *dev)
+ {
+ struct fec_enet_private *fep = netdev_priv(dev);
+- volatile fec_t *fecp;
+- unsigned char *iap, tmpaddr[ETH_ALEN];
+
+- fecp = fep->hwp;
++ DBG(0, "%s: link status changed from %d to %d\n", __FUNCTION__,
++ fep->old_linkstatus, fep->linkstatus);
++ if (fep->linkstatus) {
++ int duplex;
+
+- if (FEC_FLASHMAC) {
+- /*
+- * Get MAC address from FLASH.
+- * If it is all 1's or 0's, use the default.
+- */
+- iap = (unsigned char *)FEC_FLASHMAC;
+- if ((iap[0] == 0) && (iap[1] == 0) && (iap[2] == 0) &&
+- (iap[3] == 0) && (iap[4] == 0) && (iap[5] == 0))
+- iap = fec_mac_default;
+- if ((iap[0] == 0xff) && (iap[1] == 0xff) && (iap[2] == 0xff) &&
+- (iap[3] == 0xff) && (iap[4] == 0xff) && (iap[5] == 0xff))
+- iap = fec_mac_default;
++ duplex = 0;
++ if (fep->phy_status & (PHY_STAT_100FDX | PHY_STAT_10FDX)) {
++ duplex = 1;
++ }
++ fec_restart(dev, duplex);
++ if (fep->phy_timer) {
++ mod_timer(fep->phy_timer, jiffies + PHY_POLL_LINK_ON);
++ }
+ } else {
+- *((unsigned long *) &tmpaddr[0]) = fecp->fec_addr_low;
+- *((unsigned short *) &tmpaddr[4]) = (fecp->fec_addr_high >> 16);
+- iap = &tmpaddr[0];
++ fec_stop(dev);
++ if (fep->phy_timer) {
++ mod_timer(fep->phy_timer, jiffies + PHY_POLL_LINK_OFF);
++ }
+ }
+
+- memcpy(dev->dev_addr, iap, ETH_ALEN);
+-
+- /* Adjust MAC if using default MAC address */
+- if (iap == fec_mac_default)
+- dev->dev_addr[ETH_ALEN-1] = fec_mac_default[ETH_ALEN-1] + fep->index;
+-}
+-
+-static void __inline__ fec_enable_phy_intr(void)
+-{
+-}
+-
+-static void __inline__ fec_disable_phy_intr(void)
+-{
+- volatile unsigned long *icrp;
+- icrp = (volatile unsigned long *) (MCF_MBAR + MCFSIM_ICR1);
+- *icrp = 0x08000000;
+-}
+-
+-static void __inline__ fec_phy_ack_intr(void)
+-{
+- volatile unsigned long *icrp;
+- /* Acknowledge the interrupt */
+- icrp = (volatile unsigned long *) (MCF_MBAR + MCFSIM_ICR1);
+- *icrp = 0x0d000000;
++ fep->old_linkstatus = fep->linkstatus;
+ }
+
+-static void __inline__ fec_localhw_setup(void)
++static void fec_phy_timer(unsigned long data)
+ {
+-}
++ struct net_device *dev = (struct net_device *)data;
++ struct fec_enet_private *fep = netdev_priv(dev);
++ int link_poll_interval = fep->linkstatus ? PHY_POLL_LINK_ON : PHY_POLL_LINK_OFF;
+
+-/*
+- * Do not need to make region uncached on 5272.
+- */
+-static void __inline__ fec_uncache(unsigned long addr)
+-{
++ if (fep->old_linkstatus != fep->linkstatus) {
++ fec_link_change(dev);
++ }
++ mod_timer(fep->phy_timer, link_poll_interval);
+ }
+-
+-/* ------------------------------------------------------------------------- */
+-
+-#elif defined(CONFIG_M523x) || defined(CONFIG_M527x) || defined(CONFIG_M528x)
++#endif
+
+ /*
+- * Code specific to Coldfire 5230/5231/5232/5234/5235,
+- * the 5270/5271/5274/5275 and 5280/5282 setups.
++ * Code specific to Freescale i.MXC
+ */
+-static void __inline__ fec_request_intrs(struct net_device *dev)
++static int fec_request_intrs(struct platform_device *pdev, struct net_device *dev)
+ {
+- struct fec_enet_private *fep;
+- int b;
+- static const struct idesc {
+- char *name;
+- unsigned short irq;
+- } *idp, id[] = {
+- { "fec(TXF)", 23 },
+- { "fec(RXF)", 27 },
+- { "fec(MII)", 29 },
+- { NULL },
+- };
++ int ret;
++ struct fec_enet_private *fep = netdev_priv(dev);
+
+- fep = netdev_priv(dev);
+- b = (fep->index) ? 128 : 64;
++ fep->etn_irq = platform_get_irq(pdev, 0);
++ fep->mii_irq = platform_get_irq(pdev, 1);
+
+ /* Setup interrupt handlers. */
+- for (idp = id; idp->name; idp++) {
+- if (request_irq(b+idp->irq, fec_enet_interrupt, IRQF_DISABLED, idp->name, dev) != 0)
+- printk("FEC: Could not allocate %s IRQ(%d)!\n", idp->name, b+idp->irq);
+- }
+-
+- /* Unmask interrupts at ColdFire 5280/5282 interrupt controller */
+- {
+- volatile unsigned char *icrp;
+- volatile unsigned long *imrp;
+- int i, ilip;
+-
+- b = (fep->index) ? MCFICM_INTC1 : MCFICM_INTC0;
+- icrp = (volatile unsigned char *) (MCF_IPSBAR + b +
+- MCFINTC_ICR0);
+- for (i = 23, ilip = 0x28; (i < 36); i++)
+- icrp[i] = ilip--;
+-
+- imrp = (volatile unsigned long *) (MCF_IPSBAR + b +
+- MCFINTC_IMRH);
+- *imrp &= ~0x0000000f;
+- imrp = (volatile unsigned long *) (MCF_IPSBAR + b +
+- MCFINTC_IMRL);
+- *imrp &= ~0xff800001;
+- }
+-
+-#if defined(CONFIG_M528x)
+- /* Set up gpio outputs for MII lines */
+- {
+- volatile u16 *gpio_paspar;
+- volatile u8 *gpio_pehlpar;
+-
+- gpio_paspar = (volatile u16 *) (MCF_IPSBAR + 0x100056);
+- gpio_pehlpar = (volatile u16 *) (MCF_IPSBAR + 0x100058);
+- *gpio_paspar |= 0x0f00;
+- *gpio_pehlpar = 0xc0;
++ ret = request_irq(fep->etn_irq, fec_enet_interrupt, 0, "fec", dev);
++ if (ret != 0) {
++ printk(KERN_ERR "FEC: Could not allocate FEC IRQ(%d)!\n", fep->etn_irq);
++ return ret;
++ }
++#ifndef CONFIG_PHYLIB
++ if (fep->mii_irq >= 0) {
++ /* TODO: disable now due to CPLD issue */
++ ret = request_irq(fep->mii_irq, mii_link_interrupt, 0, "fec(MII)", dev);
++ if (ret != 0) {
++ printk(KERN_ERR "FEC: Could not allocate FEC(MII) IRQ(%d)!\n",
++ fep->mii_irq);
++ free_irq(fep->etn_irq, dev);
++ return ret;
++ }
++ /*
++ * board specific workaround should be done in board specific code
++ * This is unsafe anyway. An interrupt might have been asserted
++ * already. Use IRQ_NOAUTOEN with request_irq() to have irq initially disabled.
++ */
++ fep->phy_int_enabled = 1;
++ } else {
++ fep->phy_timer = kzalloc(sizeof(struct timer_list), GFP_KERNEL);
++ if (fep->phy_timer == NULL) {
++ free_irq(fep->etn_irq, dev);
++ return -ENOMEM;
++ }
++ init_timer(fep->phy_timer);
++ fep->phy_timer->function = fec_phy_timer;
++ fep->phy_timer->data = (unsigned long)dev;
++ fec_link_change(dev);
+ }
+ #endif
+
+-#if defined(CONFIG_M527x)
+- /* Set up gpio outputs for MII lines */
+- {
+- volatile u8 *gpio_par_fec;
+- volatile u16 *gpio_par_feci2c;
+-
+- gpio_par_feci2c = (volatile u16 *)(MCF_IPSBAR + 0x100082);
+- /* Set up gpio outputs for FEC0 MII lines */
+- gpio_par_fec = (volatile u8 *)(MCF_IPSBAR + 0x100078);
+-
+- *gpio_par_feci2c |= 0x0f00;
+- *gpio_par_fec |= 0xc0;
++ return 0;
++}
+
+-#if defined(CONFIG_FEC2)
+- /* Set up gpio outputs for FEC1 MII lines */
+- gpio_par_fec = (volatile u8 *)(MCF_IPSBAR + 0x100079);
++static void __inline__ fec_release_intrs(struct net_device *dev)
++{
++ struct fec_enet_private *fep = netdev_priv(dev);
+
+- *gpio_par_feci2c |= 0x00a0;
+- *gpio_par_fec |= 0xc0;
+-#endif /* CONFIG_FEC2 */
++ free_irq(fep->etn_irq, dev);
++#ifndef CONFIG_PHYLIB
++ if (fep->mii_irq >= 0) {
++ free_irq(fep->mii_irq, dev);
+ }
+-#endif /* CONFIG_M527x */
++#endif
+ }
+
+ static void __inline__ fec_set_mii(struct net_device *dev, struct fec_enet_private *fep)
+ {
+- volatile fec_t *fecp;
++ u32 rate;
++ struct clk *clk;
+
+- fecp = fep->hwp;
+- fecp->fec_r_cntrl = OPT_FRAME_SIZE | 0x04;
+- fecp->fec_x_cntrl = 0x00;
++ fec_reg_write(fep, FEC_RCR, OPT_FRAME_SIZE | 0x04);
++ fec_reg_write(fep, FEC_TCR, 0x00);
+
+- /*
++ /*
+ * Set MII speed to 2.5 MHz
+- * See 5282 manual section 17.5.4.7: MSCR
+ */
+- fep->phy_speed = ((((MCF_CLK / 2) / (2500000 / 10)) + 5) / 10) * 2;
+- fecp->fec_mii_speed = fep->phy_speed;
++ clk = clk_get(fep->dma_dev, "fec_clk");
++ rate = clk_get_rate(clk);
++ clk_put(clk);
+
+- fec_restart(dev, 0);
++ fep->phy_speed = ((((rate / 2 + 4999999) / 2500000) / 2) & 0x3F) << 1;
++ fec_reg_write(fep, FEC_MSCR, fep->phy_speed);
+ }
+
+-static void __inline__ fec_get_mac(struct net_device *dev)
++static const unsigned char default_mac[ETH_ALEN] = {
++ 0x00, 0x04, 0x9f, 0x00, 0x74, 0x4a,
++};
++
++#define FEC_IIM_BASE IO_ADDRESS(IIM_BASE_ADDR)
++static void fec_get_mac(struct net_device *dev)
+ {
++#if 1
++ // keep bootloader assigned MAC address
+ struct fec_enet_private *fep = netdev_priv(dev);
+- volatile fec_t *fecp;
+- unsigned char *iap, tmpaddr[ETH_ALEN];
+-
+- fecp = fep->hwp;
++ unsigned long eth_addr = fec_reg_read(fep, FEC_PALR);
++ dev->dev_addr[0] = eth_addr >> 24;
++ dev->dev_addr[1] = eth_addr >> 16;
++ dev->dev_addr[2] = eth_addr >> 8;
++ dev->dev_addr[3] = eth_addr >> 0;
++ eth_addr = fec_reg_read(fep, FEC_PAUR);
++ dev->dev_addr[5] = eth_addr >> 16;
++ dev->dev_addr[4] = eth_addr >> 24;
++#else
++ int i;
++ unsigned long fec_mac_base = FEC_IIM_BASE + MXC_IIMKEY0;
+
+- if (FEC_FLASHMAC) {
+- /*
+- * Get MAC address from FLASH.
+- * If it is all 1's or 0's, use the default.
+- */
+- iap = FEC_FLASHMAC;
+- if ((iap[0] == 0) && (iap[1] == 0) && (iap[2] == 0) &&
+- (iap[3] == 0) && (iap[4] == 0) && (iap[5] == 0))
+- iap = fec_mac_default;
+- if ((iap[0] == 0xff) && (iap[1] == 0xff) && (iap[2] == 0xff) &&
+- (iap[3] == 0xff) && (iap[4] == 0xff) && (iap[5] == 0xff))
+- iap = fec_mac_default;
+- } else {
+- *((unsigned long *) &tmpaddr[0]) = fecp->fec_addr_low;
+- *((unsigned short *) &tmpaddr[4]) = (fecp->fec_addr_high >> 16);
+- iap = &tmpaddr[0];
++ if (cpu_is_mx27_rev(CHIP_REV_2_0) > 0) {
++ fec_mac_base = FEC_IIM_BASE + MXC_IIMMAC;
+ }
+
+- memcpy(dev->dev_addr, iap, ETH_ALEN);
+-
+- /* Adjust MAC if using default MAC address */
+- if (iap == fec_mac_default)
+- dev->dev_addr[ETH_ALEN-1] = fec_mac_default[ETH_ALEN-1] + fep->index;
++ DBG(0, "%s: Reading MAC address from %08lx\n", __FUNCTION__, fec_mac_base);
++ for (i = 0; i < ETH_ALEN; i++) {
++ dev->dev_addr[ETH_ALEN - 1 - i] = __raw_readb(fec_mac_base + i * 4);
++ }
++ //memcpy(dev->dev_addr, default_mac, ETH_ALEN);
++#endif
+ }
+
+-static void __inline__ fec_enable_phy_intr(void)
++#ifdef CONFIG_PHYLIB
++static void __inline__ fec_enable_phy_intr(struct fec_enet_private *fep)
+ {
+ }
+-
+-static void __inline__ fec_disable_phy_intr(void)
++static void __inline__ fec_disable_phy_intr(struct fec_enet_private *fep)
+ {
+ }
+-
+-static void __inline__ fec_phy_ack_intr(void)
++static void __inline__ fec_phy_ack_intr(struct fec_enet_private *fep)
+ {
+ }
+-
+-static void __inline__ fec_localhw_setup(void)
++#else
++static void __inline__ fec_enable_phy_intr(struct fec_enet_private *fep)
+ {
+-}
+-
+-/*
+- * Do not need to make region uncached on 5272.
+- */
+-static void __inline__ fec_uncache(unsigned long addr)
+-{
+-}
+-
+-/* ------------------------------------------------------------------------- */
+-
+-#elif defined(CONFIG_M520x)
+-
+-/*
+- * Code specific to Coldfire 520x
+- */
+-static void __inline__ fec_request_intrs(struct net_device *dev)
+-{
+- struct fec_enet_private *fep;
+- int b;
+- static const struct idesc {
+- char *name;
+- unsigned short irq;
+- } *idp, id[] = {
+- { "fec(TXF)", 23 },
+- { "fec(RXF)", 27 },
+- { "fec(MII)", 29 },
+- { NULL },
+- };
+-
+- fep = netdev_priv(dev);
+- b = 64 + 13;
+-
+- /* Setup interrupt handlers. */
+- for (idp = id; idp->name; idp++) {
+- if (request_irq(b+idp->irq, fec_enet_interrupt, IRQF_DISABLED, idp->name,dev) != 0)
+- printk("FEC: Could not allocate %s IRQ(%d)!\n", idp->name, b+idp->irq);
+- }
+-
+- /* Unmask interrupts at ColdFire interrupt controller */
+- {
+- volatile unsigned char *icrp;
+- volatile unsigned long *imrp;
+-
+- icrp = (volatile unsigned char *) (MCF_IPSBAR + MCFICM_INTC0 +
+- MCFINTC_ICR0);
+- for (b = 36; (b < 49); b++)
+- icrp[b] = 0x04;
+- imrp = (volatile unsigned long *) (MCF_IPSBAR + MCFICM_INTC0 +
+- MCFINTC_IMRH);
+- *imrp &= ~0x0001FFF0;
++ if (!fep->phy_int_enabled) {
++ fep->phy_int_enabled = 1;
++ enable_irq(fep->mii_irq);
+ }
+- *(volatile unsigned char *)(MCF_IPSBAR + MCF_GPIO_PAR_FEC) |= 0xf0;
+- *(volatile unsigned char *)(MCF_IPSBAR + MCF_GPIO_PAR_FECI2C) |= 0x0f;
+ }
+
+-static void __inline__ fec_set_mii(struct net_device *dev, struct fec_enet_private *fep)
++static void __inline__ fec_disable_phy_intr(struct fec_enet_private *fep)
+ {
+- volatile fec_t *fecp;
+-
+- fecp = fep->hwp;
+- fecp->fec_r_cntrl = OPT_FRAME_SIZE | 0x04;
+- fecp->fec_x_cntrl = 0x00;
+-
+- /*
+- * Set MII speed to 2.5 MHz
+- * See 5282 manual section 17.5.4.7: MSCR
+- */
+- fep->phy_speed = ((((MCF_CLK / 2) / (2500000 / 10)) + 5) / 10) * 2;
+- fecp->fec_mii_speed = fep->phy_speed;
+-
+- fec_restart(dev, 0);
+-}
+-
+-static void __inline__ fec_get_mac(struct net_device *dev)
+-{
+- struct fec_enet_private *fep = netdev_priv(dev);
+- volatile fec_t *fecp;
+- unsigned char *iap, tmpaddr[ETH_ALEN];
+-
+- fecp = fep->hwp;
+-
+- if (FEC_FLASHMAC) {
+- /*
+- * Get MAC address from FLASH.
+- * If it is all 1's or 0's, use the default.
+- */
+- iap = FEC_FLASHMAC;
+- if ((iap[0] == 0) && (iap[1] == 0) && (iap[2] == 0) &&
+- (iap[3] == 0) && (iap[4] == 0) && (iap[5] == 0))
+- iap = fec_mac_default;
+- if ((iap[0] == 0xff) && (iap[1] == 0xff) && (iap[2] == 0xff) &&
+- (iap[3] == 0xff) && (iap[4] == 0xff) && (iap[5] == 0xff))
+- iap = fec_mac_default;
+- } else {
+- *((unsigned long *) &tmpaddr[0]) = fecp->fec_addr_low;
+- *((unsigned short *) &tmpaddr[4]) = (fecp->fec_addr_high >> 16);
+- iap = &tmpaddr[0];
++ if (fep->phy_int_enabled) {
++ disable_irq(fep->mii_irq);
++ fep->phy_int_enabled = 0;
+ }
+-
+- memcpy(dev->dev_addr, iap, ETH_ALEN);
+-
+- /* Adjust MAC if using default MAC address */
+- if (iap == fec_mac_default)
+- dev->dev_addr[ETH_ALEN-1] = fec_mac_default[ETH_ALEN-1] + fep->index;
+-}
+-
+-static void __inline__ fec_enable_phy_intr(void)
+-{
+-}
+-
+-static void __inline__ fec_disable_phy_intr(void)
+-{
+-}
+-
+-static void __inline__ fec_phy_ack_intr(void)
+-{
+-}
+-
+-static void __inline__ fec_localhw_setup(void)
+-{
+ }
+
+-static void __inline__ fec_uncache(unsigned long addr)
++static void __inline__ fec_phy_ack_intr(struct fec_enet_private *fep)
+ {
+-}
+-
+-/* ------------------------------------------------------------------------- */
+-
+-#elif defined(CONFIG_M532x)
+-/*
+- * Code specific for M532x
+- */
+-static void __inline__ fec_request_intrs(struct net_device *dev)
+-{
+- struct fec_enet_private *fep;
+- int b;
+- static const struct idesc {
+- char *name;
+- unsigned short irq;
+- } *idp, id[] = {
+- { "fec(TXF)", 36 },
+- { "fec(RXF)", 40 },
+- { "fec(MII)", 42 },
+- { NULL },
+- };
+-
+- fep = netdev_priv(dev);
+- b = (fep->index) ? 128 : 64;
+-
+- /* Setup interrupt handlers. */
+- for (idp = id; idp->name; idp++) {
+- if (request_irq(b+idp->irq, fec_enet_interrupt, IRQF_DISABLED, idp->name,dev) != 0)
+- printk("FEC: Could not allocate %s IRQ(%d)!\n",
+- idp->name, b+idp->irq);
+- }
+-
+- /* Unmask interrupts */
+- MCF_INTC0_ICR36 = 0x2;
+- MCF_INTC0_ICR37 = 0x2;
+- MCF_INTC0_ICR38 = 0x2;
+- MCF_INTC0_ICR39 = 0x2;
+- MCF_INTC0_ICR40 = 0x2;
+- MCF_INTC0_ICR41 = 0x2;
+- MCF_INTC0_ICR42 = 0x2;
+- MCF_INTC0_ICR43 = 0x2;
+- MCF_INTC0_ICR44 = 0x2;
+- MCF_INTC0_ICR45 = 0x2;
+- MCF_INTC0_ICR46 = 0x2;
+- MCF_INTC0_ICR47 = 0x2;
+- MCF_INTC0_ICR48 = 0x2;
+-
+- MCF_INTC0_IMRH &= ~(
+- MCF_INTC_IMRH_INT_MASK36 |
+- MCF_INTC_IMRH_INT_MASK37 |
+- MCF_INTC_IMRH_INT_MASK38 |
+- MCF_INTC_IMRH_INT_MASK39 |
+- MCF_INTC_IMRH_INT_MASK40 |
+- MCF_INTC_IMRH_INT_MASK41 |
+- MCF_INTC_IMRH_INT_MASK42 |
+- MCF_INTC_IMRH_INT_MASK43 |
+- MCF_INTC_IMRH_INT_MASK44 |
+- MCF_INTC_IMRH_INT_MASK45 |
+- MCF_INTC_IMRH_INT_MASK46 |
+- MCF_INTC_IMRH_INT_MASK47 |
+- MCF_INTC_IMRH_INT_MASK48 );
+-
+- /* Set up gpio outputs for MII lines */
+- MCF_GPIO_PAR_FECI2C |= (0 |
+- MCF_GPIO_PAR_FECI2C_PAR_MDC_EMDC |
+- MCF_GPIO_PAR_FECI2C_PAR_MDIO_EMDIO);
+- MCF_GPIO_PAR_FEC = (0 |
+- MCF_GPIO_PAR_FEC_PAR_FEC_7W_FEC |
+- MCF_GPIO_PAR_FEC_PAR_FEC_MII_FEC);
+-}
+-
+-static void __inline__ fec_set_mii(struct net_device *dev, struct fec_enet_private *fep)
+-{
+- volatile fec_t *fecp;
+-
+- fecp = fep->hwp;
+- fecp->fec_r_cntrl = OPT_FRAME_SIZE | 0x04;
+- fecp->fec_x_cntrl = 0x00;
+-
+- /*
+- * Set MII speed to 2.5 MHz
+- */
+- fep->phy_speed = ((((MCF_CLK / 2) / (2500000 / 10)) + 5) / 10) * 2;
+- fecp->fec_mii_speed = fep->phy_speed;
+-
+- fec_restart(dev, 0);
+-}
+-
+-static void __inline__ fec_get_mac(struct net_device *dev)
+-{
+- struct fec_enet_private *fep = netdev_priv(dev);
+- volatile fec_t *fecp;
+- unsigned char *iap, tmpaddr[ETH_ALEN];
+-
+- fecp = fep->hwp;
+-
+- if (FEC_FLASHMAC) {
+- /*
+- * Get MAC address from FLASH.
+- * If it is all 1's or 0's, use the default.
+- */
+- iap = FEC_FLASHMAC;
+- if ((iap[0] == 0) && (iap[1] == 0) && (iap[2] == 0) &&
+- (iap[3] == 0) && (iap[4] == 0) && (iap[5] == 0))
+- iap = fec_mac_default;
+- if ((iap[0] == 0xff) && (iap[1] == 0xff) && (iap[2] == 0xff) &&
+- (iap[3] == 0xff) && (iap[4] == 0xff) && (iap[5] == 0xff))
+- iap = fec_mac_default;
+- } else {
+- *((unsigned long *) &tmpaddr[0]) = fecp->fec_addr_low;
+- *((unsigned short *) &tmpaddr[4]) = (fecp->fec_addr_high >> 16);
+- iap = &tmpaddr[0];
++ if (fep->phy_int_enabled) {
++ disable_irq(fep->mii_irq);
++ fep->phy_int_enabled = 0;
+ }
+-
+- memcpy(dev->dev_addr, iap, ETH_ALEN);
+-
+- /* Adjust MAC if using default MAC address */
+- if (iap == fec_mac_default)
+- dev->dev_addr[ETH_ALEN-1] = fec_mac_default[ETH_ALEN-1] + fep->index;
+ }
+-
+-static void __inline__ fec_enable_phy_intr(void)
+-{
+-}
+-
+-static void __inline__ fec_disable_phy_intr(void)
+-{
+-}
+-
+-static void __inline__ fec_phy_ack_intr(void)
+-{
+-}
+-
+-static void __inline__ fec_localhw_setup(void)
+-{
+-}
+-
+-/*
+- * Do not need to make region uncached on 532x.
+- */
+-static void __inline__ fec_uncache(unsigned long addr)
+-{
+-}
+-
+-/* ------------------------------------------------------------------------- */
+-
+-
+-#else
+-
+-/*
+- * Code specific to the MPC860T setup.
+- */
+-static void __inline__ fec_request_intrs(struct net_device *dev)
+-{
+- volatile immap_t *immap;
+-
+- immap = (immap_t *)IMAP_ADDR; /* pointer to internal registers */
+-
+- if (request_8xxirq(FEC_INTERRUPT, fec_enet_interrupt, 0, "fec", dev) != 0)
+- panic("Could not allocate FEC IRQ!");
+-}
+-
+-static void __inline__ fec_get_mac(struct net_device *dev)
+-{
+- bd_t *bd;
+-
+- bd = (bd_t *)__res;
+- memcpy(dev->dev_addr, bd->bi_enetaddr, ETH_ALEN);
+-}
+-
+-static void __inline__ fec_set_mii(struct net_device *dev, struct fec_enet_private *fep)
+-{
+- extern uint _get_IMMR(void);
+- volatile immap_t *immap;
+- volatile fec_t *fecp;
+-
+- fecp = fep->hwp;
+- immap = (immap_t *)IMAP_ADDR; /* pointer to internal registers */
+-
+- /* Configure all of port D for MII.
+- */
+- immap->im_ioport.iop_pdpar = 0x1fff;
+-
+- /* Bits moved from Rev. D onward.
+- */
+- if ((_get_IMMR() & 0xffff) < 0x0501)
+- immap->im_ioport.iop_pddir = 0x1c58; /* Pre rev. D */
+- else
+- immap->im_ioport.iop_pddir = 0x1fff; /* Rev. D and later */
+-
+- /* Set MII speed to 2.5 MHz
+- */
+- fecp->fec_mii_speed = fep->phy_speed =
+- ((bd->bi_busfreq * 1000000) / 2500000) & 0x7e;
+-}
+-
+-static void __inline__ fec_enable_phy_intr(void)
+-{
+- volatile fec_t *fecp;
+-
+- fecp = fep->hwp;
+-
+- /* Enable MII command finished interrupt
+- */
+- fecp->fec_ivec = (FEC_INTERRUPT/2) << 29;
+-}
+-
+-static void __inline__ fec_disable_phy_intr(void)
+-{
+-}
+-
+-static void __inline__ fec_phy_ack_intr(void)
+-{
+-}
+-
+-static void __inline__ fec_localhw_setup(void)
+-{
+- volatile fec_t *fecp;
+-
+- fecp = fep->hwp;
+- fecp->fec_r_hash = PKT_MAXBUF_SIZE;
+- /* Enable big endian and don't care about SDMA FC.
+- */
+- fecp->fec_fun_code = 0x78000000;
+-}
+-
+-static void __inline__ fec_uncache(unsigned long addr)
+-{
+- pte_t *pte;
+- pte = va_to_pte(mem_addr);
+- pte_val(*pte) |= _PAGE_NO_CACHE;
+- flush_tlb_page(init_mm.mmap, mem_addr);
+-}
+-
+ #endif
+
+ /* ------------------------------------------------------------------------- */
+
++#ifndef CONFIG_PHYLIB
+ static void mii_display_status(struct net_device *dev)
+ {
+ struct fec_enet_private *fep = netdev_priv(dev);
+- volatile uint *s = &(fep->phy_status);
+
+- if (!fep->link && !fep->old_link) {
++ if (!fep->linkstatus && !fep->old_linkstatus) {
+ /* Link is still down - don't print anything */
+ return;
+ }
+
+ printk("%s: status: ", dev->name);
+
+- if (!fep->link) {
++ if (!fep->linkstatus) {
+ printk("link down");
+ } else {
+ printk("link up");
+
+- switch(*s & PHY_STAT_SPMASK) {
++ switch(fep->phy_status & PHY_STAT_SPMASK) {
+ case PHY_STAT_100FDX: printk(", 100MBit Full Duplex"); break;
+ case PHY_STAT_100HDX: printk(", 100MBit Half Duplex"); break;
+ case PHY_STAT_10FDX: printk(", 10MBit Full Duplex"); break;
+@@ -1880,20 +1687,19 @@ static void mii_display_status(struct ne
+ printk(", Unknown speed/duplex");
+ }
+
+- if (*s & PHY_STAT_ANC)
++ if (fep->phy_status & PHY_STAT_ANC)
+ printk(", auto-negotiation complete");
+ }
+
+- if (*s & PHY_STAT_FAULT)
++ if (fep->phy_status & PHY_STAT_FAULT)
+ printk(", remote fault");
+
+ printk(".\n");
+ }
+
+-static void mii_display_config(struct work_struct *work)
++static void mii_display_config(struct work_struct *w)
+ {
+- struct fec_enet_private *fep = container_of(work, struct fec_enet_private, phy_task);
+- struct net_device *dev = fep->netdev;
++ struct fec_enet_private *fep = container_of(w, struct fec_enet_private, phy_task);
+ uint status = fep->phy_status;
+
+ /*
+@@ -1901,7 +1707,7 @@ static void mii_display_config(struct wo
+ ** the workqueue. It is thus safe to allow to reuse it.
+ */
+ fep->mii_phy_task_queued = 0;
+- printk("%s: config: auto-negotiation ", dev->name);
++ //printk("%s: config: auto-negotiation ", dev->name);
+
+ if (status & PHY_CONF_ANE)
+ printk("on");
+@@ -1926,11 +1732,21 @@ static void mii_display_config(struct wo
+
+ fep->sequence_done = 1;
+ }
++#endif
++
++#ifndef CONFIG_PHYLIB
++static inline void *priv_netdev(struct fec_enet_private *fep)
++{
++ /* ugly hack, stolen from include linux/netdevice.h */
++ return (char *)fep - ((sizeof(struct net_device)
++ + NETDEV_ALIGN_CONST)
++ & ~NETDEV_ALIGN_CONST);
++}
+
+-static void mii_relink(struct work_struct *work)
++static void mii_relink(struct work_struct *w)
+ {
+- struct fec_enet_private *fep = container_of(work, struct fec_enet_private, phy_task);
+- struct net_device *dev = fep->netdev;
++ struct fec_enet_private *fep = container_of(w, struct fec_enet_private, phy_task);
++ struct net_device *dev = priv_netdev(fep);
+ int duplex;
+
+ /*
+@@ -1938,23 +1754,19 @@ static void mii_relink(struct work_struc
+ ** the workqueue. It is thus safe to allow to reuse it.
+ */
+ fep->mii_phy_task_queued = 0;
+- fep->link = (fep->phy_status & PHY_STAT_LINK) ? 1 : 0;
++ fep->linkstatus = (fep->phy_status & PHY_STAT_LINK) ? 1 : 0;
+ mii_display_status(dev);
+- fep->old_link = fep->link;
++ fep->old_linkstatus = fep->linkstatus;
+
+- if (fep->link) {
++ if (fep->linkstatus) {
+ duplex = 0;
+- if (fep->phy_status
+- & (PHY_STAT_100FDX | PHY_STAT_10FDX))
++ if (fep->phy_status & (PHY_STAT_100FDX | PHY_STAT_10FDX)) {
+ duplex = 1;
++ }
+ fec_restart(dev, duplex);
+- } else
++ } else {
+ fec_stop(dev);
+-
+-#if 0
+- enable_irq(fep->mii_irq);
+-#endif
+-
++ }
+ }
+
+ /* mii_queue_relink is called in interrupt context from mii_link_interrupt */
+@@ -2004,15 +1816,14 @@ phy_cmd_t const phy_cmd_config[] = {
+ static void
+ mii_discover_phy3(uint mii_reg, struct net_device *dev)
+ {
+- struct fec_enet_private *fep;
++ struct fec_enet_private *fep = netdev_priv(dev);
+ int i;
+
+- fep = netdev_priv(dev);
+ fep->phy_id |= (mii_reg & 0xffff);
+ printk("fec: PHY @ 0x%x, ID 0x%08x", fep->phy_addr, fep->phy_id);
+
+- for(i = 0; phy_info[i]; i++) {
+- if(phy_info[i]->id == (fep->phy_id >> 4))
++ for (i = 0; phy_info[i]; i++) {
++ if (phy_info[i]->id == (fep->phy_id >> 4))
+ break;
+ }
+
+@@ -2031,13 +1842,9 @@ mii_discover_phy3(uint mii_reg, struct n
+ static void
+ mii_discover_phy(uint mii_reg, struct net_device *dev)
+ {
+- struct fec_enet_private *fep;
+- volatile fec_t *fecp;
++ struct fec_enet_private *fep = netdev_priv(dev);
+ uint phytype;
+
+- fep = netdev_priv(dev);
+- fecp = fep->hwp;
+-
+ if (fep->phy_addr < 32) {
+ if ((phytype = (mii_reg & 0xffff)) != 0xffff && phytype != 0) {
+
+@@ -2045,37 +1852,41 @@ mii_discover_phy(uint mii_reg, struct ne
+ */
+ fep->phy_id = phytype << 16;
+ mii_queue(dev, mk_mii_read(MII_REG_PHYIR2),
+- mii_discover_phy3);
++ mii_discover_phy3);
+ } else {
+ fep->phy_addr++;
+ mii_queue(dev, mk_mii_read(MII_REG_PHYIR1),
+- mii_discover_phy);
++ mii_discover_phy);
+ }
+ } else {
+ printk("FEC: No PHY device found.\n");
+ /* Disable external MII interface */
+- fecp->fec_mii_speed = fep->phy_speed = 0;
+- fec_disable_phy_intr();
++ fep->phy_speed = 0;
++ fec_reg_write(fep, FEC_MSCR, fep->phy_speed);
++ fec_disable_phy_intr(fep);
+ }
+ }
++#endif
+
+-/* This interrupt occurs when the PHY detects a link change.
+-*/
+-#ifdef HAVE_mii_link_interrupt
++#ifndef CONFIG_PHYLIB
+ static irqreturn_t
+-mii_link_interrupt(int irq, void * dev_id)
++mii_link_interrupt(int irq, void *dev_id)
+ {
+- struct net_device *dev = dev_id;
++ struct net_device *dev = dev_id;
+ struct fec_enet_private *fep = netdev_priv(dev);
+
+- fec_phy_ack_intr();
++ DBG(0, "%s: \n", __FUNCTION__);
+
+-#if 0
+- disable_irq(fep->mii_irq); /* disable now, enable later */
+-#endif
++ fec_phy_ack_intr(fep);
+
+- mii_do_cmd(dev, fep->phy->ack_int);
+- mii_do_cmd(dev, phy_cmd_relink); /* restart and display status */
++ /*
++ * Some board will trigger phy interrupt before phy enable.
++ * And at that moment , fep->phy is not initialized.
++ */
++ if (fep->phy) {
++ mii_do_cmd(dev, fep->phy->ack_int);
++ mii_do_cmd(dev, phy_cmd_relink); /* restart and display status */
++ }
+
+ return IRQ_HANDLED;
+ }
+@@ -2084,16 +1895,29 @@ mii_link_interrupt(int irq, void * dev_i
+ static int
+ fec_enet_open(struct net_device *dev)
+ {
++ int ret = 0;
+ struct fec_enet_private *fep = netdev_priv(dev);
+
+ /* I should reset the ring buffers here, but I don't yet know
+ * a simple way to do that.
+ */
+- fec_set_mac_address(dev);
++ DBG(0, "%s: \n", __FUNCTION__);
++ _fec_set_mac_address(dev);
+
+- fep->sequence_done = 0;
+- fep->link = 0;
++#ifdef CONFIG_PHYLIB
++ ret = fec_connect_phy(dev, fep);
++ if (ret != 0) {
++ DBG(0, "%s: Failed to connect to PHY: %d\n", __FUNCTION__, ret);
++ return ret;
++ }
++ phy_start(fep->phy);
+
++ fep->linkstatus = fep->phy->link;
++ fec_restart(dev, 0);
++ DBG(0, "%s: Link status is: %d\n", __FUNCTION__, fep->linkstatus);
++#else
++ fep->linkstatus = 0;
++ fep->sequence_done = 0;
+ if (fep->phy) {
+ mii_do_cmd(dev, fep->phy->ack_int);
+ mii_do_cmd(dev, fep->phy->config);
+@@ -2115,16 +1939,16 @@ fec_enet_open(struct net_device *dev)
+ * based on this device does not implement a PHY interrupt,
+ * so we are never notified of link change.
+ */
+- fep->link = 1;
++ fep->linkstatus = 1;
+ } else {
+- fep->link = 1; /* lets just try it and see */
++ fep->linkstatus = 1; /* lets just try it and see */
+ /* no phy, go full duplex, it's most likely a hub chip */
+ fec_restart(dev, 1);
+ }
+-
+- netif_start_queue(dev);
++ fep->old_linkstatus = fep->linkstatus;
++#endif
+ fep->opened = 1;
+- return 0; /* Success */
++ return ret;
+ }
+
+ static int
+@@ -2132,15 +1956,30 @@ fec_enet_close(struct net_device *dev)
+ {
+ struct fec_enet_private *fep = netdev_priv(dev);
+
+- /* Don't know what to do yet.
+- */
+- fep->opened = 0;
+- netif_stop_queue(dev);
+- fec_stop(dev);
++#ifdef CONFIG_PHYLIB
++ if (fep->phy) {
++ DBG(0, "%s: Stopping PHY %p\n", __FUNCTION__, fep->phy);
++ phy_stop(fep->phy);
+
++ DBG(0, "%s: Disconnecting PHY %p\n", __FUNCTION__, fep->phy);
++ phy_disconnect(fep->phy);
++ fep->phy = NULL;
++ }
++#endif
++ fep->opened = 0;
++ if (fep->linkstatus) {
++ fec_stop(dev);
++ }
+ return 0;
+ }
+
++static struct net_device_stats *fec_enet_get_stats(struct net_device *dev)
++{
++ struct fec_enet_private *fep = netdev_priv(dev);
++
++ return &fep->stats;
++}
++
+ /* Set or clear the multicast filter for this adaptor.
+ * Skeleton taken from sunlance driver.
+ * The CPM Ethernet implementation allows Multicast as well as individual
+@@ -2156,37 +1995,32 @@ fec_enet_close(struct net_device *dev)
+
+ static void set_multicast_list(struct net_device *dev)
+ {
+- struct fec_enet_private *fep;
+- volatile fec_t *ep;
++ struct fec_enet_private *fep = netdev_priv(dev);
+ struct dev_mc_list *dmi;
+ unsigned int i, j, bit, data, crc;
+ unsigned char hash;
+
+- fep = netdev_priv(dev);
+- ep = fep->hwp;
+-
+- if (dev->flags&IFF_PROMISC) {
+- ep->fec_r_cntrl |= 0x0008;
++ if (dev->flags & IFF_PROMISC) {
++ fec_reg_write(fep, FEC_RCR, fec_reg_read(fep, FEC_RCR) | 0x0008);
+ } else {
+
+- ep->fec_r_cntrl &= ~0x0008;
++ fec_reg_write(fep, FEC_RCR, fec_reg_read(fep, FEC_RCR) & ~0x0008);
+
+ if (dev->flags & IFF_ALLMULTI) {
+ /* Catch all multicast addresses, so set the
+ * filter to all 1's.
+ */
+- ep->fec_grp_hash_table_high = 0xffffffff;
+- ep->fec_grp_hash_table_low = 0xffffffff;
++ fec_reg_write(fep, FEC_IAUR, 0xffffffff);
++ fec_reg_write(fep, FEC_IALR, 0xffffffff);
+ } else {
+ /* Clear filter and add the addresses in hash register.
+ */
+- ep->fec_grp_hash_table_high = 0;
+- ep->fec_grp_hash_table_low = 0;
++ fec_reg_write(fep, FEC_IAUR, 0);
++ fec_reg_write(fep, FEC_IALR, 0);
+
+ dmi = dev->mc_list;
+
+- for (j = 0; j < dev->mc_count; j++, dmi = dmi->next)
+- {
++ for (j = 0; j < dev->mc_count; j++, dmi = dmi->next) {
+ /* Only support group multicast for now.
+ */
+ if (!(dmi->dmi_addr[0] & 1))
+@@ -2196,11 +2030,9 @@ static void set_multicast_list(struct ne
+ */
+ crc = 0xffffffff;
+
+- for (i = 0; i < dmi->dmi_addrlen; i++)
+- {
++ for (i = 0; i < dmi->dmi_addrlen; i++) {
+ data = dmi->dmi_addr[i];
+- for (bit = 0; bit < 8; bit++, data >>= 1)
+- {
++ for (bit = 0; bit < 8; bit++, data >>= 1) {
+ crc = (crc >> 1) ^
+ (((crc ^ data) & 1) ? CRC32_POLY : 0);
+ }
+@@ -2212,9 +2044,13 @@ static void set_multicast_list(struct ne
+ hash = (crc >> (32 - HASH_BITS)) & 0x3f;
+
+ if (hash > 31)
+- ep->fec_grp_hash_table_high |= 1 << (hash - 32);
++ fec_reg_write(fep, FEC_IAUR,
++ fec_reg_read(fep, FEC_IAUR) |
++ (1 << (hash - 32)));
+ else
+- ep->fec_grp_hash_table_low |= 1 << hash;
++ fec_reg_write(fep, FEC_IALR,
++ fec_reg_read(fep, FEC_IALR) |
++ (1 << hash));
+ }
+ }
+ }
+@@ -2223,61 +2059,211 @@ static void set_multicast_list(struct ne
+ /* Set a MAC change in hardware.
+ */
+ static void
+-fec_set_mac_address(struct net_device *dev)
++_fec_set_mac_address(struct net_device *dev)
+ {
+- volatile fec_t *fecp;
+-
+- fecp = ((struct fec_enet_private *)netdev_priv(dev))->hwp;
++ struct fec_enet_private *fep = netdev_priv(dev);
+
+ /* Set station address. */
+- fecp->fec_addr_low = dev->dev_addr[3] | (dev->dev_addr[2] << 8) |
+- (dev->dev_addr[1] << 16) | (dev->dev_addr[0] << 24);
+- fecp->fec_addr_high = (dev->dev_addr[5] << 16) |
+- (dev->dev_addr[4] << 24);
++ fec_reg_write(fep, FEC_PALR, dev->dev_addr[3] | (dev->dev_addr[2] << 8) |
++ (dev->dev_addr[1] << 16) | (dev->dev_addr[0] << 24));
++ fec_reg_write(fep, FEC_PAUR, (dev->dev_addr[5] << 16) |
++ (dev->dev_addr[4] << 24));
++}
++
++static int
++fec_set_mac_address(struct net_device *dev, void *_addr)
++{
++ struct sockaddr *addr = _addr;
+
++ if (!is_valid_ether_addr((const char *)&addr->sa_data)) {
++ printk(KERN_WARNING "Bad ethernet address: %02x:%02x:%02x:%02x:%02x:%02x\n",
++ addr->sa_data[0], addr->sa_data[1], addr->sa_data[2], addr->sa_data[3],
++ addr->sa_data[4], addr->sa_data[5]);
++ return -EINVAL;
++ }
++ printk(KERN_DEBUG "Setting MAC address to %02x:%02x:%02x:%02x:%02x:%02x\n",
++ addr->sa_data[0], addr->sa_data[1], addr->sa_data[2], addr->sa_data[3],
++ addr->sa_data[4], addr->sa_data[5]);
++
++ memcpy(&dev->dev_addr, &addr->sa_data, ETH_ALEN);
++
++ _fec_set_mac_address(dev);
++
++ return 0;
+ }
+
+-/* Initialize the FEC Ethernet on 860T (or ColdFire 5272).
+- */
+- /*
+- * XXX: We need to clean up on failure exits here.
+- */
+-int __init fec_enet_init(struct net_device *dev)
++static void fec_enet_free_buffers(struct fec_enet_private *fep)
+ {
++ cbd_t *bdp = fep->rx_bd_base;
++ int i;
++
++ DBG(0, "%s: Freeing TX bounce buffers %p\n", __FUNCTION__, fep->tx_bounce[0]);
++ kfree(fep->tx_bounce[0]);
++ memset(fep->tx_bounce, 0, TX_RING_SIZE * sizeof(void*));
++ for (i = 0; i < RX_RING_SIZE; i++, bdp++) {
++ if (fep->rx_skbuff[i] != NULL) {
++ DBG(0, "%s: Freeing RX skb %p\n", __FUNCTION__, fep->rx_skbuff[i]);
++ fec_enet_rxbuf_unmap(fep, bdp, FEC_ENET_RX_FRSIZE);
++ kfree_skb(fep->rx_skbuff[i]);
++ fep->rx_skbuff[i] = NULL;
++ }
++ }
++}
++
++#ifdef CONFIG_PHYLIB
++/* called by the generic PHY layer in interrupt context */
++static int fec_mii_read(struct mii_bus *bus, int phy_id, int regnum)
++{
++ int ret;
++ struct net_device *dev = bus->priv;
+ struct fec_enet_private *fep = netdev_priv(dev);
+- unsigned long mem_addr;
+- volatile cbd_t *bdp;
+- cbd_t *cbd_base;
+- volatile fec_t *fecp;
+- int i, j;
+- static int index = 0;
++ unsigned long regval = mk_mii_read(regnum) | phy_id << 23;
++ unsigned long flags;
++ static int regs[32] = { [0 ... ARRAY_SIZE(regs) - 1] = -1};
+
+- /* Only allow us to be probed once. */
+- if (index >= FEC_MAX_PORTS)
+- return -ENXIO;
++ spin_lock_irqsave(&fep->lock, flags);
++ fep->mii_complete = 0;
++ fec_reg_write(fep, FEC_MMFR, regval);
++ spin_unlock_irqrestore(&fep->lock, flags);
++
++ while (!fep->mii_complete) {
++ cpu_relax();
++ }
++
++ ret = fec_reg_read(fep, FEC_MMFR) & 0xffff;
++ if (ret < 0) {
++ DBG(0, "%s: Failed to read PHY[%02x] reg %02x: %d\n", __FUNCTION__,
++ phy_id, regnum, ret);
++ } else if (regs[regnum] != ret) {
++ DBG(1, "%s: Read %04x from PHY[%02x] reg %02x\n", __FUNCTION__,
++ ret, phy_id, regnum);
++ regs[regnum] = ret;
++ }
++ return ret;
++}
+
+- /* Allocate memory for buffer descriptors.
+- */
+- mem_addr = __get_free_page(GFP_KERNEL);
+- if (mem_addr == 0) {
+- printk("FEC: allocate descriptor memory failed?\n");
++static int fec_mii_write(struct mii_bus *bus, int phy_id, int regnum, u16 val)
++{
++ struct net_device *dev = bus->priv;
++ struct fec_enet_private *fep = netdev_priv(dev);
++ unsigned long regval = mk_mii_write(regnum, val) | phy_id << 23;
++ unsigned long flags;
++
++ spin_lock_irqsave(&fep->lock, flags);
++ fep->mii_complete = 0;
++ fec_reg_write(fep, FEC_MMFR, regval);
++ spin_unlock_irqrestore(&fep->lock, flags);
++
++ while (!fep->mii_complete) {
++ cpu_relax();
++ }
++ DBG(1, "%s: Wrote %04x to PHY[%02x] reg %02x\n", __FUNCTION__, val, phy_id, regnum);
++ return 0;
++}
++
++static int fec_mii_reset(struct mii_bus *bus)
++{
++ return 0;
++}
++
++static int fec_init_phy(struct net_device *dev, struct fec_enet_private *fep)
++{
++ int ret;
++ int i;
++ struct mii_bus *mii;
++
++ mii = mdiobus_alloc();
++ if (mii == NULL) {
+ return -ENOMEM;
+ }
++ mii->name = "fec mii";
++ mii->read = fec_mii_read;
++ mii->write = fec_mii_write;
++ mii->reset = fec_mii_reset;
++ mii->priv = dev;
++ snprintf(mii->id, MII_BUS_ID_SIZE, "%x", 0);
++ mii->irq = kmalloc(sizeof(int) * PHY_MAX_ADDR, GFP_KERNEL);
++ for (i = 0; i < PHY_MAX_ADDR; i++) {
++ mii->irq[i] = fep->mii_irq >= 0 ? fep->mii_irq : PHY_POLL;
++ }
++
++ ret = mdiobus_register(mii);
++ if (ret != 0) {
++ DBG(0, "%s: Failed to register MII bus: %d\n", __FUNCTION__, ret);
++ kfree(mii->irq);
++ mdiobus_free(mii);
++ return ret;
++ }
++ fep->phy_addr = -1;
++ DBG(0, "%s: MII bus registered\n", __FUNCTION__);
++ for (i = 0; i < PHY_MAX_ADDR; i++) {
++ if (mii->phy_map[i] != NULL) {
++ fep->phy_addr = i;
++ break;
++ }
++ }
++ if (fep->phy_addr == -1) {
++ DBG(0, "%s: No PHY found\n", __FUNCTION__);
++ return -ENODEV;
++ }
++ DBG(0, "%s: Using PHY at addr %02x\n", __FUNCTION__, fep->phy_addr);
++ fep->mii = mii;
+
+- spin_lock_init(&fep->hw_lock);
+- spin_lock_init(&fep->mii_lock);
++ return 0;
++}
+
+- /* Create an Ethernet device instance.
+- */
+- fecp = (volatile fec_t *) fec_hw[index];
++static int fec_connect_phy(struct net_device *dev, struct fec_enet_private *fep)
++{
++ struct mii_bus *mii = fep->mii;
++
++ fep->phy = phy_connect(dev, mii->phy_map[fep->phy_addr]->dev.bus_id,
++ fec_link_change, 0, mii->phy_map[fep->phy_addr]->interface);
++ if (IS_ERR(fep->phy)) {
++ int ret = PTR_ERR(fep->phy);
++ printk(KERN_ERR "%s: Could not attach to PHY\n", dev->name);
++ fep->phy = NULL;
++ return ret;
++ }
++ DBG(0, "%s: Registered PHY %s[%02x] IRQ %d with %s\n", __FUNCTION__,
++ fep->phy->dev.bus_id, fep->phy_addr, fep->phy->irq, dev->name);
++
++ return 0;
++}
++#else
++static int fec_init_phy(struct net_device *dev, struct fec_enet_private *fep)
++{
++ /* Queue up command to detect the PHY and initialize the
++ * remainder of the interface.
++ */
++ fep->phy_id_done = 0;
++ fep->phy_addr = 0;
++ mii_queue(dev, mk_mii_read(MII_REG_PHYIR1), mii_discover_phy);
++
++ return 0;
++}
++#endif
+
+- fep->index = index;
+- fep->hwp = fecp;
+- fep->netdev = dev;
++/* Initialize the FEC Ethernet on 860T (or ColdFire 5272).
++ */
++ /*
++ * XXX: We need to clean up on failure exits here.
++ */
++#define res_len(r) ((r)->end - (r)->start + 1)
++
++int __devinit fec_enet_init(struct platform_device *pdev, struct net_device *dev)
++{
++ int ret;
++ struct fec_enet_private *fep = netdev_priv(dev);
++ cbd_t *bdp;
++ struct sk_buff *pskb;
++ int i;
++ void *mem;
++
++ spin_lock_init(&fep->lock);
+
+ /* Whack a reset. We should wait for this.
+ */
+- fecp->fec_ecntrl = 1;
++ fec_reg_write(fep, FEC_ECR, 1);
+ udelay(10);
+
+ /* Set the Ethernet address. If using multiple Enets on the 8xx,
+@@ -2288,41 +2274,36 @@ int __init fec_enet_init(struct net_devi
+ */
+ fec_get_mac(dev);
+
+- cbd_base = (cbd_t *)mem_addr;
+- /* XXX: missing check for allocation failure */
+-
+- fec_uncache(mem_addr);
+-
+- /* Set receive and transmit descriptor base.
+- */
+- fep->rx_bd_base = cbd_base;
+- fep->tx_bd_base = cbd_base + RX_RING_SIZE;
+-
+ fep->dirty_tx = fep->cur_tx = fep->tx_bd_base;
+ fep->cur_rx = fep->rx_bd_base;
+
+ fep->skb_cur = fep->skb_dirty = 0;
+
+- /* Initialize the receive buffer descriptors.
+- */
+- bdp = fep->rx_bd_base;
+- for (i=0; i<FEC_ENET_RX_PAGES; i++) {
++ /* allocate memory for TX bounce buffers */
++ mem = kzalloc(TX_RING_SIZE * FEC_ENET_TX_FRSIZE, GFP_KERNEL);
++ if (mem == NULL) {
++ return -ENOMEM;
++ }
+
+- /* Allocate a page.
+- */
+- mem_addr = __get_free_page(GFP_KERNEL);
+- /* XXX: missing check for allocation failure */
++ fec_enet_cbd_get(fep);
+
+- fec_uncache(mem_addr);
++ /* Initialize the transmit buffer descriptors.
++ */
++ bdp = fep->tx_bd_base;
++
++ DBG(0, "%s: Allocated %d byte of TX buffer memory @ %p\n", __FUNCTION__,
++ TX_RING_SIZE * FEC_ENET_TX_FRSIZE, mem);
++ for (i = 0; i < TX_RING_SIZE; i++) {
++ fep->tx_bounce[i] = mem;
++ DBG(0, "%s: TX bounce buffer[%d]=%p\n", __FUNCTION__, i, fep->tx_bounce[i]);
++ mem = (void *)((unsigned long)(mem + FEC_ENET_TX_FRSIZE));
+
+ /* Initialize the BD for every fragment in the page.
+ */
+- for (j=0; j<FEC_ENET_RX_FRPPG; j++) {
+- bdp->cbd_sc = BD_ENET_RX_EMPTY;
+- bdp->cbd_bufaddr = __pa(mem_addr);
+- mem_addr += FEC_ENET_RX_FRSIZE;
+- bdp++;
+- }
++ /* already zeroed by kzalloc */
++ //bdp->cbd_sc = 0;
++ bdp->cbd_bufaddr = ~0;
++ bdp++;
+ }
+
+ /* Set the last buffer to wrap.
+@@ -2330,80 +2311,77 @@ int __init fec_enet_init(struct net_devi
+ bdp--;
+ bdp->cbd_sc |= BD_SC_WRAP;
+
+- /* ...and the same for transmmit.
++ /* ...and the same for receive.
+ */
+- bdp = fep->tx_bd_base;
+- for (i=0, j=FEC_ENET_TX_FRPPG; i<TX_RING_SIZE; i++) {
+- if (j >= FEC_ENET_TX_FRPPG) {
+- mem_addr = __get_free_page(GFP_KERNEL);
+- j = 1;
+- } else {
+- mem_addr += FEC_ENET_TX_FRSIZE;
+- j++;
++ bdp = fep->rx_bd_base;
++ for (i = 0; i < RX_RING_SIZE; i++, bdp++) {
++ pskb = __dev_alloc_skb(FEC_ENET_RX_FRSIZE, GFP_KERNEL);
++ if (pskb == NULL) {
++ DBG(0, "%s: Failed to allocate RX skb; cleaning up\n", __FUNCTION__);
++ fec_enet_free_buffers(fep);
++ fec_enet_cbd_put(fep);
++ return -ENOMEM;
+ }
+- fep->tx_bounce[i] = (unsigned char *) mem_addr;
+-
+- /* Initialize the BD for every fragment in the page.
+- */
+- bdp->cbd_sc = 0;
+- bdp->cbd_bufaddr = 0;
+- bdp++;
++ DBG(0, "%s: RX skb allocated @ %p\n", __FUNCTION__, pskb);
++ fep->rx_skbuff[i] = pskb;
++ pskb->data = FEC_ADDR_ALIGNMENT(pskb->data);
++ bdp->cbd_sc = BD_ENET_RX_EMPTY;
++ bdp->cbd_bufaddr = ~0;
++ fec_enet_rxbuf_map(fep, bdp, pskb->data, FEC_ENET_RX_FRSIZE);
+ }
+-
+ /* Set the last buffer to wrap.
+ */
+ bdp--;
+ bdp->cbd_sc |= BD_SC_WRAP;
++ fec_enet_cbd_put(fep);
+
+ /* Set receive and transmit descriptor base.
+ */
+- fecp->fec_r_des_start = __pa((uint)(fep->rx_bd_base));
+- fecp->fec_x_des_start = __pa((uint)(fep->tx_bd_base));
+-
++ fec_reg_write(fep, FEC_ERDSR, fep->cbd_phys_base);
++ fec_reg_write(fep, FEC_ETDSR, fep->cbd_phys_base + RX_RING_SIZE * sizeof(cbd_t));
++
+ /* Install our interrupt handlers. This varies depending on
+ * the architecture.
+ */
+- fec_request_intrs(dev);
+-
+- fecp->fec_grp_hash_table_high = 0;
+- fecp->fec_grp_hash_table_low = 0;
+- fecp->fec_r_buff_size = PKT_MAXBLR_SIZE;
+- fecp->fec_ecntrl = 2;
+- fecp->fec_r_des_active = 0;
+-#ifndef CONFIG_M5272
+- fecp->fec_hash_table_high = 0;
+- fecp->fec_hash_table_low = 0;
+-#endif
+-
+- dev->base_addr = (unsigned long)fecp;
+-
++ ret = fec_request_intrs(pdev, dev);
++ if (ret != 0) {
++ return ret;
++ }
++ /* Clear and enable interrupts */
++ fec_reg_write(fep, FEC_EIR, fec_reg_read(fep, FEC_EIR));
++ fec_reg_write(fep, FEC_EIMR, FEC_ENET_TXF | FEC_ENET_TXB |
++ FEC_ENET_RXF | FEC_ENET_RXB | FEC_ENET_MII);
++
++ fec_reg_write(fep, FEC_IAUR, 0);
++ fec_reg_write(fep, FEC_IALR, 0);
++ fec_reg_write(fep, FEC_EMRBR, PKT_MAXBLR_SIZE);
++ fec_reg_write(fep, FEC_ECR, 2);
++ fec_reg_write(fep, FEC_RDAR, DONT_CARE);
++
+ /* The FEC Ethernet specific entries in the device structure. */
+ dev->open = fec_enet_open;
+ dev->hard_start_xmit = fec_enet_start_xmit;
+ dev->tx_timeout = fec_timeout;
+ dev->watchdog_timeo = TX_TIMEOUT;
+ dev->stop = fec_enet_close;
++ dev->get_stats = fec_enet_get_stats;
+ dev->set_multicast_list = set_multicast_list;
++ dev->set_mac_address = fec_set_mac_address;
+
+- for (i=0; i<NMII-1; i++)
+- mii_cmds[i].mii_next = &mii_cmds[i+1];
++#ifndef CONFIG_PHYLIB
++ for (i = 1; i < NMII; i++) {
++ mii_cmds[i - 1].mii_next = &mii_cmds[i];
++ }
+ mii_free = mii_cmds;
+-
++#endif
+ /* setup MII interface */
+ fec_set_mii(dev, fep);
+
+- /* Clear and enable interrupts */
+- fecp->fec_ievent = 0xffc00000;
+- fecp->fec_imask = (FEC_ENET_TXF | FEC_ENET_RXF | FEC_ENET_MII);
+-
+- /* Queue up command to detect the PHY and initialize the
+- * remainder of the interface.
+- */
+- fep->phy_id_done = 0;
+- fep->phy_addr = 0;
+- mii_queue(dev, mk_mii_read(MII_REG_PHYIR1), mii_discover_phy);
+-
+- index++;
++ ret = fec_init_phy(dev, fep);
++ if (ret) {
++ DBG(0, "%s: Failed to initialize PHY: %d\n", __FUNCTION__, ret);
++ return ret;
++ }
+ return 0;
+ }
+
+@@ -2414,62 +2392,67 @@ int __init fec_enet_init(struct net_devi
+ static void
+ fec_restart(struct net_device *dev, int duplex)
+ {
+- struct fec_enet_private *fep;
+- volatile cbd_t *bdp;
+- volatile fec_t *fecp;
++ struct fec_enet_private *fep = netdev_priv(dev);
++ cbd_t *bdp;
+ int i;
++ u32 rcr = OPT_FRAME_SIZE | RCR_MII_MODE; /* MII enable */
++ u32 tcr = TCR_HBC;
+
+- fep = netdev_priv(dev);
+- fecp = fep->hwp;
+-
++ DBG(0, "%s: Restarting FEC in %s-duplex mode\n", __FUNCTION__,
++ duplex ? "full" : "half");
+ /* Whack a reset. We should wait for this.
+ */
+- fecp->fec_ecntrl = 1;
++ fec_reg_write(fep, FEC_ECR, 1);
+ udelay(10);
+
++ /* Enable interrupts we wish to service.
++ */
++ fec_reg_write(fep, FEC_EIMR, FEC_ENET_TXF | FEC_ENET_TXB |
++ FEC_ENET_RXF | FEC_ENET_RXB | FEC_ENET_MII);
++
+ /* Clear any outstanding interrupt.
+- */
+- fecp->fec_ievent = 0xffc00000;
+- fec_enable_phy_intr();
++ *
++ */
++ fec_reg_write(fep, FEC_EIR, FEC_ENET_MASK);
++
++ fec_enable_phy_intr(fep);
+
+ /* Set station address.
+ */
+- fec_set_mac_address(dev);
++ _fec_set_mac_address(dev);
+
+ /* Reset all multicast.
+ */
+- fecp->fec_grp_hash_table_high = 0;
+- fecp->fec_grp_hash_table_low = 0;
++ fec_reg_write(fep, FEC_IAUR, 0);
++ fec_reg_write(fep, FEC_IALR, 0);
+
+ /* Set maximum receive buffer size.
+ */
+- fecp->fec_r_buff_size = PKT_MAXBLR_SIZE;
+-
+- fec_localhw_setup();
++ fec_reg_write(fep, FEC_EMRBR, PKT_MAXBLR_SIZE);
+
+ /* Set receive and transmit descriptor base.
+ */
+- fecp->fec_r_des_start = __pa((uint)(fep->rx_bd_base));
+- fecp->fec_x_des_start = __pa((uint)(fep->tx_bd_base));
+-
++ fec_reg_write(fep, FEC_ERDSR, fep->cbd_phys_base);
++ fec_reg_write(fep, FEC_ETDSR, fep->cbd_phys_base + RX_RING_SIZE * sizeof(cbd_t));
++
+ fep->dirty_tx = fep->cur_tx = fep->tx_bd_base;
+ fep->cur_rx = fep->rx_bd_base;
+
+ /* Reset SKB transmit buffers.
+ */
+ fep->skb_cur = fep->skb_dirty = 0;
+- for (i=0; i<=TX_RING_MOD_MASK; i++) {
++ bdp = fep->tx_bd_base;
++ for (i = 0; i <= TX_RING_MOD_MASK; i++) {
+ if (fep->tx_skbuff[i] != NULL) {
+- dev_kfree_skb_any(fep->tx_skbuff[i]);
+- fep->tx_skbuff[i] = NULL;
++ fec_free_skb(fep, bdp, &fep->tx_skbuff[i]);
++ bdp++;
+ }
+ }
+
+ /* Initialize the receive buffer descriptors.
+ */
+ bdp = fep->rx_bd_base;
+- for (i=0; i<RX_RING_SIZE; i++) {
+-
++ for (i = 0; i < RX_RING_SIZE; i++) {
+ /* Initialize the BD for every fragment in the page.
+ */
+ bdp->cbd_sc = BD_ENET_RX_EMPTY;
+@@ -2484,12 +2467,11 @@ fec_restart(struct net_device *dev, int
+ /* ...and the same for transmmit.
+ */
+ bdp = fep->tx_bd_base;
+- for (i=0; i<TX_RING_SIZE; i++) {
+-
++ for (i = 0; i < TX_RING_SIZE; i++) {
+ /* Initialize the BD for every fragment in the page.
+ */
+ bdp->cbd_sc = 0;
+- bdp->cbd_bufaddr = 0;
++ bdp->cbd_bufaddr = ~0;
+ bdp++;
+ }
+
+@@ -2501,92 +2483,321 @@ fec_restart(struct net_device *dev, int
+ /* Enable MII mode.
+ */
+ if (duplex) {
+- fecp->fec_r_cntrl = OPT_FRAME_SIZE | 0x04;/* MII enable */
+- fecp->fec_x_cntrl = 0x04; /* FD enable */
++ tcr |= TCR_FDEN; /* FD enable */
+ } else {
+- /* MII enable|No Rcv on Xmit */
+- fecp->fec_r_cntrl = OPT_FRAME_SIZE | 0x06;
+- fecp->fec_x_cntrl = 0x00;
++ rcr |= RCR_DRT; /* No Rcv on Xmit */
+ }
++ fec_reg_write(fep, FEC_RCR, rcr);
++ fec_reg_write(fep, FEC_TCR, tcr);
+ fep->full_duplex = duplex;
+
+ /* Set MII speed.
+ */
+- fecp->fec_mii_speed = fep->phy_speed;
++ fec_reg_write(fep, FEC_MSCR, fep->phy_speed);
+
+ /* And last, enable the transmit and receive processing.
+ */
+- fecp->fec_ecntrl = 2;
+- fecp->fec_r_des_active = 0;
+-
+- /* Enable interrupts we wish to service.
+- */
+- fecp->fec_imask = (FEC_ENET_TXF | FEC_ENET_RXF | FEC_ENET_MII);
++ fec_reg_write(fep, FEC_ECR, 2);
++ fec_reg_write(fep, FEC_RDAR, DONT_CARE);
++
++ DBG(0, "%s: Starting netif queue\n", __FUNCTION__);
++ netif_start_queue(dev);
+ }
+
+ static void
+ fec_stop(struct net_device *dev)
+ {
+- volatile fec_t *fecp;
+- struct fec_enet_private *fep;
++ struct fec_enet_private *fep = netdev_priv(dev);
+
+- fep = netdev_priv(dev);
+- fecp = fep->hwp;
++ DBG(0, "%s: Stopping netif queue\n", __FUNCTION__);
++ netif_stop_queue(dev);
+
+ /*
+ ** We cannot expect a graceful transmit stop without link !!!
+ */
+- if (fep->link)
+- {
+- fecp->fec_x_cntrl = 0x01; /* Graceful transmit stop */
++ if (fep->linkstatus) {
++ fec_reg_write(fep, FEC_TCR, 0x01); /* Graceful transmit stop */
+ udelay(10);
+- if (!(fecp->fec_ievent & FEC_ENET_GRA))
++ if (!(fec_reg_read(fep, FEC_EIR) & FEC_ENET_GRA))
+ printk("fec_stop : Graceful transmit stop did not complete !\n");
+- }
++ }
+
+ /* Whack a reset. We should wait for this.
+ */
+- fecp->fec_ecntrl = 1;
++ fec_reg_write(fep, FEC_ECR, 1);
+ udelay(10);
+
+- /* Clear outstanding MII command interrupts.
++ /* Mask and clear outstanding MII command interrupts.
+ */
+- fecp->fec_ievent = FEC_ENET_MII;
+- fec_enable_phy_intr();
++ fec_reg_write(fep, FEC_EIMR, FEC_ENET_MII);
++ fec_reg_write(fep, FEC_EIR, FEC_ENET_MII);
++ fec_enable_phy_intr(fep);
+
+- fecp->fec_imask = FEC_ENET_MII;
+- fecp->fec_mii_speed = fep->phy_speed;
++ fec_reg_write(fep, FEC_MSCR, fep->phy_speed);
+ }
+
+-static int __init fec_enet_module_init(void)
++static int __devinit fec_enet_probe(struct platform_device *pdev)
+ {
++ int ret;
++ struct fec_enet_private *fep;
+ struct net_device *dev;
+- int i, err;
+- DECLARE_MAC_BUF(mac);
++ struct fec_enet_platform_data *pdata = pdev->dev.platform_data;
++ struct resource *res_mem1;
++ struct resource *res_mem2;
++
++ res_mem1 = platform_get_resource(pdev, IORESOURCE_MEM, 0);
++ if (res_mem1 == NULL) {
++ return -ENODEV;
++ }
++
++ res_mem1 = request_mem_region(res_mem1->start, res_len(res_mem1), DRV_NAME);
++ if (res_mem1 == NULL) {
++ return -EBUSY;
++ }
++ res_mem2 = platform_get_resource(pdev, IORESOURCE_MEM, 1);
++ if (res_mem2 != NULL) {
++ res_mem2 = request_mem_region(res_mem2->start, res_len(res_mem2), DRV_NAME);
++ if (res_mem2 == NULL) {
++ ret = -EBUSY;
++ goto release1;
++ }
++ }
++
++ dev = alloc_etherdev(sizeof(struct fec_enet_private));
++ if (dev == NULL) {
++ ret = -ENOMEM;
++ goto release2;
++ }
++ platform_set_drvdata(pdev, dev);
++ fep = netdev_priv(dev);
++ fep->res_mem1 = res_mem1;
++ fep->res_mem2 = res_mem2;
++
++ fep->reg_base = ioremap(res_mem1->start, res_len(res_mem1));
++ if (fep->reg_base == NULL) {
++ printk("FEC: Mapping FEC registers failed\n");
++ ret = -ENOMEM;
++ goto free_netdev;
++ }
++ DBG(0, "%s: FEC registers @ %08lx mapped to %p\n", __FUNCTION__,
++ (unsigned long)res_mem1->start, fep->reg_base);
++
++ /* Allocate memory for buffer descriptors. */
++ fep->cbd_mem_base = dma_alloc_coherent(&pdev->dev, CBD_BUF_SIZE, &fep->cbd_phys_base,
++ GFP_KERNEL);
++ if (fep->cbd_mem_base == NULL) {
++ printk("FEC: allocate descriptor memory failed\n");
++ ret = -ENOMEM;
++ goto unmap;
++ }
++ DBG(0, "%s: Allocated %lu [(%u + %lu) * %d] byte for CBD buffer @ %p[%08lx]\n",
++ __FUNCTION__, CBD_BUF_SIZE, TX_RING_SIZE, RX_RING_SIZE, sizeof(cbd_t),
++ fep->cbd_mem_base, (unsigned long)fep->cbd_phys_base);
++
++ /* Set receive and transmit descriptor base.
++ */
++ fep->rx_bd_base = fep->cbd_mem_base;
++ fep->tx_bd_base = fep->rx_bd_base + RX_RING_SIZE;
+
+ printk("FEC ENET Version 0.2\n");
++ ret = platform_func(pdata->arch_init, pdev);
++ if (ret != 0) {
++ printk(KERN_ERR "%s: platform init failed: %d\n", __FUNCTION__, ret);
++ goto free_dma;
++ }
++
++ ret = fec_enet_init(pdev, dev);
++ if (ret != 0) {
++ goto fec_disable;
++ }
++
++ /* Enable most messages by default */
++ fep->msg_enable = (NETIF_MSG_IFUP << 1) - 1;
++ ret = register_netdev(dev);
++ if (ret != 0) {
++ /* XXX: missing cleanup here */
++ goto free_buffers;
++ }
++
++ printk(KERN_INFO "%s: ethernet %02x:%02x:%02x:%02x:%02x:%02x\n", dev->name,
++ dev->dev_addr[0], dev->dev_addr[1], dev->dev_addr[2],
++ dev->dev_addr[3], dev->dev_addr[4], dev->dev_addr[5]);
+
+- for (i = 0; (i < FEC_MAX_PORTS); i++) {
+- dev = alloc_etherdev(sizeof(struct fec_enet_private));
+- if (!dev)
+- return -ENOMEM;
+- err = fec_enet_init(dev);
+- if (err) {
+- free_netdev(dev);
+- continue;
+- }
+- if (register_netdev(dev) != 0) {
+- /* XXX: missing cleanup here */
+- free_netdev(dev);
+- return -EIO;
++ return 0;
++
++ free_buffers:
++ fec_enet_free_buffers(fep);
++
++ fec_disable:
++ platform_func(pdata->arch_exit, pdev);
++
++ free_dma:
++ dma_free_coherent(&pdev->dev, CBD_BUF_SIZE, fep->cbd_mem_base, fep->cbd_phys_base);
++
++ unmap:
++ iounmap(fep->reg_base);
++
++ free_netdev:
++ free_netdev(dev);
++
++ release2:
++ if (res_mem2 != NULL) {
++ release_resource(res_mem2);
++ }
++
++ release1:
++ release_resource(res_mem1);
++
++ return ret;
++}
++
++static int __devexit fec_enet_remove(struct platform_device *pdev)
++{
++ struct net_device *dev = platform_get_drvdata(pdev);
++ struct fec_enet_private *fep = netdev_priv(dev);
++
++ unregister_netdev(dev);
++ free_netdev(dev);
++
++#ifdef CONFIG_PHYLIB
++ if (fep->mii != NULL) {
++ kfree(fep->mii->irq);
++ mdiobus_unregister(fep->mii);
++ }
++ mdiobus_free(fep->mii);
++#endif
++ fec_release_intrs(dev);
++
++ DBG(0, "%s: Unmapping FEC registers %p\n", __FUNCTION__, fep->reg_base);
++ iounmap(fep->reg_base);
++
++ fec_enet_free_buffers(fep);
++
++ DBG(0, "%s: Freeing CBD buffer area %p[%08lx]\n", __FUNCTION__,
++ fep->cbd_mem_base, (unsigned long)fep->cbd_phys_base);
++ dma_free_coherent(&pdev->dev, CBD_BUF_SIZE, fep->cbd_mem_base, fep->cbd_phys_base);
++
++ release_resource(fep->res_mem1);
++ if (fep->res_mem2 != NULL) {
++ release_resource(fep->res_mem2);
++ }
++ return 0;
++}
++
++static void fec_enet_shutdown(struct platform_device *pdev)
++{
++ struct fec_enet_platform_data *pdata = pdev->dev.platform_data;
++
++ DBG(0, "%s: Shutting down FEC Hardware\n", __FUNCTION__);
++ platform_func(pdata->arch_exit, pdev);
++}
++
++#ifdef CONFIG_PM
++static int fec_enet_suspend(struct platform_device *pdev, pm_message_t state)
++{
++ int ret;
++ struct fec_enet_platform_data *pdata = pdev->dev.platform_data;
++ struct net_device *ndev = platform_get_drvdata(pdev);
++ struct fec_enet_private *fep = netdev_priv(ndev);
++
++ if (netif_running(ndev)) {
++ DBG(0, "%s: Detaching netif\n", __FUNCTION__);
++ netif_device_detach(ndev);
++#ifdef CONFIG_PHYLIB
++ DBG(0, "%s: Disconnecting PHY %p\n", __FUNCTION__, fep->phy);
++ phy_disconnect(fep->phy);
++ fep->phy = NULL;
++#endif
++ }
++#ifndef CONFIG_PHYLIB
++ if (fep->phy_timer) {
++ ret = del_timer_sync(fep->phy_timer);
++ if (ret != 0) {
++ DBG(0, "%s: Failed to delete PHY timer: %d\n", __FUNCTION__, ret);
++ return ret;
+ }
++ }
++#endif
++ DBG(0, "%s: Shutting down FEC Hardware %d\n", __FUNCTION__,
++ netif_running(ndev));
++ ret = platform_func(pdata->suspend, pdev);
++ if (ret != 0 && netif_running(ndev)) {
++ DBG(0, "%s: Failed to suspend: %d\n", __FUNCTION__, ret);
++ /* Undo suspend */
++#ifdef CONFIG_PHYLIB
++ DBG(0, "%s: Reconnecting PHY\n", __FUNCTION__);
++ if (fec_connect_phy(ndev, fep) != 0) {
++ DBG(0, "%s: Failed to connect to PHY\n", __FUNCTION__);
++ return ret;
++ }
++ phy_start(fep->phy);
++#endif
++ fec_link_change(ndev);
++ netif_device_attach(ndev);
++ }
++ return ret;
++}
+
+- printk("%s: ethernet %s\n",
+- dev->name, print_mac(mac, dev->dev_addr));
++static int fec_enet_resume(struct platform_device *pdev)
++{
++ int ret;
++ struct fec_enet_platform_data *pdata = pdev->dev.platform_data;
++ struct net_device *ndev = platform_get_drvdata(pdev);
++
++ DBG(0, "%s: Powering up FEC Hardware %d\n", __FUNCTION__,
++ netif_running(ndev));
++ ret = platform_func(pdata->resume, pdev);
++ if (ret != 0) {
++ DBG(0, "%s: Failed to resume: %d\n", __FUNCTION__, ret);
++ return ret;
++ }
++ if (netif_running(ndev)) {
++#ifdef CONFIG_PHYLIB
++ struct fec_enet_private *fep = netdev_priv(ndev);
++
++ DBG(0, "%s: Reconnecting PHY\n", __FUNCTION__);
++ ret = fec_connect_phy(ndev, fep);
++ if (ret != 0) {
++ DBG(0, "%s: Failed to connect to PHY: %d\n", __FUNCTION__, ret);
++ return ret;
++ }
++ phy_start(fep->phy);
++#endif
++ fec_link_change(ndev);
++ netif_device_attach(ndev);
+ }
+ return 0;
+ }
++#else
++#define fec_enet_suspend NULL
++#define fec_enet_resume NULL
++#endif
++
++static struct platform_driver fec_enet_driver = {
++ .driver = {
++ .name = DRV_NAME,
++ },
++ .probe = fec_enet_probe,
++ .remove = __devexit_p(fec_enet_remove),
++ .shutdown = fec_enet_shutdown,
++ .suspend = fec_enet_suspend,
++ .resume = fec_enet_resume,
++};
++
++static int __init fec_enet_module_init(void)
++{
++ int ret;
+
++ ret = platform_driver_register(&fec_enet_driver);
++
++ return ret;
++}
+ module_init(fec_enet_module_init);
+
++static void __exit fec_enet_module_cleanup(void)
++{
++ platform_driver_unregister(&fec_enet_driver);
++}
++module_exit(fec_enet_module_cleanup);
++
+ MODULE_LICENSE("GPL");
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/net/fec.h linux-2.6.28-karo/drivers/net/fec.h
+--- linux-2.6.28/drivers/net/fec.h 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/net/fec.h 2009-03-11 13:16:24.000000000 +0100
+@@ -13,13 +13,50 @@
+ #define FEC_H
+ /****************************************************************************/
+
++/*
++ * dummy value to write into RDAR,TDAR. FEC hardware will scan the TX/RX
++ * descriptors in memory upon any write access to those registers.
++ * The actual value written to those registers does not matter.
++*/
++#define DONT_CARE 0
++#define RDAR_BUSY (1 << 24)
++#define TDAR_BUSY (1 << 24)
++
+ #if defined(CONFIG_M523x) || defined(CONFIG_M527x) || defined(CONFIG_M528x) || \
+- defined(CONFIG_M520x) || defined(CONFIG_M532x)
++ defined(CONFIG_M520x) || defined(CONFIG_M532x) || defined(CONFIG_ARCH_MXC)
+ /*
+ * Just figures, Motorola would have to change the offsets for
+ * registers in the same peripheral device on different models
+ * of the ColdFire!
+ */
++// relying on structure alignment for hardware register is just evil
++#ifndef GARBAGE
++#define FEC_EIR 0x004
++#define FEC_EIMR 0x008
++#define FEC_RDAR 0x010
++#define FEC_TDAR 0x014
++#define FEC_ECR 0x024
++#define FEC_MMFR 0x040
++#define FEC_MSCR 0x044
++#define FEC_MIBC 0x064
++#define FEC_RCR 0x084
++#define FEC_TCR 0x0c4
++#define FEC_PALR 0x0e4
++#define FEC_PAUR 0x0e8
++#define FEC_OPD 0x0ec
++#define FEC_IAUR 0x118
++#define FEC_IALR 0x11c
++#define FEC_GAUR 0x120
++#define FEC_GALR 0x124
++#define FEC_TFWR 0x144
++#define FEC_FRBR 0x14c
++#define FEC_FRSR 0x150
++#define FEC_ERDSR 0x180
++#define FEC_ETDSR 0x184
++#define FEC_EMRBR 0x188
++
++#else
++
+ typedef struct fec {
+ unsigned long fec_reserved0;
+ unsigned long fec_ievent; /* Interrupt event reg */
+@@ -57,6 +94,7 @@ typedef struct fec {
+ unsigned long fec_x_des_start; /* Transmit descriptor ring */
+ unsigned long fec_r_buff_size; /* Maximum receive buff size */
+ } fec_t;
++#endif
+
+ #else
+
+@@ -88,8 +126,8 @@ typedef struct fec {
+ unsigned long fec_reserved7[158];
+ unsigned long fec_addr_low; /* Low 32bits MAC address */
+ unsigned long fec_addr_high; /* High 16bits MAC address */
+- unsigned long fec_grp_hash_table_high;/* High 32bits hash table */
+- unsigned long fec_grp_hash_table_low; /* Low 32bits hash table */
++ unsigned long fec_hash_table_high; /* High 32bits hash table */
++ unsigned long fec_hash_table_low; /* Low 32bits hash table */
+ unsigned long fec_r_des_start; /* Receive descriptor ring */
+ unsigned long fec_x_des_start; /* Transmit descriptor ring */
+ unsigned long fec_r_buff_size; /* Maximum receive buff size */
+@@ -103,18 +141,28 @@ typedef struct fec {
+ /*
+ * Define the buffer descriptor structure.
+ */
++/* Please see "Receive Buffer Descriptor Field Definitions" in Specification.
++ * It's LE.
++ */
++#ifdef CONFIG_ARCH_MXC
++typedef struct bufdesc {
++ unsigned short cbd_datlen; /* Data length */
++ unsigned short cbd_sc; /* Control and status info */
++ dma_addr_t cbd_bufaddr; /* Buffer address as seen by FEC Hardware */
++} cbd_t;
++#else
+ typedef struct bufdesc {
+ unsigned short cbd_sc; /* Control and status info */
+ unsigned short cbd_datlen; /* Data length */
+- unsigned long cbd_bufaddr; /* Buffer address */
++ dma_addr_t cbd_bufaddr; /* Buffer address */
+ } cbd_t;
+-
++#endif
+
+ /*
+ * The following definitions courtesy of commproc.h, which where
+ * Copyright (c) 1997 Dan Malek (dmalek@jlc.net).
+ */
+-#define BD_SC_EMPTY ((ushort)0x8000) /* Recieve is empty */
++#define BD_SC_EMPTY ((ushort)0x8000) /* Receive is empty */
+ #define BD_SC_READY ((ushort)0x8000) /* Transmit is ready */
+ #define BD_SC_WRAP ((ushort)0x2000) /* Last buffer descriptor */
+ #define BD_SC_INTRPT ((ushort)0x1000) /* Interrupt on change */
+@@ -161,5 +209,22 @@ typedef struct bufdesc {
+ #define BD_ENET_TX_STATS ((ushort)0x03ff) /* All status bits */
+
+
++#define RCR_LOOP (1 << 0)
++#define RCR_DRT (1 << 1)
++#define RCR_MII_MODE (1 << 2)
++#define RCR_PROM (1 << 3)
++#define RCR_BC_REJ (1 << 4)
++#define RCR_FCE (1 << 5)
++#define RCR_MAX_FL_SHIFT 16
++#define RCR_MAX_FL_MASK (0x7ff << (RCR_MAX_FL_SHIFT))
++#define RCR_MAX_FL_set(n) (((n) << (RCR_MAX_FL_SHIFT)) & (RCR_MAX_FL_MASK))
++#define RCR_MAX_FL_get(n) (((n) & (RCR_MAX_FL_MASK)) >> (RCR_MAX_FL_SHIFT))
++
++#define TCR_GTS (1 << 0)
++#define TCR_HBC (1 << 1)
++#define TCR_FDEN (1 << 2)
++#define TCR_TFCPAUSE (1 << 3)
++#define TCR_RFCPAUSE (1 << 4)
++
+ /****************************************************************************/
+ #endif /* FEC_H */
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/power/Kconfig linux-2.6.28-karo/drivers/power/Kconfig
+--- linux-2.6.28/drivers/power/Kconfig 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/power/Kconfig 2009-03-11 13:16:24.000000000 +0100
+@@ -68,4 +68,16 @@ config BATTERY_BQ27x00
+ help
+ Say Y here to enable support for batteries with BQ27200(I2C) chip.
+
++config LP3972
++ tristate "National Semiconductor LP3972 Power Management Unit"
++ depends on EXPERIMENTAL
++ depends on I2C
++ default n
++ help
++ If you say yes here you get support for the National Semiconductor
++ Power Management Chip.
++
++ This driver can also be built as a module. If so, the module
++ will be called lp3972.
++
+ endif # POWER_SUPPLY
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/power/Makefile linux-2.6.28-karo/drivers/power/Makefile
+--- linux-2.6.28/drivers/power/Makefile 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/power/Makefile 2009-03-11 13:16:24.000000000 +0100
+@@ -23,3 +23,5 @@ obj-$(CONFIG_BATTERY_OLPC) += olpc_batte
+ obj-$(CONFIG_BATTERY_TOSA) += tosa_battery.o
+ obj-$(CONFIG_BATTERY_WM97XX) += wm97xx_battery.o
+ obj-$(CONFIG_BATTERY_BQ27x00) += bq27x00_battery.o
++
++obj-$(CONFIG_LP3972) += lp3972.o
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/power/lp3972.c linux-2.6.28-karo/drivers/power/lp3972.c
+--- linux-2.6.28/drivers/power/lp3972.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/power/lp3972.c 2009-03-11 13:52:22.000000000 +0100
+@@ -0,0 +1,267 @@
++/*
++ * Copyright (C) 2008 Lothar Wassmann <LW@KARO-electronics.de>
++ *
++ * 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 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301
++ */
++
++#include <linux/module.h>
++#include <linux/init.h>
++#include <linux/slab.h>
++#include <linux/i2c.h>
++
++/* Addresses to scan */
++static unsigned short normal_i2c[] = { 0x34,
++ I2C_CLIENT_END };
++
++I2C_CLIENT_INSMOD_1(lp3972);
++
++struct lp3972_data {
++ struct i2c_client client;
++};
++
++static int lp3972_attach_adapter(struct i2c_adapter *adapter);
++static int lp3972_detect(struct i2c_adapter *adapter, int address, int kind);
++static int lp3972_detach_client(struct i2c_client *client);
++static void lp3972_init_client(struct i2c_client *client);
++
++static struct i2c_driver lp3972_driver = {
++ .driver = {
++ .name = "lp3972",
++ },
++ //.id = I2C_DRIVERID_LP3972,
++ .attach_adapter = lp3972_attach_adapter,
++ .detach_client = lp3972_detach_client,
++};
++
++struct lp3972_dev_attr {
++ struct device_attribute dev_attr;
++ u8 reg;
++ u8 val;
++};
++
++#define to_lp3972_attr(d) container_of(d, struct lp3972_dev_attr, dev_attr)
++
++static ssize_t lp3972_get_val(struct device *dev, struct device_attribute *attr, char *buf)
++{
++ int ret;
++ struct i2c_client *client = to_i2c_client(dev);
++ struct lp3972_dev_attr *lp3972_attr = to_lp3972_attr(attr);
++
++ ret = i2c_smbus_read_byte_data(client, lp3972_attr->reg);
++ if (ret < 0) {
++ return ret;
++ }
++ lp3972_attr->val = ret;
++ return sprintf(buf, "0x%02x\n", ret);
++}
++
++static ssize_t lp3972_set_val(struct device *dev, struct device_attribute *attr, const char *buf,
++ size_t count)
++{
++ int ret;
++ struct i2c_client *client = to_i2c_client(dev);
++ struct lp3972_dev_attr *lp3972_attr = to_lp3972_attr(attr);
++ unsigned long val = simple_strtoul(buf, NULL, 0);
++
++ if (val > 0xff) {
++ printk(KERN_WARNING "value %ld is out of range\n", val);
++ return -EINVAL;
++ }
++ if (val != lp3972_attr->val) {
++ lp3972_attr->val = val;
++ ret = i2c_smbus_write_byte_data(client, lp3972_attr->reg, lp3972_attr->val);
++ if (ret < 0) {
++ return ret;
++ }
++ }
++ return count;
++}
++
++#define LP3972_DEV_ATTR(_name, _mode, _reg, _read, _write) \
++ struct lp3972_dev_attr lp3972_dev_attr_##_name = { \
++ .dev_attr = __ATTR(_name,_mode,_read,_write), \
++ .reg = _reg, \
++ }
++
++static LP3972_DEV_ATTR(scr, S_IWUSR | S_IRUGO, 0x07, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(over1, S_IWUSR | S_IRUGO, 0x10, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(ovsr1, S_IRUGO, 0x11, lp3972_get_val, NULL);
++static LP3972_DEV_ATTR(over2, S_IWUSR | S_IRUGO, 0x12, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(ovsr2, S_IRUGO, 0x13, lp3972_get_val, NULL);
++static LP3972_DEV_ATTR(vcc1, S_IWUSR | S_IRUGO, 0x20, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(adtv1, S_IWUSR | S_IRUGO, 0x23, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(adtv2, S_IWUSR | S_IRUGO, 0x24, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(avrc, S_IWUSR | S_IRUGO, 0x25, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(cdtc1, S_IWUSR, 0x26, NULL, lp3972_set_val);
++static LP3972_DEV_ATTR(cdtc2, S_IWUSR, 0x27, NULL, lp3972_set_val);
++static LP3972_DEV_ATTR(sdtv1, S_IWUSR | S_IRUGO, 0x29, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(sdtv2, S_IWUSR | S_IRUGO, 0x2a, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(mdtv1, S_IWUSR | S_IRUGO, 0x32, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(mdtv2, S_IWUSR | S_IRUGO, 0x33, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(l12vcr, S_IWUSR | S_IRUGO, 0x39, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(l34vcr, S_IWUSR | S_IRUGO, 0x3a, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(scr1, S_IWUSR | S_IRUGO, 0x80, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(scr2, S_IWUSR | S_IRUGO, 0x81, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(oen3, S_IWUSR | S_IRUGO, 0x82, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(osr3, S_IWUSR | S_IRUGO, 0x83, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(loer, S_IWUSR | S_IRUGO, 0x84, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(b2tv, S_IWUSR | S_IRUGO, 0x85, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(b3tv, S_IWUSR | S_IRUGO, 0x86, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(c32rc, S_IWUSR | S_IRUGO, 0x87, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(isra, S_IRUGO, 0x88, lp3972_get_val, NULL);
++static LP3972_DEV_ATTR(bccr, S_IWUSR | S_IRUGO, 0x89, lp3972_get_val, lp3972_set_val);
++static LP3972_DEV_ATTR(ii1rr, S_IRUGO, 0x8e, lp3972_get_val, NULL);
++static LP3972_DEV_ATTR(ii2rr, S_IRUGO, 0x8f, lp3972_get_val, NULL);
++
++static struct attribute *lp3972_attributes[] = {
++ &lp3972_dev_attr_scr.dev_attr.attr,
++ &lp3972_dev_attr_over1.dev_attr.attr,
++ &lp3972_dev_attr_ovsr1.dev_attr.attr,
++ &lp3972_dev_attr_over2.dev_attr.attr,
++ &lp3972_dev_attr_ovsr2.dev_attr.attr,
++ &lp3972_dev_attr_vcc1.dev_attr.attr,
++ &lp3972_dev_attr_adtv1.dev_attr.attr,
++ &lp3972_dev_attr_adtv2.dev_attr.attr,
++ &lp3972_dev_attr_avrc.dev_attr.attr,
++ &lp3972_dev_attr_cdtc1.dev_attr.attr,
++ &lp3972_dev_attr_cdtc2.dev_attr.attr,
++ &lp3972_dev_attr_sdtv1.dev_attr.attr,
++ &lp3972_dev_attr_sdtv2.dev_attr.attr,
++ &lp3972_dev_attr_mdtv1.dev_attr.attr,
++ &lp3972_dev_attr_mdtv2.dev_attr.attr,
++ &lp3972_dev_attr_l12vcr.dev_attr.attr,
++ &lp3972_dev_attr_l34vcr.dev_attr.attr,
++ &lp3972_dev_attr_scr1.dev_attr.attr,
++ &lp3972_dev_attr_scr2.dev_attr.attr,
++ &lp3972_dev_attr_oen3.dev_attr.attr,
++ &lp3972_dev_attr_osr3.dev_attr.attr,
++ &lp3972_dev_attr_loer.dev_attr.attr,
++ &lp3972_dev_attr_b2tv.dev_attr.attr,
++ &lp3972_dev_attr_b3tv.dev_attr.attr,
++ &lp3972_dev_attr_c32rc.dev_attr.attr,
++ &lp3972_dev_attr_isra.dev_attr.attr,
++ &lp3972_dev_attr_bccr.dev_attr.attr,
++ &lp3972_dev_attr_ii1rr.dev_attr.attr,
++ &lp3972_dev_attr_ii2rr.dev_attr.attr,
++ NULL
++};
++
++static const struct attribute_group lp3972_attr_group = {
++ .attrs = lp3972_attributes,
++};
++
++/*
++ * Real code
++ */
++
++static int lp3972_attach_adapter(struct i2c_adapter *adapter)
++{
++ return i2c_probe(adapter, &addr_data, lp3972_detect);
++}
++
++/* This function is called by i2c_probe */
++static int lp3972_detect(struct i2c_adapter *adapter, int address, int kind)
++{
++ int ret;
++ struct i2c_client *new_client;
++ struct lp3972_data *data;
++ const char *client_name = "";
++
++ if (!i2c_check_functionality(adapter, I2C_FUNC_SMBUS_BYTE)) {
++ return -ENOTSUPP;
++ }
++
++ data = kzalloc(sizeof(struct lp3972_data), GFP_KERNEL);
++ if (data == NULL) {
++ return -ENOMEM;
++ }
++
++ new_client = &data->client;
++ i2c_set_clientdata(new_client, data);
++ new_client->addr = address;
++ new_client->adapter = adapter;
++ new_client->driver = &lp3972_driver;
++ new_client->flags = 0;
++
++ /* Now, we would do the remaining detection. But the LP3972 is plainly
++ impossible to detect! Stupid chip. */
++
++ /* Determine the chip type */
++ if (kind <= 0) {
++ kind = lp3972;
++ }
++
++ client_name = "lp3972";
++
++ /* Fill in the remaining client fields and put it into the global list */
++ strlcpy(new_client->name, client_name, I2C_NAME_SIZE);
++
++ /* Tell the I2C layer a new client has arrived */
++ ret = i2c_attach_client(new_client);
++ if (ret != 0) {
++ goto free;
++ }
++
++ /* Initialize the LP3972 chip */
++ lp3972_init_client(new_client);
++
++ /* Register sysfs hooks */
++ ret = sysfs_create_group(&new_client->dev.kobj, &lp3972_attr_group);
++ if (ret == 0) {
++ return ret;
++ }
++
++ i2c_detach_client(new_client);
++ free:
++ kfree(data);
++ return ret;
++}
++
++static int lp3972_detach_client(struct i2c_client *client)
++{
++ int ret;
++
++ sysfs_remove_group(&client->dev.kobj, &lp3972_attr_group);
++
++ ret = i2c_detach_client(client);
++ if (ret != 0) {
++ return ret;
++ }
++
++ kfree(i2c_get_clientdata(client));
++ return 0;
++}
++
++/* Called when we have found a new LP3972. */
++static void lp3972_init_client(struct i2c_client *client)
++{
++}
++
++static int __init lp3972_init(void)
++{
++ return i2c_add_driver(&lp3972_driver);
++}
++
++static void __exit lp3972_exit(void)
++{
++ i2c_del_driver(&lp3972_driver);
++}
++
++
++MODULE_AUTHOR("Lothar Wassmann <LW@KARO-electronics.de>");
++MODULE_DESCRIPTION("LP3972 I2C PMIC driver");
++MODULE_LICENSE("GPL v2");
++
++module_init(lp3972_init);
++module_exit(lp3972_exit);
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/rtc/Kconfig linux-2.6.28-karo/drivers/rtc/Kconfig
+--- linux-2.6.28/drivers/rtc/Kconfig 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/rtc/Kconfig 2009-03-11 13:16:24.000000000 +0100
+@@ -141,6 +141,24 @@ config RTC_DRV_DS1307
+ This driver can also be built as a module. If so, the module
+ will be called rtc-ds1307.
+
++config RTC_DRV_DS13XX
++ tristate "Dallas/Maxim DS13xx I2C RTC chips"
++ depends on RTC_CLASS && I2C
++ help
++ If you say yes here you get support for various compatible RTC
++ chips (often with battery backup) connected with I2C. This driver
++ should handle DS1307, DS1337, DS1338, DS1339, DS1340, ST M41T00,
++ and probably other chips. Platform specific code may provide a
++ callback function to properly initialize the chip.
++
++ The first seven registers on these chips hold an RTC, and other
++ registers may add features such as NVRAM, a trickle charger for
++ the RTC/NVRAM backup power, and alarms. This driver may not
++ expose all those available chip features.
++
++ This driver can also be built as a module. If so, the module
++ will be called rtc-ds13xx.
++
+ config RTC_DRV_DS1374
+ tristate "Dallas/Maxim DS1374"
+ depends on RTC_CLASS && I2C
+@@ -649,6 +667,13 @@ config RTC_DRV_RS5C313
+ help
+ If you say yes here you get support for the Ricoh RS5C313 RTC chips.
+
++config RTC_MXC
++ tristate "Freescale MXC Real Time Clock"
++ depends on ARCH_MXC
++ depends on RTC_CLASS
++ help
++ If you say yes here you get support for the Freescale i.MX RTC
++
+ config RTC_DRV_PARISC
+ tristate "PA-RISC firmware RTC support"
+ depends on PARISC
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/rtc/Makefile linux-2.6.28-karo/drivers/rtc/Makefile
+--- linux-2.6.28/drivers/rtc/Makefile 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/rtc/Makefile 2009-03-11 13:16:24.000000000 +0100
+@@ -28,6 +28,7 @@ obj-$(CONFIG_RTC_DRV_DS1302) += rtc-ds13
+ obj-$(CONFIG_RTC_DRV_DS1305) += rtc-ds1305.o
+ obj-$(CONFIG_RTC_DRV_DS1307) += rtc-ds1307.o
+ obj-$(CONFIG_RTC_DRV_DS1374) += rtc-ds1374.o
++obj-$(CONFIG_RTC_DRV_DS13XX) += rtc-ds13xx.o
+ obj-$(CONFIG_RTC_DRV_DS1390) += rtc-ds1390.o
+ obj-$(CONFIG_RTC_DRV_DS1511) += rtc-ds1511.o
+ obj-$(CONFIG_RTC_DRV_DS1553) += rtc-ds1553.o
+@@ -62,6 +63,7 @@ obj-$(CONFIG_RTC_DRV_RX8581) += rtc-rx85
+ obj-$(CONFIG_RTC_DRV_S35390A) += rtc-s35390a.o
+ obj-$(CONFIG_RTC_DRV_S3C) += rtc-s3c.o
+ obj-$(CONFIG_RTC_DRV_SA1100) += rtc-sa1100.o
++obj-$(CONFIG_RTC_MXC) += rtc-mxc.o
+ obj-$(CONFIG_RTC_DRV_SH) += rtc-sh.o
+ obj-$(CONFIG_RTC_DRV_STK17TA8) += rtc-stk17ta8.o
+ obj-$(CONFIG_RTC_DRV_TEST) += rtc-test.o
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/rtc/rtc-ds13xx.c linux-2.6.28-karo/drivers/rtc/rtc-ds13xx.c
+--- linux-2.6.28/drivers/rtc/rtc-ds13xx.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/rtc/rtc-ds13xx.c 2009-03-11 18:38:58.000000000 +0100
+@@ -0,0 +1,590 @@
++/*
++ * rtc-ds13xx.c - RTC driver for some mostly-compatible I2C chips.
++ *
++ * Copyright (C) 2005 James Chapman (ds1337 core)
++ * Copyright (C) 2006 David Brownell
++ * Copyright (C) 2007 Lothar Wassmann <LW@karo-electronics.de>
++ * rewritten to support initialization via platform_device
++ *
++ * 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/slab.h>
++#include <linux/platform_device.h>
++#include <linux/i2c.h>
++#include <linux/rtc.h>
++#include <linux/bcd.h>
++#include <linux/rtc/ds13xx.h>
++
++
++/* We can't determine type by probing, but if we expect pre-Linux code
++ * to have set the chip up as a clock (turning on the oscillator and
++ * setting the date and time), Linux can ignore the non-clock features.
++ * That's a natural job for a factory or repair bench.
++ *
++ * If the I2C "force" mechanism is used, we assume the chip is a ds1337.
++ * (Much better would be board-specific tables of I2C devices, along with
++ * the platform_data drivers would use to sort such issues out.)
++ */
++
++static unsigned short normal_i2c[] = { 0x68, I2C_CLIENT_END };
++
++I2C_CLIENT_INSMOD;
++
++// the RTC epoch starts at 2000
++#define CENTURY 2000
++
++/* RTC registers don't differ much, except for the century flag */
++#define DS13XX_REG_SECS 0x00 /* 00-59 */
++# define DS1307_BIT_CH (1 << 7)
++#define DS13XX_REG_MIN 0x01 /* 00-59 */
++#define DS13XX_REG_HOUR 0x02 /* 00-23, or 1-12{am,pm} */
++# define DS1340_BIT_CENTURY_EN (1 << 7) /* in REG_HOUR */
++# define DS1340_BIT_CENTURY (1 << 6) /* in REG_HOUR */
++#define DS13XX_REG_WDAY 0x03 /* 01-07 */
++#define DS13XX_REG_MDAY 0x04 /* 01-31 */
++#define DS13XX_REG_MONTH 0x05 /* 01-12 */
++# define DS133X_BIT_CENTURY (1 << 7) /* in REG_MONTH */
++#define DS13XX_REG_YEAR 0x06 /* 00-99 */
++
++/* Other registers (control, status, alarms, trickle charge, NVRAM, etc)
++ * start at 7, and they differ a lot. Only control and status matter for RTC;
++ * be careful using them.
++ */
++#define DS1307_REG_CONTROL 0x07
++#define DS1339_REG_AL1SEC 0x07
++#define DS1339_REG_AL1MIN 0x08
++#define DS1339_REG_AL1HRS 0x09
++#define DS1339_REG_AL1DAY 0x0a
++#define DS1339_REG_AL2MIN 0x0b
++#define DS1339_REG_AL2HRS 0x0c
++#define DS1339_REG_AL2DAY 0x0d
++# define DS1339_BIT_ALMSK (1 << 7)
++# define DS1339_BIT_DYDT (1 << 6)
++#define DS133X_REG_CONTROL 0x0e
++# define DS1338_BIT_OSF (1 << 5)
++# define DS133X_BIT_nEOSC (1 << 7)
++# define DS133X_BIT_A1IE (1 << 0)
++# define DS133X_BIT_A2IE (1 << 1)
++#define DS133X_REG_STATUS 0x0f
++# define DS133X_BIT_OSF (1 << 7)
++# define DS133X_BIT_A1I (1 << 0)
++# define DS133X_BIT_A2I (1 << 1)
++#define DS1339_REG_TRC 0x10
++# define DS1339_TRC_MAGIC 0xa0
++#define DS1340_REG_STATUS 0x09
++# define DS1340_BIT_OSF (1 << 7)
++
++struct ds13xx {
++ u8 reg_addr;
++ u8 regs[8];
++ enum ds13xx_type type;
++ struct i2c_msg msg[2];
++ struct i2c_client client;
++ struct rtc_device *rtc;
++ int valid_time;
++};
++
++static const struct i2c_device_id ds13xx_id[] = {
++ { "ds1307", ds_1307 },
++ { "ds1337", ds_1337 },
++ { "ds1338", ds_1338 },
++ { "ds1339", ds_1339 },
++ { "ds1340", ds_1340 },
++ { "m41t00", m41t00 },
++ { }
++};
++MODULE_DEVICE_TABLE(i2c, ds13xx_id);
++
++static inline int ds13xx_write_reg(struct i2c_client *client, u8 regno, u8 val)
++{
++ dev_dbg(&client->dev, "Writing %02x to reg %02x\n", val, regno);
++ return i2c_smbus_write_byte_data(client, regno, val);
++}
++
++static inline int ds13xx_read_reg(struct i2c_client *client, u8 regno)
++{
++ int ret;
++
++ ret = i2c_smbus_read_byte_data(client, regno);
++
++ return ret;
++}
++
++static int __devinit ds13xx_i2c_init(struct platform_device *pdev, struct i2c_client *client)
++{
++ int ret = 0;
++ struct ds13xx_platform_data *pdata = pdev->dev.platform_data;
++ struct ds13xx *ds13xx = i2c_get_clientdata(client);
++
++ if (pdata != NULL && pdata->type > 0) {
++ ds13xx->type = pdata->type;
++ switch (ds13xx->type) {
++ case unknown:
++ strlcpy(client->name, "unknown", I2C_NAME_SIZE);
++ break;
++ case ds_1307:
++ strlcpy(client->name, "ds1307", I2C_NAME_SIZE);
++ break;
++ case ds_1337:
++ strlcpy(client->name, "ds1337", I2C_NAME_SIZE);
++ break;
++ case ds_1339:
++ strlcpy(client->name, "ds1339", I2C_NAME_SIZE);
++ break;
++ case ds_1340:
++ strlcpy(client->name, "ds1340", I2C_NAME_SIZE);
++ break;
++ default:
++ return -ENODEV;
++ }
++ } else {
++ dev_dbg(&client->dev, "No platform_data\n");
++ }
++ return ret;
++}
++
++static int __devinit ds13xx_rtc_init(struct ds13xx_platform_data *pdata, struct i2c_client *client)
++{
++ int ret = 0;
++ struct ds13xx *ds13xx = i2c_get_clientdata(client);
++ int oscstart = 0;
++
++ ds13xx->valid_time = 1;
++ dev_dbg(&client->dev, "Inititialzing RTC %s\n", client->name);
++ switch (ds13xx->type) {
++ case ds_1307:
++ case m41t00:
++ if (ds13xx->regs[DS13XX_REG_SECS] & DS1307_BIT_CH) {
++ oscstart = 1;
++ ds13xx_write_reg(client, DS13XX_REG_SECS, 0);
++ }
++ if (pdata && pdata->ctrl >= 0) {
++ ds13xx_write_reg(client, DS1307_REG_CONTROL, pdata->ctrl);
++ }
++ break;
++ case ds_1337:
++ ret = ds13xx_read_reg(client, DS133X_REG_CONTROL);
++ if (ret > 0 && ret & DS133X_BIT_nEOSC) {
++ oscstart = 1;
++ }
++ if (pdata && pdata->ctrl >= 0) {
++ ds13xx_write_reg(client, DS133X_REG_CONTROL, pdata->ctrl);
++ }
++ break;
++ case ds_1338:
++ /* clock halted? turn it on, so clock can tick. */
++ if (ds13xx->regs[DS13XX_REG_SECS] & DS1307_BIT_CH) {
++ ds13xx_write_reg(client, DS13XX_REG_SECS, 0);
++ }
++ /* oscillator fault? clear flag, and warn */
++ if (ds13xx->regs[DS1307_REG_CONTROL] & DS1338_BIT_OSF) {
++ ret = ds13xx_write_reg(client, DS1307_REG_CONTROL,
++ ds13xx->regs[DS1307_REG_CONTROL] &
++ ~DS1338_BIT_OSF);
++ ds13xx->valid_time = 0;
++ }
++ break;
++ case ds_1339:
++ ret = ds13xx_read_reg(client, DS133X_REG_CONTROL);
++ if (ret < 0) {
++ dev_warn(&client->dev, "failed to read DS1339 control register: %d\n",
++ ret);
++ break;
++ } else if (ret & DS133X_BIT_nEOSC) {
++ oscstart = 1;
++ }
++ if (pdata && pdata->ctrl >= 0) {
++ ds13xx_write_reg(client, DS133X_REG_CONTROL, pdata->ctrl);
++ }
++ if (pdata && pdata->trc >= 0) {
++ ds13xx_write_reg(client, DS1339_REG_TRC, pdata->trc);
++ }
++ ret = ds13xx_read_reg(&ds13xx->client, DS133X_REG_STATUS);
++ if (ret < 0) {
++ dev_warn(&client->dev, "failed to read DS1339 status: %d\n",
++ ret);
++ break;
++ } else {
++ dev_dbg(&client->dev, "DS1339 status: %02x\n", ret);
++ }
++ if (ret > 0 && ret & DS133X_BIT_OSF) {
++ ds13xx->valid_time = 0;
++ }
++ break;
++ case ds_1340:
++ if (ds13xx->regs[DS13XX_REG_SECS] & DS133X_BIT_nEOSC) {
++ oscstart = 1;
++ ds13xx_write_reg(client, DS13XX_REG_SECS, 0);
++ }
++ if (pdata && pdata->ctrl >= 0) {
++ ds13xx_write_reg(client, DS133X_REG_CONTROL, pdata->ctrl);
++ }
++ ret = ds13xx_read_reg(&ds13xx->client, DS1340_REG_STATUS);
++ if (ret > 0 && ret & DS1340_BIT_OSF) {
++ ds13xx->valid_time = 0;
++ }
++ break;
++ default:
++ dev_warn(&client->dev, "Unknown DS13xx chip type: %d\n", ds13xx->type);
++ break;
++ }
++ if (!ds13xx->valid_time) {
++ dev_warn(&client->dev, "%s oscillator failure; RTC time must be set!\n",
++ client->name);
++ } else if (oscstart) {
++ ds13xx->valid_time = 0;
++ dev_warn(&client->dev, "%s oscillator has been started; RTC time must be set!\n",
++ client->name);
++ }
++
++ return ret < 0 ? ret : 0;
++}
++
++static int ds13xx_get_time(struct device *dev, struct rtc_time *t)
++{
++ int ret;
++ struct ds13xx *ds13xx = dev_get_drvdata(dev);
++ static int once = 1;
++
++ if (!ds13xx->valid_time) {
++ if (once) {
++ dev_warn(dev, "RTC Oscillator failure; RTC time must be set!\n");
++ once = 0;
++ }
++ }
++
++ /* read the RTC registers all at once */
++ ds13xx->msg[1].flags = I2C_M_RD;
++ ds13xx->msg[1].len = 7;
++
++ ret = i2c_transfer(ds13xx->client.adapter, ds13xx->msg, 2);
++ if (ret != 2) {
++ dev_err(dev, "%s error %d\n", "read", ret);
++ return ret < 0 ? ret : -EIO;
++ }
++
++ dev_dbg(dev, "%s: %02x %02x %02x %02x %02x %02x %02x\n",
++ "read",
++ ds13xx->regs[0], ds13xx->regs[1],
++ ds13xx->regs[2], ds13xx->regs[3],
++ ds13xx->regs[4], ds13xx->regs[5],
++ ds13xx->regs[6]);
++
++ t->tm_sec = bcd2bin(ds13xx->regs[DS13XX_REG_SECS] & 0x7f);
++ t->tm_min = bcd2bin(ds13xx->regs[DS13XX_REG_MIN] & 0x7f);
++ ret = ds13xx->regs[DS13XX_REG_HOUR] & 0x3f;
++ t->tm_hour = bcd2bin(ret);
++ t->tm_wday = bcd2bin(ds13xx->regs[DS13XX_REG_WDAY] & 0x07) - 1;
++ t->tm_mday = bcd2bin(ds13xx->regs[DS13XX_REG_MDAY] & 0x3f);
++ ret = ds13xx->regs[DS13XX_REG_MONTH] & 0x1f;
++ t->tm_mon = bcd2bin(ret) - 1;
++
++ t->tm_year = bcd2bin(ds13xx->regs[DS13XX_REG_YEAR]) + 100;
++ switch (ds13xx->type) {
++ case ds_1339:
++ if (ds13xx->regs[DS13XX_REG_MONTH] & DS133X_BIT_CENTURY) {
++ t->tm_year += 100;
++ }
++ default:
++ break;
++ }
++ dev_dbg(dev, "%s secs=%d, mins=%d, "
++ "hours=%d, mday=%d, mon=%d, year=%d, wday=%d\n",
++ "read", t->tm_sec, t->tm_min,
++ t->tm_hour, t->tm_mday,
++ t->tm_mon, t->tm_year - 100 + CENTURY, t->tm_wday);
++
++ return 0;
++}
++
++static int ds13xx_set_time(struct device *dev, struct rtc_time *t)
++{
++ int ret;
++ struct ds13xx *ds13xx = dev_get_drvdata(dev);
++ u8 *buf = ds13xx->regs;
++
++ dev_dbg(dev, "%s secs=%d, mins=%d, "
++ "hours=%d, mday=%d, mon=%d, year=%d, wday=%d\n",
++ "write", t->tm_sec, t->tm_min,
++ t->tm_hour, t->tm_mday,
++ t->tm_mon, t->tm_year - 100 + CENTURY, t->tm_wday);
++
++ *buf++ = 0; /* first register addr */
++ buf[DS13XX_REG_SECS] = bin2bcd(t->tm_sec);
++ buf[DS13XX_REG_MIN] = bin2bcd(t->tm_min);
++ buf[DS13XX_REG_HOUR] = bin2bcd(t->tm_hour);
++ buf[DS13XX_REG_WDAY] = bin2bcd(t->tm_wday + 1);
++ buf[DS13XX_REG_MDAY] = bin2bcd(t->tm_mday);
++ buf[DS13XX_REG_MONTH] = bin2bcd(t->tm_mon + 1);
++
++ ret = t->tm_year - 100;
++ buf[DS13XX_REG_YEAR] = bin2bcd(ret);
++
++ if (!ds13xx->valid_time) {
++ switch (ds13xx->type) {
++ case ds_1307:
++ break;
++ case ds_1337:
++ case ds_1339:
++ dev_dbg(dev, "Clearing OSF\n");
++ // clear oscillator fail flag, in case it was set
++ ret = ds13xx_write_reg(&ds13xx->client, DS133X_REG_STATUS,
++ (u8)~DS133X_BIT_OSF);
++ break;
++ case ds_1340:
++ buf[DS13XX_REG_HOUR] |= DS1340_BIT_CENTURY_EN;
++ // clear oscillator fail flag, in case it was set
++ ret = ds13xx_write_reg(&ds13xx->client, DS1340_REG_STATUS,
++ (u8)~DS1340_BIT_OSF);
++ break;
++ default:
++ BUG();
++ }
++ }
++ ds13xx->msg[1].flags = 0;
++ ds13xx->msg[1].len = sizeof(ds13xx->regs);
++
++ dev_dbg(dev, "%s: %02x %02x %02x %02x %02x %02x %02x\n",
++ "write", buf[0], buf[1], buf[2], buf[3],
++ buf[4], buf[5], buf[6]);
++
++ ret = i2c_transfer(ds13xx->client.adapter, &ds13xx->msg[1], 1);
++ if (ret != 1) {
++ dev_err(dev, "%s error %d\n", "write", ret);
++ return ret < 0 ? ret : -EIO;
++ }
++ ds13xx->valid_time = 1;
++ return 0;
++}
++
++static const struct rtc_class_ops ds13xx_rtc_ops = {
++ .read_time = ds13xx_get_time,
++ .set_time = ds13xx_set_time,
++};
++
++static struct i2c_driver ds13xx_i2c_driver;
++static struct platform_device *ds13xx_dev;
++
++static int __devinit ds13xx_detect(struct i2c_adapter *adapter, int address, int kind)
++{
++ struct ds13xx *ds13xx;
++ int err = -ENODEV;
++ struct i2c_client *client;
++ int ret;
++
++ if (ds13xx_dev == NULL) {
++ return -ENODEV;
++ }
++
++ ds13xx = kzalloc(sizeof(struct ds13xx), GFP_KERNEL);
++ if (ds13xx == NULL) {
++ err = -ENOMEM;
++ goto exit;
++ }
++
++ client = &ds13xx->client;
++ client->addr = address;
++ client->adapter = adapter;
++ client->driver = &ds13xx_i2c_driver;
++ client->flags = 0;
++
++ i2c_set_clientdata(client, ds13xx);
++
++ ds13xx->msg[0].addr = client->addr;
++ ds13xx->msg[0].flags = 0;
++ ds13xx->msg[0].len = 1;
++ ds13xx->msg[0].buf = &ds13xx->reg_addr;
++
++ ds13xx->msg[1].addr = client->addr;
++ ds13xx->msg[1].flags = I2C_M_RD;
++ ds13xx->msg[1].len = sizeof(ds13xx->regs);
++ ds13xx->msg[1].buf = ds13xx->regs;
++
++ /* HACK: "force" implies "needs ds1337-style-oscillator setup" */
++ if (kind >= 0) {
++ ds13xx->type = ds_1337;
++
++ ds13xx->reg_addr = DS133X_REG_CONTROL;
++ ds13xx->msg[1].len = 2;
++
++ ret = i2c_transfer(client->adapter, ds13xx->msg, 2);
++ if (ret != 2) {
++ pr_debug("read error %d\n", ret);
++ err = ret < 0 ? ret : -EIO;
++ goto exit_free;
++ }
++
++ ds13xx->reg_addr = 0;
++ ds13xx->msg[1].len = sizeof(ds13xx->regs);
++
++ /* oscillator is off; need to turn it on */
++ if ((ds13xx->regs[0] & DS133X_BIT_nEOSC)
++ || (ds13xx->regs[1] & DS133X_BIT_OSF)) {
++ dev_err(&ds13xx_dev->dev, "no ds1337 oscillator code\n");
++ goto exit_free;
++ }
++ } else {
++ ds13xx->type = ds_1307;
++ }
++
++ err = ds13xx_i2c_init(ds13xx_dev, client);
++ if (err) {
++ goto exit_free;
++ }
++
++read_rtc:
++ /* read RTC registers */
++
++ ret = i2c_transfer(client->adapter, ds13xx->msg, 2);
++ if (ret != 2) {
++ dev_dbg(&ds13xx_dev->dev, "read error %d\n", ret);
++ err = ret < 0 ? ret : -EIO;
++ goto exit_free;
++ }
++
++ err = ds13xx_rtc_init(ds13xx_dev->dev.platform_data, client);
++ if (err) {
++ goto exit_free;
++ }
++
++ /* minimal sanity checking; some chips (like DS1340) don't
++ * specify the extra bits as must-be-zero, but there are
++ * still a few values that are clearly out-of-range.
++ */
++ ret = ds13xx->regs[DS13XX_REG_SECS];
++ if (ret & DS1307_BIT_CH) {
++ if (ds13xx->type && ds13xx->type != ds_1307) {
++ pr_debug("not a ds1307?\n");
++ goto exit_free;
++ }
++ ds13xx->type = ds_1307;
++
++ /* this partial initialization should work for ds13xx,
++ * ds1338, ds1340, st m41t00, and more.
++ */
++ dev_warn(&client->dev, "oscillator started; SET TIME!\n");
++ ds13xx_write_reg(client, DS13XX_REG_SECS, 0);
++ goto read_rtc;
++ }
++ ret = bcd2bin(ret & 0x7f);
++ if (ret > 60)
++ goto exit_free;
++ ret = bcd2bin(ds13xx->regs[DS13XX_REG_MIN] & 0x7f);
++ if (ret > 60)
++ goto exit_free;
++
++ ret = bcd2bin(ds13xx->regs[DS13XX_REG_MDAY] & 0x3f);
++ if (ret == 0 || ret > 31)
++ goto exit_free;
++
++ ret = bcd2bin(ds13xx->regs[DS13XX_REG_MONTH] & 0x1f);
++ if (ret == 0 || ret > 12)
++ goto exit_free;
++
++ /* force into in 24 hour mode (most chips) or
++ * disable century bit (ds1340)
++ */
++ ret = ds13xx->regs[DS13XX_REG_HOUR];
++ if (ret & (1 << 6)) {
++ if (ret & (1 << 5))
++ ret = bcd2bin(ret & 0x1f) + 12;
++ else
++ ret = bcd2bin(ret);
++ ds13xx_write_reg(client,
++ DS13XX_REG_HOUR,
++ bin2bcd(ret));
++ }
++
++ /* Tell the I2C layer a new client has arrived */
++ if ((err = i2c_attach_client(client)))
++ goto exit_free;
++
++ dev_dbg(&client->dev, "Registering RTC %s\n", client->name);
++ ds13xx->rtc = rtc_device_register(client->name, &client->dev,
++ &ds13xx_rtc_ops, THIS_MODULE);
++ if (IS_ERR(ds13xx->rtc)) {
++ err = PTR_ERR(ds13xx->rtc);
++ dev_err(&client->dev,
++ "unable to register the class device\n");
++ goto exit_detach;
++ }
++
++ return 0;
++
++exit_detach:
++ i2c_detach_client(client);
++exit_free:
++ kfree(ds13xx);
++exit:
++ return err;
++}
++
++static int __devinit ds13xx_attach_adapter(struct i2c_adapter *adapter)
++{
++ if (!i2c_check_functionality(adapter, I2C_FUNC_I2C))
++ return 0;
++ return i2c_probe(adapter, &addr_data, ds13xx_detect);
++}
++
++static int __devexit ds13xx_detach_client(struct i2c_client *client)
++{
++ int err;
++ struct ds13xx *ds13xx = i2c_get_clientdata(client);
++
++ rtc_device_unregister(ds13xx->rtc);
++ if ((err = i2c_detach_client(client)))
++ return err;
++ kfree(ds13xx);
++ return 0;
++}
++
++static struct i2c_driver ds13xx_i2c_driver = {
++ .driver = {
++ .name = "ds13xx",
++ .owner = THIS_MODULE,
++ },
++ .id_table = ds13xx_id,
++ .attach_adapter = ds13xx_attach_adapter,
++ .detach_client = __devexit_p(ds13xx_detach_client),
++};
++
++static int __devinit ds13xx_probe(struct platform_device *pdev)
++{
++ ds13xx_dev = pdev;
++ return i2c_add_driver(&ds13xx_i2c_driver);
++}
++
++static int __devexit ds13xx_remove(struct platform_device *pdev)
++{
++ i2c_del_driver(&ds13xx_i2c_driver);
++ ds13xx_dev = NULL;
++ return 0;
++}
++
++static struct platform_driver ds13xx_driver = {
++ .probe = ds13xx_probe,
++ .remove = __devexit_p(ds13xx_remove),
++ .driver = {
++ .owner = THIS_MODULE,
++ .name = "rtc-ds13xx",
++ },
++};
++
++static int __init ds13xx_init(void)
++{
++ return platform_driver_register(&ds13xx_driver);
++}
++module_init(ds13xx_init);
++
++static void __exit ds13xx_exit(void)
++{
++ platform_driver_unregister(&ds13xx_driver);
++}
++module_exit(ds13xx_exit);
++
++MODULE_DESCRIPTION("RTC driver for DS1307,1337,1339,1340 and similar chips");
++MODULE_LICENSE("GPL");
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/rtc/rtc-mxc.c linux-2.6.28-karo/drivers/rtc/rtc-mxc.c
+--- linux-2.6.28/drivers/rtc/rtc-mxc.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/rtc/rtc-mxc.c 2009-03-11 18:47:09.000000000 +0100
+@@ -0,0 +1,835 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ */
++
++/*
++ * The code contained herein is licensed under the GNU General Public
++ * License. You may obtain a copy of the GNU General Public License
++ * Version 2 or later at the following locations:
++ *
++ * http://www.opensource.org/licenses/gpl-license.html
++ * http://www.gnu.org/copyleft/gpl.html
++ */
++/*
++ * Implementation based on rtc-ds1553.c
++ */
++
++/*!
++ * @defgroup RTC Real Time Clock (RTC) Driver
++ */
++/*!
++ * @file rtc-mxc.c
++ * @brief Real Time Clock interface
++ *
++ * This file contains Real Time Clock interface for Linux.
++ *
++ * @ingroup RTC
++ */
++
++#include <linux/rtc.h>
++#include <linux/module.h>
++#include <linux/fs.h>
++#include <linux/init.h>
++#include <linux/interrupt.h>
++#include <linux/platform_device.h>
++#include <linux/clk.h>
++#include <linux/uaccess.h>
++
++#include <mach/hardware.h>
++#include <asm/io.h>
++
++#define RTC_INPUT_CLK_32768HZ (0x00 << 5)
++#define RTC_INPUT_CLK_32000HZ (0x01 << 5)
++#define RTC_INPUT_CLK_38400HZ (0x02 << 5)
++
++#define RTC_SW_BIT (1 << 0)
++#define RTC_ALM_BIT (1 << 2)
++#define RTC_1HZ_BIT (1 << 4)
++#define RTC_2HZ_BIT (1 << 7)
++#define RTC_SAM0_BIT (1 << 8)
++#define RTC_SAM1_BIT (1 << 9)
++#define RTC_SAM2_BIT (1 << 10)
++#define RTC_SAM3_BIT (1 << 11)
++#define RTC_SAM4_BIT (1 << 12)
++#define RTC_SAM5_BIT (1 << 13)
++#define RTC_SAM6_BIT (1 << 14)
++#define RTC_SAM7_BIT (1 << 15)
++#define PIT_ALL_ON (RTC_2HZ_BIT | RTC_SAM0_BIT | RTC_SAM1_BIT | \
++ RTC_SAM2_BIT | RTC_SAM3_BIT | RTC_SAM4_BIT | \
++ RTC_SAM5_BIT | RTC_SAM6_BIT | RTC_SAM7_BIT)
++
++#define RTC_ENABLE_BIT (1 << 7)
++
++#define MAX_PIE_NUM 9
++#define MAX_PIE_FREQ 512
++const u32 PIE_BIT_DEF[MAX_PIE_NUM][2] = {
++ {2, RTC_2HZ_BIT},
++ {4, RTC_SAM0_BIT},
++ {8, RTC_SAM1_BIT},
++ {16, RTC_SAM2_BIT},
++ {32, RTC_SAM3_BIT},
++ {64, RTC_SAM4_BIT},
++ {128, RTC_SAM5_BIT},
++ {256, RTC_SAM6_BIT},
++ {MAX_PIE_FREQ, RTC_SAM7_BIT},
++};
++
++/* Those are the bits from a classic RTC we want to mimic */
++#define RTC_IRQF 0x80 /* any of the following 3 is active */
++#define RTC_PF 0x40 /* Periodic interrupt */
++#define RTC_AF 0x20 /* Alarm interrupt */
++#define RTC_UF 0x10 /* Update interrupt for 1Hz RTC */
++
++#define MXC_RTC_TIME 0
++#define MXC_RTC_ALARM 1
++
++#define RTC_HOURMIN 0x00 /* 32bit rtc hour/min counter reg */
++#define RTC_SECOND 0x04 /* 32bit rtc seconds counter reg */
++#define RTC_ALRM_HM 0x08 /* 32bit rtc alarm hour/min reg */
++#define RTC_ALRM_SEC 0x0C /* 32bit rtc alarm seconds reg */
++#define RTC_RTCCTL 0x10 /* 32bit rtc control reg */
++#define RTC_RTCISR 0x14 /* 32bit rtc interrupt status reg */
++#define RTC_RTCIENR 0x18 /* 32bit rtc interrupt enable reg */
++#define RTC_STPWCH 0x1C /* 32bit rtc stopwatch min reg */
++#define RTC_DAYR 0x20 /* 32bit rtc days counter reg */
++#define RTC_DAYALARM 0x24 /* 32bit rtc day alarm reg */
++#define RTC_TEST1 0x28 /* 32bit rtc test reg 1 */
++#define RTC_TEST2 0x2C /* 32bit rtc test reg 2 */
++#define RTC_TEST3 0x30 /* 32bit rtc test reg 3 */
++
++struct rtc_plat_data {
++ struct rtc_device *rtc;
++ void __iomem *ioaddr;
++ unsigned long baseaddr;
++ int irq;
++ struct clk *clk;
++ unsigned int irqen;
++ int alrm_sec;
++ int alrm_min;
++ int alrm_hour;
++ int alrm_mday;
++};
++
++/*!
++ * @defgroup RTC Real Time Clock (RTC) Driver
++ */
++/*!
++ * @file rtc-mxc.c
++ * @brief Real Time Clock interface
++ *
++ * This file contains Real Time Clock interface for Linux.
++ *
++ * @ingroup RTC
++ */
++
++#if defined(CONFIG_MXC_MC13783_RTC)
++#include <asm/arch/pmic_rtc.h>
++#else
++#define pmic_rtc_get_time(args) MXC_EXTERNAL_RTC_NONE
++#define pmic_rtc_set_time(args) MXC_EXTERNAL_RTC_NONE
++#define pmic_rtc_loaded() 0
++#endif
++
++#define RTC_VERSION "1.0"
++#define MXC_EXTERNAL_RTC_OK 0
++#define MXC_EXTERNAL_RTC_ERR -1
++#define MXC_EXTERNAL_RTC_NONE -2
++
++/*!
++ * This function reads the RTC value from some external source.
++ *
++ * @param second pointer to the returned value in second
++ *
++ * @return 0 if successful; non-zero otherwise
++ */
++int get_ext_rtc_time(u32 * second)
++{
++ int ret = 0;
++ struct timeval tmp;
++ if (!pmic_rtc_loaded()) {
++ return MXC_EXTERNAL_RTC_NONE;
++ }
++
++ ret = pmic_rtc_get_time(&tmp);
++
++ if (0 == ret)
++ *second = tmp.tv_sec;
++ else
++ ret = MXC_EXTERNAL_RTC_ERR;
++
++ return ret;
++}
++
++/*!
++ * This function sets external RTC
++ *
++ * @param second value in second to be set to external RTC
++ *
++ * @return 0 if successful; non-zero otherwise
++ */
++int set_ext_rtc_time(u32 second)
++{
++ int ret = 0;
++ struct timeval tmp;
++
++ if (!pmic_rtc_loaded()) {
++ return MXC_EXTERNAL_RTC_NONE;
++ }
++
++ tmp.tv_sec = second;
++
++ ret = pmic_rtc_set_time(&tmp);
++
++ if (0 != ret)
++ ret = MXC_EXTERNAL_RTC_ERR;
++
++ return ret;
++}
++
++static u32 rtc_freq = 2; /* minimun value for PIE */
++static unsigned long rtc_status;
++
++static struct rtc_time g_rtc_alarm = {
++ .tm_year = 0,
++ .tm_mon = 0,
++ .tm_mday = 0,
++ .tm_hour = 0,
++ .tm_mon = 0,
++ .tm_sec = 0,
++};
++
++static DEFINE_SPINLOCK(rtc_lock);
++
++/*!
++ * This function is used to obtain the RTC time or the alarm value in
++ * second.
++ *
++ * @param time_alarm use MXC_RTC_TIME for RTC time value; MXC_RTC_ALARM for alarm value
++ *
++ * @return The RTC time or alarm time in second.
++ */
++static u32 get_alarm_or_time(struct device *dev, int time_alarm)
++{
++ struct platform_device *pdev = to_platform_device(dev);
++ struct rtc_plat_data *pdata = platform_get_drvdata(pdev);
++ void __iomem *ioaddr = pdata->ioaddr;
++ u32 day, hr, min, sec, hr_min;
++ if (time_alarm == MXC_RTC_TIME) {
++ day = readw(ioaddr + RTC_DAYR);
++ hr_min = readw(ioaddr + RTC_HOURMIN);
++ sec = readw(ioaddr + RTC_SECOND);
++ } else if (time_alarm == MXC_RTC_ALARM) {
++ day = readw(ioaddr + RTC_DAYALARM);
++ hr_min = (0x0000FFFF) & readw(ioaddr + RTC_ALRM_HM);
++ sec = readw(ioaddr + RTC_ALRM_SEC);
++ } else {
++ panic("wrong value for time_alarm=%d\n", time_alarm);
++ }
++
++ hr = hr_min >> 8;
++ min = hr_min & 0x00FF;
++
++ return ((((day * 24 + hr) * 60) + min) * 60 + sec);
++}
++
++/*!
++ * This function sets the RTC alarm value or the time value.
++ *
++ * @param time_alarm the new alarm value to be updated in the RTC
++ * @param time use MXC_RTC_TIME for RTC time value; MXC_RTC_ALARM for alarm value
++ */
++static void set_alarm_or_time(struct device *dev, int time_alarm, u32 time)
++{
++ u32 day, hr, min, sec, temp;
++ struct platform_device *pdev = to_platform_device(dev);
++ struct rtc_plat_data *pdata = platform_get_drvdata(pdev);
++ void __iomem *ioaddr = pdata->ioaddr;
++ day = time / 86400;
++ time -= day * 86400;
++ /* time is within a day now */
++ hr = time / 3600;
++ time -= hr * 3600;
++ /* time is within an hour now */
++ min = time / 60;
++ sec = time - min * 60;
++
++ temp = (hr << 8) + min;
++
++ if (time_alarm == MXC_RTC_TIME) {
++ writew(day, ioaddr + RTC_DAYR);
++ writew(sec, ioaddr + RTC_SECOND);
++ writew(temp, ioaddr + RTC_HOURMIN);
++ } else if (time_alarm == MXC_RTC_ALARM) {
++ writew(day, ioaddr + RTC_DAYALARM);
++ writew(sec, ioaddr + RTC_ALRM_SEC);
++ writew(temp, ioaddr + RTC_ALRM_HM);
++ } else {
++ panic("wrong value for time_alarm=%d\n", time_alarm);
++ }
++}
++
++/*!
++ * This function updates the RTC alarm registers and then clears all the
++ * interrupt status bits.
++ *
++ * @param alrm the new alarm value to be updated in the RTC
++ *
++ * @return 0 if successful; non-zero otherwise.
++ */
++static int rtc_update_alarm(struct device *dev, struct rtc_time *alrm)
++{
++ struct rtc_time alarm_tm, now_tm;
++ unsigned long now, time;
++ int ret;
++ struct platform_device *pdev = to_platform_device(dev);
++ struct rtc_plat_data *pdata = platform_get_drvdata(pdev);
++ void __iomem *ioaddr = pdata->ioaddr;
++
++ now = get_alarm_or_time(dev, MXC_RTC_TIME);
++ rtc_time_to_tm(now, &now_tm);
++ alarm_tm.tm_year = now_tm.tm_year;
++ alarm_tm.tm_mon = now_tm.tm_mon;
++ alarm_tm.tm_mday = now_tm.tm_mday;
++ alarm_tm.tm_hour = alrm->tm_hour;
++ alarm_tm.tm_min = alrm->tm_min;
++ alarm_tm.tm_sec = alrm->tm_sec;
++ rtc_tm_to_time(&now_tm, &now);
++ rtc_tm_to_time(&alarm_tm, &time);
++ if (time < now) {
++ time += 60 * 60 * 24;
++ rtc_time_to_tm(time, &alarm_tm);
++ }
++ ret = rtc_tm_to_time(&alarm_tm, &time);
++
++ /* clear all the interrupt status bits */
++ writew(readw(ioaddr + RTC_RTCISR), ioaddr + RTC_RTCISR);
++
++ set_alarm_or_time(dev, MXC_RTC_ALARM, time);
++
++ return ret;
++}
++
++/*!
++ * This function is the RTC interrupt service routine.
++ *
++ * @param irq RTC IRQ number
++ * @param dev_id device ID which is not used
++ *
++ * @return IRQ_HANDLED as defined in the include/linux/interrupt.h file.
++ */
++static irqreturn_t mxc_rtc_interrupt(int irq, void *dev_id)
++{
++ struct platform_device *pdev = dev_id;
++ struct rtc_plat_data *pdata = platform_get_drvdata(pdev);
++ void __iomem *ioaddr = pdata->ioaddr;
++ u32 status;
++ u32 events = 0;
++ spin_lock(&rtc_lock);
++ status = readw(ioaddr + RTC_RTCISR) & readw(ioaddr + RTC_RTCIENR);
++ /* clear interrupt sources */
++ writew(status, ioaddr + RTC_RTCISR);
++
++ /* clear alarm interrupt if it has occurred */
++ if (status & RTC_ALM_BIT) {
++ status &= ~RTC_ALM_BIT;
++ }
++
++ /* update irq data & counter */
++ if (status & RTC_ALM_BIT) {
++ events |= (RTC_AF | RTC_IRQF);
++ }
++ if (status & RTC_1HZ_BIT) {
++ events |= (RTC_UF | RTC_IRQF);
++ }
++ if (status & PIT_ALL_ON) {
++ events |= (RTC_PF | RTC_IRQF);
++ }
++
++ if ((status & RTC_ALM_BIT) && rtc_valid_tm(&g_rtc_alarm)) {
++ rtc_update_alarm(&pdev->dev, &g_rtc_alarm);
++ }
++
++ spin_unlock(&rtc_lock);
++ rtc_update_irq(pdata->rtc, 1, events);
++ return IRQ_HANDLED;
++}
++
++/*!
++ * This function is used to open the RTC driver by registering the RTC
++ * interrupt service routine.
++ *
++ * @return 0 if successful; non-zero otherwise.
++ */
++static int mxc_rtc_open(struct device *dev)
++{
++ if (test_and_set_bit(1, &rtc_status))
++ return -EBUSY;
++ return 0;
++}
++
++/*!
++ * clear all interrupts and release the IRQ
++ */
++static void mxc_rtc_release(struct device *dev)
++{
++ struct platform_device *pdev = to_platform_device(dev);
++ struct rtc_plat_data *pdata = platform_get_drvdata(pdev);
++ void __iomem *ioaddr = pdata->ioaddr;
++
++ spin_lock_irq(&rtc_lock);
++ writew(0, ioaddr + RTC_RTCIENR); /* Disable all rtc interrupts */
++ writew(0xFFFFFFFF, ioaddr + RTC_RTCISR); /* Clear all interrupt status */
++ spin_unlock_irq(&rtc_lock);
++ rtc_status = 0;
++}
++
++/*!
++ * This function is used to support some ioctl calls directly.
++ * Other ioctl calls are supported indirectly through the
++ * arm/common/rtctime.c file.
++ *
++ * @param cmd ioctl command as defined in include/linux/rtc.h
++ * @param arg value for the ioctl command
++ *
++ * @return 0 if successful or negative value otherwise.
++ */
++static int mxc_rtc_ioctl(struct device *dev, unsigned int cmd,
++ unsigned long arg)
++{
++ struct platform_device *pdev = to_platform_device(dev);
++ struct rtc_plat_data *pdata = platform_get_drvdata(pdev);
++ void __iomem *ioaddr = pdata->ioaddr;
++ int i;
++ switch (cmd) {
++ case RTC_PIE_OFF:
++ writew((readw(ioaddr + RTC_RTCIENR) & ~PIT_ALL_ON),
++ ioaddr + RTC_RTCIENR);
++ return 0;
++ case RTC_IRQP_SET:
++ if (arg < 2 || arg > MAX_PIE_FREQ || (arg % 2) != 0)
++ return -EINVAL; /* Also make sure a power of 2Hz */
++ if ((arg > 64) && (!capable(CAP_SYS_RESOURCE)))
++ return -EACCES;
++ rtc_freq = arg;
++ return 0;
++ case RTC_IRQP_READ:
++ return put_user(rtc_freq, (u32 *) arg);
++ case RTC_PIE_ON:
++ for (i = 0; i < MAX_PIE_NUM; i++) {
++ if (PIE_BIT_DEF[i][0] == rtc_freq) {
++ break;
++ }
++ }
++ if (i == MAX_PIE_NUM) {
++ return -EACCES;
++ }
++ spin_lock_irq(&rtc_lock);
++ writew((readw(ioaddr + RTC_RTCIENR) | PIE_BIT_DEF[i][1]),
++ ioaddr + RTC_RTCIENR);
++ spin_unlock_irq(&rtc_lock);
++ return 0;
++ case RTC_AIE_OFF:
++ spin_lock_irq(&rtc_lock);
++ writew((readw(ioaddr + RTC_RTCIENR) & ~RTC_ALM_BIT),
++ ioaddr + RTC_RTCIENR);
++ spin_unlock_irq(&rtc_lock);
++ return 0;
++
++ case RTC_AIE_ON:
++ spin_lock_irq(&rtc_lock);
++ writew((readw(ioaddr + RTC_RTCIENR) | RTC_ALM_BIT),
++ ioaddr + RTC_RTCIENR);
++ spin_unlock_irq(&rtc_lock);
++ return 0;
++
++ case RTC_UIE_OFF: /* UIE is for the 1Hz interrupt */
++ spin_lock_irq(&rtc_lock);
++ writew((readw(ioaddr + RTC_RTCIENR) & ~RTC_1HZ_BIT),
++ ioaddr + RTC_RTCIENR);
++ spin_unlock_irq(&rtc_lock);
++ return 0;
++
++ case RTC_UIE_ON:
++ spin_lock_irq(&rtc_lock);
++ writew((readw(ioaddr + RTC_RTCIENR) | RTC_1HZ_BIT),
++ ioaddr + RTC_RTCIENR);
++ spin_unlock_irq(&rtc_lock);
++ return 0;
++ }
++ return -ENOIOCTLCMD;
++}
++
++/*!
++ * This function reads the current RTC time into tm in Gregorian date.
++ *
++ * @param tm contains the RTC time value upon return
++ *
++ * @return 0 if successful; non-zero otherwise.
++ */
++static int mxc_rtc_read_time(struct device *dev, struct rtc_time *tm)
++{
++ u32 val;
++
++ /* Avoid roll-over from reading the different registers */
++ do {
++ val = get_alarm_or_time(dev, MXC_RTC_TIME);
++ } while (val != get_alarm_or_time(dev, MXC_RTC_TIME));
++
++ rtc_time_to_tm(val, tm);
++ return 0;
++}
++
++/*!
++ * This function sets the internal RTC time based on tm in Gregorian date.
++ *
++ * @param tm the time value to be set in the RTC
++ *
++ * @return 0 if successful; non-zero otherwise.
++ */
++static int mxc_rtc_set_time(struct device *dev, struct rtc_time *tm)
++{
++ unsigned long time;
++ int ret;
++ ret = rtc_tm_to_time(tm, &time);
++ if (ret != 0) {
++ return ret;
++ }
++
++ /* Avoid roll-over from reading the different registers */
++ do {
++ set_alarm_or_time(dev, MXC_RTC_TIME, time);
++ } while (time != get_alarm_or_time(dev, MXC_RTC_TIME));
++
++ ret = set_ext_rtc_time(time);
++
++ if (ret != MXC_EXTERNAL_RTC_OK) {
++ if (ret == MXC_EXTERNAL_RTC_NONE) {
++ pr_info("No external RTC\n");
++ ret = 0;
++ } else
++ pr_info("Failed to set external RTC\n");
++ }
++
++ return ret;
++}
++
++/*!
++ * This function reads the current alarm value into the passed in \b alrm
++ * argument. It updates the \b alrm's pending field value based on the whether
++ * an alarm interrupt occurs or not.
++ *
++ * @param alrm contains the RTC alarm value upon return
++ *
++ * @return 0 if successful; non-zero otherwise.
++ */
++static int mxc_rtc_read_alarm(struct device *dev, struct rtc_wkalrm *alrm)
++{
++ struct platform_device *pdev = to_platform_device(dev);
++ struct rtc_plat_data *pdata = platform_get_drvdata(pdev);
++ void __iomem *ioaddr = pdata->ioaddr;
++
++ rtc_time_to_tm(get_alarm_or_time(dev, MXC_RTC_ALARM), &alrm->time);
++ alrm->enabled = !!(readw(ioaddr + RTC_RTCIENR) & RTC_ALM_BIT);
++ alrm->pending = !!(readw(ioaddr + RTC_RTCISR) & RTC_ALM_BIT);
++ return 0;
++}
++
++/*!
++ * This function sets the RTC alarm based on passed in alrm.
++ *
++ * @param alrm the alarm value to be set in the RTC
++ *
++ * @return 0 if successful; non-zero otherwise.
++ */
++static int mxc_rtc_set_alarm(struct device *dev, struct rtc_wkalrm *alrm)
++{
++ struct platform_device *pdev = to_platform_device(dev);
++ struct rtc_plat_data *pdata = platform_get_drvdata(pdev);
++ void __iomem *ioaddr = pdata->ioaddr;
++ int ret;
++
++ spin_lock_irq(&rtc_lock);
++ if (rtc_valid_tm(&alrm->time)) {
++ if (alrm->time.tm_sec > 59 ||
++ alrm->time.tm_hour > 23 || alrm->time.tm_min > 59) {
++ ret = -EINVAL;
++ goto out;
++ }
++ ret = rtc_update_alarm(dev, &alrm->time);
++ } else {
++ if ((ret = rtc_valid_tm(&alrm->time)))
++ goto out;
++ ret = rtc_update_alarm(dev, &alrm->time);
++ }
++
++ if (ret == 0) {
++ memcpy(&g_rtc_alarm, &alrm->time, sizeof(struct rtc_time));
++
++ if (alrm->enabled) {
++ writew((readw(ioaddr + RTC_RTCIENR) | RTC_ALM_BIT),
++ ioaddr + RTC_RTCIENR);
++ } else {
++ writew((readw(ioaddr + RTC_RTCIENR) & ~RTC_ALM_BIT),
++ ioaddr + RTC_RTCIENR);
++ }
++ device_set_wakeup_enable(dev, alrm->enabled);
++ }
++ out:
++ spin_unlock_irq(&rtc_lock);
++
++ return ret;
++}
++
++/*!
++ * This function is used to provide the content for the /proc/driver/rtc
++ * file.
++ *
++ * @param buf the buffer to hold the information that the driver wants to write
++ *
++ * @return The number of bytes written into the rtc file.
++ */
++static int mxc_rtc_proc(struct device *dev, struct seq_file *sq)
++{
++ struct platform_device *pdev = to_platform_device(dev);
++ struct rtc_plat_data *pdata = platform_get_drvdata(pdev);
++ void __iomem *ioaddr = pdata->ioaddr;
++ char *p = sq->buf;
++
++ p += sprintf(p, "alarm_IRQ\t: %s\n",
++ (((readw(ioaddr + RTC_RTCIENR)) & RTC_ALM_BIT) !=
++ 0) ? "yes" : "no");
++ p += sprintf(p, "update_IRQ\t: %s\n",
++ (((readw(ioaddr + RTC_RTCIENR)) & RTC_1HZ_BIT) !=
++ 0) ? "yes" : "no");
++ p += sprintf(p, "periodic_IRQ\t: %s\n",
++ (((readw(ioaddr + RTC_RTCIENR)) & PIT_ALL_ON) !=
++ 0) ? "yes" : "no");
++ p += sprintf(p, "periodic_freq\t: %d\n", rtc_freq);
++
++ return p - (sq->buf);
++}
++
++/*!
++ * The RTC driver structure
++ */
++static struct rtc_class_ops mxc_rtc_ops = {
++ .open = mxc_rtc_open,
++ .release = mxc_rtc_release,
++ .ioctl = mxc_rtc_ioctl,
++ .read_time = mxc_rtc_read_time,
++ .set_time = mxc_rtc_set_time,
++ .read_alarm = mxc_rtc_read_alarm,
++ .set_alarm = mxc_rtc_set_alarm,
++ .proc = mxc_rtc_proc,
++};
++
++/*! MXC RTC Power management control */
++
++static struct timespec mxc_rtc_delta;
++
++static int mxc_rtc_probe(struct platform_device *pdev)
++{
++ struct clk *clk;
++ struct timespec tv;
++ struct resource *res;
++ struct rtc_time temp_time;
++ struct rtc_device *rtc;
++ struct rtc_plat_data *pdata = NULL;
++ u32 sec, reg;
++ int ret;
++ int rtc_input_clk;
++
++ res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
++ if (!res)
++ return -ENODEV;
++
++ pdata = kzalloc(sizeof(*pdata), GFP_KERNEL);
++ if (!pdata)
++ return -ENOMEM;
++
++ pdata->clk = clk_get(&pdev->dev, "rtc_clk");
++ clk_enable(pdata->clk);
++
++ pdata->baseaddr = res->start;
++ pdata->ioaddr = ((void *)(IO_ADDRESS(pdata->baseaddr)));
++ /* Configure and enable the RTC */
++ rtc = rtc_device_register(pdev->name, &pdev->dev, &mxc_rtc_ops,
++ THIS_MODULE);
++ if (IS_ERR(rtc)) {
++ ret = PTR_ERR(rtc);
++ if (pdata->irq >= 0)
++ free_irq(pdata->irq, pdev);
++ kfree(pdata);
++ return ret;
++ }
++ pdata->rtc = rtc;
++ platform_set_drvdata(pdev, pdata);
++
++ pdata->irq = platform_get_irq(pdev, 0);
++ if (pdata->irq >= 0) {
++ if (request_irq(pdata->irq, mxc_rtc_interrupt, IRQF_SHARED,
++ pdev->name, pdev) < 0) {
++ dev_warn(&pdev->dev, "interrupt not available.\n");
++ pdata->irq = -1;
++ }
++ }
++ device_set_wakeup_capable(&pdev->dev, 1);
++
++ ret = get_ext_rtc_time(&sec);
++ if (ret == MXC_EXTERNAL_RTC_OK) {
++ rtc_time_to_tm(sec, &temp_time);
++ mxc_rtc_set_time(&pdev->dev, &temp_time);
++
++ } else if (ret == MXC_EXTERNAL_RTC_NONE) {
++ pr_info("No external RTC device\n");
++ } else {
++ pr_info("Reading external RTC device failed\n");
++ }
++ tv.tv_nsec = 0;
++ tv.tv_sec = get_alarm_or_time(&pdev->dev, MXC_RTC_TIME);
++ clk = clk_get(NULL, "ckil");
++ rtc_input_clk = clk_get_rate(clk);
++ if (rtc_input_clk == 32768)
++ reg = RTC_INPUT_CLK_32768HZ;
++ else if (rtc_input_clk == 32000)
++ reg = RTC_INPUT_CLK_32000HZ;
++ else if (rtc_input_clk == 38400)
++ reg = RTC_INPUT_CLK_38400HZ;
++ else {
++ printk(KERN_ALERT "rtc clock is not valid");
++ return -EINVAL;
++ }
++ clk_put(clk);
++ reg |= RTC_ENABLE_BIT;
++ writew(reg, (pdata->ioaddr + RTC_RTCCTL));
++ if (((readw(pdata->ioaddr + RTC_RTCCTL)) & RTC_ENABLE_BIT) == 0) {
++ printk(KERN_ALERT "rtc : hardware module can't be enabled!\n");
++ return -EPERM;
++ }
++ printk("Real Time clock Driver v%s %ukHz base clock\n", RTC_VERSION,
++ rtc_input_clk);
++ return 0;
++}
++
++static int __exit mxc_rtc_remove(struct platform_device *pdev)
++{
++ struct rtc_plat_data *pdata = platform_get_drvdata(pdev);
++ rtc_device_unregister(pdata->rtc);
++ if (pdata->irq >= 0) {
++ free_irq(pdata->irq, pdev);
++ }
++ clk_disable(pdata->clk);
++ clk_put(pdata->clk);
++ kfree(pdata);
++ mxc_rtc_release(NULL);
++ return 0;
++}
++
++/*!
++ * This function is called to save the system time delta relative to
++ * the MXC RTC when enterring a low power state. This time delta is
++ * then used on resume to adjust the system time to account for time
++ * loss while suspended.
++ *
++ * @param pdev not used
++ * @param state Power state to enter.
++ *
++ * @return The function always returns 0.
++ */
++static int mxc_rtc_suspend(struct platform_device *pdev, pm_message_t state)
++{
++ struct rtc_plat_data *pdata = platform_get_drvdata(pdev);
++ struct timespec tv;
++
++ /* calculate time delta for suspend */
++ /* RTC precision is 1 second; adjust delta for avg 1/2 sec err */
++ tv.tv_nsec = NSEC_PER_SEC >> 1;
++ tv.tv_sec = get_alarm_or_time(&pdev->dev, MXC_RTC_TIME);
++ set_normalized_timespec(&mxc_rtc_delta,
++ xtime.tv_sec - tv.tv_sec,
++ xtime.tv_nsec - tv.tv_nsec);
++
++ if (device_may_wakeup(&pdev->dev)) {
++ int ret;
++
++ ret = enable_irq_wake(pdata->irq);
++ if (ret != 0) {
++ dev_warn(&pdev->dev, "Failed to enable IRQ wake for IRQ %d: %d\n",
++ pdata->irq, ret);
++ return ret;
++ }
++ }
++ return 0;
++}
++
++/*!
++ * This function is called to correct the system time based on the
++ * current MXC RTC time relative to the time delta saved during
++ * suspend.
++ *
++ * @param pdev not used
++ *
++ * @return The function always returns 0.
++ */
++static int mxc_rtc_resume(struct platform_device *pdev)
++{
++ struct rtc_plat_data *pdata = platform_get_drvdata(pdev);
++ struct timespec tv;
++ struct timespec ts;
++
++ tv.tv_nsec = 0;
++ tv.tv_sec = get_alarm_or_time(&pdev->dev, MXC_RTC_TIME);
++
++ /* restore wall clock using delta against this RTC;
++ * adjust again for avg 1/2 second RTC sampling error
++ */
++ set_normalized_timespec(&ts,
++ tv.tv_sec + mxc_rtc_delta.tv_sec,
++ (NSEC_PER_SEC >> 1) + mxc_rtc_delta.tv_nsec);
++ do_settimeofday(&ts);
++ if (device_may_wakeup(&pdev->dev))
++ disable_irq_wake(pdata->irq);
++ return 0;
++}
++
++/*!
++ * Contains pointers to the power management callback functions.
++ */
++static struct platform_driver mxc_rtc_driver = {
++ .driver = {
++ .name = "mxc_rtc",
++ },
++ .probe = mxc_rtc_probe,
++ .remove = __exit_p(mxc_rtc_remove),
++ .suspend = mxc_rtc_suspend,
++ .resume = mxc_rtc_resume,
++};
++
++/*!
++ * This function creates the /proc/driver/rtc file and registers the device RTC
++ * in the /dev/misc directory. It also reads the RTC value from external source
++ * and setup the internal RTC properly.
++ *
++ * @return -1 if RTC is failed to initialize; 0 is successful.
++ */
++static int __init mxc_rtc_init(void)
++{
++ return platform_driver_register(&mxc_rtc_driver);
++}
++
++/*!
++ * This function removes the /proc/driver/rtc file and un-registers the
++ * device RTC from the /dev/misc directory.
++ */
++static void __exit mxc_rtc_exit(void)
++{
++ platform_driver_unregister(&mxc_rtc_driver);
++
++}
++
++module_init(mxc_rtc_init);
++module_exit(mxc_rtc_exit);
++
++MODULE_AUTHOR("Freescale Semiconductor, Inc.");
++MODULE_DESCRIPTION("Realtime Clock Driver (RTC)");
++MODULE_LICENSE("GPL");
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/serial/imx.c linux-2.6.28-karo/drivers/serial/imx.c
+--- linux-2.6.28/drivers/serial/imx.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/serial/imx.c 2009-03-11 13:16:24.000000000 +0100
+@@ -31,6 +31,7 @@
+ #endif
+
+ #include <linux/module.h>
++#include <linux/io.h>
+ #include <linux/ioport.h>
+ #include <linux/init.h>
+ #include <linux/console.h>
+@@ -42,7 +43,6 @@
+ #include <linux/serial.h>
+ #include <linux/clk.h>
+
+-#include <asm/io.h>
+ #include <asm/irq.h>
+ #include <mach/hardware.h>
+ #include <mach/imx-uart.h>
+@@ -66,7 +66,7 @@
+ #define ONEMS 0xb0 /* One Millisecond register */
+ #define UTS 0xb4 /* UART Test Register */
+ #endif
+-#ifdef CONFIG_ARCH_IMX
++#if defined(CONFIG_ARCH_IMX) || defined(CONFIG_ARCH_MX1)
+ #define BIPR1 0xb0 /* Incremental Preset Register 1 */
+ #define BIPR2 0xb4 /* Incremental Preset Register 2 */
+ #define BIPR3 0xb8 /* Incremental Preset Register 3 */
+@@ -79,105 +79,105 @@
+ #endif
+
+ /* UART Control Register Bit Fields.*/
+-#define URXD_CHARRDY (1<<15)
+-#define URXD_ERR (1<<14)
+-#define URXD_OVRRUN (1<<13)
+-#define URXD_FRMERR (1<<12)
+-#define URXD_BRK (1<<11)
+-#define URXD_PRERR (1<<10)
+-#define UCR1_ADEN (1<<15) /* Auto dectect interrupt */
+-#define UCR1_ADBR (1<<14) /* Auto detect baud rate */
+-#define UCR1_TRDYEN (1<<13) /* Transmitter ready interrupt enable */
+-#define UCR1_IDEN (1<<12) /* Idle condition interrupt */
+-#define UCR1_RRDYEN (1<<9) /* Recv ready interrupt enable */
+-#define UCR1_RDMAEN (1<<8) /* Recv ready DMA enable */
+-#define UCR1_IREN (1<<7) /* Infrared interface enable */
+-#define UCR1_TXMPTYEN (1<<6) /* Transimitter empty interrupt enable */
+-#define UCR1_RTSDEN (1<<5) /* RTS delta interrupt enable */
+-#define UCR1_SNDBRK (1<<4) /* Send break */
+-#define UCR1_TDMAEN (1<<3) /* Transmitter ready DMA enable */
+-#ifdef CONFIG_ARCH_IMX
+-#define UCR1_UARTCLKEN (1<<2) /* UART clock enabled */
++#define URXD_CHARRDY (1 << 15)
++#define URXD_ERR (1 << 14)
++#define URXD_OVRRUN (1 << 13)
++#define URXD_FRMERR (1 << 12)
++#define URXD_BRK (1 << 11)
++#define URXD_PRERR (1 << 10)
++#define UCR1_ADEN (1 << 15) /* Auto dectect interrupt */
++#define UCR1_ADBR (1 << 14) /* Auto detect baud rate */
++#define UCR1_TRDYEN (1 << 13) /* Transmitter ready interrupt enable */
++#define UCR1_IDEN (1 << 12) /* Idle condition interrupt */
++#define UCR1_RRDYEN (1 << 9) /* Recv ready interrupt enable */
++#define UCR1_RDMAEN (1 << 8) /* Recv ready DMA enable */
++#define UCR1_IREN (1 << 7) /* Infrared interface enable */
++#define UCR1_TXMPTYEN (1 << 6) /* Transimitter empty interrupt enable */
++#define UCR1_RTSDEN (1 << 5) /* RTS delta interrupt enable */
++#define UCR1_SNDBRK (1 << 4) /* Send break */
++#define UCR1_TDMAEN (1 << 3) /* Transmitter ready DMA enable */
++#if defined(CONFIG_ARCH_IMX) || defined(CONFIG_ARCH_MX1)
++#define UCR1_UARTCLKEN (1 << 2) /* UART clock enabled */
+ #endif
+ #if defined CONFIG_ARCH_MX3 || defined CONFIG_ARCH_MX2
+-#define UCR1_UARTCLKEN (0) /* not present on mx2/mx3 */
++#define UCR1_UARTCLKEN 0 /* not present on mx2/mx3 */
+ #endif
+-#define UCR1_DOZE (1<<1) /* Doze */
+-#define UCR1_UARTEN (1<<0) /* UART enabled */
+-#define UCR2_ESCI (1<<15) /* Escape seq interrupt enable */
+-#define UCR2_IRTS (1<<14) /* Ignore RTS pin */
+-#define UCR2_CTSC (1<<13) /* CTS pin control */
+-#define UCR2_CTS (1<<12) /* Clear to send */
+-#define UCR2_ESCEN (1<<11) /* Escape enable */
+-#define UCR2_PREN (1<<8) /* Parity enable */
+-#define UCR2_PROE (1<<7) /* Parity odd/even */
+-#define UCR2_STPB (1<<6) /* Stop */
+-#define UCR2_WS (1<<5) /* Word size */
+-#define UCR2_RTSEN (1<<4) /* Request to send interrupt enable */
+-#define UCR2_TXEN (1<<2) /* Transmitter enabled */
+-#define UCR2_RXEN (1<<1) /* Receiver enabled */
+-#define UCR2_SRST (1<<0) /* SW reset */
+-#define UCR3_DTREN (1<<13) /* DTR interrupt enable */
+-#define UCR3_PARERREN (1<<12) /* Parity enable */
+-#define UCR3_FRAERREN (1<<11) /* Frame error interrupt enable */
+-#define UCR3_DSR (1<<10) /* Data set ready */
+-#define UCR3_DCD (1<<9) /* Data carrier detect */
+-#define UCR3_RI (1<<8) /* Ring indicator */
+-#define UCR3_TIMEOUTEN (1<<7) /* Timeout interrupt enable */
+-#define UCR3_RXDSEN (1<<6) /* Receive status interrupt enable */
+-#define UCR3_AIRINTEN (1<<5) /* Async IR wake interrupt enable */
+-#define UCR3_AWAKEN (1<<4) /* Async wake interrupt enable */
++#define UCR1_DOZE (1 << 1) /* Doze */
++#define UCR1_UARTEN (1 << 0) /* UART enabled */
++#define UCR2_ESCI (1 << 15) /* Escape seq interrupt enable */
++#define UCR2_IRTS (1 << 14) /* Ignore RTS pin */
++#define UCR2_CTSC (1 << 13) /* CTS pin control */
++#define UCR2_CTS (1 << 12) /* Clear to send */
++#define UCR2_ESCEN (1 << 11) /* Escape enable */
++#define UCR2_PREN (1 << 8) /* Parity enable */
++#define UCR2_PROE (1 << 7) /* Parity odd/even */
++#define UCR2_STPB (1 << 6) /* Stop */
++#define UCR2_WS (1 << 5) /* Word size */
++#define UCR2_RTSEN (1 << 4) /* Request to send interrupt enable */
++#define UCR2_TXEN (1 << 2) /* Transmitter enabled */
++#define UCR2_RXEN (1 << 1) /* Receiver enabled */
++#define UCR2_SRST (1 << 0) /* SW reset */
++#define UCR3_DTREN (1 << 13) /* DTR interrupt enable */
++#define UCR3_PARERREN (1 << 12) /* Parity enable */
++#define UCR3_FRAERREN (1 << 11) /* Frame error interrupt enable */
++#define UCR3_DSR (1 << 10) /* Data set ready */
++#define UCR3_DCD (1 << 9) /* Data carrier detect */
++#define UCR3_RI (1 << 8) /* Ring indicator */
++#define UCR3_TIMEOUTEN (1 << 7) /* Timeout interrupt enable */
++#define UCR3_RXDSEN (1 << 6) /* Receive status interrupt enable */
++#define UCR3_AIRINTEN (1 << 5) /* Async IR wake interrupt enable */
++#define UCR3_AWAKEN (1 << 4) /* Async wake interrupt enable */
+ #ifdef CONFIG_ARCH_IMX
+-#define UCR3_REF25 (1<<3) /* Ref freq 25 MHz, only on mx1 */
+-#define UCR3_REF30 (1<<2) /* Ref Freq 30 MHz, only on mx1 */
++#define UCR3_REF25 (1 << 3) /* Ref freq 25 MHz, only on mx1 */
++#define UCR3_REF30 (1 << 2) /* Ref Freq 30 MHz, only on mx1 */
+ #endif
+ #if defined CONFIG_ARCH_MX2 || defined CONFIG_ARCH_MX3
+-#define UCR3_RXDMUXSEL (1<<2) /* RXD Muxed Input Select, on mx2/mx3 */
++#define UCR3_RXDMUXSEL (1 << 2) /* RXD Muxed Input Select, on mx2/mx3 */
+ #endif
+-#define UCR3_INVT (1<<1) /* Inverted Infrared transmission */
+-#define UCR3_BPEN (1<<0) /* Preset registers enable */
+-#define UCR4_CTSTL_32 (32<<10) /* CTS trigger level (32 chars) */
+-#define UCR4_INVR (1<<9) /* Inverted infrared reception */
+-#define UCR4_ENIRI (1<<8) /* Serial infrared interrupt enable */
+-#define UCR4_WKEN (1<<7) /* Wake interrupt enable */
+-#define UCR4_REF16 (1<<6) /* Ref freq 16 MHz */
+-#define UCR4_IRSC (1<<5) /* IR special case */
+-#define UCR4_TCEN (1<<3) /* Transmit complete interrupt enable */
+-#define UCR4_BKEN (1<<2) /* Break condition interrupt enable */
+-#define UCR4_OREN (1<<1) /* Receiver overrun interrupt enable */
+-#define UCR4_DREN (1<<0) /* Recv data ready interrupt enable */
+-#define UFCR_RXTL_SHF 0 /* Receiver trigger level shift */
+-#define UFCR_RFDIV (7<<7) /* Reference freq divider mask */
+-#define UFCR_TXTL_SHF 10 /* Transmitter trigger level shift */
+-#define USR1_PARITYERR (1<<15) /* Parity error interrupt flag */
+-#define USR1_RTSS (1<<14) /* RTS pin status */
+-#define USR1_TRDY (1<<13) /* Transmitter ready interrupt/dma flag */
+-#define USR1_RTSD (1<<12) /* RTS delta */
+-#define USR1_ESCF (1<<11) /* Escape seq interrupt flag */
+-#define USR1_FRAMERR (1<<10) /* Frame error interrupt flag */
+-#define USR1_RRDY (1<<9) /* Receiver ready interrupt/dma flag */
+-#define USR1_TIMEOUT (1<<7) /* Receive timeout interrupt status */
+-#define USR1_RXDS (1<<6) /* Receiver idle interrupt flag */
+-#define USR1_AIRINT (1<<5) /* Async IR wake interrupt flag */
+-#define USR1_AWAKE (1<<4) /* Aysnc wake interrupt flag */
+-#define USR2_ADET (1<<15) /* Auto baud rate detect complete */
+-#define USR2_TXFE (1<<14) /* Transmit buffer FIFO empty */
+-#define USR2_DTRF (1<<13) /* DTR edge interrupt flag */
+-#define USR2_IDLE (1<<12) /* Idle condition */
+-#define USR2_IRINT (1<<8) /* Serial infrared interrupt flag */
+-#define USR2_WAKE (1<<7) /* Wake */
+-#define USR2_RTSF (1<<4) /* RTS edge interrupt flag */
+-#define USR2_TXDC (1<<3) /* Transmitter complete */
+-#define USR2_BRCD (1<<2) /* Break condition */
+-#define USR2_ORE (1<<1) /* Overrun error */
+-#define USR2_RDR (1<<0) /* Recv data ready */
+-#define UTS_FRCPERR (1<<13) /* Force parity error */
+-#define UTS_LOOP (1<<12) /* Loop tx and rx */
+-#define UTS_TXEMPTY (1<<6) /* TxFIFO empty */
+-#define UTS_RXEMPTY (1<<5) /* RxFIFO empty */
+-#define UTS_TXFULL (1<<4) /* TxFIFO full */
+-#define UTS_RXFULL (1<<3) /* RxFIFO full */
+-#define UTS_SOFTRST (1<<0) /* Software reset */
++#define UCR3_INVT (1 << 1) /* Inverted Infrared transmission */
++#define UCR3_BPEN (1 << 0) /* Preset registers enable */
++#define UCR4_CTSTL_32 (32 << 10) /* CTS trigger level (32 chars) */
++#define UCR4_INVR (1 << 9) /* Inverted infrared reception */
++#define UCR4_ENIRI (1 << 8) /* Serial infrared interrupt enable */
++#define UCR4_WKEN (1 << 7) /* Wake interrupt enable */
++#define UCR4_REF16 (1 << 6) /* Ref freq 16 MHz */
++#define UCR4_IRSC (1 << 5) /* IR special case */
++#define UCR4_TCEN (1 << 3) /* Transmit complete interrupt enable */
++#define UCR4_BKEN (1 << 2) /* Break condition interrupt enable */
++#define UCR4_OREN (1 << 1) /* Receiver overrun interrupt enable */
++#define UCR4_DREN (1 << 0) /* Recv data ready interrupt enable */
++#define UFCR_RXTL_SHF 0 /* Receiver trigger level shift */
++#define UFCR_RFDIV (7 << 7) /* Reference freq divider mask */
++#define UFCR_TXTL_SHF 10 /* Transmitter trigger level shift */
++#define USR1_PARITYERR (1 << 15) /* Parity error interrupt flag */
++#define USR1_RTSS (1 << 14) /* RTS pin status */
++#define USR1_TRDY (1 << 13) /* Transmitter ready interrupt/dma flag */
++#define USR1_RTSD (1 << 12) /* RTS delta */
++#define USR1_ESCF (1 << 11) /* Escape seq interrupt flag */
++#define USR1_FRAMERR (1 << 10) /* Frame error interrupt flag */
++#define USR1_RRDY (1 << 9) /* Receiver ready interrupt/dma flag */
++#define USR1_TIMEOUT (1 << 7) /* Receive timeout interrupt status */
++#define USR1_RXDS (1 << 6) /* Receiver idle interrupt flag */
++#define USR1_AIRINT (1 << 5) /* Async IR wake interrupt flag */
++#define USR1_AWAKE (1 << 4) /* Aysnc wake interrupt flag */
++#define USR2_ADET (1 << 15) /* Auto baud rate detect complete */
++#define USR2_TXFE (1 << 14) /* Transmit buffer FIFO empty */
++#define USR2_DTRF (1 << 13) /* DTR edge interrupt flag */
++#define USR2_IDLE (1 << 12) /* Idle condition */
++#define USR2_IRINT (1 << 8) /* Serial infrared interrupt flag */
++#define USR2_WAKE (1 << 7) /* Wake */
++#define USR2_RTSF (1 << 4) /* RTS edge interrupt flag */
++#define USR2_TXDC (1 << 3) /* Transmitter complete */
++#define USR2_BRCD (1 << 2) /* Break condition */
++#define USR2_ORE (1 << 1) /* Overrun error */
++#define USR2_RDR (1 << 0) /* Recv data ready */
++#define UTS_FRCPERR (1 << 13) /* Force parity error */
++#define UTS_LOOP (1 << 12) /* Loop tx and rx */
++#define UTS_TXEMPTY (1 << 6) /* TxFIFO empty */
++#define UTS_RXEMPTY (1 << 5) /* RxFIFO empty */
++#define UTS_TXFULL (1 << 4) /* TxFIFO full */
++#define UTS_RXFULL (1 << 3) /* RxFIFO full */
++#define UTS_SOFTRST (1 << 0) /* Software reset */
+
+ /* We've been assigned a range on the "Low-density serial ports" major */
+ #ifdef CONFIG_ARCH_IMX
+@@ -187,9 +187,9 @@
+ #define MAX_INTERNAL_IRQ IMX_IRQS
+ #endif
+
+-#if defined CONFIG_ARCH_MX3 || defined CONFIG_ARCH_MX2
+-#define SERIAL_IMX_MAJOR 207
+-#define MINOR_START 16
++#ifdef CONFIG_ARCH_MXC
++#define SERIAL_IMX_MAJOR 207
++#define MINOR_START 16
+ #define DEV_NAME "ttymxc"
+ #define MAX_INTERNAL_IRQ MXC_MAX_INT_LINES
+ #endif
+@@ -200,7 +200,7 @@
+ * so we have to poll them. We also check immediately before
+ * filling the TX fifo incase CTS has been dropped.
+ */
+-#define MCTRL_TIMEOUT (250*HZ/1000)
++#define MCTRL_TIMEOUT (250 * HZ / 1000)
+
+ #define DRIVER_NAME "IMX-uart"
+
+@@ -210,7 +210,7 @@ struct imx_port {
+ struct uart_port port;
+ struct timer_list timer;
+ unsigned int old_status;
+- int txirq,rxirq,rtsirq;
++ int txirq, rxirq, rtsirq;
+ int have_rtscts:1;
+ struct clk *clk;
+ };
+@@ -268,8 +268,8 @@ static void imx_stop_tx(struct uart_port
+ struct imx_port *sport = (struct imx_port *)port;
+ unsigned long temp;
+
+- temp = readl(sport->port.membase + UCR1);
+- writel(temp & ~UCR1_TXMPTYEN, sport->port.membase + UCR1);
++ temp = __raw_readl(sport->port.membase + UCR1);
++ __raw_writel(temp & ~UCR1_TXMPTYEN, sport->port.membase + UCR1);
+ }
+
+ /*
+@@ -280,8 +280,8 @@ static void imx_stop_rx(struct uart_port
+ struct imx_port *sport = (struct imx_port *)port;
+ unsigned long temp;
+
+- temp = readl(sport->port.membase + UCR2);
+- writel(temp &~ UCR2_RXEN, sport->port.membase + UCR2);
++ temp = __raw_readl(sport->port.membase + UCR2);
++ __raw_writel(temp & ~UCR2_RXEN, sport->port.membase + UCR2);
+ }
+
+ /*
+@@ -298,12 +298,12 @@ static inline void imx_transmit_buffer(s
+ {
+ struct circ_buf *xmit = &sport->port.info->xmit;
+
+- while (!(readl(sport->port.membase + UTS) & UTS_TXFULL)) {
++ while (!(__raw_readl(sport->port.membase + UTS) & UTS_TXFULL)) {
+ /* send xmit->buf[xmit->tail]
+ * out the port here */
+- writel(xmit->buf[xmit->tail], sport->port.membase + URTX0);
+- xmit->tail = (xmit->tail + 1) &
+- (UART_XMIT_SIZE - 1);
++ __raw_writel(xmit->buf[xmit->tail],
++ sport->port.membase + URTX0);
++ xmit->tail = (xmit->tail + 1) & (UART_XMIT_SIZE - 1);
+ sport->port.icount.tx++;
+ if (uart_circ_empty(xmit))
+ break;
+@@ -321,22 +321,22 @@ static void imx_start_tx(struct uart_por
+ struct imx_port *sport = (struct imx_port *)port;
+ unsigned long temp;
+
+- temp = readl(sport->port.membase + UCR1);
+- writel(temp | UCR1_TXMPTYEN, sport->port.membase + UCR1);
++ temp = __raw_readl(sport->port.membase + UCR1);
++ __raw_writel(temp | UCR1_TXMPTYEN, sport->port.membase + UCR1);
+
+- if (readl(sport->port.membase + UTS) & UTS_TXEMPTY)
++ if (__raw_readl(sport->port.membase + UTS) & UTS_TXEMPTY)
+ imx_transmit_buffer(sport);
+ }
+
+ static irqreturn_t imx_rtsint(int irq, void *dev_id)
+ {
+ struct imx_port *sport = dev_id;
+- unsigned int val = readl(sport->port.membase + USR1) & USR1_RTSS;
++ unsigned int val = __raw_readl(sport->port.membase + USR1) & USR1_RTSS;
+ unsigned long flags;
+
+ spin_lock_irqsave(&sport->port.lock, flags);
+
+- writel(USR1_RTSD, sport->port.membase + USR1);
++ __raw_writel(USR1_RTSD, sport->port.membase + USR1);
+ uart_handle_cts_change(&sport->port, !!val);
+ wake_up_interruptible(&sport->port.info->delta_msr_wait);
+
+@@ -350,11 +350,10 @@ static irqreturn_t imx_txint(int irq, vo
+ struct circ_buf *xmit = &sport->port.info->xmit;
+ unsigned long flags;
+
+- spin_lock_irqsave(&sport->port.lock,flags);
+- if (sport->port.x_char)
+- {
++ spin_lock_irqsave(&sport->port.lock, flags);
++ if (sport->port.x_char) {
+ /* Send next char */
+- writel(sport->port.x_char, sport->port.membase + URTX0);
++ __raw_writel(sport->port.x_char, sport->port.membase + URTX0);
+ goto out;
+ }
+
+@@ -369,37 +368,37 @@ static irqreturn_t imx_txint(int irq, vo
+ uart_write_wakeup(&sport->port);
+
+ out:
+- spin_unlock_irqrestore(&sport->port.lock,flags);
++ spin_unlock_irqrestore(&sport->port.lock, flags);
+ return IRQ_HANDLED;
+ }
+
+ static irqreturn_t imx_rxint(int irq, void *dev_id)
+ {
+ struct imx_port *sport = dev_id;
+- unsigned int rx,flg,ignored = 0;
++ unsigned int rx, flg, ignored = 0;
+ struct tty_struct *tty = sport->port.info->port.tty;
+ unsigned long flags, temp;
+
+- spin_lock_irqsave(&sport->port.lock,flags);
++ spin_lock_irqsave(&sport->port.lock, flags);
+
+- while (readl(sport->port.membase + USR2) & USR2_RDR) {
++ while (__raw_readl(sport->port.membase + USR2) & USR2_RDR) {
+ flg = TTY_NORMAL;
+ sport->port.icount.rx++;
+
+- rx = readl(sport->port.membase + URXD0);
++ rx = __raw_readl(sport->port.membase + URXD0);
+
+- temp = readl(sport->port.membase + USR2);
++ temp = __raw_readl(sport->port.membase + USR2);
+ if (temp & USR2_BRCD) {
+- writel(temp | USR2_BRCD, sport->port.membase + USR2);
++ __raw_writel(temp | USR2_BRCD,
++ sport->port.membase + USR2);
+ if (uart_handle_break(&sport->port))
+ continue;
+ }
+
+- if (uart_handle_sysrq_char
+- (&sport->port, (unsigned char)rx))
++ if (uart_handle_sysrq_char(&sport->port, (unsigned char)rx))
+ continue;
+
+- if (rx & (URXD_PRERR | URXD_OVRRUN | URXD_FRMERR) ) {
++ if (rx & (URXD_PRERR | URXD_OVRRUN | URXD_FRMERR)) {
+ if (rx & URXD_PRERR)
+ sport->port.icount.parity++;
+ else if (rx & URXD_FRMERR)
+@@ -431,7 +430,7 @@ static irqreturn_t imx_rxint(int irq, vo
+ }
+
+ out:
+- spin_unlock_irqrestore(&sport->port.lock,flags);
++ spin_unlock_irqrestore(&sport->port.lock, flags);
+ tty_flip_buffer_push(tty);
+ return IRQ_HANDLED;
+ }
+@@ -441,13 +440,13 @@ static irqreturn_t imx_int(int irq, void
+ struct imx_port *sport = dev_id;
+ unsigned int sts;
+
+- sts = readl(sport->port.membase + USR1);
++ sts = __raw_readl(sport->port.membase + USR1);
+
+ if (sts & USR1_RRDY)
+ imx_rxint(irq, dev_id);
+
+ if (sts & USR1_TRDY &&
+- readl(sport->port.membase + UCR1) & UCR1_TXMPTYEN)
++ __raw_readl(sport->port.membase + UCR1) & UCR1_TXMPTYEN)
+ imx_txint(irq, dev_id);
+
+ if (sts & USR1_RTSD)
+@@ -463,7 +462,8 @@ static unsigned int imx_tx_empty(struct
+ {
+ struct imx_port *sport = (struct imx_port *)port;
+
+- return (readl(sport->port.membase + USR2) & USR2_TXDC) ? TIOCSER_TEMT : 0;
++ return (__raw_readl(sport->port.membase + USR2) & USR2_TXDC) ?
++ TIOCSER_TEMT : 0;
+ }
+
+ /*
+@@ -471,29 +471,29 @@ static unsigned int imx_tx_empty(struct
+ */
+ static unsigned int imx_get_mctrl(struct uart_port *port)
+ {
+- struct imx_port *sport = (struct imx_port *)port;
+- unsigned int tmp = TIOCM_DSR | TIOCM_CAR;
++ struct imx_port *sport = (struct imx_port *)port;
++ unsigned int tmp = TIOCM_DSR | TIOCM_CAR;
+
+- if (readl(sport->port.membase + USR1) & USR1_RTSS)
+- tmp |= TIOCM_CTS;
++ if (__raw_readl(sport->port.membase + USR1) & USR1_RTSS)
++ tmp |= TIOCM_CTS;
+
+- if (readl(sport->port.membase + UCR2) & UCR2_CTS)
+- tmp |= TIOCM_RTS;
++ if (__raw_readl(sport->port.membase + UCR2) & UCR2_CTS)
++ tmp |= TIOCM_RTS;
+
+- return tmp;
++ return tmp;
+ }
+
+ static void imx_set_mctrl(struct uart_port *port, unsigned int mctrl)
+ {
+- struct imx_port *sport = (struct imx_port *)port;
++ struct imx_port *sport = (struct imx_port *)port;
+ unsigned long temp;
+
+- temp = readl(sport->port.membase + UCR2) & ~UCR2_CTS;
++ temp = __raw_readl(sport->port.membase + UCR2) & ~UCR2_CTS;
+
+- if (mctrl & TIOCM_RTS)
++ if (mctrl & TIOCM_RTS)
+ temp |= UCR2_CTS;
+
+- writel(temp, sport->port.membase + UCR2);
++ __raw_writel(temp, sport->port.membase + UCR2);
+ }
+
+ /*
+@@ -506,12 +506,12 @@ static void imx_break_ctl(struct uart_po
+
+ spin_lock_irqsave(&sport->port.lock, flags);
+
+- temp = readl(sport->port.membase + UCR1) & ~UCR1_SNDBRK;
++ temp = __raw_readl(sport->port.membase + UCR1) & ~UCR1_SNDBRK;
+
+- if ( break_state != 0 )
++ if (break_state != 0)
+ temp |= UCR1_SNDBRK;
+
+- writel(temp, sport->port.membase + UCR1);
++ __raw_writel(temp, sport->port.membase + UCR1);
+
+ spin_unlock_irqrestore(&sport->port.lock, flags);
+ }
+@@ -531,17 +531,17 @@ static int imx_setup_ufcr(struct imx_por
+ ufcr_rfdiv = (clk_get_rate(sport->clk) + sport->port.uartclk / 2)
+ / sport->port.uartclk;
+
+- if(!ufcr_rfdiv)
++ if (!ufcr_rfdiv)
+ ufcr_rfdiv = 1;
+
+- if(ufcr_rfdiv >= 7)
++ if (ufcr_rfdiv >= 7)
+ ufcr_rfdiv = 6;
+ else
+ ufcr_rfdiv = 6 - ufcr_rfdiv;
+
+ val |= UFCR_RFDIV & (ufcr_rfdiv << 7);
+
+- writel(val, sport->port.membase + UFCR);
++ __raw_writel(val, sport->port.membase + UFCR);
+
+ return 0;
+ }
+@@ -557,8 +557,8 @@ static int imx_startup(struct uart_port
+ /* disable the DREN bit (Data Ready interrupt enable) before
+ * requesting IRQs
+ */
+- temp = readl(sport->port.membase + UCR4);
+- writel(temp & ~UCR4_DREN, sport->port.membase + UCR4);
++ temp = __raw_readl(sport->port.membase + UCR4);
++ __raw_writel(temp & ~UCR4_DREN, sport->port.membase + UCR4);
+
+ /*
+ * Allocate the IRQ(s) i.MX1 has three interrupts whereas later
+@@ -576,8 +576,8 @@ static int imx_startup(struct uart_port
+ goto error_out2;
+
+ retval = request_irq(sport->rtsirq, imx_rtsint,
+- (sport->rtsirq < MAX_INTERNAL_IRQ) ? 0 :
+- IRQF_TRIGGER_FALLING | IRQF_TRIGGER_RISING,
++ (sport->rtsirq < MAX_INTERNAL_IRQ) ? 0 :
++ IRQF_TRIGGER_FALLING | IRQF_TRIGGER_RISING,
+ DRIVER_NAME, sport);
+ if (retval)
+ goto error_out3;
+@@ -593,28 +593,28 @@ static int imx_startup(struct uart_port
+ /*
+ * Finally, clear and enable interrupts
+ */
+- writel(USR1_RTSD, sport->port.membase + USR1);
++ __raw_writel(USR1_RTSD, sport->port.membase + USR1);
+
+- temp = readl(sport->port.membase + UCR1);
++ temp = __raw_readl(sport->port.membase + UCR1);
+ temp |= UCR1_RRDYEN | UCR1_RTSDEN | UCR1_UARTEN;
+- writel(temp, sport->port.membase + UCR1);
++ __raw_writel(temp, sport->port.membase + UCR1);
+
+- temp = readl(sport->port.membase + UCR2);
++ temp = __raw_readl(sport->port.membase + UCR2);
+ temp |= (UCR2_RXEN | UCR2_TXEN);
+- writel(temp, sport->port.membase + UCR2);
++ __raw_writel(temp, sport->port.membase + UCR2);
+
+ #if defined CONFIG_ARCH_MX2 || defined CONFIG_ARCH_MX3
+- temp = readl(sport->port.membase + UCR3);
++ temp = __raw_readl(sport->port.membase + UCR3);
+ temp |= UCR3_RXDMUXSEL;
+- writel(temp, sport->port.membase + UCR3);
++ __raw_writel(temp, sport->port.membase + UCR3);
+ #endif
+
+ /*
+ * Enable modem status interrupts
+ */
+- spin_lock_irqsave(&sport->port.lock,flags);
++ spin_lock_irqsave(&sport->port.lock, flags);
+ imx_enable_ms(&sport->port);
+- spin_unlock_irqrestore(&sport->port.lock,flags);
++ spin_unlock_irqrestore(&sport->port.lock, flags);
+
+ return 0;
+
+@@ -652,14 +652,14 @@ static void imx_shutdown(struct uart_por
+ * Disable all interrupts, port and break condition.
+ */
+
+- temp = readl(sport->port.membase + UCR1);
++ temp = __raw_readl(sport->port.membase + UCR1);
+ temp &= ~(UCR1_TXMPTYEN | UCR1_RRDYEN | UCR1_RTSDEN | UCR1_UARTEN);
+- writel(temp, sport->port.membase + UCR1);
++ __raw_writel(temp, sport->port.membase + UCR1);
+ }
+
+ static void
+ imx_set_termios(struct uart_port *port, struct ktermios *termios,
+- struct ktermios *old)
++ struct ktermios *old)
+ {
+ struct imx_port *sport = (struct imx_port *)port;
+ unsigned long flags;
+@@ -680,7 +680,7 @@ imx_set_termios(struct uart_port *port,
+ * We only support CS7 and CS8.
+ */
+ while ((termios->c_cflag & CSIZE) != CS7 &&
+- (termios->c_cflag & CSIZE) != CS8) {
++ (termios->c_cflag & CSIZE) != CS8) {
+ termios->c_cflag &= ~CSIZE;
+ termios->c_cflag |= old_csize;
+ old_csize = CS8;
+@@ -692,7 +692,7 @@ imx_set_termios(struct uart_port *port,
+ ucr2 = UCR2_SRST | UCR2_IRTS;
+
+ if (termios->c_cflag & CRTSCTS) {
+- if( sport->have_rtscts ) {
++ if (sport->have_rtscts) {
+ ucr2 &= ~UCR2_IRTS;
+ ucr2 |= UCR2_CTSC;
+ } else {
+@@ -748,17 +748,17 @@ imx_set_termios(struct uart_port *port,
+ /*
+ * disable interrupts and drain transmitter
+ */
+- old_ucr1 = readl(sport->port.membase + UCR1);
+- writel(old_ucr1 & ~(UCR1_TXMPTYEN | UCR1_RRDYEN | UCR1_RTSDEN),
++ old_ucr1 = __raw_readl(sport->port.membase + UCR1);
++ __raw_writel(old_ucr1 & ~(UCR1_TXMPTYEN | UCR1_RRDYEN | UCR1_RTSDEN),
+ sport->port.membase + UCR1);
+
+- while ( !(readl(sport->port.membase + USR2) & USR2_TXDC))
++ while (!(__raw_readl(sport->port.membase + USR2) & USR2_TXDC))
+ barrier();
+
+ /* then, disable everything */
+- old_txrxen = readl(sport->port.membase + UCR2);
+- writel(old_txrxen & ~( UCR2_TXEN | UCR2_RXEN),
+- sport->port.membase + UCR2);
++ old_txrxen = __raw_readl(sport->port.membase + UCR2);
++ __raw_writel(old_txrxen & ~(UCR2_TXEN | UCR2_RXEN),
++ sport->port.membase + UCR2);
+ old_txrxen &= (UCR2_TXEN | UCR2_RXEN);
+
+ div = sport->port.uartclk / (baud * 16);
+@@ -780,27 +780,27 @@ imx_set_termios(struct uart_port *port,
+ if (denom > 0)
+ denom -= 1;
+
+- writel(num, sport->port.membase + UBIR);
+- writel(denom, sport->port.membase + UBMR);
++ __raw_writel(num, sport->port.membase + UBIR);
++ __raw_writel(denom, sport->port.membase + UBMR);
+
+ if (div == 7)
+ div = 6; /* 6 in RFDIV means divide by 7 */
+ else
+ div = 6 - div;
+
+- ufcr = readl(sport->port.membase + UFCR);
+- ufcr = (ufcr & (~UFCR_RFDIV)) |
+- (div << 7);
+- writel(ufcr, sport->port.membase + UFCR);
++ ufcr = __raw_readl(sport->port.membase + UFCR);
++ ufcr = (ufcr & ~UFCR_RFDIV) | (div << 7);
++ __raw_writel(ufcr, sport->port.membase + UFCR);
+
+ #ifdef ONEMS
+- writel(sport->port.uartclk / div / 1000, sport->port.membase + ONEMS);
++ __raw_writel(sport->port.uartclk / div / 1000,
++ sport->port.membase + ONEMS);
+ #endif
+
+- writel(old_ucr1, sport->port.membase + UCR1);
++ __raw_writel(old_ucr1, sport->port.membase + UCR1);
+
+ /* set the parity, stop bits and data size */
+- writel(ucr2 | old_txrxen, sport->port.membase + UCR2);
++ __raw_writel(ucr2 | old_txrxen, sport->port.membase + UCR2);
+
+ if (UART_ENABLE_MS(&sport->port, termios->c_cflag))
+ imx_enable_ms(&sport->port);
+@@ -854,7 +854,7 @@ static void imx_config_port(struct uart_
+ struct imx_port *sport = (struct imx_port *)port;
+
+ if (flags & UART_CONFIG_TYPE &&
+- imx_request_port(&sport->port) == 0)
++ imx_request_port(&sport->port) == 0)
+ sport->port.type = PORT_IMX;
+ }
+
+@@ -912,10 +912,10 @@ static void imx_console_putchar(struct u
+ {
+ struct imx_port *sport = (struct imx_port *)port;
+
+- while (readl(sport->port.membase + UTS) & UTS_TXFULL)
++ while (__raw_readl(sport->port.membase + UTS) & UTS_TXFULL)
+ barrier();
+
+- writel(ch, sport->port.membase + URTX0);
++ __raw_writel(ch, sport->port.membase + URTX0);
+ }
+
+ /*
+@@ -930,14 +930,14 @@ imx_console_write(struct console *co, co
+ /*
+ * First, save UCR1/2 and then disable interrupts
+ */
+- old_ucr1 = readl(sport->port.membase + UCR1);
+- old_ucr2 = readl(sport->port.membase + UCR2);
++ old_ucr1 = __raw_readl(sport->port.membase + UCR1);
++ old_ucr2 = __raw_readl(sport->port.membase + UCR2);
+
+- writel((old_ucr1 | UCR1_UARTCLKEN | UCR1_UARTEN) &
++ __raw_writel((old_ucr1 | UCR1_UARTCLKEN | UCR1_UARTEN) &
+ ~(UCR1_TXMPTYEN | UCR1_RRDYEN | UCR1_RTSDEN),
+ sport->port.membase + UCR1);
+
+- writel(old_ucr2 | UCR2_TXEN, sport->port.membase + UCR2);
++ __raw_writel(old_ucr2 | UCR2_TXEN, sport->port.membase + UCR2);
+
+ uart_console_write(&sport->port, s, count, imx_console_putchar);
+
+@@ -945,10 +945,11 @@ imx_console_write(struct console *co, co
+ * Finally, wait for transmitter to become empty
+ * and restore UCR1/2
+ */
+- while (!(readl(sport->port.membase + USR2) & USR2_TXDC));
++ while (!(__raw_readl(sport->port.membase + USR2) & USR2_TXDC))
++ ;
+
+- writel(old_ucr1, sport->port.membase + UCR1);
+- writel(old_ucr2, sport->port.membase + UCR2);
++ __raw_writel(old_ucr1, sport->port.membase + UCR1);
++ __raw_writel(old_ucr2, sport->port.membase + UCR2);
+ }
+
+ /*
+@@ -957,16 +958,15 @@ imx_console_write(struct console *co, co
+ */
+ static void __init
+ imx_console_get_options(struct imx_port *sport, int *baud,
+- int *parity, int *bits)
++ int *parity, int *bits)
+ {
+-
+- if ( readl(sport->port.membase + UCR1) | UCR1_UARTEN ) {
++ if (__raw_readl(sport->port.membase + UCR1) | UCR1_UARTEN) {
+ /* ok, the port was enabled */
+- unsigned int ucr2, ubir,ubmr, uartclk;
++ unsigned int ucr2, ubir, ubmr, uartclk;
+ unsigned int baud_raw;
+ unsigned int ucfr_rfdiv;
+
+- ucr2 = readl(sport->port.membase + UCR2);
++ ucr2 = __raw_readl(sport->port.membase + UCR2);
+
+ *parity = 'n';
+ if (ucr2 & UCR2_PREN) {
+@@ -981,10 +981,11 @@ imx_console_get_options(struct imx_port
+ else
+ *bits = 7;
+
+- ubir = readl(sport->port.membase + UBIR) & 0xffff;
+- ubmr = readl(sport->port.membase + UBMR) & 0xffff;
++ ubir = __raw_readl(sport->port.membase + UBIR) & 0xffff;
++ ubmr = __raw_readl(sport->port.membase + UBMR) & 0xffff;
+
+- ucfr_rfdiv = (readl(sport->port.membase + UFCR) & UFCR_RFDIV) >> 7;
++ ucfr_rfdiv = (__raw_readl(sport->port.membase + UFCR) &
++ UFCR_RFDIV) >> 7;
+ if (ucfr_rfdiv == 6)
+ ucfr_rfdiv = 7;
+ else
+@@ -993,7 +994,8 @@ imx_console_get_options(struct imx_port
+ uartclk = clk_get_rate(sport->clk);
+ uartclk /= ucfr_rfdiv;
+
+- { /*
++ {
++ /*
+ * The next code provides exact computation of
+ * baud_raw = round(((uartclk/16) * (ubir + 1)) / (ubmr + 1))
+ * without need of float support or long long division,
+@@ -1008,7 +1010,7 @@ imx_console_get_options(struct imx_port
+ *baud = (baud_raw + 50) / 100 * 100;
+ }
+
+- if(*baud != baud_raw)
++ if (*baud != baud_raw)
+ printk(KERN_INFO "Serial: Console IMX rounded baud rate from %d to %d\n",
+ baud_raw, *baud);
+ }
+@@ -1028,9 +1030,13 @@ imx_console_setup(struct console *co, ch
+ * if so, search for the first available port that does have
+ * console support.
+ */
+- if (co->index == -1 || co->index >= ARRAY_SIZE(imx_ports))
++ if (co->index < 0 || co->index >= ARRAY_SIZE(imx_ports))
+ co->index = 0;
+- sport = imx_ports[co->index];
++ do {
++ sport = imx_ports[co->index];
++ } while (sport == NULL && ++co->index < ARRAY_SIZE(imx_ports));
++ if (!sport)
++ return -ENODEV;
+
+ if (options)
+ uart_parse_options(options, &baud, &parity, &bits, &flow);
+@@ -1070,22 +1076,22 @@ static struct uart_driver imx_reg = {
+
+ static int serial_imx_suspend(struct platform_device *dev, pm_message_t state)
+ {
+- struct imx_port *sport = platform_get_drvdata(dev);
++ struct imx_port *sport = platform_get_drvdata(dev);
+
+- if (sport)
+- uart_suspend_port(&imx_reg, &sport->port);
++ if (sport)
++ uart_suspend_port(&imx_reg, &sport->port);
+
+- return 0;
++ return 0;
+ }
+
+ static int serial_imx_resume(struct platform_device *dev)
+ {
+- struct imx_port *sport = platform_get_drvdata(dev);
++ struct imx_port *sport = platform_get_drvdata(dev);
+
+- if (sport)
+- uart_resume_port(&imx_reg, &sport->port);
++ if (sport)
++ uart_resume_port(&imx_reg, &sport->port);
+
+- return 0;
++ return 0;
+ }
+
+ static int serial_imx_probe(struct platform_device *pdev)
+@@ -1141,7 +1147,7 @@ static int serial_imx_probe(struct platf
+ imx_ports[pdev->id] = sport;
+
+ pdata = pdev->dev.platform_data;
+- if(pdata && (pdata->flags & IMXUART_HAVE_RTSCTS))
++ if (pdata && (pdata->flags & IMXUART_HAVE_RTSCTS))
+ sport->have_rtscts = 1;
+
+ if (pdata->init) {
+@@ -1150,13 +1156,13 @@ static int serial_imx_probe(struct platf
+ goto clkput;
+ }
+
+- uart_add_one_port(&imx_reg, &sport->port);
+ platform_set_drvdata(pdev, &sport->port);
++ uart_add_one_port(&imx_reg, &sport->port);
+
+ return 0;
+ clkput:
+- clk_put(sport->clk);
+ clk_disable(sport->clk);
++ clk_put(sport->clk);
+ unmap:
+ iounmap(sport->port.membase);
+ free:
+@@ -1172,15 +1178,12 @@ static int serial_imx_remove(struct plat
+
+ pdata = pdev->dev.platform_data;
+
+- platform_set_drvdata(pdev, NULL);
+-
+- if (sport) {
+- uart_remove_one_port(&imx_reg, &sport->port);
+- clk_put(sport->clk);
+- }
++ uart_remove_one_port(&imx_reg, &sport->port);
+
+ clk_disable(sport->clk);
++ clk_put(sport->clk);
+
++ platform_set_drvdata(pdev, NULL);
+ if (pdata->exit)
+ pdata->exit(pdev);
+
+@@ -1191,13 +1194,13 @@ static int serial_imx_remove(struct plat
+ }
+
+ static struct platform_driver serial_imx_driver = {
+- .probe = serial_imx_probe,
+- .remove = serial_imx_remove,
++ .probe = serial_imx_probe,
++ .remove = serial_imx_remove,
+
+ .suspend = serial_imx_suspend,
+ .resume = serial_imx_resume,
+ .driver = {
+- .name = "imx-uart",
++ .name = "imx-uart",
+ .owner = THIS_MODULE,
+ },
+ };
+@@ -1209,14 +1212,14 @@ static int __init imx_serial_init(void)
+ printk(KERN_INFO "Serial: IMX driver\n");
+
+ ret = uart_register_driver(&imx_reg);
+- if (ret)
++ if (ret != 0)
+ return ret;
+
+ ret = platform_driver_register(&serial_imx_driver);
+ if (ret != 0)
+ uart_unregister_driver(&imx_reg);
+
+- return 0;
++ return ret;
+ }
+
+ static void __exit imx_serial_exit(void)
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/spi/Kconfig linux-2.6.28-karo/drivers/spi/Kconfig
+--- linux-2.6.28/drivers/spi/Kconfig 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/spi/Kconfig 2009-03-11 13:16:24.000000000 +0100
+@@ -135,6 +135,55 @@ config SPI_MPC83xx
+ technology. This driver uses a simple set of shift registers for data
+ (opposed to the CPM based descriptor model).
+
++config SPI_MXC
++ tristate "MXC CSPI controller as SPI Master"
++ depends on ARCH_MXC && SPI_MASTER
++ select SPI_BITBANG
++ help
++ This implements the SPI master mode using MXC CSPI.
++
++config SPI_MXC_TEST_LOOPBACK
++ bool "LOOPBACK Testing of CSPIs"
++ depends on SPI_MXC
++ default n
++
++config SPI_MXC_SELECT1
++ bool "CSPI1"
++ depends on SPI_MXC
++ default y
++
++config SPI_MXC_SELECT2
++ bool "CSPI2"
++ depends on SPI_MXC
++ default n
++
++config SPI_MXC_SELECT3
++ bool "CSPI3"
++ depends on SPI_MXC && (ARCH_MX3 || ARCH_MX2)
++ default n
++
++choice
++ prompt "SPI controller hardware revision"
++ default SPI_MXC_REV0
++ depends on SPI_MXC
++ config SPI_MXC_REV0
++ bool "Revision 0"
++ help
++ hardware revision 0, mostly on MX27
++ config SPI_MXC_REV4
++ bool "Revision 4"
++ help
++ hardware revision 4, mostly on MX31
++ config SPI_MXC_REV5
++ bool "Revision 5"
++ help
++ hardware revision 5
++ config SPI_MXC_REV7
++ bool "Revision 7"
++ help
++ hardware revision 7
++endchoice
++
+ config SPI_OMAP_UWIRE
+ tristate "OMAP1 MicroWire"
+ depends on ARCH_OMAP1
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/spi/Makefile linux-2.6.28-karo/drivers/spi/Makefile
+--- linux-2.6.28/drivers/spi/Makefile 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/spi/Makefile 2009-03-11 13:16:24.000000000 +0100
+@@ -24,6 +24,7 @@ obj-$(CONFIG_SPI_OMAP24XX) += omap2_mcs
+ obj-$(CONFIG_SPI_ORION) += orion_spi.o
+ obj-$(CONFIG_SPI_MPC52xx_PSC) += mpc52xx_psc_spi.o
+ obj-$(CONFIG_SPI_MPC83xx) += spi_mpc83xx.o
++obj-$(CONFIG_SPI_MXC) += mxc_spi.o
+ obj-$(CONFIG_SPI_S3C24XX_GPIO) += spi_s3c24xx_gpio.o
+ obj-$(CONFIG_SPI_S3C24XX) += spi_s3c24xx.o
+ obj-$(CONFIG_SPI_TXX9) += spi_txx9.o
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/spi/mxc_spi.c linux-2.6.28-karo/drivers/spi/mxc_spi.c
+--- linux-2.6.28/drivers/spi/mxc_spi.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/spi/mxc_spi.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,748 @@
++/*
++ * Copyright 2004-2007 Freescale Semiconductor, Inc. All Rights Reserved.
++ * Copyright (C) 2008 Juergen Beisert
++ *
++ * 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
++ */
++
++#include <linux/completion.h>
++#include <linux/platform_device.h>
++#include <linux/kernel.h>
++#include <linux/init.h>
++#include <linux/interrupt.h>
++#include <linux/irq.h>
++#include <linux/module.h>
++#include <linux/delay.h>
++#include <linux/types.h>
++#include <linux/clk.h>
++
++#include <linux/spi/spi.h>
++#include <linux/spi/spi_bitbang.h>
++
++#include <asm/irq.h>
++#include <asm/io.h>
++#include <mach/gpio.h>
++#include <mach/imx_spi.h>
++
++#ifdef CONFIG_SPI_MXC_TEST_LOOPBACK
++struct spi_chip_info {
++ int lb_enable;
++};
++
++static struct spi_chip_info lb_chip_info = {
++ .lb_enable = 1,
++};
++
++static struct spi_board_info loopback_info[] = {
++#ifdef CONFIG_SPI_MXC_SELECT1
++ {
++ .modalias = "loopback_spi",
++ .controller_data = &lb_chip_info,
++ .irq = 0,
++ .max_speed_hz = 4000000,
++ .bus_num = 0,
++ .chip_select = 4,
++ },
++#endif
++#ifdef CONFIG_SPI_MXC_SELECT2
++ {
++ .modalias = "loopback_spi",
++ .controller_data = &lb_chip_info,
++ .irq = 0,
++ .max_speed_hz = 4000000,
++ .bus_num = 1,
++ .chip_select = 4,
++ },
++#endif
++#ifdef CONFIG_SPI_MXC_SELECT3
++ {
++ .modalias = "loopback_spi",
++ .controller_data = &lb_chip_info,
++ .irq = 0,
++ .max_speed_hz = 4000000,
++ .bus_num = 2,
++ .chip_select = 4,
++ },
++#endif
++};
++#endif
++
++static struct mxc_spi_unique_def spi_ver_0_7 = {
++ .intr_bit_shift = MXC_CSPIINT_IRQSHIFT_0_7,
++ .cs_shift = MXC_CSPICTRL_CSSHIFT_0_7,
++ .bc_shift = MXC_CSPICTRL_BCSHIFT_0_7,
++ .bc_mask = MXC_CSPICTRL_BCMASK_0_7,
++ .drctrl_shift = MXC_CSPICTRL_DRCTRLSHIFT_0_7,
++ .xfer_complete = MXC_CSPISTAT_TC_0_7,
++ .bc_overflow = MXC_CSPISTAT_BO_0_7,
++};
++
++static struct mxc_spi_unique_def spi_ver_0_5 = {
++ .intr_bit_shift = MXC_CSPIINT_IRQSHIFT_0_5,
++ .cs_shift = MXC_CSPICTRL_CSSHIFT_0_5,
++ .bc_shift = MXC_CSPICTRL_BCSHIFT_0_5,
++ .bc_mask = MXC_CSPICTRL_BCMASK_0_5,
++ .drctrl_shift = MXC_CSPICTRL_DRCTRLSHIFT_0_5,
++ .xfer_complete = MXC_CSPISTAT_TC_0_5,
++ .bc_overflow = MXC_CSPISTAT_BO_0_5,
++};
++
++static struct mxc_spi_unique_def spi_ver_0_4 = {
++ .intr_bit_shift = MXC_CSPIINT_IRQSHIFT_0_4,
++ .cs_shift = MXC_CSPICTRL_CSSHIFT_0_4,
++ .bc_shift = MXC_CSPICTRL_BCSHIFT_0_4,
++ .bc_mask = MXC_CSPICTRL_BCMASK_0_4,
++ .drctrl_shift = MXC_CSPICTRL_DRCTRLSHIFT_0_4,
++ .xfer_complete = MXC_CSPISTAT_TC_0_4,
++ .bc_overflow = MXC_CSPISTAT_BO_0_4,
++};
++
++static struct mxc_spi_unique_def spi_ver_0_0 = {
++ .intr_bit_shift = MXC_CSPIINT_IRQSHIFT_0_0,
++ .cs_shift = MXC_CSPICTRL_CSSHIFT_0_0,
++ .bc_shift = MXC_CSPICTRL_BCSHIFT_0_0,
++ .bc_mask = MXC_CSPICTRL_BCMASK_0_0,
++ .drctrl_shift = MXC_CSPICTRL_DRCTRLSHIFT_0_0,
++ .xfer_complete = MXC_CSPISTAT_TC_0_0,
++ .bc_overflow = MXC_CSPISTAT_BO_0_0,
++};
++
++struct mxc_spi;
++/*
++ * Structure to group together all the data buffers and functions
++ * used in data transfers.
++ */
++struct mxc_spi_xfer {
++ const void *tx_buf; /* Transmit buffer. */
++ void *rx_buf; /* Receive buffer. */
++ unsigned int count; /* Data transfered count. */
++ void (*rx_get) (struct mxc_spi *, u32 val); /* Function to read the FIFO data to rx_buf. */
++ u32(*tx_get) (struct mxc_spi *); /* Function to get the data to be written to FIFO. */
++};
++
++/*
++ * This structure is a way for the low level driver to define their own
++ * spi_master structure. This structure includes the core spi_master
++ * structure that is provided by Linux SPI Framework/driver as an
++ * element and has other elements that are specifically required by this
++ * low-level driver.
++ */
++struct mxc_spi {
++ struct spi_bitbang mxc_bitbang; /* SPI Master and a simple I/O queue runner. */
++ struct completion xfer_done; /* Completion flags used in data transfers. */
++ struct mxc_spi_xfer transfer; /* Data transfer structure. */
++ struct resource *res; /* Resource structure, which will maintain base addresses and IRQs. */
++ void *base; /* Base address of CSPI, used in readl and writel. */
++ int irq; /* CSPI IRQ number. */
++ struct clk *clk; /* CSPI Clock id. */
++ /*!
++ * CSPI input clock SCLK.
++ */
++ unsigned long spi_ipg_clk;
++ /*!
++ * CSPI registers' bit pattern.
++ */
++ struct mxc_spi_unique_def *spi_ver_def;
++};
++
++#define MXC_SPI_BUF_RX(type) \
++void mxc_spi_buf_rx_##type(struct mxc_spi *master_drv_data, u32 val)\
++{\
++ type *rx = master_drv_data->transfer.rx_buf;\
++ *rx++ = (type)val;\
++ master_drv_data->transfer.rx_buf = rx;\
++}
++
++#define MXC_SPI_BUF_TX(type) \
++u32 mxc_spi_buf_tx_##type(struct mxc_spi *master_drv_data)\
++{\
++ u32 val;\
++ const type *tx = master_drv_data->transfer.tx_buf;\
++ val = *tx++;\
++ master_drv_data->transfer.tx_buf = tx;\
++ return val;\
++}
++MXC_SPI_BUF_RX(u8)
++ MXC_SPI_BUF_TX(u8)
++ MXC_SPI_BUF_RX(u16)
++ MXC_SPI_BUF_TX(u16)
++ MXC_SPI_BUF_RX(u32)
++ MXC_SPI_BUF_TX(u32)
++
++static int spi_enable_interrupt(struct mxc_spi *master_data, unsigned int irqs)
++{
++ if (irqs & ~((1 << master_data->spi_ver_def->intr_bit_shift) - 1)) {
++ return -1;
++ }
++
++ writel((irqs | readl(master_data->base + MXC_CSPIINT)),
++ master_data->base + MXC_CSPIINT);
++
++ return 0;
++}
++
++static int spi_disable_interrupt(struct mxc_spi *master_data, unsigned int irqs)
++{
++ if (irqs & ~((1 << master_data->spi_ver_def->intr_bit_shift) - 1)) {
++ return -1;
++ }
++
++ writel((~irqs & readl(master_data->base + MXC_CSPIINT)),
++ master_data->base + MXC_CSPIINT);
++ return 0;
++}
++
++static unsigned int spi_find_baudrate(struct mxc_spi *master_data,
++ unsigned int baud)
++{
++ unsigned int divisor;
++ unsigned int shift = 0;
++
++ /* Calculate required divisor (rounded) */
++ divisor = (master_data->spi_ipg_clk + baud / 2) / baud;
++ while (divisor >>= 1)
++ shift++;
++ MXC_CSPICTRL_ADJUST_SHIFT(shift);
++ if (shift > MXC_CSPICTRL_MAXDATRATE)
++ shift = MXC_CSPICTRL_MAXDATRATE;
++
++ return shift << MXC_CSPICTRL_DATASHIFT;
++}
++
++static unsigned int spi_get_rx_data(void *base)
++{
++ return readl(base + MXC_CSPIRXDATA);
++}
++
++static void spi_put_tx_data(void *base, unsigned int val)
++{
++ unsigned int ctrl_reg;
++
++ writel(val, base + MXC_CSPITXDATA);
++
++ ctrl_reg = readl(base + MXC_CSPICTRL);
++
++ ctrl_reg |= MXC_CSPICTRL_XCH;
++
++ writel(ctrl_reg, base + MXC_CSPICTRL);
++
++ return;
++}
++
++void mxc_spi_chipselect(struct spi_device *spi, int is_active)
++{
++ struct mxc_spi *master_drv_data;
++ struct mxc_spi_xfer *ptransfer;
++ struct mxc_spi_unique_def *spi_ver_def;
++ unsigned int ctrl_reg;
++ unsigned int ctrl_mask;
++ unsigned int xfer_len;
++
++ if (is_active == BITBANG_CS_INACTIVE) {
++ /*Need to deselect the slave */
++ return;
++ }
++
++ /* Get the master controller driver data from spi device's master */
++
++ master_drv_data = spi_master_get_devdata(spi->master);
++ spi_ver_def = master_drv_data->spi_ver_def;
++
++ xfer_len = spi->bits_per_word;
++
++ /* Control Register Settings for transfer to this slave */
++
++ ctrl_reg = readl(master_drv_data->base + MXC_CSPICTRL);
++
++ ctrl_mask =
++ (MXC_CSPICTRL_LOWPOL | MXC_CSPICTRL_PHA | MXC_CSPICTRL_HIGHSSPOL |
++ MXC_CSPICTRL_CSMASK << spi_ver_def->cs_shift |
++ MXC_CSPICTRL_DATAMASK << MXC_CSPICTRL_DATASHIFT |
++ spi_ver_def->bc_mask << spi_ver_def->bc_shift);
++ ctrl_reg &= ~ctrl_mask;
++
++ ctrl_reg |=
++ ((spi->chip_select & MXC_CSPICTRL_CSMASK) << spi_ver_def->cs_shift);
++ ctrl_reg |= spi_find_baudrate(master_drv_data, spi->max_speed_hz);
++ ctrl_reg |=
++ (((xfer_len - 1) & spi_ver_def->bc_mask) << spi_ver_def->bc_shift);
++ if (spi->mode & SPI_CPHA)
++ ctrl_reg |= MXC_CSPICTRL_PHA;
++ if (!(spi->mode & SPI_CPOL))
++ ctrl_reg |= MXC_CSPICTRL_LOWPOL;
++ if (spi->mode & SPI_CS_HIGH)
++ ctrl_reg |= MXC_CSPICTRL_HIGHSSPOL;
++
++ writel(ctrl_reg, master_drv_data->base + MXC_CSPICTRL);
++
++ /* Initialize the functions for transfer */
++ ptransfer = &master_drv_data->transfer;
++ if (xfer_len <= 8) {
++ ptransfer->rx_get = mxc_spi_buf_rx_u8;
++ ptransfer->tx_get = mxc_spi_buf_tx_u8;
++ } else if (xfer_len <= 16) {
++ ptransfer->rx_get = mxc_spi_buf_rx_u16;
++ ptransfer->tx_get = mxc_spi_buf_tx_u16;
++ } else if (xfer_len <= 32) {
++ ptransfer->rx_get = mxc_spi_buf_rx_u32;
++ ptransfer->tx_get = mxc_spi_buf_tx_u32;
++ }
++#ifdef CONFIG_SPI_MXC_TEST_LOOPBACK
++ {
++ struct spi_chip_info *lb_chip =
++ spi->controller_data;
++ if (!lb_chip)
++ writel(0, master_drv_data->base + MXC_CSPITEST);
++ else if (lb_chip->lb_enable)
++ writel(MXC_CSPITEST_LBC,
++ master_drv_data->base + MXC_CSPITEST);
++ }
++#endif
++ return;
++}
++
++static irqreturn_t mxc_spi_isr(int irq, void *dev_id)
++{
++ struct mxc_spi *master_drv_data = dev_id;
++ irqreturn_t ret = IRQ_NONE;
++ unsigned int status;
++
++ /* Read the interrupt status register to determine the source */
++ status = readl(master_drv_data->base + MXC_CSPISTAT);
++
++ /* Rx is given higher priority - Handle it first */
++ if (status & MXC_CSPISTAT_RR) {
++ u32 rx_tmp = spi_get_rx_data(master_drv_data->base);
++
++ if (master_drv_data->transfer.rx_buf)
++ master_drv_data->transfer.rx_get(master_drv_data,
++ rx_tmp);
++
++ ret = IRQ_HANDLED;
++ }
++
++ master_drv_data->transfer.count--;
++ /* Handle the Tx now */
++ if (master_drv_data->transfer.count) {
++ if (master_drv_data->transfer.tx_buf) {
++ u32 tx_tmp =
++ master_drv_data->transfer.tx_get(master_drv_data);
++
++ spi_put_tx_data(master_drv_data->base, tx_tmp);
++ }
++ } else {
++ complete(&master_drv_data->xfer_done);
++ }
++
++ return ret;
++}
++
++int mxc_spi_setup(struct spi_device *spi)
++{
++ struct mxc_spi *master_data = spi_master_get_devdata(spi->master);
++
++ if ((spi->max_speed_hz < 0)
++ || (spi->max_speed_hz > (master_data->spi_ipg_clk / 4))) {
++ dev_err(&spi->dev, "Cannot set required SPI clock speed."
++ " Requested: %uHz, possible: %luHz\n",
++ spi->max_speed_hz, master_data->spi_ipg_clk / 4);
++ return -EINVAL;
++ }
++
++ if (!spi->bits_per_word)
++ spi->bits_per_word = 8;
++
++ pr_debug("%s: mode %d, %u bpw, %d hz\n", __FUNCTION__,
++ spi->mode, spi->bits_per_word, spi->max_speed_hz);
++
++ return 0;
++}
++
++int mxc_spi_transfer(struct spi_device *spi, struct spi_transfer *t)
++{
++ struct mxc_spi *master_drv_data = NULL;
++
++ /* Get the master controller driver data from spi device's master */
++
++ master_drv_data = spi_master_get_devdata(spi->master);
++
++ /* Modify the Tx, Rx, Count */
++
++ master_drv_data->transfer.tx_buf = t->tx_buf;
++ master_drv_data->transfer.rx_buf = t->rx_buf;
++ master_drv_data->transfer.count = t->len;
++ INIT_COMPLETION(master_drv_data->xfer_done);
++
++ /* Enable the Rx Interrupts */
++
++ spi_enable_interrupt(master_drv_data, MXC_CSPIINT_RREN);
++
++ /* Perform single Tx transaction */
++
++ spi_put_tx_data(master_drv_data->base,
++ master_drv_data->transfer.tx_get(master_drv_data));
++
++ /* Wait for transfer completion */
++
++ wait_for_completion(&master_drv_data->xfer_done);
++
++ /* Disable the Rx Interrupts */
++
++ spi_disable_interrupt(master_drv_data, MXC_CSPIINT_RREN);
++
++ return t->len - master_drv_data->transfer.count;
++}
++
++void mxc_spi_cleanup(struct spi_device *spi)
++{
++ return;
++}
++
++static int mxc_spi_probe(struct platform_device *pdev)
++{
++ struct mxc_spi_master *mxc_platform_info;
++ struct spi_master *master;
++ struct mxc_spi *master_drv_data;
++ unsigned int spi_ver;
++ int ret = -ENODEV;
++
++ dev_dbg(&pdev->dev, "Enter: %s\n", __FUNCTION__);
++
++ /* Get the platform specific data for this master device */
++
++ mxc_platform_info = pdev->dev.platform_data;
++ if (!mxc_platform_info) {
++ dev_err(&pdev->dev, "can't get the platform data for CSPI\n");
++ return -ENODEV;
++ }
++
++ /* Allocate SPI master controller */
++
++ master = spi_alloc_master(&pdev->dev, sizeof(struct mxc_spi));
++ if (!master) {
++ dev_err(&pdev->dev, "can't alloc for spi_master\n");
++ return -ENOMEM;
++ }
++
++ /* Set this device's driver data to master */
++
++ platform_set_drvdata(pdev, master);
++
++ /* Set this master's data from platform_info */
++
++ master->bus_num = pdev->id;
++ master->num_chipselect = mxc_platform_info->maxchipselect;
++
++ /* Set the master controller driver data for this master */
++
++ master_drv_data = spi_master_get_devdata(master);
++ if (master_drv_data == NULL)
++ goto err;
++ master_drv_data->mxc_bitbang.master = spi_master_get(master);
++
++ /* Set the master bitbang data */
++
++ master_drv_data->mxc_bitbang.chipselect = mxc_spi_chipselect;
++ master_drv_data->mxc_bitbang.txrx_bufs = mxc_spi_transfer;
++ master_drv_data->mxc_bitbang.master->setup = mxc_spi_setup;
++ master_drv_data->mxc_bitbang.master->cleanup = mxc_spi_cleanup;
++
++ /* Initialize the completion object */
++
++ init_completion(&master_drv_data->xfer_done);
++
++ /* Set the master controller register addresses and irqs */
++
++ master_drv_data->res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
++ if (!master_drv_data->res) {
++ dev_err(&pdev->dev, "can't get platform resource for CSPI%d\n",
++ master->bus_num);
++ ret = -ENOMEM;
++ goto err;
++ }
++
++ if (!request_mem_region(master_drv_data->res->start,
++ master_drv_data->res->end -
++ master_drv_data->res->start + 1, pdev->name)) {
++ dev_err(&pdev->dev, "request_mem_region failed for CSPI%d\n",
++ master->bus_num);
++ ret = -EBUSY;
++ goto err;
++ }
++
++ master_drv_data->base = ioremap(master_drv_data->res->start,
++ master_drv_data->res->end -
++ master_drv_data->res->start + 1);
++
++ if (!master_drv_data->base) {
++ dev_err(&pdev->dev, "Cannot map iomem for CSPI%d\n",
++ master->bus_num);
++ ret = -EINVAL;
++ goto err1;
++ }
++
++ ret = platform_get_irq(pdev, 0);
++ if (ret < 0) {
++ dev_err(&pdev->dev, "can't get IRQ for CSPI%d\n",
++ master->bus_num);
++ goto err2;
++ }
++ master_drv_data->irq = ret;
++
++ /* Register for SPI Interrupt */
++
++ ret = request_irq(master_drv_data->irq, mxc_spi_isr,
++ 0, "CSPI_IRQ", master_drv_data);
++ if (ret != 0) {
++ dev_err(&pdev->dev, "request_irq failed for CSPI%d\n",
++ master->bus_num);
++ goto err2;
++ }
++
++ /* Setup any GPIO active */
++ if (mxc_platform_info->init)
++ mxc_platform_info->init(pdev);
++
++ /* Identify SPI version */
++
++ spi_ver = mxc_platform_info->spi_version;
++ if (spi_ver == 7) {
++ master_drv_data->spi_ver_def = &spi_ver_0_7;
++ } else if (spi_ver == 5) {
++ master_drv_data->spi_ver_def = &spi_ver_0_5;
++ } else if (spi_ver == 4) {
++ master_drv_data->spi_ver_def = &spi_ver_0_4;
++ } else if (spi_ver == 0) {
++ master_drv_data->spi_ver_def = &spi_ver_0_0;
++ }
++
++ /* Enable the CSPI Clock, CSPI Module, set as a master */
++
++ master_drv_data->clk = clk_get(&pdev->dev, "cspi_clk");
++ clk_enable(master_drv_data->clk);
++ master_drv_data->spi_ipg_clk = clk_get_rate(master_drv_data->clk);
++
++ writel(0x00008000, master_drv_data->base + MXC_CSPICTRL); /* reset default */
++ writel(MXC_CSPICTRL_ENABLE | MXC_CSPICTRL_MASTER,
++ master_drv_data->base + MXC_CSPICTRL);
++ writel(MXC_CSPIPERIOD_32KHZ,
++ master_drv_data->base + MXC_CSPIPERIOD);
++ writel(0, master_drv_data->base + MXC_CSPIINT);
++
++ /* Start the SPI Master Controller driver */
++
++ ret = spi_bitbang_start(&master_drv_data->mxc_bitbang);
++ if (ret != 0) {
++ dev_err(&pdev->dev, "Cannot start bitbang driver\n");
++ goto err3;
++ }
++
++ printk(KERN_INFO "CSPI: %s-%d probed\n", pdev->name, pdev->id);
++
++#ifdef CONFIG_SPI_MXC_TEST_LOOPBACK
++ {
++ int i;
++ struct spi_board_info *bi = &loopback_info[0];
++ for (i = 0; i < ARRAY_SIZE(loopback_info); i++, bi++) {
++ if (bi->bus_num != master->bus_num)
++ continue;
++
++ dev_info(&pdev->dev,
++ "registering loopback device '%s'\n",
++ bi->modalias);
++
++ spi_new_device(master, bi);
++ }
++ }
++#endif
++ return ret;
++
++err3:
++ clk_disable(master_drv_data->clk);
++ clk_put(master_drv_data->clk);
++ if (mxc_platform_info->exit)
++ mxc_platform_info->exit(pdev);
++ free_irq(master_drv_data->irq, master_drv_data);
++err2:
++ iounmap(master_drv_data->base);
++err1:
++ release_mem_region(pdev->resource[0].start,
++ pdev->resource[0].end - pdev->resource[0].start + 1);
++err:
++ spi_master_put(master);
++ kfree(master);
++ platform_set_drvdata(pdev, NULL);
++ return ret;
++}
++
++static int mxc_spi_remove(struct platform_device *pdev)
++{
++ struct spi_master *master = platform_get_drvdata(pdev);
++ struct mxc_spi_master *mxc_platform_info =
++ pdev->dev.platform_data;
++
++ if (master) {
++ struct mxc_spi *master_drv_data =
++ spi_master_get_devdata(master);
++
++ if (mxc_platform_info->exit)
++ mxc_platform_info->exit(pdev);
++ clk_disable(master_drv_data->clk);
++ /* clk_put(master_drv_data->clk); */ /* FIXME required? */
++
++ /* Disable the CSPI module */
++
++ writel(MXC_CSPICTRL_DISABLE,
++ master_drv_data->base + MXC_CSPICTRL);
++
++ spi_bitbang_stop(&master_drv_data->mxc_bitbang);
++
++ /* free resources */
++ spi_master_put(master);
++ free_irq(master_drv_data->irq, master_drv_data);
++ iounmap(master_drv_data->base);
++ release_mem_region(master_drv_data->res->start,
++ master_drv_data->res->end -
++ master_drv_data->res->start + 1);
++ }
++
++ printk(KERN_INFO "CSPI: %s-%d removed\n", pdev->name, pdev->id);
++ platform_set_drvdata(pdev, NULL);
++
++ return 0;
++}
++
++#ifdef CONFIG_PM
++static int suspend_devices(struct device *dev, void *pm_message)
++{
++ pm_message_t *state = pm_message;
++
++ if (dev->power.power_state.event != state->event) {
++ dev_warn(dev, "mismatch in pm state request\n");
++ return -1;
++ }
++
++ return 0;
++}
++
++static int spi_bitbang_suspend(struct spi_bitbang *bitbang)
++{
++ unsigned long flags;
++ unsigned limit = 500;
++
++ spin_lock_irqsave(&bitbang->lock, flags);
++ //bitbang->shutdown = 0;
++ while (!list_empty(&bitbang->queue) && limit--) {
++ spin_unlock_irqrestore(&bitbang->lock, flags);
++
++ dev_dbg(&bitbang->master->dev, "wait for queue\n");
++ msleep(10);
++
++ spin_lock_irqsave(&bitbang->lock, flags);
++ }
++ if (!list_empty(&bitbang->queue)) {
++ dev_err(&bitbang->master->dev, "queue didn't empty\n");
++ return -EBUSY;
++ }
++ spin_unlock_irqrestore(&bitbang->lock, flags);
++
++ //bitbang->shutdown = 1;
++
++ return 0;
++}
++
++static void spi_bitbang_resume(struct spi_bitbang *bitbang)
++{
++ spin_lock_init(&bitbang->lock);
++ INIT_LIST_HEAD(&bitbang->queue);
++
++ bitbang->busy = 0;
++ //bitbang->shutdown = 0;
++}
++
++static int mxc_spi_suspend(struct platform_device *pdev, pm_message_t state)
++{
++ struct spi_master *master = platform_get_drvdata(pdev);
++ struct mxc_spi *master_drv_data = spi_master_get_devdata(master);
++ struct mxc_spi_master *mxc_platform_info =
++ pdev->dev.platform_data;
++ int ret = 0;
++
++ if (device_for_each_child(&pdev->dev, &state, suspend_devices) != 0) {
++ dev_warn(&pdev->dev, "suspend aborted\n");
++ return -EINVAL;
++ }
++
++ spi_bitbang_suspend(&master_drv_data->mxc_bitbang);
++ writel(MXC_CSPICTRL_DISABLE,
++ master_drv_data->base + MXC_CSPICTRL);
++
++ clk_disable(master_drv_data->clk);
++ if (mxc_platform_info->exit)
++ mxc_platform_info->exit(pdev);
++
++ return ret;
++}
++
++static int mxc_spi_resume(struct platform_device *pdev)
++{
++ struct spi_master *master = platform_get_drvdata(pdev);
++ struct mxc_spi *master_drv_data = spi_master_get_devdata(master);
++ struct mxc_spi_master *mxc_platform_info =
++ pdev->dev.platform_data;
++
++ if (mxc_platform_info->init)
++ mxc_platform_info->init(pdev);
++ clk_enable(master_drv_data->clk);
++
++ spi_bitbang_resume(&master_drv_data->mxc_bitbang);
++ writel(MXC_CSPICTRL_ENABLE | MXC_CSPICTRL_MASTER,
++ master_drv_data->base + MXC_CSPICTRL);
++
++ return 0;
++}
++#else
++# define mxc_spi_suspend NULL
++# define mxc_spi_resume NULL
++#endif /* CONFIG_PM */
++
++static struct platform_driver mxc_spi_driver = {
++ .driver = {
++ .name = "mxc_spi",
++ .owner = THIS_MODULE,
++ },
++ .probe = mxc_spi_probe,
++ .remove = mxc_spi_remove,
++ .suspend = mxc_spi_suspend,
++ .resume = mxc_spi_resume,
++};
++
++static int __init mxc_spi_init(void)
++{
++ pr_debug("Registering the SPI Controller Driver\n");
++ return platform_driver_register(&mxc_spi_driver);
++}
++
++static void __exit mxc_spi_exit(void)
++{
++ pr_debug("Unregistering the SPI Controller Driver\n");
++ platform_driver_unregister(&mxc_spi_driver);
++}
++
++subsys_initcall(mxc_spi_init);
++module_exit(mxc_spi_exit);
++
++MODULE_DESCRIPTION("SPI Master Controller driver");
++MODULE_AUTHOR("Freescale Semiconductor, Inc.");
++MODULE_LICENSE("GPL");
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/usb/Kconfig linux-2.6.28-karo/drivers/usb/Kconfig
+--- linux-2.6.28/drivers/usb/Kconfig 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/usb/Kconfig 2009-03-11 13:16:24.000000000 +0100
+@@ -56,6 +56,7 @@ config USB_ARCH_HAS_EHCI
+ default y if PPC_83xx
+ default y if SOC_AU1200
+ default y if ARCH_IXP4XX
++ default y if ARCH_MXC
+ default PCI
+
+ # ARM SA1111 chips have a non-PCI based "OHCI-compatible" USB host interface.
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/usb/host/Kconfig linux-2.6.28-karo/drivers/usb/host/Kconfig
+--- linux-2.6.28/drivers/usb/host/Kconfig 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/usb/host/Kconfig 2009-03-11 13:16:24.000000000 +0100
+@@ -96,6 +96,13 @@ config USB_EHCI_HCD_PPC_OF
+ Enables support for the USB controller present on the PowerPC
+ OpenFirmware platform bus.
+
++config USB_EHCI_MXC
++ bool "Support for Freescale on-chip EHCI USB controller"
++ depends on USB_EHCI_HCD && ARCH_MXC
++ select USB_EHCI_ROOT_HUB_TT
++ ---help---
++ Variation of ARC USB block used in some Freescale chips.
++
+ config USB_ISP116X_HCD
+ tristate "ISP116X HCD support"
+ depends on USB
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/usb/host/ehci-hcd.c linux-2.6.28-karo/drivers/usb/host/ehci-hcd.c
+--- linux-2.6.28/drivers/usb/host/ehci-hcd.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/usb/host/ehci-hcd.c 2009-03-11 13:16:24.000000000 +0100
+@@ -1009,6 +1009,11 @@ MODULE_LICENSE ("GPL");
+ #define PLATFORM_DRIVER ehci_fsl_driver
+ #endif
+
++#ifdef CONFIG_USB_EHCI_MXC
++#include "ehci-mxc.c"
++#define PLATFORM_DRIVER ehci_mxc_driver
++#endif
++
+ #ifdef CONFIG_SOC_AU1200
+ #include "ehci-au1xxx.c"
+ #define PLATFORM_DRIVER ehci_hcd_au1xxx_driver
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/usb/host/ehci-mxc.c linux-2.6.28-karo/drivers/usb/host/ehci-mxc.c
+--- linux-2.6.28/drivers/usb/host/ehci-mxc.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/usb/host/ehci-mxc.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,241 @@
++/*
++ * Copyright (c) 2008 Sascha Hauer <s.hauer@pengutronix.de>, Pengutronix
++ *
++ * 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., 675 Mass Ave, Cambridge, MA 02139, USA.
++ *
++ */
++
++#include <linux/platform_device.h>
++#include <linux/clk.h>
++#include <mach/mxc_ehci.h>
++
++/* called during probe() after chip reset completes */
++static int ehci_mxc_setup(struct usb_hcd *hcd)
++{
++ struct ehci_hcd *ehci = hcd_to_ehci(hcd);
++ int retval;
++
++ /* EHCI registers start at offset 0x100 */
++ ehci->caps = hcd->regs + 0x100;
++ ehci->regs = hcd->regs + 0x100 +
++ HC_LENGTH(ehci_readl(ehci, &ehci->caps->hc_capbase));
++ dbg_hcs_params(ehci, "reset");
++ dbg_hcc_params(ehci, "reset");
++
++ /* cache this readonly data; minimize chip reads */
++ ehci->hcs_params = ehci_readl(ehci, &ehci->caps->hcs_params);
++
++ retval = ehci_halt(ehci);
++ if (retval)
++ return retval;
++
++ /* data structure init */
++ retval = ehci_init(hcd);
++ if (retval)
++ return retval;
++
++ hcd->has_tt = 1;
++
++ ehci->sbrn = 0x20;
++
++ ehci_reset(ehci);
++
++ ehci_port_power(ehci, 0);
++ return 0;
++}
++
++static const struct hc_driver ehci_mxc_hc_driver = {
++ .description = hcd_name,
++ .product_desc = "Freescale On-Chip EHCI Host Controller",
++ .hcd_priv_size = sizeof(struct ehci_hcd),
++
++ /*
++ * generic hardware linkage
++ */
++ .irq = ehci_irq,
++ .flags = HCD_USB2 | HCD_MEMORY,
++
++ /*
++ * basic lifecycle operations
++ */
++ .reset = ehci_mxc_setup,
++ .start = ehci_run,
++ .stop = ehci_stop,
++ .shutdown = ehci_shutdown,
++
++ /*
++ * managing i/o requests and associated device resources
++ */
++ .urb_enqueue = ehci_urb_enqueue,
++ .urb_dequeue = ehci_urb_dequeue,
++ .endpoint_disable = ehci_endpoint_disable,
++
++ /*
++ * scheduling support
++ */
++ .get_frame_number = ehci_get_frame,
++
++ /*
++ * root hub support
++ */
++ .hub_status_data = ehci_hub_status_data,
++ .hub_control = ehci_hub_control,
++ .bus_suspend = ehci_bus_suspend,
++ .bus_resume = ehci_bus_resume,
++ .relinquish_port = ehci_relinquish_port,
++ .port_handed_over = ehci_port_handed_over,
++};
++
++static int ehci_mxc_drv_probe(struct platform_device *pdev)
++{
++ struct mxc_usbh_platform_data *pdata = pdev->dev.platform_data;
++ struct usb_hcd *hcd;
++ struct resource *res;
++ int irq, ret, temp;
++ struct clk *usbclk, *ahbclk;
++
++ dev_info(&pdev->dev, "initializing i.MX USB Controller\n");
++
++ /* Need platform data for setup */
++ if (!pdata) {
++ dev_err(&pdev->dev,
++ "No platform data for %s.\n", pdev->dev.bus_id);
++ return -ENODEV;
++ }
++
++ irq = platform_get_irq(pdev, 0);
++
++ hcd = usb_create_hcd(&ehci_mxc_hc_driver, &pdev->dev, pdev->dev.bus_id);
++ if (!hcd) {
++ ret = -ENOMEM;
++ goto err1;
++ }
++
++ res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
++ if (!res) {
++ dev_err(&pdev->dev,
++ "Found HC with no register addr. Check %s setup!\n",
++ pdev->dev.bus_id);
++ ret = -ENODEV;
++ goto err1;
++ }
++
++ hcd->rsrc_start = res->start;
++ hcd->rsrc_len = resource_size(res);
++
++ if (!request_mem_region(hcd->rsrc_start, hcd->rsrc_len, hcd_name)) {
++ dev_dbg(&pdev->dev, "controller already in use\n");
++ ret = -EBUSY;
++ goto err1;
++ }
++
++ hcd->regs = ioremap(hcd->rsrc_start, hcd->rsrc_len);
++ if (!hcd->regs) {
++ dev_err(&pdev->dev, "error mapping memory\n");
++ ret = -EFAULT;
++ goto err2;
++ }
++
++ ahbclk = clk_get(NULL, "usb_ahb_clk");
++ if (IS_ERR(ahbclk)) {
++ ret = PTR_ERR(ahbclk);
++ printk(KERN_ERR "Failed to get usb_ahb_clk: %d\n", ret);
++ goto err3;
++ }
++ clk_enable(ahbclk);
++
++ usbclk = clk_get(NULL, "usb_clk");
++ if (IS_ERR(usbclk)) {
++ ret = PTR_ERR(usbclk);
++ printk(KERN_ERR "Failed to get usb_clk: %d\n", ret);
++ goto err4;
++ }
++ clk_enable(usbclk);
++
++ if (pdata->init) {
++ ret = pdata->init(pdev);
++ if (ret) {
++ dev_err(&pdev->dev, "platform init failed\n");
++ goto err5;
++ }
++ }
++
++ /* Set to Host mode */
++ temp = readl(hcd->regs + 0x1a8);
++ writel(temp | 0x3, hcd->regs + 0x1a8);
++
++ ret = usb_add_hcd(hcd, irq, IRQF_DISABLED | IRQF_SHARED);
++ if (ret)
++ goto err6;
++
++ platform_set_drvdata(pdev, hcd);
++
++ return 0;
++err6:
++ if (pdata->exit)
++ pdata->exit(pdev);
++err5:
++ clk_disable(usbclk);
++ clk_put(usbclk);
++err4:
++ clk_disable(ahbclk);
++ clk_put(ahbclk);
++err3:
++ iounmap(hcd->regs);
++err2:
++ release_mem_region(hcd->rsrc_start, hcd->rsrc_len);
++err1:
++ usb_put_hcd(hcd);
++ return ret;
++}
++
++static int ehci_mxc_drv_remove(struct platform_device *pdev)
++{
++ struct usb_hcd *hcd = platform_get_drvdata(pdev);
++ struct mxc_usbh_platform_data *pdata = pdev->dev.platform_data;
++ struct clk *usbclk, *ahbclk;
++
++ usb_remove_hcd(hcd);
++ iounmap(hcd->regs);
++ release_mem_region(hcd->rsrc_start, hcd->rsrc_len);
++ usb_put_hcd(hcd);
++ platform_set_drvdata(pdev, NULL);
++
++ if (pdata->exit)
++ pdata->exit(pdev);
++
++ usbclk = clk_get(NULL, "usb_clk");
++ if (!IS_ERR(usbclk)) {
++ clk_disable(usbclk);
++ clk_put(usbclk);
++ }
++ ahbclk = clk_get(NULL, "usb_ahb_clk");
++ if (!IS_ERR(ahbclk)) {
++ clk_disable(ahbclk);
++ clk_put(ahbclk);
++ }
++ return 0;
++}
++
++MODULE_ALIAS("platform:mxc-ehci");
++
++static struct platform_driver ehci_mxc_driver = {
++ .probe = ehci_mxc_drv_probe,
++ .remove = ehci_mxc_drv_remove,
++ .shutdown = usb_hcd_platform_shutdown,
++ .driver = {
++ .name = "mxc-ehci",
++ },
++};
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/video/Kconfig linux-2.6.28-karo/drivers/video/Kconfig
+--- linux-2.6.28/drivers/video/Kconfig 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/video/Kconfig 2009-03-11 13:16:24.000000000 +0100
+@@ -397,7 +397,7 @@ config FB_SA1100
+
+ config FB_IMX
+ tristate "Motorola i.MX LCD support"
+- depends on FB && ARM && ARCH_IMX
++ depends on FB && ARM && (ARCH_IMX || ARCH_MX2)
+ select FB_CFB_FILLRECT
+ select FB_CFB_COPYAREA
+ select FB_CFB_IMAGEBLIT
+@@ -2115,6 +2115,18 @@ config FB_PRE_INIT_FB
+ Select this option if display contents should be inherited as set by
+ the bootloader.
+
++config FB_MXC
++ tristate "MXC Framebuffer support"
++ depends on FB && MXC_IPU
++ select FB_CFB_FILLRECT
++ select FB_CFB_COPYAREA
++ select FB_CFB_IMAGEBLIT
++ default y
++ help
++ This is a framebuffer device for the i.MX31 LCD Controller. So
++ far only synchronous displays are supported. If you plan to use
++ an LCD display with your i.MX31 system, say Y here.
++
+ source "drivers/video/omap/Kconfig"
+
+ source "drivers/video/backlight/Kconfig"
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/video/Makefile linux-2.6.28-karo/drivers/video/Makefile
+--- linux-2.6.28/drivers/video/Makefile 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/video/Makefile 2009-03-11 13:16:24.000000000 +0100
+@@ -132,6 +132,7 @@ obj-$(CONFIG_FB_VGA16) += vga
+ obj-$(CONFIG_FB_OF) += offb.o
+ obj-$(CONFIG_FB_BF54X_LQ043) += bf54x-lq043fb.o
+ obj-$(CONFIG_FB_BFIN_T350MCQB) += bfin-t350mcqb-fb.o
++obj-$(CONFIG_FB_MXC) += mxcfb.o
+
+ # the test framebuffer is last
+ obj-$(CONFIG_FB_VIRTUAL) += vfb.o
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/video/imxfb.c linux-2.6.28-karo/drivers/video/imxfb.c
+--- linux-2.6.28/drivers/video/imxfb.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/video/imxfb.c 2009-03-11 13:16:24.000000000 +0100
+@@ -1,6 +1,4 @@
+ /*
+- * linux/drivers/video/imxfb.c
+- *
+ * Freescale i.MX Frame Buffer device driver
+ *
+ * Copyright (C) 2004 Sascha Hauer, Pengutronix
+@@ -16,8 +14,6 @@
+ * linux-arm-kernel@lists.arm.linux.org.uk
+ */
+
+-//#define DEBUG 1
+-
+ #include <linux/module.h>
+ #include <linux/kernel.h>
+ #include <linux/errno.h>
+@@ -30,11 +26,11 @@
+ #include <linux/init.h>
+ #include <linux/ioport.h>
+ #include <linux/cpufreq.h>
++#include <linux/clk.h>
+ #include <linux/platform_device.h>
+ #include <linux/dma-mapping.h>
++#include <linux/io.h>
+
+-#include <mach/hardware.h>
+-#include <asm/io.h>
+ #include <mach/imxfb.h>
+
+ /*
+@@ -42,23 +38,170 @@
+ */
+ #define DEBUG_VAR 1
+
+-#include "imxfb.h"
++#define DRIVER_NAME "imx-fb"
++
++#define LCDC_SSA 0x00
++
++#define LCDC_SIZE 0x04
++#define SIZE_XMAX(x) ((((x) >> 4) & 0x3f) << 20)
++#ifdef CONFIG_ARCH_IMX
++#define SIZE_YMAX(y) ((y) & 0x1ff)
++#else
++#define SIZE_YMAX(y) ((y) & 0x3ff)
++#endif
++
++#define LCDC_VPW 0x08
++#define VPW_VPW(x) ((x) & 0x3ff)
++
++#define LCDC_CPOS 0x0C
++#define CPOS_CC1 (1<<31)
++#define CPOS_CC0 (1<<30)
++#define CPOS_OP (1<<28)
++#define CPOS_CXP(x) (((x) & 3ff) << 16)
++#ifdef CONFIG_ARCH_IMX
++#define CPOS_CYP(y) ((y) & 0x1ff)
++#else
++#define CPOS_CYP(y) ((y) & 0x3ff)
++#endif
++
++#define LCDC_LCWHB 0x10
++#define LCWHB_BK_EN (1<<31)
++#define LCWHB_CW(w) (((w) & 0x1f) << 24)
++#define LCWHB_CH(h) (((h) & 0x1f) << 16)
++#define LCWHB_BD(x) ((x) & 0xff)
++
++#define LCDC_LCHCC 0x14
++#ifdef CONFIG_ARCH_IMX
++#define LCHCC_CUR_COL_R(r) (((r) & 0x1f) << 11)
++#define LCHCC_CUR_COL_G(g) (((g) & 0x3f) << 5)
++#define LCHCC_CUR_COL_B(b) ((b) & 0x1f)
++#else
++#define LCHCC_CUR_COL_R(r) (((r) & 0x3f) << 12)
++#define LCHCC_CUR_COL_G(g) (((g) & 0x3f) << 6)
++#define LCHCC_CUR_COL_B(b) ((b) & 0x3f)
++#endif
++
++#define LCDC_PCR 0x18
++
++#define LCDC_HCR 0x1C
++#define HCR_H_WIDTH(x) (((x) & 0x3f) << 26)
++#define HCR_H_WAIT_1(x) (((x) & 0xff) << 8)
++#define HCR_H_WAIT_2(x) ((x) & 0xff)
++
++#define LCDC_VCR 0x20
++#define VCR_V_WIDTH(x) (((x) & 0x3f) << 26)
++#define VCR_V_WAIT_1(x) (((x) & 0xff) << 8)
++#define VCR_V_WAIT_2(x) ((x) & 0xff)
++
++#define LCDC_POS 0x24
++#define POS_POS(x) ((x) & 1f)
++
++#define LCDC_LSCR1 0x28
++/* bit fields in imxfb.h */
++
++#define LCDC_PWMR 0x2C
++/* bit fields in imxfb.h */
++
++#define LCDC_DMACR 0x30
++/* bit fields in imxfb.h */
++
++#define LCDC_RMCR 0x34
++#define RMCR_LCDC_EN (1<<1)
++#define RMCR_SELF_REF (1<<0)
++
++#define LCDC_LCDICR 0x38
++#define LCDICR_INT_SYN (1<<2)
++#define LCDICR_INT_CON (1)
++
++#define LCDC_LCDISR 0x40
++#define LCDISR_UDR_ERR (1<<3)
++#define LCDISR_ERR_RES (1<<2)
++#define LCDISR_EOF (1<<1)
++#define LCDISR_BOF (1<<0)
+
+-static struct imxfb_rgb def_rgb_16 = {
+- .red = { .offset = 8, .length = 4, },
+- .green = { .offset = 4, .length = 4, },
+- .blue = { .offset = 0, .length = 4, },
+- .transp = { .offset = 0, .length = 0, },
++/*
++ * These are the bitfields for each
++ * display depth that we support.
++ */
++struct imxfb_rgb {
++ struct fb_bitfield red;
++ struct fb_bitfield green;
++ struct fb_bitfield blue;
++ struct fb_bitfield transp;
+ };
+
+-static struct imxfb_rgb def_rgb_8 = {
+- .red = { .offset = 0, .length = 8, },
+- .green = { .offset = 0, .length = 8, },
+- .blue = { .offset = 0, .length = 8, },
+- .transp = { .offset = 0, .length = 0, },
++struct imxfb_info {
++ struct platform_device *pdev;
++ void __iomem *regs;
++ struct clk *clk;
++ char enabled;
++
++ u_int max_bpp;
++ u_int max_xres;
++ u_int max_yres;
++
++ /*
++ * These are the addresses we mapped
++ * the framebuffer memory region to.
++ */
++ dma_addr_t map_dma;
++ u_char *map_cpu;
++ u_int map_size;
++
++ u_char *screen_cpu;
++ dma_addr_t screen_dma;
++ u_int palette_size;
++
++ dma_addr_t dbar1;
++ dma_addr_t dbar2;
++
++ u_int pcr;
++ u_int pwmr;
++ u_int lscr1;
++ u_int dmacr;
++ u_int cmap_inverse:1,
++ cmap_static:1,
++ unused:30;
++
++ void (*lcd_power)(int);
++ void (*backlight_power)(int);
+ };
+
+-static int imxfb_activate_var(struct fb_var_screeninfo *var, struct fb_info *info);
++#define IMX_NAME "IMX"
++
++/*
++ * Minimum X and Y resolutions
++ */
++#define MIN_XRES 64
++#define MIN_YRES 64
++
++static struct imxfb_rgb def_rgb_18 = {
++ .red = {.offset = 18, .length = 6,},
++ .green = {.offset = 10, .length = 6,},
++ .blue = {.offset = 2, .length = 6,},
++ .transp = {.offset = 0, .length = 0,},
++};
++
++static struct imxfb_rgb def_rgb_16_tft = {
++ .red = {.offset = 11, .length = 5,},
++ .green = {.offset = 5, .length = 6,},
++ .blue = {.offset = 0, .length = 5,},
++ .transp = {.offset = 0, .length = 0,},
++};
++
++static struct imxfb_rgb def_rgb_16_stn = {
++ .red = {.offset = 8, .length = 4,},
++ .green = {.offset = 4, .length = 4,},
++ .blue = {.offset = 0, .length = 4,},
++ .transp = {.offset = 0, .length = 0,},
++};
++
++static struct imxfb_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 inline u_int chan_to_field(u_int chan, struct fb_bitfield *bf)
+ {
+@@ -67,10 +210,8 @@ static inline u_int chan_to_field(u_int
+ return chan << bf->offset;
+ }
+
+-#define LCDC_PALETTE(x) __REG2(IMX_LCDC_BASE+0x800, (x)<<2)
+-static int
+-imxfb_setpalettereg(u_int regno, u_int red, u_int green, u_int blue,
+- u_int trans, struct fb_info *info)
++static int imxfb_setpalettereg(u_int regno, u_int red, u_int green, u_int blue,
++ u_int trans, struct fb_info *info)
+ {
+ struct imxfb_info *fbi = info->par;
+ u_int val, ret = 1;
+@@ -81,14 +222,13 @@ imxfb_setpalettereg(u_int regno, u_int r
+ (CNVT_TOHW(green,4) << 4) |
+ CNVT_TOHW(blue, 4);
+
+- LCDC_PALETTE(regno) = val;
++ writel(val, fbi->regs + 0x800 + (regno << 2));
+ ret = 0;
+ }
+ return ret;
+ }
+
+-static int
+-imxfb_setcolreg(u_int regno, u_int red, u_int green, u_int blue,
++static int imxfb_setcolreg(u_int regno, u_int red, u_int green, u_int blue,
+ u_int trans, struct fb_info *info)
+ {
+ struct imxfb_info *fbi = info->par;
+@@ -148,11 +288,10 @@ imxfb_setcolreg(u_int regno, u_int red,
+ * yres, xres_virtual, yres_virtual, xoffset, yoffset, grayscale,
+ * bitfields, horizontal timing, vertical timing.
+ */
+-static int
+-imxfb_check_var(struct fb_var_screeninfo *var, struct fb_info *info)
++static int imxfb_check_var(struct fb_var_screeninfo *var, struct fb_info *info)
+ {
+ struct imxfb_info *fbi = info->par;
+- int rgbidx;
++ struct imxfb_rgb *rgb;
+
+ if (var->xres < MIN_XRES)
+ var->xres = MIN_XRES;
+@@ -167,24 +306,29 @@ imxfb_check_var(struct fb_var_screeninfo
+
+ pr_debug("var->bits_per_pixel=%d\n", var->bits_per_pixel);
+ switch (var->bits_per_pixel) {
++ case 32:
++ rgb = &def_rgb_18;
++ break;
+ case 16:
+- rgbidx = RGB_16;
++ default:
++ if (readl(fbi->regs + LCDC_PCR) & PCR_TFT)
++ rgb = &def_rgb_16_tft;
++ else
++ rgb = &def_rgb_16_stn;
+ break;
+ case 8:
+- rgbidx = RGB_8;
++ rgb = &def_rgb_8;
+ break;
+- default:
+- rgbidx = RGB_16;
+ }
+
+ /*
+ * Copy the RGB parameters for this display
+ * from the machine specific parameters.
+ */
+- var->red = fbi->rgb[rgbidx]->red;
+- var->green = fbi->rgb[rgbidx]->green;
+- var->blue = fbi->rgb[rgbidx]->blue;
+- var->transp = fbi->rgb[rgbidx]->transp;
++ var->red = rgb->red;
++ var->green = rgb->green;
++ var->blue = rgb->blue;
++ var->transp = rgb->transp;
+
+ pr_debug("RGBT length = %d:%d:%d:%d\n",
+ var->red.length, var->green.length, var->blue.length,
+@@ -206,9 +350,7 @@ static int imxfb_set_par(struct fb_info
+ struct imxfb_info *fbi = info->par;
+ struct fb_var_screeninfo *var = &info->var;
+
+- pr_debug("set_par\n");
+-
+- if (var->bits_per_pixel == 16)
++ if (var->bits_per_pixel == 16 || var->bits_per_pixel == 32)
+ info->fix.visual = FB_VISUAL_TRUECOLOR;
+ else if (!fbi->cmap_static)
+ info->fix.visual = FB_VISUAL_PSEUDOCOLOR;
+@@ -221,49 +363,72 @@ static int imxfb_set_par(struct fb_info
+ info->fix.visual = FB_VISUAL_STATIC_PSEUDOCOLOR;
+ }
+
+- info->fix.line_length = var->xres_virtual *
+- var->bits_per_pixel / 8;
++ info->fix.line_length = var->xres_virtual * var->bits_per_pixel / 8;
+ fbi->palette_size = var->bits_per_pixel == 8 ? 256 : 16;
+
+- imxfb_activate_var(var, info);
++ if (var->bits_per_pixel == 16)
++ fb_dealloc_cmap(&info->cmap);
++ else
++ fb_alloc_cmap(&info->cmap, 1 << info->var.bits_per_pixel, 0);
+
+ return 0;
+ }
+
+ static void imxfb_enable_controller(struct imxfb_info *fbi)
+ {
+- pr_debug("Enabling LCD controller\n");
+-
+- /* initialize LCDC */
+- LCDC_RMCR &= ~RMCR_LCDC_EN; /* just to be safe... */
+-
+- LCDC_SSA = fbi->screen_dma;
+- /* physical screen start address */
+- LCDC_VPW = VPW_VPW(fbi->max_xres * fbi->max_bpp / 8 / 4);
++ struct imx_fb_platform_data *pdata = fbi->pdev->dev.platform_data;
++ int ret;
+
+- LCDC_POS = 0x00000000; /* panning offset 0 (0 pixel offset) */
++ if (!fbi->enabled) {
++ if (clk_enable(fbi->clk) != 0)
++ printk("Warning: Could not enable clock!");
++ pr_debug("LCD clock enabled\n");
++
++ if (pdata->init) {
++ ret = (pdata->init)(fbi->pdev);
++ pr_debug("GPIOs enabled\n");
++ if (ret != 0)
++ pr_err("Failing to enable LCD pins\n");
++ }
+
+- /* disable hardware cursor */
+- LCDC_CPOS &= ~(CPOS_CC0 | CPOS_CC1);
++ if (pdata->lcd_power) {
++ pdata->lcd_power(1);
++ pr_debug("Display's power enabled\n");
++ }
+
+- LCDC_RMCR = RMCR_LCDC_EN;
++ if (pdata->backlight_power) {
++ pdata->backlight_power(1);
++ pr_debug("Backlight's power enabled\n");
++ }
+
+- if(fbi->backlight_power)
+- fbi->backlight_power(1);
+- if(fbi->lcd_power)
+- fbi->lcd_power(1);
++ fbi->enabled = 1;
++ }
+ }
+
+ static void imxfb_disable_controller(struct imxfb_info *fbi)
+ {
+- pr_debug("Disabling LCD controller\n");
++ struct imx_fb_platform_data *pdata = fbi->pdev->dev.platform_data;
++ int ret;
++
++ if (fbi->enabled) {
++ if (pdata->backlight_power)
++ pdata->backlight_power(0);
++ if (pdata->lcd_power)
++ pdata->lcd_power(0);
+
+- if(fbi->backlight_power)
+- fbi->backlight_power(0);
+- if(fbi->lcd_power)
+- fbi->lcd_power(0);
++#ifdef CONFIG_ARCH_IMX
++ writel(0, fbi->regs + LCDC_RMCR); /* for IMX only */
++#endif
++ clk_disable(fbi->clk);
++
++ if (pdata->exit) {
++ ret = (pdata->exit)(fbi->pdev);
++ if (ret != 0)
++ pr_err("Failed to disable LCD pins\n");
++ }
+
+- LCDC_RMCR = 0;
++ fbi->enabled = 0;
++ }
+ }
+
+ static int imxfb_blank(int blank, struct fb_info *info)
+@@ -298,114 +463,68 @@ static struct fb_ops imxfb_ops = {
+ .fb_blank = imxfb_blank,
+ };
+
+-/*
+- * imxfb_activate_var():
+- * Configures LCD Controller based on entries in var parameter. Settings are
+- * only written to the controller if changes were made.
+- */
+-static int imxfb_activate_var(struct fb_var_screeninfo *var, struct fb_info *info)
+-{
+- struct imxfb_info *fbi = info->par;
+- pr_debug("var: xres=%d hslen=%d lm=%d rm=%d\n",
+- var->xres, var->hsync_len,
+- var->left_margin, var->right_margin);
+- pr_debug("var: yres=%d vslen=%d um=%d bm=%d\n",
+- var->yres, var->vsync_len,
+- var->upper_margin, var->lower_margin);
+-
+-#if DEBUG_VAR
+- if (var->xres < 16 || var->xres > 1024)
+- printk(KERN_ERR "%s: invalid xres %d\n",
+- info->fix.id, var->xres);
+- if (var->hsync_len < 1 || var->hsync_len > 64)
+- printk(KERN_ERR "%s: invalid hsync_len %d\n",
+- info->fix.id, var->hsync_len);
+- if (var->left_margin > 255)
+- printk(KERN_ERR "%s: invalid left_margin %d\n",
+- info->fix.id, var->left_margin);
+- if (var->right_margin > 255)
+- printk(KERN_ERR "%s: invalid right_margin %d\n",
+- info->fix.id, var->right_margin);
+- if (var->yres < 1 || var->yres > 511)
+- printk(KERN_ERR "%s: invalid yres %d\n",
+- info->fix.id, var->yres);
+- if (var->vsync_len > 100)
+- printk(KERN_ERR "%s: invalid vsync_len %d\n",
+- info->fix.id, var->vsync_len);
+- if (var->upper_margin > 63)
+- printk(KERN_ERR "%s: invalid upper_margin %d\n",
+- info->fix.id, var->upper_margin);
+- if (var->lower_margin > 255)
+- printk(KERN_ERR "%s: invalid lower_margin %d\n",
+- info->fix.id, var->lower_margin);
+-#endif
+-
+- LCDC_HCR = HCR_H_WIDTH(var->hsync_len) |
+- HCR_H_WAIT_1(var->left_margin) |
+- HCR_H_WAIT_2(var->right_margin);
+-
+- LCDC_VCR = VCR_V_WIDTH(var->vsync_len) |
+- VCR_V_WAIT_1(var->upper_margin) |
+- VCR_V_WAIT_2(var->lower_margin);
+-
+- LCDC_SIZE = SIZE_XMAX(var->xres) | SIZE_YMAX(var->yres);
+- LCDC_PCR = fbi->pcr;
+- LCDC_PWMR = fbi->pwmr;
+- LCDC_LSCR1 = fbi->lscr1;
+- LCDC_DMACR = fbi->dmacr;
+-
+- return 0;
+-}
+-
+-static void imxfb_setup_gpio(struct imxfb_info *fbi)
++/* enable the LCD controller with basic setup for the connected display. */
++static void imxfb_setup_display(struct imxfb_info *fbi)
+ {
+- int width;
++ u32 pixel_clk, pcr;
++ struct imx_fb_platform_data *pinfo = fbi->pdev->dev.platform_data;
+
+- LCDC_RMCR &= ~(RMCR_LCDC_EN | RMCR_SELF_REF);
++#ifdef CONFIG_ARCH_IMX
++ writel(readl(fbi->regs + LCDC_RMCR) & ~RMCR_LCDC_EN,
++ fbi->regs + LCDC_RMCR); /* just to be safe... */
++#endif
+
+- if( fbi->pcr & PCR_TFT )
+- width = 16;
+- else
+- width = 1 << ((fbi->pcr >> 28) & 0x3);
++ /* physical screen start address */
++ writel(fbi->screen_dma, fbi->regs + LCDC_SSA);
+
+- switch(width) {
+- case 16:
+- imx_gpio_mode(PD30_PF_LD15);
+- imx_gpio_mode(PD29_PF_LD14);
+- imx_gpio_mode(PD28_PF_LD13);
+- imx_gpio_mode(PD27_PF_LD12);
+- imx_gpio_mode(PD26_PF_LD11);
+- imx_gpio_mode(PD25_PF_LD10);
+- imx_gpio_mode(PD24_PF_LD9);
+- imx_gpio_mode(PD23_PF_LD8);
+- case 8:
+- imx_gpio_mode(PD22_PF_LD7);
+- imx_gpio_mode(PD21_PF_LD6);
+- imx_gpio_mode(PD20_PF_LD5);
+- imx_gpio_mode(PD19_PF_LD4);
+- case 4:
+- imx_gpio_mode(PD18_PF_LD3);
+- imx_gpio_mode(PD17_PF_LD2);
+- case 2:
+- imx_gpio_mode(PD16_PF_LD1);
+- case 1:
+- imx_gpio_mode(PD15_PF_LD0);
+- }
+-
+- /* initialize GPIOs */
+- imx_gpio_mode(PD6_PF_LSCLK);
+- imx_gpio_mode(PD11_PF_CONTRAST);
+- imx_gpio_mode(PD14_PF_FLM_VSYNC);
+- imx_gpio_mode(PD13_PF_LP_HSYNC);
+- imx_gpio_mode(PD12_PF_ACD_OE);
+-
+- /* These are only needed for Sharp HR TFT displays */
+- if (fbi->pcr & PCR_SHARP) {
+- imx_gpio_mode(PD7_PF_REV);
+- imx_gpio_mode(PD8_PF_CLS);
+- imx_gpio_mode(PD9_PF_PS);
+- imx_gpio_mode(PD10_PF_SPL_SPR);
+- }
++ /* dimension of this display */
++ writel(SIZE_XMAX(pinfo->xres) | SIZE_YMAX(pinfo->yres),
++ fbi->regs + LCDC_SIZE);
++
++ /* virtual page width */
++ writel(VPW_VPW(fbi->max_xres * fbi->max_bpp / 8 / 4),
++ fbi->regs + LCDC_VPW);
++
++ /* setup the pixel clock */
++ pixel_clk = clk_get_rate(fbi->clk);
++
++ pcr = (pixel_clk + ((pinfo->pixclock * 1000UL) / 2)) / (pinfo->pixclock * 1000UL);
++ if (--pcr > 0x3F) {
++ pcr = 0x3F;
++ printk("Must limit pixel clock to %luHz\n", clk_get_rate(fbi->clk) / pcr);
++ }
++
++ /* add sync polarities */
++ pcr |= fbi->pcr & ~0x3F;
++ writel(pcr, fbi->regs + LCDC_PCR);
++
++ /* setup sync behaviour */
++ writel(HCR_H_WIDTH(pinfo->hsync_len - 1) |
++ HCR_H_WAIT_1(pinfo->right_margin - 1) |
++ HCR_H_WAIT_2(pinfo->left_margin - 3),
++ fbi->regs + LCDC_HCR);
++
++ writel(VCR_V_WIDTH(pinfo->vsync_len) |
++ VCR_V_WAIT_1(pinfo->lower_margin) |
++ VCR_V_WAIT_2(pinfo->upper_margin),
++ fbi->regs + LCDC_VCR);
++
++ /* PWM contrast control register */
++ writel(fbi->pwmr, fbi->regs + LCDC_PWMR);
++
++ /* special setup for Sharp displays */
++ writel(fbi->lscr1, fbi->regs + LCDC_LSCR1);
++
++ /* dma watermarks for this mode */
++ writel(fbi->dmacr, fbi->regs + LCDC_DMACR);
++
++ /* activate refresh */
++#ifdef CONFIG_ARCH_IMX
++ writel(readl(fbi->regs + LCDC_RMCR) | RMCR_LCDC_EN,
++ fbi->regs + LCDC_RMCR);
++#else
++ writel(0x00000000, fbi->regs + LCDC_RMCR);
++#endif
+ }
+
+ #ifdef CONFIG_PM
+@@ -413,19 +532,23 @@ static void imxfb_setup_gpio(struct imxf
+ * Power management hooks. Note that we won't be called from IRQ context,
+ * unlike the blank functions above, so we may sleep.
+ */
+-static int imxfb_suspend(struct platform_device *dev, pm_message_t state)
++static int imxfb_suspend(struct platform_device *pdev, pm_message_t state)
+ {
+- struct imxfb_info *fbi = platform_get_drvdata(dev);
+- pr_debug("%s\n",__func__);
++ struct fb_info *info = platform_get_drvdata(pdev);
++ struct imxfb_info *fbi = info->par;
++
++ pr_debug("%s\n", __func__);
+
+ imxfb_disable_controller(fbi);
+ return 0;
+ }
+
+-static int imxfb_resume(struct platform_device *dev)
++static int imxfb_resume(struct platform_device *pdev)
+ {
+- struct imxfb_info *fbi = platform_get_drvdata(dev);
+- pr_debug("%s\n",__func__);
++ struct fb_info *info = platform_get_drvdata(pdev);
++ struct imxfb_info *fbi = info->par;
++
++ pr_debug("%s\n", __func__);
+
+ imxfb_enable_controller(fbi);
+ return 0;
+@@ -435,149 +558,148 @@ static int imxfb_resume(struct platform_
+ #define imxfb_resume NULL
+ #endif
+
+-static int __init imxfb_init_fbinfo(struct device *dev)
++static int __init imxfb_init_fbinfo(struct platform_device *pdev)
+ {
+- struct imxfb_mach_info *inf = dev->platform_data;
+- struct fb_info *info = dev_get_drvdata(dev);
++ struct imx_fb_platform_data *pdata = pdev->dev.platform_data;
++ struct fb_info *info = dev_get_drvdata(&pdev->dev);
+ struct imxfb_info *fbi = info->par;
+
+- pr_debug("%s\n",__func__);
++ pr_debug("%s\n", __func__);
+
+- info->pseudo_palette = kmalloc( sizeof(u32) * 16, GFP_KERNEL);
++ info->pseudo_palette = kmalloc(sizeof(u32) * 16, GFP_KERNEL);
+ if (!info->pseudo_palette)
+ return -ENOMEM;
+
+ memset(fbi, 0, sizeof(struct imxfb_info));
+- fbi->dev = dev;
+
+ strlcpy(info->fix.id, IMX_NAME, sizeof(info->fix.id));
+
+- info->fix.type = FB_TYPE_PACKED_PIXELS;
++ info->fix.type = FB_TYPE_PACKED_PIXELS;
+ info->fix.type_aux = 0;
+ info->fix.xpanstep = 0;
+ info->fix.ypanstep = 0;
+ info->fix.ywrapstep = 0;
+- info->fix.accel = FB_ACCEL_NONE;
++ info->fix.accel = FB_ACCEL_NONE;
+
+ info->var.nonstd = 0;
+ info->var.activate = FB_ACTIVATE_NOW;
+ info->var.height = -1;
+ info->var.width = -1;
+ info->var.accel_flags = 0;
+- info->var.vmode = FB_VMODE_NONINTERLACED;
++ info->var.vmode = FB_VMODE_NONINTERLACED;
+
+ info->fbops = &imxfb_ops;
+- info->flags = FBINFO_FLAG_DEFAULT | FBINFO_READS_FAST;
+-
+- fbi->rgb[RGB_16] = &def_rgb_16;
+- fbi->rgb[RGB_8] = &def_rgb_8;
++ info->flags = FBINFO_FLAG_DEFAULT |
++ FBINFO_READS_FAST;
+
+- fbi->max_xres = inf->xres;
+- info->var.xres = inf->xres;
+- info->var.xres_virtual = inf->xres;
+- fbi->max_yres = inf->yres;
+- info->var.yres = inf->yres;
+- info->var.yres_virtual = inf->yres;
+- fbi->max_bpp = inf->bpp;
+- info->var.bits_per_pixel = inf->bpp;
+- info->var.nonstd = inf->nonstd;
+- info->var.pixclock = inf->pixclock;
+- info->var.hsync_len = inf->hsync_len;
+- info->var.left_margin = inf->left_margin;
+- info->var.right_margin = inf->right_margin;
+- info->var.vsync_len = inf->vsync_len;
+- info->var.upper_margin = inf->upper_margin;
+- info->var.lower_margin = inf->lower_margin;
+- info->var.sync = inf->sync;
+- info->var.grayscale = inf->cmap_greyscale;
+- fbi->cmap_inverse = inf->cmap_inverse;
+- fbi->cmap_static = inf->cmap_static;
+- fbi->pcr = inf->pcr;
+- fbi->lscr1 = inf->lscr1;
+- fbi->dmacr = inf->dmacr;
+- fbi->pwmr = inf->pwmr;
+- fbi->lcd_power = inf->lcd_power;
+- fbi->backlight_power = inf->backlight_power;
++ fbi->max_xres = pdata->xres;
++ info->var.xres = pdata->xres;
++ info->var.xres_virtual = pdata->xres;
++ fbi->max_yres = pdata->yres;
++ info->var.yres = pdata->yres;
++ info->var.yres_virtual = pdata->yres;
++ fbi->max_bpp = pdata->bpp;
++ info->var.bits_per_pixel = pdata->bpp;
++ info->var.nonstd = pdata->nonstd;
++ info->var.pixclock = pdata->pixclock;
++ info->var.hsync_len = pdata->hsync_len;
++ info->var.left_margin = pdata->left_margin;
++ info->var.right_margin = pdata->right_margin;
++ info->var.vsync_len = pdata->vsync_len;
++ info->var.upper_margin = pdata->upper_margin;
++ info->var.lower_margin = pdata->lower_margin;
++ info->var.sync = pdata->sync;
++ info->var.grayscale = pdata->cmap_greyscale;
++ fbi->cmap_inverse = pdata->cmap_inverse;
++ fbi->cmap_static = pdata->cmap_static;
++ fbi->pcr = pdata->pcr;
++ fbi->lscr1 = pdata->lscr1;
++ fbi->dmacr = pdata->dmacr;
++ fbi->pwmr = pdata->pwmr;
++ fbi->lcd_power = pdata->lcd_power;
++ fbi->backlight_power = pdata->backlight_power;
+ info->fix.smem_len = fbi->max_xres * fbi->max_yres *
+ fbi->max_bpp / 8;
+
+ return 0;
+ }
+
+-/*
+- * Allocates the DRAM memory for the frame buffer. This buffer is
+- * remapped into a non-cached, non-buffered, memory region to
+- * allow pixel writes to occur without flushing the cache.
+- * Once this area is remapped, all virtual memory access to the
+- * video memory should occur at the new region.
+- */
+-static int __init imxfb_map_video_memory(struct fb_info *info)
+-{
+- struct imxfb_info *fbi = info->par;
+-
+- fbi->map_size = PAGE_ALIGN(info->fix.smem_len);
+- fbi->map_cpu = dma_alloc_writecombine(fbi->dev, fbi->map_size,
+- &fbi->map_dma,GFP_KERNEL);
+-
+- if (fbi->map_cpu) {
+- info->screen_base = fbi->map_cpu;
+- fbi->screen_cpu = fbi->map_cpu;
+- fbi->screen_dma = fbi->map_dma;
+- info->fix.smem_start = fbi->screen_dma;
+- }
+-
+- return fbi->map_cpu ? 0 : -ENOMEM;
+-}
+-
+ static int __init imxfb_probe(struct platform_device *pdev)
+ {
+ struct imxfb_info *fbi;
+ struct fb_info *info;
+- struct imxfb_mach_info *inf;
++ struct imx_fb_platform_data *pdata;
+ struct resource *res;
+ int ret;
+
+ printk("i.MX Framebuffer driver\n");
+
+ res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
+- if(!res)
++ if (!res)
+ return -ENODEV;
+
+- inf = pdev->dev.platform_data;
+- if(!inf) {
++ pdata = pdev->dev.platform_data;
++ if (!pdata) {
+ dev_err(&pdev->dev,"No platform_data available\n");
+ return -ENOMEM;
+ }
+
+ info = framebuffer_alloc(sizeof(struct imxfb_info), &pdev->dev);
+- if(!info)
++ if (!info)
+ return -ENOMEM;
+
+ fbi = info->par;
+
+ platform_set_drvdata(pdev, info);
+
+- ret = imxfb_init_fbinfo(&pdev->dev);
+- if( ret < 0 )
++ ret = imxfb_init_fbinfo(pdev);
++ if (ret < 0)
+ goto failed_init;
+
+- res = request_mem_region(res->start, res->end - res->start + 1, "IMXFB");
++ fbi->pdev = pdev;
++
++ res = request_mem_region(res->start, resource_size(res),
++ DRIVER_NAME);
+ if (!res) {
+ ret = -EBUSY;
+- goto failed_regs;
++ goto failed_req;
++ }
++
++ fbi->regs = ioremap(res->start, resource_size(res));
++ if (fbi->regs == NULL) {
++ printk(KERN_ERR"Cannot map frame buffer registers\n");
++ goto failed_ioremap;
+ }
+
+- if (!inf->fixed_screen_cpu) {
+- ret = imxfb_map_video_memory(info);
+- if (ret) {
+- dev_err(&pdev->dev, "Failed to allocate video RAM: %d\n", ret);
++ fbi->clk = clk_get(&pdev->dev, "lcdc_clk");
++ if (IS_ERR(fbi->clk)) {
++ ret = PTR_ERR(fbi->clk);
++ dev_err(&pdev->dev,
++ "Cannot get the clock for IMX LCD unit %d: %d\n",
++ pdev->id, ret);
++ goto failed_getclock;
++ }
++
++ if (!pdata->fixed_screen_cpu) {
++ fbi->map_size = PAGE_ALIGN(info->fix.smem_len);
++ fbi->map_cpu = dma_alloc_writecombine(&pdev->dev,
++ fbi->map_size, &fbi->map_dma, GFP_KERNEL);
++
++ if (!fbi->map_cpu) {
++ dev_err(&pdev->dev, "Failed to allocate %u byte for video RAM\n",
++ fbi->map_size);
+ ret = -ENOMEM;
+ goto failed_map;
+ }
++
++ info->screen_base = fbi->map_cpu;
++ fbi->screen_cpu = fbi->map_cpu;
++ fbi->screen_dma = fbi->map_dma;
++ info->fix.smem_start = fbi->screen_dma;
+ } else {
+ /* Fixed framebuffer mapping enables location of the screen in eSRAM */
+- fbi->map_cpu = inf->fixed_screen_cpu;
+- fbi->map_dma = inf->fixed_screen_dma;
++ fbi->map_cpu = pdata->fixed_screen_cpu;
++ fbi->map_dma = pdata->fixed_screen_dma;
+ info->screen_base = fbi->map_cpu;
+ fbi->screen_cpu = fbi->map_cpu;
+ fbi->screen_dma = fbi->map_dma;
+@@ -588,17 +710,26 @@ static int __init imxfb_probe(struct pla
+ * This makes sure that our colour bitfield
+ * descriptors are correctly initialised.
+ */
+- imxfb_check_var(&info->var, info);
++ imxfb_setup_display(fbi);
++ ret = imxfb_check_var(&info->var, info);
++ if (ret) {
++ dev_err(&pdev->dev, "failed to get suitable mode\n");
++ goto failed_cmap;
++ }
+
+- ret = fb_alloc_cmap(&info->cmap, 1<<info->var.bits_per_pixel, 0);
+- if (ret < 0)
++ ret = imxfb_set_par(info);
++ if (ret) {
++ dev_err(&pdev->dev, "Failed to set parameters\n");
+ goto failed_cmap;
++ }
+
+- imxfb_setup_gpio(fbi);
++ ret = fb_alloc_cmap(&info->cmap, 1 << info->var.bits_per_pixel, 0);
++ if (ret < 0)
++ goto failed_cmap;
+
+ imxfb_set_par(info);
+ ret = register_framebuffer(info);
+- if (ret < 0) {
++ if (ret) {
+ dev_err(&pdev->dev, "failed to register framebuffer\n");
+ goto failed_register;
+ }
+@@ -610,20 +741,24 @@ static int __init imxfb_probe(struct pla
+ failed_register:
+ fb_dealloc_cmap(&info->cmap);
+ failed_cmap:
+- if (!inf->fixed_screen_cpu)
++ if (!pdata->fixed_screen_cpu)
+ dma_free_writecombine(&pdev->dev,fbi->map_size,fbi->map_cpu,
+- fbi->map_dma);
++ fbi->map_dma);
+ failed_map:
+- kfree(info->pseudo_palette);
+-failed_regs:
++ clk_put(fbi->clk);
++failed_getclock:
++ iounmap(fbi->regs);
++failed_ioremap:
+ release_mem_region(res->start, res->end - res->start);
++failed_req:
++ kfree(info->pseudo_palette);
+ failed_init:
+ platform_set_drvdata(pdev, NULL);
+ framebuffer_release(info);
+ return ret;
+ }
+
+-static int imxfb_remove(struct platform_device *pdev)
++static int __devexit imxfb_remove(struct platform_device *pdev)
+ {
+ struct fb_info *info = platform_get_drvdata(pdev);
+ struct imxfb_info *fbi = info->par;
+@@ -639,7 +774,12 @@ static int imxfb_remove(struct platform_
+ kfree(info->pseudo_palette);
+ framebuffer_release(info);
+
++ iounmap(fbi->regs);
+ release_mem_region(res->start, res->end - res->start + 1);
++
++ clk_disable(fbi->clk);
++ clk_put(fbi->clk);
++
+ platform_set_drvdata(pdev, NULL);
+
+ return 0;
+@@ -653,19 +793,18 @@ void imxfb_shutdown(struct platform_dev
+ }
+
+ static struct platform_driver imxfb_driver = {
+- .probe = imxfb_probe,
+ .suspend = imxfb_suspend,
+ .resume = imxfb_resume,
+- .remove = imxfb_remove,
++ .remove = __devexit_p(imxfb_remove),
+ .shutdown = imxfb_shutdown,
+ .driver = {
+- .name = "imx-fb",
++ .name = DRIVER_NAME,
+ },
+ };
+
+ int __init imxfb_init(void)
+ {
+- return platform_driver_register(&imxfb_driver);
++ return platform_driver_probe(&imxfb_driver, imxfb_probe);
+ }
+
+ static void __exit imxfb_cleanup(void)
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/video/imxfb.h linux-2.6.28-karo/drivers/video/imxfb.h
+--- linux-2.6.28/drivers/video/imxfb.h 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/video/imxfb.h 1970-01-01 01:00:00.000000000 +0100
+@@ -1,73 +0,0 @@
+-/*
+- * linux/drivers/video/imxfb.h
+- *
+- * Freescale i.MX Frame Buffer device driver
+- *
+- * Copyright (C) 2004 S.Hauer, Pengutronix
+- *
+- * Copyright (C) 1999 Eric A. Thomas
+- * Based on acornfb.c Copyright (C) Russell King.
+- *
+- * 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.
+- */
+-
+-/*
+- * These are the bitfields for each
+- * display depth that we support.
+- */
+-struct imxfb_rgb {
+- struct fb_bitfield red;
+- struct fb_bitfield green;
+- struct fb_bitfield blue;
+- struct fb_bitfield transp;
+-};
+-
+-#define RGB_16 (0)
+-#define RGB_8 (1)
+-#define NR_RGB 2
+-
+-struct imxfb_info {
+- struct device *dev;
+- struct imxfb_rgb *rgb[NR_RGB];
+-
+- u_int max_bpp;
+- u_int max_xres;
+- u_int max_yres;
+-
+- /*
+- * These are the addresses we mapped
+- * the framebuffer memory region to.
+- */
+- dma_addr_t map_dma;
+- u_char * map_cpu;
+- u_int map_size;
+-
+- u_char * screen_cpu;
+- dma_addr_t screen_dma;
+- u_int palette_size;
+-
+- dma_addr_t dbar1;
+- dma_addr_t dbar2;
+-
+- u_int pcr;
+- u_int pwmr;
+- u_int lscr1;
+- u_int dmacr;
+- u_int cmap_inverse:1,
+- cmap_static:1,
+- unused:30;
+-
+- void (*lcd_power)(int);
+- void (*backlight_power)(int);
+-};
+-
+-#define IMX_NAME "IMX"
+-
+-/*
+- * Minimum X and Y resolutions
+- */
+-#define MIN_XRES 64
+-#define MIN_YRES 64
+-
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/w1/masters/Kconfig linux-2.6.28-karo/drivers/w1/masters/Kconfig
+--- linux-2.6.28/drivers/w1/masters/Kconfig 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/w1/masters/Kconfig 2009-03-11 13:16:24.000000000 +0100
+@@ -34,6 +34,12 @@ config W1_MASTER_DS2482
+ This driver can also be built as a module. If so, the module
+ will be called ds2482.
+
++config W1_MASTER_MXC
++ tristate "Freescale MXC 1-wire busmaster"
++ depends on W1 && ARCH_MXC
++ help
++ Say Y here to enable MXC 1-wire host
++
+ config W1_MASTER_DS1WM
+ tristate "Maxim DS1WM 1-wire busmaster"
+ depends on W1 && ARM && HAVE_CLK
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/w1/masters/Makefile linux-2.6.28-karo/drivers/w1/masters/Makefile
+--- linux-2.6.28/drivers/w1/masters/Makefile 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/drivers/w1/masters/Makefile 2009-03-11 13:16:24.000000000 +0100
+@@ -5,6 +5,8 @@
+ obj-$(CONFIG_W1_MASTER_MATROX) += matrox_w1.o
+ obj-$(CONFIG_W1_MASTER_DS2490) += ds2490.o
+ obj-$(CONFIG_W1_MASTER_DS2482) += ds2482.o
++obj-$(CONFIG_W1_MASTER_MXC) += mxc_w1.o
++
+ obj-$(CONFIG_W1_MASTER_DS1WM) += ds1wm.o
+ obj-$(CONFIG_W1_MASTER_GPIO) += w1-gpio.o
+ obj-$(CONFIG_HDQ_MASTER_OMAP) += omap_hdq.o
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/drivers/w1/masters/mxc_w1.c linux-2.6.28-karo/drivers/w1/masters/mxc_w1.c
+--- linux-2.6.28/drivers/w1/masters/mxc_w1.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/drivers/w1/masters/mxc_w1.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,211 @@
++/*
++ * Copyright 2005-2008 Freescale Semiconductor, Inc. All Rights Reserved.
++ * Copyright 2008 Luotao Fu, kernel@pengutronix.de
++ *
++ * 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
++ *
++ */
++
++#include <linux/module.h>
++#include <linux/interrupt.h>
++#include <linux/platform_device.h>
++#include <linux/clk.h>
++#include <linux/delay.h>
++#include <linux/io.h>
++
++#include "../w1.h"
++#include "../w1_int.h"
++#include "../w1_log.h"
++
++/* According to the mx27 Datasheet the reset procedure should take up to about
++ * 1350us. We set the timeout to 500*100us = 50ms for sure */
++#define MXC_W1_RESET_TIMEOUT 500
++
++/*
++ * MXC W1 Register offsets
++ */
++#define MXC_W1_CONTROL 0x00
++#define MXC_W1_TIME_DIVIDER 0x02
++#define MXC_W1_RESET 0x04
++#define MXC_W1_COMMAND 0x06
++#define MXC_W1_TXRX 0x08
++#define MXC_W1_INTERRUPT 0x0A
++#define MXC_W1_INTERRUPT_EN 0x0C
++
++struct mxc_w1_device {
++ void __iomem *regs;
++ unsigned int clkdiv;
++ struct clk *clk;
++ struct w1_bus_master bus_master;
++};
++
++/*
++ * this is the low level routine to
++ * reset the device on the One Wire interface
++ * on the hardware
++ */
++static u8 mxc_w1_ds2_reset_bus(void *data)
++{
++ u8 reg_val;
++ unsigned int timeout_cnt = 0;
++ struct mxc_w1_device *dev = data;
++
++ __raw_writeb(0x80, (dev->regs + MXC_W1_CONTROL));
++
++ while (1) {
++ reg_val = __raw_readb(dev->regs + MXC_W1_CONTROL);
++
++ if (((reg_val >> 7) & 0x1) == 0 ||
++ timeout_cnt > MXC_W1_RESET_TIMEOUT)
++ break;
++ else
++ timeout_cnt++;
++
++ udelay(100);
++ }
++ return (reg_val >> 7) & 0x1;
++}
++
++/*
++ * this is the low level routine to read/write a bit on the One Wire
++ * interface on the hardware. It does write 0 if parameter bit is set
++ * to 0, otherwise a write 1/read.
++ */
++static u8 mxc_w1_ds2_touch_bit(void *data, u8 bit)
++{
++ struct mxc_w1_device *mdev = data;
++ void __iomem *ctrl_addr = mdev->regs + MXC_W1_CONTROL;
++ unsigned int timeout_cnt = 400; /* Takes max. 120us according to
++ * datasheet.
++ */
++
++ __raw_writeb((1 << (5 - bit)), ctrl_addr);
++
++ while (timeout_cnt--) {
++ if (!((__raw_readb(ctrl_addr) >> (5 - bit)) & 0x1))
++ break;
++
++ udelay(1);
++ }
++
++ return ((__raw_readb(ctrl_addr)) >> 3) & 0x1;
++}
++
++static int __init mxc_w1_probe(struct platform_device *pdev)
++{
++ struct mxc_w1_device *mdev;
++ struct resource *res;
++ int err = 0;
++
++ res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
++ if (!res)
++ return -ENODEV;
++
++ mdev = kzalloc(sizeof(struct mxc_w1_device), GFP_KERNEL);
++ if (!mdev)
++ return -ENOMEM;
++
++ mdev->clk = clk_get(&pdev->dev, "owire_clk");
++ if (!mdev->clk) {
++ err = -ENODEV;
++ goto failed_clk;
++ }
++
++ mdev->clkdiv = (clk_get_rate(mdev->clk) / 1000000) - 1;
++
++ res = request_mem_region(res->start, resource_size(res),
++ "mxc_w1");
++ if (!res) {
++ err = -EBUSY;
++ goto failed_req;
++ }
++
++ mdev->regs = ioremap(res->start, resource_size(res));
++ if (!mdev->regs) {
++ printk(KERN_ERR "Cannot map frame buffer registers\n");
++ goto failed_ioremap;
++ }
++
++ clk_enable(mdev->clk);
++ __raw_writeb(mdev->clkdiv, mdev->regs + MXC_W1_TIME_DIVIDER);
++
++ mdev->bus_master.data = mdev;
++ mdev->bus_master.reset_bus = mxc_w1_ds2_reset_bus;
++ mdev->bus_master.touch_bit = mxc_w1_ds2_touch_bit;
++
++ err = w1_add_master_device(&mdev->bus_master);
++
++ if (err)
++ goto failed_add;
++
++ platform_set_drvdata(pdev, mdev);
++ return 0;
++
++failed_add:
++ iounmap(mdev->regs);
++failed_ioremap:
++ release_mem_region(res->start, resource_size(res));
++failed_req:
++ clk_put(mdev->clk);
++failed_clk:
++ kfree(mdev);
++ return err;
++}
++
++/*
++ * disassociate the w1 device from the driver
++ */
++static int mxc_w1_remove(struct platform_device *pdev)
++{
++ struct mxc_w1_device *mdev = platform_get_drvdata(pdev);
++ struct resource *res;
++
++ res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
++
++ w1_remove_master_device(&mdev->bus_master);
++
++ iounmap(mdev->regs);
++ release_mem_region(res->start, resource_size(res));
++ clk_disable(mdev->clk);
++ clk_put(mdev->clk);
++
++ platform_set_drvdata(pdev, NULL);
++
++ return 0;
++}
++
++static struct platform_driver mxc_w1_driver = {
++ .driver = {
++ .name = "mxc_w1",
++ },
++ .probe = mxc_w1_probe,
++ .remove = mxc_w1_remove,
++};
++
++static int __init mxc_w1_init(void)
++{
++ return platform_driver_register(&mxc_w1_driver);
++}
++
++static void mxc_w1_exit(void)
++{
++ platform_driver_unregister(&mxc_w1_driver);
++}
++
++module_init(mxc_w1_init);
++module_exit(mxc_w1_exit);
++
++MODULE_LICENSE("GPL");
++MODULE_AUTHOR("Freescale Semiconductors Inc");
++MODULE_DESCRIPTION("Driver for One-Wire on MXC");
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/include/linux/dmaengine.h linux-2.6.28-karo/include/linux/dmaengine.h
+--- linux-2.6.28/include/linux/dmaengine.h 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/include/linux/dmaengine.h 2009-03-11 13:16:24.000000000 +0100
+@@ -285,6 +285,7 @@ struct dma_async_tx_descriptor {
+ struct list_head tx_list;
+ struct dma_chan *chan;
+ dma_cookie_t (*tx_submit)(struct dma_async_tx_descriptor *tx);
++ void (*tx_free)(struct dma_async_tx_descriptor *tx);
+ dma_async_tx_callback callback;
+ void *callback_param;
+ struct dma_async_tx_descriptor *next;
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/include/linux/fec_enet.h linux-2.6.28-karo/include/linux/fec_enet.h
+--- linux-2.6.28/include/linux/fec_enet.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/include/linux/fec_enet.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,26 @@
++/*
++ * Copyright (C) 2007 Lothar Wassmann <LW@KARO-electronics.de>
++ *
++ * platform_data definitions for fec_enet device
++ *
++ * 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 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301
++ */
++
++struct fec_enet_platform_data {
++ /* callback for platform specific initialization */
++ int (*arch_init)(struct platform_device *dev);
++ void (*arch_exit)(struct platform_device *dev);
++ int (*suspend)(struct platform_device *dev);
++ int (*resume)(struct platform_device *dev);
++};
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/include/linux/mtd/mtd.h linux-2.6.28-karo/include/linux/mtd/mtd.h
+--- linux-2.6.28/include/linux/mtd/mtd.h 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/include/linux/mtd/mtd.h 2009-03-11 13:16:24.000000000 +0100
+@@ -268,13 +268,13 @@ static inline void mtd_erase_callback(st
+ #define MTD_DEBUG_LEVEL3 (3) /* Noisy */
+
+ #ifdef CONFIG_MTD_DEBUG
+-#define DEBUG(n, args...) \
++#define MTD_DEBUG(n, args...) \
+ do { \
+ if (n <= CONFIG_MTD_DEBUG_VERBOSE) \
+ printk(KERN_INFO args); \
+ } while(0)
+ #else /* CONFIG_MTD_DEBUG */
+-#define DEBUG(n, args...) \
++#define MTD_DEBUG(n, args...) \
+ do { \
+ if (0) \
+ printk(KERN_INFO args); \
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/include/linux/rtc/ds13xx.h linux-2.6.28-karo/include/linux/rtc/ds13xx.h
+--- linux-2.6.28/include/linux/rtc/ds13xx.h 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/include/linux/rtc/ds13xx.h 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,73 @@
++/*
++ * include/linux/rtc/rtc-ds1339.h
++ *
++ * platform specific definitions for DS1339 RTC driver
++ *
++ *
++ * Copyright (C) 2007 Lothar Wassmann <LW@KARO-electronics.de>
++ *
++ * 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 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.
++ */
++
++enum ds13xx_type {
++ unknown,
++ ds_1307, /* or ds1338, ... */
++ ds_1337,
++ ds_1338,
++ ds_1339,
++ ds_1340,
++ m41t00,
++ // rs5c372 too? different address...
++};
++
++/* DS1307 control reg (0x07) bit definitions */
++#define DS1307_BIT_OUT 0x80
++#define DS1307_BIT_SQWE 0x10
++#define DS1307_BIT_RS1 0x02
++#define DS1307_BIT_RS0 0x01
++/* DS1337/DS1339 control reg (0x0e) bit definitions */
++#define DS133X_BIT_INTCN 0x04
++#define DS133X_BIT_RS1 0x08
++#define DS133X_BIT_RS2 0x10
++#define DS1339_BIT_BBSQI 0x20
++/* DS1339 trickle charge reg (0x10) bit definitions */
++#define DS1339_TRC_ENABLE 0xa0
++#define DS1339_TRC_250R 0x01
++#define DS1339_TRC_2K 0x02
++#define DS1339_TRC_4K 0x03
++#define DS1339_DIODE_DISABLE 0x04
++#define DS1339_DIODE_ENABLE 0x08
++/* DS1340 trickle charge reg (0x08) bit definitions */
++#define DS1340_TRC_ENABLE 0xa0
++#define DS1340_TRC_250R 0x01
++#define DS1340_TRC_2K 0x02
++#define DS1340_TRC_4K 0x03
++#define DS1340_DIODE_DISABLE 0x04
++#define DS1340_DIODE_ENABLE 0x08
++/* DS1340 control reg (0x09) bit definitions */
++#define DS1340_BIT_OUT 0x80
++#define DS1340_BIT_FT 0x0
++#define DS1340_CAL_SIGN 0x0
++#define DS1340_CAL_MASK 0x1f
++#define DS1340_CAL(v) ((v) & (DS1340_CAL_MASK))
++
++struct ds13xx_platform_data {
++ /* type of DS13XX chip */
++ enum ds13xx_type type;
++ /* value for Control register on DS13xx; < 0 means don't care */
++ short ctrl;
++ /* value for Trickle charge register on DS1339; < 0 means don't care */
++ short trc;
++};
++
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/kernel/printk.c linux-2.6.28-karo/kernel/printk.c
+--- linux-2.6.28/kernel/printk.c 2008-12-25 00:26:37.000000000 +0100
++++ linux-2.6.28-karo/kernel/printk.c 2009-03-11 13:16:24.000000000 +0100
+@@ -622,6 +622,9 @@ static int recursion_bug;
+ static int new_text_line = 1;
+ static char printk_buf[1024];
+
++#ifdef CONFIG_DEBUG_LL
++extern void asmlinkage printascii(const char *);
++#endif
+ asmlinkage int vprintk(const char *fmt, va_list args)
+ {
+ int printed_len = 0;
+@@ -669,6 +672,9 @@ asmlinkage int vprintk(const char *fmt,
+ sizeof(printk_buf) - printed_len, fmt, args);
+
+
++#ifdef CONFIG_DEBUG_LL
++ printascii(printk_buf);
++#endif
+ /*
+ * Copy the output into log_buf. If the caller didn't provide
+ * appropriate log level tags, we insert them here
+diff -purN -X linux-2.6.28/Documentation/dontdiff linux-2.6.28/sound/arm/soc-wrapper.c linux-2.6.28-karo/sound/arm/soc-wrapper.c
+--- linux-2.6.28/sound/arm/soc-wrapper.c 1970-01-01 01:00:00.000000000 +0100
++++ linux-2.6.28-karo/sound/arm/soc-wrapper.c 2009-03-11 13:16:24.000000000 +0100
+@@ -0,0 +1,222 @@
++
++#define SOC_SHIFT_LEFT 8
++#define SOC_SHIFT_RIGHT 13
++#define SOC_SHIFT_MAX 18
++#define SOC_SHIFT_INVERT 26
++
++#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 }
++
++#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_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_SINGLE_VALUE(reg, shift, max, invert) ((reg) | ((shift) << SOC_SHIFT_LEFT) |\
++ ((shift) << SOC_SHIFT_RIGHT) | ((max) << SOC_SHIFT_MAX) | ((invert) << SOC_SHIFT_INVERT))
++
++#define SOC_DOUBLE(xname, reg, shift_left, shift_right, max, 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) << SOC_SHIFT_LEFT) | \
++ ((shift_right) << SOC_SHIFT_RIGHT) | ((max) << SOC_SHIFT_MAX) | ((invert) << SOC_SHIFT_INVERT) }
++
++#define SOC_SINGLE_EXT(xname, xreg, xshift, xmask, xinvert,\
++ xhandler_get, xhandler_put) \
++{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, \
++ .info = snd_soc_info_volsw, \
++ .get = xhandler_get, .put = xhandler_put, \
++ .private_value = SOC_SINGLE_VALUE(xreg, xshift, xmask, xinvert) }
++
++#define SOC_SINGLE(xname, reg, shift, max, 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, max, invert) }
++
++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;
++};
++
++static unsigned int snd_soc_read(int reg);
++static void snd_soc_write(int reg, int val);
++
++static 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;
++}
++
++static int snd_soc_info_volsw(struct snd_kcontrol *kcontrol,
++ struct snd_ctl_elem_info *uinfo)
++{
++ int max = (kcontrol->private_value >> SOC_SHIFT_MAX) & 0xff;
++ int shift = (kcontrol->private_value >> SOC_SHIFT_LEFT) & 0x1f;
++ int rshift = (kcontrol->private_value >> SOC_SHIFT_RIGHT) & 0x1f;
++
++ if (max == 1)
++ uinfo->type = SNDRV_CTL_ELEM_TYPE_BOOLEAN;
++ else
++ uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
++
++ uinfo->count = shift == rshift ? 1 : 2;
++ uinfo->value.integer.min = 0;
++ uinfo->value.integer.max = max;
++ return 0;
++}
++
++static int snd_soc_get_volsw(struct snd_kcontrol *kcontrol,
++ struct snd_ctl_elem_value *ucontrol)
++{
++ int reg = kcontrol->private_value & 0xff;
++ int shift = (kcontrol->private_value >> SOC_SHIFT_LEFT) & 0x1f;
++ int rshift = (kcontrol->private_value >> SOC_SHIFT_RIGHT) & 0x1f;
++ int max = (kcontrol->private_value >> SOC_SHIFT_MAX) & 0xff;
++ int mask = (1 << fls(max)) - 1;
++ int invert = (kcontrol->private_value >> SOC_SHIFT_INVERT) & 0x01;
++
++ ucontrol->value.integer.value[0] =
++ (snd_soc_read(reg) >> shift) & mask;
++ if (shift != rshift)
++ ucontrol->value.integer.value[1] =
++ (snd_soc_read(reg) >> rshift) & mask;
++ if (invert) {
++ ucontrol->value.integer.value[0] =
++ max - ucontrol->value.integer.value[0];
++ if (shift != rshift)
++ ucontrol->value.integer.value[1] =
++ max - ucontrol->value.integer.value[1];
++ }
++
++ return 0;
++}
++
++static int snd_soc_update_bits(unsigned short reg,
++ unsigned int mask, unsigned int value)
++{
++ int change;
++ unsigned int old, new;
++
++// mutex_lock(&io_mutex);
++ old = snd_soc_read(reg);
++ new = (old & ~mask) | value;
++ change = old != new;
++ if (change)
++ snd_soc_write(reg, new);
++
++// mutex_unlock(&io_mutex);
++ return change;
++}
++
++static int snd_soc_put_volsw(struct snd_kcontrol *kcontrol,
++ struct snd_ctl_elem_value *ucontrol)
++{
++ int reg = kcontrol->private_value & 0xff;
++ int shift = (kcontrol->private_value >> SOC_SHIFT_LEFT) & 0x1f;
++ int rshift = (kcontrol->private_value >> SOC_SHIFT_RIGHT) & 0x1f;
++ int max = (kcontrol->private_value >> SOC_SHIFT_MAX) & 0xff;
++ int mask = (1 << fls(max)) - 1;
++ int invert = (kcontrol->private_value >> SOC_SHIFT_INVERT) & 0x01;
++ unsigned int val, val2, val_mask;
++
++ val = (ucontrol->value.integer.value[0] & mask);
++ if (invert)
++ val = max - val;
++ val_mask = mask << shift;
++ val = val << shift;
++ if (shift != rshift) {
++ val2 = (ucontrol->value.integer.value[1] & mask);
++ if (invert)
++ val2 = max - val2;
++ val_mask |= mask << rshift;
++ val |= val2 << rshift;
++ }
++ return snd_soc_update_bits(reg, val_mask, val);
++}
++
++static 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;
++}
++
++static int snd_soc_get_enum_double(struct snd_kcontrol *kcontrol,
++ struct snd_ctl_elem_value *ucontrol)
++{
++ struct soc_enum *e = (struct soc_enum *)kcontrol->private_value;
++ unsigned int val, bitmask;
++
++ for (bitmask = 1; bitmask < e->mask; bitmask <<= 1)
++ ;
++ val = snd_soc_read(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;
++}
++
++static int snd_soc_put_enum_double(struct snd_kcontrol *kcontrol,
++ struct snd_ctl_elem_value *ucontrol)
++{
++ struct soc_enum *e = (struct soc_enum *)kcontrol->private_value;
++ unsigned int val;
++ unsigned int 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(e->reg, mask, val);
++}