diff options
author | Srikant Patnaik | 2015-01-13 15:08:24 +0530 |
---|---|---|
committer | Srikant Patnaik | 2015-01-13 15:08:24 +0530 |
commit | 97327692361306d1e6259021bc425e32832fdb50 (patch) | |
tree | fe9088f3248ec61e24f404f21b9793cb644b7f01 /drivers/video/wmt/mali.c | |
parent | 2d05a8f663478a44e088d122e0d62109bbc801d0 (diff) | |
parent | a3a8b90b61e21be3dde9101c4e86c881e0f06210 (diff) | |
download | FOSSEE-netbook-kernel-source-97327692361306d1e6259021bc425e32832fdb50.tar.gz FOSSEE-netbook-kernel-source-97327692361306d1e6259021bc425e32832fdb50.tar.bz2 FOSSEE-netbook-kernel-source-97327692361306d1e6259021bc425e32832fdb50.zip |
dirty fix to merging
Diffstat (limited to 'drivers/video/wmt/mali.c')
-rw-r--r-- | drivers/video/wmt/mali.c | 759 |
1 files changed, 759 insertions, 0 deletions
diff --git a/drivers/video/wmt/mali.c b/drivers/video/wmt/mali.c new file mode 100644 index 00000000..4ae9509c --- /dev/null +++ b/drivers/video/wmt/mali.c @@ -0,0 +1,759 @@ +/* + * Copyright (c) 2008-2011 WonderMedia Technologies, 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 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, see <http://www.gnu.org/licenses/>. + * + * WonderMedia Technologies, Inc. + * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C + */ + +/* Written by Vincent Chen, WonderMedia Technologies, Inc., 2008-2011 */ + +#include <asm/cacheflush.h> +#include <linux/io.h> +#include <linux/slab.h> +#include <linux/mm.h> +#include <linux/spinlock.h> +#include <linux/module.h> +#include <linux/platform_device.h> +#include <linux/delay.h> +#include "mali.h" +#include <mach/hardware.h> + +/* Mali-400 Power Management Kernel Driver */ + +#define REG_MALI_BADDR (0xD8080000 + WMT_MMAP_OFFSET) +/* +#define MALI_PMU_CONTROL +#define MALI_CLK_CONTROL +#define USE_PMC_POWER_STATUS +*/ + +static spinlock_t mali_spinlock; + +static unsigned int mali_max_freq; +static unsigned int mali_cur_freq; + +static int on = 1; +static int off; +static int acpi; +static int mem_size = 0x100000; /* 1 MiB */ +static int debug; + +#define MALI_POWER_MASK 0xf + +#define OPT_EQUAL(opt, name) (!strncmp(opt, name, strlen(name))) +#define OPT_INTVAL(opt, name) kstrtoul(opt + strlen(name) + 1, 0, NULL) +#define OPT_STRVAL(opt, name) (opt + strlen(name)) + +static inline char *get_opt_string(const char *this_opt, const char *name) +{ + const char *p; + int i; + char *ret; + + p = OPT_STRVAL(this_opt, name); + i = 0; + while (p[i] && p[i] != ' ' && p[i] != ',') + i++; + ret = kmalloc(i + 1, GFP_KERNEL); + if (ret) { + strncpy(ret, p, i); + ret[i] = '\0'; + } + return ret; +} + +static inline int get_opt_int(const char *this_opt, const char *name, + int *ret) +{ + if (!ret) + return 0; + + if (!OPT_EQUAL(this_opt, name)) + return 0; + + *ret = OPT_INTVAL(this_opt, name); + + return 1; +} + +static inline int get_opt_bool(const char *this_opt, const char *name, + int *ret) +{ + if (!ret) + return 0; + + if (OPT_EQUAL(this_opt, name)) { + if (this_opt[strlen(name)] == '=') + *ret = kstrtoul(this_opt + strlen(name) + 1, + 0, NULL); + else + *ret = 1; + } else { + if (OPT_EQUAL(this_opt, "no") && OPT_EQUAL(this_opt + 2, name)) + *ret = 0; + else + return 0; + } + return 1; +} + +static int __init malipm_setup(char *options) +{ + char *this_opt; + + if (!options || !*options) + return 0; + + /* The syntax is: + * malipm=[<param>][,<param>=<val>] ... + * e.g., + * malipm=on:acpi,mem_size=0x100000 + */ + + while ((this_opt = strsep(&options, ",")) != NULL) { + if (!*this_opt) + continue; + if (get_opt_bool(this_opt, "on", &on)) + ; + else if (get_opt_bool(this_opt, "off", &off)) + ; + else if (get_opt_bool(this_opt, "acpi", &acpi)) + ; + else if (get_opt_int(this_opt, "mem_size", &mem_size)) + ; + else if (get_opt_bool(this_opt, "debug", &debug)) + ; + } + + on = !off; + printk(KERN_DEBUG + "malipm: on %d, off %d, acpi %d, debug %d, %d MiB\n", + on, off, acpi, debug, mem_size >> 20); + + return 0; +} +__setup("malipm=", malipm_setup); + +static int mali_suspend(u32 cores); +static int mali_resume(u32 cores); +static void mali_enable_clock(int enab1le); +static void mali_enable_power(int enable); +static void mali_set_memory_base(unsigned int val); +static void mali_set_memory_size(unsigned int val); +static void mali_set_mem_validation_base(unsigned int val); +static void mali_set_mem_validation_size(unsigned int val); +static void mali_get_memory_base(unsigned int *val); +static void mali_get_memory_size(unsigned int *val); +static void mali_get_mem_validation_base(unsigned int *val); +static void mali_get_mem_validation_size(unsigned int *val); + +/* symbols for ARM's Mali.ko */ +unsigned int mali_memory_base; +EXPORT_SYMBOL(mali_memory_base); +unsigned int mali_memory_size; +EXPORT_SYMBOL(mali_memory_size); +unsigned int mali_mem_validation_base; +EXPORT_SYMBOL(mali_mem_validation_base); +unsigned int mali_mem_validation_size; +EXPORT_SYMBOL(mali_mem_validation_size); +unsigned int mali_ump_secure_id; +EXPORT_SYMBOL(mali_ump_secure_id); +unsigned int (*mali_get_ump_secure_id)(unsigned int addr, unsigned int size); +EXPORT_SYMBOL(mali_get_ump_secure_id); +void (*mali_put_ump_secure_id)(unsigned int ump_id); +EXPORT_SYMBOL(mali_put_ump_secure_id); + +/* PMU */ + +#define REG_MALI400_BASE (0xd8080000 + WMT_MMAP_OFFSET) +#define REG_MALI400_GP (REG_MALI400_BASE) +#define REG_MALI400_L2 (REG_MALI400_BASE + 0x1000) +#define REG_MALI400_PMU (REG_MALI400_BASE + 0x2000) +#define REG_MALI400_MMU_GP (REG_MALI400_BASE + 0x3000) +#define REG_MALI400_MMU_PP0 (REG_MALI400_BASE + 0x4000) +#define REG_MALI400_MMU_PP1 (REG_MALI400_BASE + 0x5000) +#define REG_MALI400_MMU_PP2 (REG_MALI400_BASE + 0x6000) +#define REG_MALI400_MMU_PP3 (REG_MALI400_BASE + 0x7000) +#define REG_MALI400_PP0 (REG_MALI400_BASE + 0x8000) +#define REG_MALI400_PP1 (REG_MALI400_BASE + 0xa000) +#define REG_MALI400_PP2 (REG_MALI400_BASE + 0xc000) +#define REG_MALI400_PP3 (REG_MALI400_BASE + 0xe000) + +#define MMU_DTE_ADDR 0x00 +#define MMU_STATUS 0x04 +#define MMU_COMMAND 0x08 +#define MMU_PAGE_FAULT_ADDR 0x0c +#define MMU_ZAP_ONE_LINE 0x10 +#define MMU_INT_RAWSTAT 0x14 +#define MMU_INT_CLEAR 0x18 +#define MMU_INT_MASK 0x1c +#define MMU_INT_STATUS 0x20 + +#ifdef USE_PMC_POWER_STATUS +#define REG_PMC_BASE (0xd8130000 + WMT_MMAP_OFFSET) +#define REG_MALI_GP_SHUT_OFF_CONTROL (REG_PMC_BASE + 0x0600) +#define REG_MALI_L2C_SHUT_OFF_CONTROL (REG_PMC_BASE + 0x0620) +#define REG_MALI_PP0_SHUT_OFF_CONTROL (REG_PMC_BASE + 0x0624) +#define REG_MALI_PP1_SHUT_OFF_CONTROL (REG_PMC_BASE + 0x0628) + +#define PWR_SEQ_MSK 0x6 +#define PWR_STA_MSK 0xf0 + +#ifndef REG_VAL32 +#define REG_VAL32 REG_GET32 +#endif + +static int wait_powerup(unsigned int reg) +{ + int i = 10; /* 10 * 100 us = 1 ms */ + unsigned int val; + + while (i) { + val = ioread32(reg); + if ((val & PWR_SEQ_MSK) == 0) + break; + udelay(100); + i--; + } + if (i == 0) + printk(KERN_ERR "%s %d: *0x%08x = %08x\n", + __func__, __LINE__, reg, val); + while (i) { + val = ioread32(reg); + if (val & PWR_STA_MSK) + break; + udelay(100); + i--; + } + if (i == 0) + printk(KERN_ERR "%s %d: *0x%08x = %08x\n", + __func__, __LINE__, reg, val); + + return i ? 0 : -1; +} + +static int wait_powerdown(unsigned int reg) +{ + int i = 10; /* 10 * 100 us = 1 ms */ + unsigned int val; + + while (i) { + val = ioread32(reg); + if ((val & PWR_SEQ_MSK) == 0) + break; + udelay(100); + i--; + } + if (i == 0) + printk(KERN_ERR "%s %d: *0x%08x = %08x\n", + __func__, __LINE__, reg, val); + while (i) { + val = ioread32(reg); + if ((val & PWR_STA_MSK) == 0) + break; + udelay(100); + i--; + } + if (i == 0) + printk(KERN_ERR "%s %d: *0x%08x = %08x\n", + __func__, __LINE__, reg, val); + + return i ? 0 : -1; +} +#endif /* USE_PMC_POWER_STATUS */ + +int mali_platform_wait_powerup(int msk) +{ +#ifdef USE_PMC_POWER_STATUS + int err; + + if (debug) + printk(KERN_DEBUG "%s\n", __func__); + + err = 0; + + if (msk & BIT0) + err += wait_powerup(REG_MALI_GP_SHUT_OFF_CONTROL); + if (msk & BIT1) + err += wait_powerup(REG_MALI_L2C_SHUT_OFF_CONTROL); + if (msk & BIT2) + err += wait_powerup(REG_MALI_PP0_SHUT_OFF_CONTROL); + if (msk & BIT3) + err += wait_powerup(REG_MALI_PP1_SHUT_OFF_CONTROL); + + if (err) + printk(KERN_ERR "%s error\n", __func__); + + return err; +#else + return 0; +#endif /* USE_PMC_POWER_STATUS */ +} +EXPORT_SYMBOL(mali_platform_wait_powerup); + +int mali_platform_wait_powerdown(int msk) +{ +#ifdef USE_PMC_POWER_STATUS + int err; + + if (debug) + printk(KERN_DEBUG "%s\n", __func__); + + err = 0; + + if (msk & BIT0) + err += wait_powerdown(REG_MALI_GP_SHUT_OFF_CONTROL); + if (msk & BIT1) + err += wait_powerdown(REG_MALI_L2C_SHUT_OFF_CONTROL); + if (msk & BIT2) + err += wait_powerdown(REG_MALI_PP0_SHUT_OFF_CONTROL); + if (msk & BIT3) + err += wait_powerdown(REG_MALI_PP1_SHUT_OFF_CONTROL); + + if (err) + printk(KERN_ERR "%s error\n", __func__); + + return err; +#else + return 0; +#endif /* USE_PMC_POWER_STATUS */ +} +EXPORT_SYMBOL(mali_platform_wait_powerdown); + +int mali_pmu_power_up(unsigned int msk) +{ + int timeout; + unsigned int pmu0c; + unsigned int pmu08; + unsigned int val; + + val = ioread32(REG_MALI400_PMU + 8); + if ((val & msk) == 0) + return 0; + + pmu08 = ioread32(REG_MALI400_PMU + 8); + pmu0c = ioread32(REG_MALI400_PMU + 0xc); + iowrite32(0, REG_MALI400_PMU + 0xc); + + asm volatile("" : : : "memory"); + + /* PMU_POWER_UP */ + iowrite32(msk, REG_MALI400_PMU); + + asm volatile("" : : : "memory"); + + timeout = 10; + + do { + val = ioread32(REG_MALI400_PMU + 8); + if ((val & msk) == 0) + break; + msleep_interruptible(1); + timeout--; + } while (timeout > 0); + + if (debug) { + val = ioread32(REG_MALI400_PMU + 8); + if (timeout == 0) + printk(KERN_DEBUG "%s: 0x%x, 0x%08x -> 0x%08x: fail\n", + __func__, msk, pmu08, val); + else + printk(KERN_DEBUG "%s: 0x%x, 0x%08x -> 0x%08x: pass\n", + __func__, msk, pmu08, val); + } + + iowrite32(pmu0c, REG_MALI400_PMU + 0xc); + + msleep_interruptible(10); + + if (timeout == 0) { + printk(KERN_DEBUG "mali pmu power up failure\n"); + return -1; + } + + return 0; +} + +int mali_pmu_power_down(unsigned int msk) +{ + unsigned int pmu08; + unsigned int pmu0c; + int timeout; + unsigned int val; + + val = ioread32(REG_MALI400_PMU + 8); + if ((val & msk) == msk) + return 0; + + pmu08 = ioread32(REG_MALI400_PMU + 8); + pmu0c = ioread32(REG_MALI400_PMU + 0xc); + iowrite32(0, REG_MALI400_PMU + 0xc); + + asm volatile("" : : : "memory"); + + /* PMU_POWER_DOWN */ + iowrite32(msk, REG_MALI400_PMU + 4); + + asm volatile("" : : : "memory"); + + timeout = 10; + + do { + val = ioread32(REG_MALI400_PMU + 8); + if ((val & msk) == msk) + break; + msleep_interruptible(1); + timeout--; + } while (timeout > 0); + + if (debug) { + val = ioread32(REG_MALI400_PMU + 8); + if (timeout == 0) + printk(KERN_DEBUG "%s: 0x%x, 0x%08x -> 0x%08x: fail\n", + __func__, msk, pmu08, val); + else + printk(KERN_DEBUG "%s: 0x%x, 0x%08x -> 0x%08x: pass\n", + __func__, msk, pmu08, val); + } + + iowrite32(pmu0c, REG_MALI400_PMU + 0xc); + + msleep_interruptible(10); + + if (timeout == 0) { + printk(KERN_DEBUG "mali pmu power down failure\n"); + return -1; + } + + return 0; +} + +static void mali_show_info(void) +{ + /* GP_CONTR_REG_VERSION */ + printk(KERN_INFO "maligp: version = 0x%08x\n", + ioread32(REG_MALI400_GP + 0x6c)); + + /* PP0_VERSION */ + printk(KERN_INFO "malipp: version = 0x%08x\n", + ioread32(REG_MALI400_PP0 + 0x1000)); +} + +struct mali_device *create_mali_device(void) +{ + struct mali_device *dev; + + dev = kcalloc(1, sizeof(struct mali_device), GFP_KERNEL); + dev->suspend = &mali_suspend; + dev->resume = &mali_resume; + dev->enable_clock = &mali_enable_clock; + dev->enable_power = &mali_enable_power; + dev->set_memory_base = &mali_set_memory_base; + dev->set_memory_size = &mali_set_memory_size; + dev->set_mem_validation_base = &mali_set_mem_validation_base; + dev->set_mem_validation_size = &mali_set_mem_validation_size; + dev->get_memory_base = &mali_get_memory_base; + dev->get_memory_size = &mali_get_memory_size; + dev->get_mem_validation_base = &mali_get_mem_validation_base; + dev->get_mem_validation_size = &mali_get_mem_validation_size; + + return dev; +} +EXPORT_SYMBOL(create_mali_device); + +void release_mali_device(struct mali_device *dev) +{ + kfree(dev); +} +EXPORT_SYMBOL(release_mali_device); + +static int mali_suspend(u32 cores) +{ + if (debug) + printk(KERN_DEBUG "mali_suspend(%d)\n", cores); + + return 0; +} + +static int mali_resume(u32 cores) +{ + if (debug) + printk(KERN_DEBUG "mali_resume(%d)\n", cores); + + return 0; +} + +static void mali_enable_clock(int enable) +{ + int clk_en; + + /* + * if your enable clock with auto_pll_divisor() twice, + * then you have to call it at least twice to disable clock. + * It is really bad. + */ + + if (enable) { + auto_pll_divisor(DEV_MALI, CLK_ENABLE, 0, 0); + if (debug) + printk(KERN_DEBUG "Mali clock enabled\n"); + } else { + do { + clk_en = auto_pll_divisor(DEV_MALI, CLK_DISABLE, 0, 0); + } while (clk_en); + if (debug) + printk(KERN_DEBUG "Mali clock disabled\n"); + } +} + +static void mali_enable_power(int enable) +{ + /* Mali-400's power was always enabled on WM3481. */ +} + +static void mali_set_memory_base(unsigned int val) +{ + spin_lock(&mali_spinlock); + mali_memory_base = val; + spin_unlock(&mali_spinlock); +} + +static void mali_set_memory_size(unsigned int val) +{ + spin_lock(&mali_spinlock); + mali_memory_size = val; + spin_unlock(&mali_spinlock); +} + +static void mali_set_mem_validation_base(unsigned int val) +{ + spin_lock(&mali_spinlock); + mali_mem_validation_base = val; + spin_unlock(&mali_spinlock); +} + +static void mali_set_mem_validation_size(unsigned int val) +{ + spin_lock(&mali_spinlock); + mali_mem_validation_size = val; + spin_unlock(&mali_spinlock); +} + +static void mali_get_memory_base(unsigned int *val) +{ + spin_lock(&mali_spinlock); + *val = mali_memory_base; + spin_unlock(&mali_spinlock); +} + +static void mali_get_memory_size(unsigned int *val) +{ + spin_lock(&mali_spinlock); + *val = mali_memory_size; + spin_unlock(&mali_spinlock); +} + +static void mali_get_mem_validation_base(unsigned int *val) +{ + spin_lock(&mali_spinlock); + *val = mali_mem_validation_base; + spin_unlock(&mali_spinlock); +} + +static void mali_get_mem_validation_size(unsigned int *val) +{ + spin_lock(&mali_spinlock); + *val = mali_mem_validation_size; + spin_unlock(&mali_spinlock); +} + +/* Export functions */ + +int mali_platform_init_impl(void *data) +{ + if (debug) + printk(KERN_DEBUG "mali_platform_init_impl\n"); + + return 0; +} +EXPORT_SYMBOL(mali_platform_init_impl); + +int mali_platform_deinit_impl(void *data) +{ + if (debug) + printk(KERN_DEBUG "mali_platform_deinit_impl\n"); + + return 0; +} +EXPORT_SYMBOL(mali_platform_deinit_impl); + +int mali_platform_powerdown_impl(u32 cores) +{ + unsigned int status; + + if (debug) + printk(KERN_DEBUG "mali_platform_powerdown_impl(%d)\n", cores); + + status = MALI_POWER_MASK; + + if (acpi == 0) + return 0; + +#ifdef MALI_PMU_CONTROL + status = ioread32(REG_MALI400_PMU + 8); + + if ((status & MALI_POWER_MASK) != MALI_POWER_MASK) + mali_pmu_power_down(MALI_POWER_MASK); +#endif + +#ifdef MALI_CLK_CONTROL + spin_lock(&mali_spinlock); + mali_enable_clock(0); + mali_enable_power(0); + spin_unlock(&mali_spinlock); +#endif + + return 0; +} +EXPORT_SYMBOL(mali_platform_powerdown_impl); + +int mali_platform_powerup_impl(u32 cores) +{ + unsigned int status; + + if (debug) + printk(KERN_DEBUG "mali_platform_powerup_impl(%d)\n", cores); + +#ifdef MALI_PMU_CONTROL + status = ioread32(REG_MALI400_PMU + 8); + + /* printk("mali pmu: status = 0x08%x\n", status); */ + + if ((status & MALI_POWER_MASK) != 0) + mali_pmu_power_up(MALI_POWER_MASK); +#else + status = 0; +#endif + +#ifdef MALI_CLK_CONTROL + spin_lock(&mali_spinlock); + mali_enable_power(1); + mali_enable_clock(1); + spin_unlock(&mali_spinlock); +#endif + + return 0; +} +EXPORT_SYMBOL(mali_platform_powerup_impl); + +void mali_gpu_utilization_handler_impl(u32 utilization) +{ + unsigned int freq; + unsigned int oldfreq; + int ret; + + if (acpi < 2 || mali_max_freq == 0) + return; + + utilization = (utilization + 63) & ~63; + oldfreq = mali_cur_freq; + freq = ((utilization * mali_max_freq) >> 8) + 64; + ret = auto_pll_divisor(DEV_MALI, SET_DIV, 2, freq); /* MHz */ + if (ret > 0) + mali_cur_freq = (ret >> 20) + 1; /* MHz */ + printk(KERN_DEBUG "%s: %d, %d -> %d (%d) MHz\n", + __func__, utilization, oldfreq, mali_cur_freq, freq); +} +EXPORT_SYMBOL(mali_gpu_utilization_handler_impl); + +void set_mali_parent_power_domain(struct platform_device *dev) +{ + /* No implemented yet */ +} +EXPORT_SYMBOL(set_mali_parent_power_domain); + +static int __init mali_init(void) +{ + unsigned long smem_start; + unsigned long smem_len; + int err = 0; + int ret; + + spin_lock_init(&mali_spinlock); + + smem_start = num_physpages << PAGE_SHIFT; + + smem_len = mem_size; + + mali_set_memory_base(smem_start); + mali_set_memory_size(smem_len); + mali_set_mem_validation_base(0); + mali_set_mem_validation_size(0); + + mali_ump_secure_id = (unsigned int) -1; + mali_get_ump_secure_id = NULL; + mali_put_ump_secure_id = NULL; + + if (off) + return -1; + + mali_enable_power(1); + mali_enable_clock(1); + + /* Wait for power stable */ + msleep_interruptible(1); + + /* Verify Mali-400 PMU */ + err += mali_pmu_power_down(MALI_POWER_MASK); + if (!err) + err += mali_pmu_power_up(MALI_POWER_MASK); + if (!err) + mali_show_info(); + + if (acpi) + err += mali_pmu_power_down(MALI_POWER_MASK); + + if (acpi > 1) { + ret = auto_pll_divisor(DEV_MALI, GET_FREQ, 0, 0); + if (ret > 0) { + mali_max_freq = (ret >> 20) + 1; /* MHz */ + mali_cur_freq = mali_max_freq; + } + } else { + mali_max_freq = mali_cur_freq = 0; + } + + /* Power on all Mali core at bootup, otherwise Mali driver will fail + * at driver/src/devicedrv/mali/common/mali_pp.c: mali_pp_reset_wait(). + */ + mali_pmu_power_up(MALI_POWER_MASK); + + return err; +} + +static void __exit mali_exit(void) +{ + mali_enable_clock(0); + mali_enable_power(0); +} + +module_init(mali_init); +module_exit(mali_exit); + +MODULE_AUTHOR("WonderMedia Technologies, Inc."); +MODULE_DESCRIPTION("Mali PM Kernel Driver"); +MODULE_LICENSE("GPL"); + |