2 * Copyright (c) 2013, The Linux Foundation. All rights reserved.
3 * Copyright (c) 2015, Sony Mobile Communications AB
5 * This software is licensed under the terms of the GNU General Public
6 * License version 2, as published by the Free Software Foundation, and
7 * may be copied, distributed, and modified under those terms.
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
15 #include <linux/hwspinlock.h>
17 #include <linux/kernel.h>
18 #include <linux/mfd/syscon.h>
19 #include <linux/module.h>
21 #include <linux/of_device.h>
22 #include <linux/platform_device.h>
23 #include <linux/pm_runtime.h>
24 #include <linux/regmap.h>
26 #include "hwspinlock_internal.h"
28 #define QCOM_MUTEX_APPS_PROC_ID 1
29 #define QCOM_MUTEX_NUM_LOCKS 32
31 static int qcom_hwspinlock_trylock(struct hwspinlock
*lock
)
33 struct regmap_field
*field
= lock
->priv
;
37 ret
= regmap_field_write(field
, QCOM_MUTEX_APPS_PROC_ID
);
41 ret
= regmap_field_read(field
, &lock_owner
);
45 return lock_owner
== QCOM_MUTEX_APPS_PROC_ID
;
48 static void qcom_hwspinlock_unlock(struct hwspinlock
*lock
)
50 struct regmap_field
*field
= lock
->priv
;
54 ret
= regmap_field_read(field
, &lock_owner
);
56 pr_err("%s: unable to query spinlock owner\n", __func__
);
60 if (lock_owner
!= QCOM_MUTEX_APPS_PROC_ID
) {
61 pr_err("%s: spinlock not owned by us (actual owner is %d)\n",
62 __func__
, lock_owner
);
65 ret
= regmap_field_write(field
, 0);
67 pr_err("%s: failed to unlock spinlock\n", __func__
);
70 static const struct hwspinlock_ops qcom_hwspinlock_ops
= {
71 .trylock
= qcom_hwspinlock_trylock
,
72 .unlock
= qcom_hwspinlock_unlock
,
75 static const struct of_device_id qcom_hwspinlock_of_match
[] = {
76 { .compatible
= "qcom,sfpb-mutex" },
77 { .compatible
= "qcom,tcsr-mutex" },
80 MODULE_DEVICE_TABLE(of
, qcom_hwspinlock_of_match
);
82 static int qcom_hwspinlock_probe(struct platform_device
*pdev
)
84 struct hwspinlock_device
*bank
;
85 struct device_node
*syscon
;
86 struct reg_field field
;
87 struct regmap
*regmap
;
94 syscon
= of_parse_phandle(pdev
->dev
.of_node
, "syscon", 0);
96 dev_err(&pdev
->dev
, "no syscon property\n");
100 regmap
= syscon_node_to_regmap(syscon
);
102 return PTR_ERR(regmap
);
104 ret
= of_property_read_u32_index(pdev
->dev
.of_node
, "syscon", 1, &base
);
106 dev_err(&pdev
->dev
, "no offset in syscon\n");
110 ret
= of_property_read_u32_index(pdev
->dev
.of_node
, "syscon", 2, &stride
);
112 dev_err(&pdev
->dev
, "no stride syscon\n");
116 array_size
= QCOM_MUTEX_NUM_LOCKS
* sizeof(struct hwspinlock
);
117 bank
= devm_kzalloc(&pdev
->dev
, sizeof(*bank
) + array_size
, GFP_KERNEL
);
121 platform_set_drvdata(pdev
, bank
);
123 for (i
= 0; i
< QCOM_MUTEX_NUM_LOCKS
; i
++) {
124 field
.reg
= base
+ i
* stride
;
128 bank
->lock
[i
].priv
= devm_regmap_field_alloc(&pdev
->dev
,
132 pm_runtime_enable(&pdev
->dev
);
134 ret
= hwspin_lock_register(bank
, &pdev
->dev
, &qcom_hwspinlock_ops
,
135 0, QCOM_MUTEX_NUM_LOCKS
);
137 pm_runtime_disable(&pdev
->dev
);
142 static int qcom_hwspinlock_remove(struct platform_device
*pdev
)
144 struct hwspinlock_device
*bank
= platform_get_drvdata(pdev
);
147 ret
= hwspin_lock_unregister(bank
);
149 dev_err(&pdev
->dev
, "%s failed: %d\n", __func__
, ret
);
153 pm_runtime_disable(&pdev
->dev
);
158 static struct platform_driver qcom_hwspinlock_driver
= {
159 .probe
= qcom_hwspinlock_probe
,
160 .remove
= qcom_hwspinlock_remove
,
162 .name
= "qcom_hwspinlock",
163 .of_match_table
= qcom_hwspinlock_of_match
,
167 static int __init
qcom_hwspinlock_init(void)
169 return platform_driver_register(&qcom_hwspinlock_driver
);
171 /* board init code might need to reserve hwspinlocks for predefined purposes */
172 postcore_initcall(qcom_hwspinlock_init
);
174 static void __exit
qcom_hwspinlock_exit(void)
176 platform_driver_unregister(&qcom_hwspinlock_driver
);
178 module_exit(qcom_hwspinlock_exit
);
180 MODULE_LICENSE("GPL v2");
181 MODULE_DESCRIPTION("Hardware spinlock driver for Qualcomm SoCs");