diff options
Diffstat (limited to 'arch/arm/mach-qcom')
| -rw-r--r-- | arch/arm/mach-qcom/Kconfig | 29 | ||||
| -rw-r--r-- | arch/arm/mach-qcom/Makefile | 5 | ||||
| -rw-r--r-- | arch/arm/mach-qcom/board.c | 28 | ||||
| -rw-r--r-- | arch/arm/mach-qcom/platsmp.c | 378 | ||||
| -rw-r--r-- | arch/arm/mach-qcom/scm-boot.c | 39 | ||||
| -rw-r--r-- | arch/arm/mach-qcom/scm-boot.h | 24 | ||||
| -rw-r--r-- | arch/arm/mach-qcom/scm.c | 299 | ||||
| -rw-r--r-- | arch/arm/mach-qcom/scm.h | 25 | 
8 files changed, 827 insertions, 0 deletions
diff --git a/arch/arm/mach-qcom/Kconfig b/arch/arm/mach-qcom/Kconfig new file mode 100644 index 00000000000..ee5697ba05b --- /dev/null +++ b/arch/arm/mach-qcom/Kconfig @@ -0,0 +1,29 @@ +menuconfig ARCH_QCOM +	bool "Qualcomm Support" if ARCH_MULTI_V7 +	select ARCH_REQUIRE_GPIOLIB +	select ARM_GIC +	select ARM_AMBA +	select CLKSRC_OF +	select PINCTRL +	select QCOM_SCM if SMP +	help +	  Support for Qualcomm's devicetree based systems. + +if ARCH_QCOM + +config ARCH_MSM8X60 +	bool "Enable support for MSM8X60" +	select CLKSRC_QCOM + +config ARCH_MSM8960 +	bool "Enable support for MSM8960" +	select CLKSRC_QCOM + +config ARCH_MSM8974 +	bool "Enable support for MSM8974" +	select HAVE_ARM_ARCH_TIMER + +config QCOM_SCM +	bool + +endif diff --git a/arch/arm/mach-qcom/Makefile b/arch/arm/mach-qcom/Makefile new file mode 100644 index 00000000000..8f756ae1ae3 --- /dev/null +++ b/arch/arm/mach-qcom/Makefile @@ -0,0 +1,5 @@ +obj-y			:= board.o +obj-$(CONFIG_SMP)	+= platsmp.o +obj-$(CONFIG_QCOM_SCM)	+= scm.o scm-boot.o + +CFLAGS_scm.o :=$(call as-instr,.arch_extension sec,-DREQUIRES_SEC=1) diff --git a/arch/arm/mach-qcom/board.c b/arch/arm/mach-qcom/board.c new file mode 100644 index 00000000000..c437a994172 --- /dev/null +++ b/arch/arm/mach-qcom/board.c @@ -0,0 +1,28 @@ +/* Copyright (c) 2010-2014 The Linux Foundation. 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 version 2 and + * only 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. + */ + +#include <linux/init.h> + +#include <asm/mach/arch.h> + +static const char * const qcom_dt_match[] __initconst = { +	"qcom,apq8064", +	"qcom,apq8074-dragonboard", +	"qcom,apq8084", +	"qcom,msm8660-surf", +	"qcom,msm8960-cdp", +	NULL +}; + +DT_MACHINE_START(QCOM_DT, "Qualcomm (Flattened Device Tree)") +	.dt_compat = qcom_dt_match, +MACHINE_END diff --git a/arch/arm/mach-qcom/platsmp.c b/arch/arm/mach-qcom/platsmp.c new file mode 100644 index 00000000000..d6908569eca --- /dev/null +++ b/arch/arm/mach-qcom/platsmp.c @@ -0,0 +1,378 @@ +/* + *  Copyright (C) 2002 ARM Ltd. + *  All Rights Reserved + *  Copyright (c) 2010, Code Aurora Forum. All rights reserved. + *  Copyright (c) 2014 The Linux Foundation. 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 version 2 as + * published by the Free Software Foundation. + */ + +#include <linux/init.h> +#include <linux/errno.h> +#include <linux/delay.h> +#include <linux/device.h> +#include <linux/of.h> +#include <linux/of_address.h> +#include <linux/smp.h> +#include <linux/io.h> + +#include <asm/smp_plat.h> + +#include "scm-boot.h" + +#define VDD_SC1_ARRAY_CLAMP_GFS_CTL	0x35a0 +#define SCSS_CPU1CORE_RESET		0x2d80 +#define SCSS_DBG_STATUS_CORE_PWRDUP	0x2e64 + +#define APCS_CPU_PWR_CTL	0x04 +#define PLL_CLAMP		BIT(8) +#define CORE_PWRD_UP		BIT(7) +#define COREPOR_RST		BIT(5) +#define CORE_RST		BIT(4) +#define L2DT_SLP		BIT(3) +#define CLAMP			BIT(0) + +#define APC_PWR_GATE_CTL	0x14 +#define BHS_CNT_SHIFT		24 +#define LDO_PWR_DWN_SHIFT	16 +#define LDO_BYP_SHIFT		8 +#define BHS_SEG_SHIFT		1 +#define BHS_EN			BIT(0) + +#define APCS_SAW2_VCTL		0x14 +#define APCS_SAW2_2_VCTL	0x1c + +extern void secondary_startup(void); + +static DEFINE_SPINLOCK(boot_lock); + +#ifdef CONFIG_HOTPLUG_CPU +static void __ref qcom_cpu_die(unsigned int cpu) +{ +	wfi(); +} +#endif + +static void qcom_secondary_init(unsigned int cpu) +{ +	/* +	 * Synchronise with the boot thread. +	 */ +	spin_lock(&boot_lock); +	spin_unlock(&boot_lock); +} + +static int scss_release_secondary(unsigned int cpu) +{ +	struct device_node *node; +	void __iomem *base; + +	node = of_find_compatible_node(NULL, NULL, "qcom,gcc-msm8660"); +	if (!node) { +		pr_err("%s: can't find node\n", __func__); +		return -ENXIO; +	} + +	base = of_iomap(node, 0); +	of_node_put(node); +	if (!base) +		return -ENOMEM; + +	writel_relaxed(0, base + VDD_SC1_ARRAY_CLAMP_GFS_CTL); +	writel_relaxed(0, base + SCSS_CPU1CORE_RESET); +	writel_relaxed(3, base + SCSS_DBG_STATUS_CORE_PWRDUP); +	mb(); +	iounmap(base); + +	return 0; +} + +static int kpssv1_release_secondary(unsigned int cpu) +{ +	int ret = 0; +	void __iomem *reg, *saw_reg; +	struct device_node *cpu_node, *acc_node, *saw_node; +	u32 val; + +	cpu_node = of_get_cpu_node(cpu, NULL); +	if (!cpu_node) +		return -ENODEV; + +	acc_node = of_parse_phandle(cpu_node, "qcom,acc", 0); +	if (!acc_node) { +		ret = -ENODEV; +		goto out_acc; +	} + +	saw_node = of_parse_phandle(cpu_node, "qcom,saw", 0); +	if (!saw_node) { +		ret = -ENODEV; +		goto out_saw; +	} + +	reg = of_iomap(acc_node, 0); +	if (!reg) { +		ret = -ENOMEM; +		goto out_acc_map; +	} + +	saw_reg = of_iomap(saw_node, 0); +	if (!saw_reg) { +		ret = -ENOMEM; +		goto out_saw_map; +	} + +	/* Turn on CPU rail */ +	writel_relaxed(0xA4, saw_reg + APCS_SAW2_VCTL); +	mb(); +	udelay(512); + +	/* Krait bring-up sequence */ +	val = PLL_CLAMP | L2DT_SLP | CLAMP; +	writel_relaxed(val, reg + APCS_CPU_PWR_CTL); +	val &= ~L2DT_SLP; +	writel_relaxed(val, reg + APCS_CPU_PWR_CTL); +	mb(); +	ndelay(300); + +	val |= COREPOR_RST; +	writel_relaxed(val, reg + APCS_CPU_PWR_CTL); +	mb(); +	udelay(2); + +	val &= ~CLAMP; +	writel_relaxed(val, reg + APCS_CPU_PWR_CTL); +	mb(); +	udelay(2); + +	val &= ~COREPOR_RST; +	writel_relaxed(val, reg + APCS_CPU_PWR_CTL); +	mb(); +	udelay(100); + +	val |= CORE_PWRD_UP; +	writel_relaxed(val, reg + APCS_CPU_PWR_CTL); +	mb(); + +	iounmap(saw_reg); +out_saw_map: +	iounmap(reg); +out_acc_map: +	of_node_put(saw_node); +out_saw: +	of_node_put(acc_node); +out_acc: +	of_node_put(cpu_node); +	return ret; +} + +static int kpssv2_release_secondary(unsigned int cpu) +{ +	void __iomem *reg; +	struct device_node *cpu_node, *l2_node, *acc_node, *saw_node; +	void __iomem *l2_saw_base; +	unsigned reg_val; +	int ret; + +	cpu_node = of_get_cpu_node(cpu, NULL); +	if (!cpu_node) +		return -ENODEV; + +	acc_node = of_parse_phandle(cpu_node, "qcom,acc", 0); +	if (!acc_node) { +		ret = -ENODEV; +		goto out_acc; +	} + +	l2_node = of_parse_phandle(cpu_node, "next-level-cache", 0); +	if (!l2_node) { +		ret = -ENODEV; +		goto out_l2; +	} + +	saw_node = of_parse_phandle(l2_node, "qcom,saw", 0); +	if (!saw_node) { +		ret = -ENODEV; +		goto out_saw; +	} + +	reg = of_iomap(acc_node, 0); +	if (!reg) { +		ret = -ENOMEM; +		goto out_map; +	} + +	l2_saw_base = of_iomap(saw_node, 0); +	if (!l2_saw_base) { +		ret = -ENOMEM; +		goto out_saw_map; +	} + +	/* Turn on the BHS, turn off LDO Bypass and power down LDO */ +	reg_val = (64 << BHS_CNT_SHIFT) | (0x3f << LDO_PWR_DWN_SHIFT) | BHS_EN; +	writel_relaxed(reg_val, reg + APC_PWR_GATE_CTL); +	mb(); +	/* wait for the BHS to settle */ +	udelay(1); + +	/* Turn on BHS segments */ +	reg_val |= 0x3f << BHS_SEG_SHIFT; +	writel_relaxed(reg_val, reg + APC_PWR_GATE_CTL); +	mb(); +	 /* wait for the BHS to settle */ +	udelay(1); + +	/* Finally turn on the bypass so that BHS supplies power */ +	reg_val |= 0x3f << LDO_BYP_SHIFT; +	writel_relaxed(reg_val, reg + APC_PWR_GATE_CTL); + +	/* enable max phases */ +	writel_relaxed(0x10003, l2_saw_base + APCS_SAW2_2_VCTL); +	mb(); +	udelay(50); + +	reg_val = COREPOR_RST | CLAMP; +	writel_relaxed(reg_val, reg + APCS_CPU_PWR_CTL); +	mb(); +	udelay(2); + +	reg_val &= ~CLAMP; +	writel_relaxed(reg_val, reg + APCS_CPU_PWR_CTL); +	mb(); +	udelay(2); + +	reg_val &= ~COREPOR_RST; +	writel_relaxed(reg_val, reg + APCS_CPU_PWR_CTL); +	mb(); + +	reg_val |= CORE_PWRD_UP; +	writel_relaxed(reg_val, reg + APCS_CPU_PWR_CTL); +	mb(); + +	ret = 0; + +	iounmap(l2_saw_base); +out_saw_map: +	iounmap(reg); +out_map: +	of_node_put(saw_node); +out_saw: +	of_node_put(l2_node); +out_l2: +	of_node_put(acc_node); +out_acc: +	of_node_put(cpu_node); + +	return ret; +} + +static DEFINE_PER_CPU(int, cold_boot_done); + +static int qcom_boot_secondary(unsigned int cpu, int (*func)(unsigned int)) +{ +	int ret = 0; + +	if (!per_cpu(cold_boot_done, cpu)) { +		ret = func(cpu); +		if (!ret) +			per_cpu(cold_boot_done, cpu) = true; +	} + +	/* +	 * set synchronisation state between this boot processor +	 * and the secondary one +	 */ +	spin_lock(&boot_lock); + +	/* +	 * Send the secondary CPU a soft interrupt, thereby causing +	 * the boot monitor to read the system wide flags register, +	 * and branch to the address found there. +	 */ +	arch_send_wakeup_ipi_mask(cpumask_of(cpu)); + +	/* +	 * now the secondary core is starting up let it run its +	 * calibrations, then wait for it to finish +	 */ +	spin_unlock(&boot_lock); + +	return ret; +} + +static int msm8660_boot_secondary(unsigned int cpu, struct task_struct *idle) +{ +	return qcom_boot_secondary(cpu, scss_release_secondary); +} + +static int kpssv1_boot_secondary(unsigned int cpu, struct task_struct *idle) +{ +	return qcom_boot_secondary(cpu, kpssv1_release_secondary); +} + +static int kpssv2_boot_secondary(unsigned int cpu, struct task_struct *idle) +{ +	return qcom_boot_secondary(cpu, kpssv2_release_secondary); +} + +static void __init qcom_smp_prepare_cpus(unsigned int max_cpus) +{ +	int cpu, map; +	unsigned int flags = 0; +	static const int cold_boot_flags[] = { +		0, +		SCM_FLAG_COLDBOOT_CPU1, +		SCM_FLAG_COLDBOOT_CPU2, +		SCM_FLAG_COLDBOOT_CPU3, +	}; + +	for_each_present_cpu(cpu) { +		map = cpu_logical_map(cpu); +		if (WARN_ON(map >= ARRAY_SIZE(cold_boot_flags))) { +			set_cpu_present(cpu, false); +			continue; +		} +		flags |= cold_boot_flags[map]; +	} + +	if (scm_set_boot_addr(virt_to_phys(secondary_startup), flags)) { +		for_each_present_cpu(cpu) { +			if (cpu == smp_processor_id()) +				continue; +			set_cpu_present(cpu, false); +		} +		pr_warn("Failed to set CPU boot address, disabling SMP\n"); +	} +} + +static struct smp_operations smp_msm8660_ops __initdata = { +	.smp_prepare_cpus	= qcom_smp_prepare_cpus, +	.smp_secondary_init	= qcom_secondary_init, +	.smp_boot_secondary	= msm8660_boot_secondary, +#ifdef CONFIG_HOTPLUG_CPU +	.cpu_die		= qcom_cpu_die, +#endif +}; +CPU_METHOD_OF_DECLARE(qcom_smp, "qcom,gcc-msm8660", &smp_msm8660_ops); + +static struct smp_operations qcom_smp_kpssv1_ops __initdata = { +	.smp_prepare_cpus	= qcom_smp_prepare_cpus, +	.smp_secondary_init	= qcom_secondary_init, +	.smp_boot_secondary	= kpssv1_boot_secondary, +#ifdef CONFIG_HOTPLUG_CPU +	.cpu_die		= qcom_cpu_die, +#endif +}; +CPU_METHOD_OF_DECLARE(qcom_smp_kpssv1, "qcom,kpss-acc-v1", &qcom_smp_kpssv1_ops); + +static struct smp_operations qcom_smp_kpssv2_ops __initdata = { +	.smp_prepare_cpus	= qcom_smp_prepare_cpus, +	.smp_secondary_init	= qcom_secondary_init, +	.smp_boot_secondary	= kpssv2_boot_secondary, +#ifdef CONFIG_HOTPLUG_CPU +	.cpu_die		= qcom_cpu_die, +#endif +}; +CPU_METHOD_OF_DECLARE(qcom_smp_kpssv2, "qcom,kpss-acc-v2", &qcom_smp_kpssv2_ops); diff --git a/arch/arm/mach-qcom/scm-boot.c b/arch/arm/mach-qcom/scm-boot.c new file mode 100644 index 00000000000..45cee3e469a --- /dev/null +++ b/arch/arm/mach-qcom/scm-boot.c @@ -0,0 +1,39 @@ +/* Copyright (c) 2010, Code Aurora Forum. 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 version 2 and + * only 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, USA. + */ + +#include <linux/module.h> +#include <linux/slab.h> + +#include "scm.h" +#include "scm-boot.h" + +/* + * Set the cold/warm boot address for one of the CPU cores. + */ +int scm_set_boot_addr(phys_addr_t addr, int flags) +{ +	struct { +		unsigned int flags; +		phys_addr_t  addr; +	} cmd; + +	cmd.addr = addr; +	cmd.flags = flags; +	return scm_call(SCM_SVC_BOOT, SCM_BOOT_ADDR, +			&cmd, sizeof(cmd), NULL, 0); +} +EXPORT_SYMBOL(scm_set_boot_addr); diff --git a/arch/arm/mach-qcom/scm-boot.h b/arch/arm/mach-qcom/scm-boot.h new file mode 100644 index 00000000000..6aabb242817 --- /dev/null +++ b/arch/arm/mach-qcom/scm-boot.h @@ -0,0 +1,24 @@ +/* Copyright (c) 2010, Code Aurora Forum. 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 version 2 and + * only 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. + */ +#ifndef __MACH_SCM_BOOT_H +#define __MACH_SCM_BOOT_H + +#define SCM_BOOT_ADDR			0x1 +#define SCM_FLAG_COLDBOOT_CPU1		0x01 +#define SCM_FLAG_COLDBOOT_CPU2		0x08 +#define SCM_FLAG_COLDBOOT_CPU3		0x20 +#define SCM_FLAG_WARMBOOT_CPU0		0x04 +#define SCM_FLAG_WARMBOOT_CPU1		0x02 + +int scm_set_boot_addr(phys_addr_t addr, int flags); + +#endif diff --git a/arch/arm/mach-qcom/scm.c b/arch/arm/mach-qcom/scm.c new file mode 100644 index 00000000000..c536fd6bf82 --- /dev/null +++ b/arch/arm/mach-qcom/scm.c @@ -0,0 +1,299 @@ +/* Copyright (c) 2010, Code Aurora Forum. 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 version 2 and + * only 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, USA. + */ + +#include <linux/slab.h> +#include <linux/io.h> +#include <linux/module.h> +#include <linux/mutex.h> +#include <linux/errno.h> +#include <linux/err.h> + +#include <asm/cacheflush.h> + +#include "scm.h" + +/* Cache line size for msm8x60 */ +#define CACHELINESIZE 32 + +#define SCM_ENOMEM		-5 +#define SCM_EOPNOTSUPP		-4 +#define SCM_EINVAL_ADDR		-3 +#define SCM_EINVAL_ARG		-2 +#define SCM_ERROR		-1 +#define SCM_INTERRUPTED		1 + +static DEFINE_MUTEX(scm_lock); + +/** + * struct scm_command - one SCM command buffer + * @len: total available memory for command and response + * @buf_offset: start of command buffer + * @resp_hdr_offset: start of response buffer + * @id: command to be executed + * @buf: buffer returned from scm_get_command_buffer() + * + * An SCM command is laid out in memory as follows: + * + *	------------------- <--- struct scm_command + *	| command header  | + *	------------------- <--- scm_get_command_buffer() + *	| command buffer  | + *	------------------- <--- struct scm_response and + *	| response header |      scm_command_to_response() + *	------------------- <--- scm_get_response_buffer() + *	| response buffer | + *	------------------- + * + * There can be arbitrary padding between the headers and buffers so + * you should always use the appropriate scm_get_*_buffer() routines + * to access the buffers in a safe manner. + */ +struct scm_command { +	u32	len; +	u32	buf_offset; +	u32	resp_hdr_offset; +	u32	id; +	u32	buf[0]; +}; + +/** + * struct scm_response - one SCM response buffer + * @len: total available memory for response + * @buf_offset: start of response data relative to start of scm_response + * @is_complete: indicates if the command has finished processing + */ +struct scm_response { +	u32	len; +	u32	buf_offset; +	u32	is_complete; +}; + +/** + * alloc_scm_command() - Allocate an SCM command + * @cmd_size: size of the command buffer + * @resp_size: size of the response buffer + * + * Allocate an SCM command, including enough room for the command + * and response headers as well as the command and response buffers. + * + * Returns a valid &scm_command on success or %NULL if the allocation fails. + */ +static struct scm_command *alloc_scm_command(size_t cmd_size, size_t resp_size) +{ +	struct scm_command *cmd; +	size_t len = sizeof(*cmd) + sizeof(struct scm_response) + cmd_size + +		resp_size; + +	cmd = kzalloc(PAGE_ALIGN(len), GFP_KERNEL); +	if (cmd) { +		cmd->len = len; +		cmd->buf_offset = offsetof(struct scm_command, buf); +		cmd->resp_hdr_offset = cmd->buf_offset + cmd_size; +	} +	return cmd; +} + +/** + * free_scm_command() - Free an SCM command + * @cmd: command to free + * + * Free an SCM command. + */ +static inline void free_scm_command(struct scm_command *cmd) +{ +	kfree(cmd); +} + +/** + * scm_command_to_response() - Get a pointer to a scm_response + * @cmd: command + * + * Returns a pointer to a response for a command. + */ +static inline struct scm_response *scm_command_to_response( +		const struct scm_command *cmd) +{ +	return (void *)cmd + cmd->resp_hdr_offset; +} + +/** + * scm_get_command_buffer() - Get a pointer to a command buffer + * @cmd: command + * + * Returns a pointer to the command buffer of a command. + */ +static inline void *scm_get_command_buffer(const struct scm_command *cmd) +{ +	return (void *)cmd->buf; +} + +/** + * scm_get_response_buffer() - Get a pointer to a response buffer + * @rsp: response + * + * Returns a pointer to a response buffer of a response. + */ +static inline void *scm_get_response_buffer(const struct scm_response *rsp) +{ +	return (void *)rsp + rsp->buf_offset; +} + +static int scm_remap_error(int err) +{ +	switch (err) { +	case SCM_ERROR: +		return -EIO; +	case SCM_EINVAL_ADDR: +	case SCM_EINVAL_ARG: +		return -EINVAL; +	case SCM_EOPNOTSUPP: +		return -EOPNOTSUPP; +	case SCM_ENOMEM: +		return -ENOMEM; +	} +	return -EINVAL; +} + +static u32 smc(u32 cmd_addr) +{ +	int context_id; +	register u32 r0 asm("r0") = 1; +	register u32 r1 asm("r1") = (u32)&context_id; +	register u32 r2 asm("r2") = cmd_addr; +	do { +		asm volatile( +			__asmeq("%0", "r0") +			__asmeq("%1", "r0") +			__asmeq("%2", "r1") +			__asmeq("%3", "r2") +#ifdef REQUIRES_SEC +			".arch_extension sec\n" +#endif +			"smc	#0	@ switch to secure world\n" +			: "=r" (r0) +			: "r" (r0), "r" (r1), "r" (r2) +			: "r3"); +	} while (r0 == SCM_INTERRUPTED); + +	return r0; +} + +static int __scm_call(const struct scm_command *cmd) +{ +	int ret; +	u32 cmd_addr = virt_to_phys(cmd); + +	/* +	 * Flush the entire cache here so callers don't have to remember +	 * to flush the cache when passing physical addresses to the secure +	 * side in the buffer. +	 */ +	flush_cache_all(); +	ret = smc(cmd_addr); +	if (ret < 0) +		ret = scm_remap_error(ret); + +	return ret; +} + +/** + * scm_call() - Send an SCM command + * @svc_id: service identifier + * @cmd_id: command identifier + * @cmd_buf: command buffer + * @cmd_len: length of the command buffer + * @resp_buf: response buffer + * @resp_len: length of the response buffer + * + * Sends a command to the SCM and waits for the command to finish processing. + */ +int scm_call(u32 svc_id, u32 cmd_id, const void *cmd_buf, size_t cmd_len, +		void *resp_buf, size_t resp_len) +{ +	int ret; +	struct scm_command *cmd; +	struct scm_response *rsp; + +	cmd = alloc_scm_command(cmd_len, resp_len); +	if (!cmd) +		return -ENOMEM; + +	cmd->id = (svc_id << 10) | cmd_id; +	if (cmd_buf) +		memcpy(scm_get_command_buffer(cmd), cmd_buf, cmd_len); + +	mutex_lock(&scm_lock); +	ret = __scm_call(cmd); +	mutex_unlock(&scm_lock); +	if (ret) +		goto out; + +	rsp = scm_command_to_response(cmd); +	do { +		u32 start = (u32)rsp; +		u32 end = (u32)scm_get_response_buffer(rsp) + resp_len; +		start &= ~(CACHELINESIZE - 1); +		while (start < end) { +			asm ("mcr p15, 0, %0, c7, c6, 1" : : "r" (start) +			     : "memory"); +			start += CACHELINESIZE; +		} +	} while (!rsp->is_complete); + +	if (resp_buf) +		memcpy(resp_buf, scm_get_response_buffer(rsp), resp_len); +out: +	free_scm_command(cmd); +	return ret; +} +EXPORT_SYMBOL(scm_call); + +u32 scm_get_version(void) +{ +	int context_id; +	static u32 version = -1; +	register u32 r0 asm("r0"); +	register u32 r1 asm("r1"); + +	if (version != -1) +		return version; + +	mutex_lock(&scm_lock); + +	r0 = 0x1 << 8; +	r1 = (u32)&context_id; +	do { +		asm volatile( +			__asmeq("%0", "r0") +			__asmeq("%1", "r1") +			__asmeq("%2", "r0") +			__asmeq("%3", "r1") +#ifdef REQUIRES_SEC +			".arch_extension sec\n" +#endif +			"smc	#0	@ switch to secure world\n" +			: "=r" (r0), "=r" (r1) +			: "r" (r0), "r" (r1) +			: "r2", "r3"); +	} while (r0 == SCM_INTERRUPTED); + +	version = r1; +	mutex_unlock(&scm_lock); + +	return version; +} +EXPORT_SYMBOL(scm_get_version); diff --git a/arch/arm/mach-qcom/scm.h b/arch/arm/mach-qcom/scm.h new file mode 100644 index 00000000000..00b31ea58f2 --- /dev/null +++ b/arch/arm/mach-qcom/scm.h @@ -0,0 +1,25 @@ +/* Copyright (c) 2010, Code Aurora Forum. 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 version 2 and + * only 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. + */ +#ifndef __MACH_SCM_H +#define __MACH_SCM_H + +#define SCM_SVC_BOOT			0x1 +#define SCM_SVC_PIL			0x2 + +extern int scm_call(u32 svc_id, u32 cmd_id, const void *cmd_buf, size_t cmd_len, +		void *resp_buf, size_t resp_len); + +#define SCM_VERSION(major, minor) (((major) << 16) | ((minor) & 0xFF)) + +extern u32 scm_get_version(void); + +#endif  | 
