1 /*
2  * linux/arch/unicore32/kernel/pwm.c
3  *
4  * Code specific to PKUnity SoC and UniCore ISA
5  *
6  *	Maintained by GUAN Xue-tao <gxt@mprc.pku.edu.cn>
7  *	Copyright (C) 2001-2010 Guan Xuetao
8  *
9  * This program is free software; you can redistribute it and/or modify
10  * it under the terms of the GNU General Public License version 2 as
11  * published by the Free Software Foundation.
12  */
13 
14 #include <linux/module.h>
15 #include <linux/kernel.h>
16 #include <linux/platform_device.h>
17 #include <linux/slab.h>
18 #include <linux/err.h>
19 #include <linux/clk.h>
20 #include <linux/io.h>
21 #include <linux/pwm.h>
22 
23 #include <asm/div64.h>
24 #include <mach/hardware.h>
25 
26 struct pwm_device {
27 	struct list_head	node;
28 	struct platform_device *pdev;
29 
30 	const char	*label;
31 	struct clk	*clk;
32 	int		clk_enabled;
33 
34 	unsigned int	use_count;
35 	unsigned int	pwm_id;
36 };
37 
38 /*
39  * period_ns = 10^9 * (PRESCALE + 1) * (PV + 1) / PWM_CLK_RATE
40  * duty_ns   = 10^9 * (PRESCALE + 1) * DC / PWM_CLK_RATE
41  */
pwm_config(struct pwm_device * pwm,int duty_ns,int period_ns)42 int pwm_config(struct pwm_device *pwm, int duty_ns, int period_ns)
43 {
44 	unsigned long long c;
45 	unsigned long period_cycles, prescale, pv, dc;
46 
47 	if (pwm == NULL || period_ns == 0 || duty_ns > period_ns)
48 		return -EINVAL;
49 
50 	c = clk_get_rate(pwm->clk);
51 	c = c * period_ns;
52 	do_div(c, 1000000000);
53 	period_cycles = c;
54 
55 	if (period_cycles < 1)
56 		period_cycles = 1;
57 	prescale = (period_cycles - 1) / 1024;
58 	pv = period_cycles / (prescale + 1) - 1;
59 
60 	if (prescale > 63)
61 		return -EINVAL;
62 
63 	if (duty_ns == period_ns)
64 		dc = OST_PWMDCCR_FDCYCLE;
65 	else
66 		dc = (pv + 1) * duty_ns / period_ns;
67 
68 	/* NOTE: the clock to PWM has to be enabled first
69 	 * before writing to the registers
70 	 */
71 	clk_enable(pwm->clk);
72 	OST_PWMPWCR = prescale;
73 	OST_PWMDCCR = pv - dc;
74 	OST_PWMPCR  = pv;
75 	clk_disable(pwm->clk);
76 
77 	return 0;
78 }
79 EXPORT_SYMBOL(pwm_config);
80 
pwm_enable(struct pwm_device * pwm)81 int pwm_enable(struct pwm_device *pwm)
82 {
83 	int rc = 0;
84 
85 	if (!pwm->clk_enabled) {
86 		rc = clk_enable(pwm->clk);
87 		if (!rc)
88 			pwm->clk_enabled = 1;
89 	}
90 	return rc;
91 }
92 EXPORT_SYMBOL(pwm_enable);
93 
pwm_disable(struct pwm_device * pwm)94 void pwm_disable(struct pwm_device *pwm)
95 {
96 	if (pwm->clk_enabled) {
97 		clk_disable(pwm->clk);
98 		pwm->clk_enabled = 0;
99 	}
100 }
101 EXPORT_SYMBOL(pwm_disable);
102 
103 static DEFINE_MUTEX(pwm_lock);
104 static LIST_HEAD(pwm_list);
105 
pwm_request(int pwm_id,const char * label)106 struct pwm_device *pwm_request(int pwm_id, const char *label)
107 {
108 	struct pwm_device *pwm;
109 	int found = 0;
110 
111 	mutex_lock(&pwm_lock);
112 
113 	list_for_each_entry(pwm, &pwm_list, node) {
114 		if (pwm->pwm_id == pwm_id) {
115 			found = 1;
116 			break;
117 		}
118 	}
119 
120 	if (found) {
121 		if (pwm->use_count == 0) {
122 			pwm->use_count++;
123 			pwm->label = label;
124 		} else
125 			pwm = ERR_PTR(-EBUSY);
126 	} else
127 		pwm = ERR_PTR(-ENOENT);
128 
129 	mutex_unlock(&pwm_lock);
130 	return pwm;
131 }
132 EXPORT_SYMBOL(pwm_request);
133 
pwm_free(struct pwm_device * pwm)134 void pwm_free(struct pwm_device *pwm)
135 {
136 	mutex_lock(&pwm_lock);
137 
138 	if (pwm->use_count) {
139 		pwm->use_count--;
140 		pwm->label = NULL;
141 	} else
142 		pr_warning("PWM device already freed\n");
143 
144 	mutex_unlock(&pwm_lock);
145 }
146 EXPORT_SYMBOL(pwm_free);
147 
__add_pwm(struct pwm_device * pwm)148 static inline void __add_pwm(struct pwm_device *pwm)
149 {
150 	mutex_lock(&pwm_lock);
151 	list_add_tail(&pwm->node, &pwm_list);
152 	mutex_unlock(&pwm_lock);
153 }
154 
pwm_probe(struct platform_device * pdev,unsigned int pwm_id,struct pwm_device * parent_pwm)155 static struct pwm_device *pwm_probe(struct platform_device *pdev,
156 		unsigned int pwm_id, struct pwm_device *parent_pwm)
157 {
158 	struct pwm_device *pwm;
159 	struct resource *r;
160 	int ret = 0;
161 
162 	pwm = kzalloc(sizeof(struct pwm_device), GFP_KERNEL);
163 	if (pwm == NULL) {
164 		dev_err(&pdev->dev, "failed to allocate memory\n");
165 		return ERR_PTR(-ENOMEM);
166 	}
167 
168 	pwm->clk = clk_get(NULL, "OST_CLK");
169 	if (IS_ERR(pwm->clk)) {
170 		ret = PTR_ERR(pwm->clk);
171 		goto err_free;
172 	}
173 	pwm->clk_enabled = 0;
174 
175 	pwm->use_count = 0;
176 	pwm->pwm_id = pwm_id;
177 	pwm->pdev = pdev;
178 
179 	r = platform_get_resource(pdev, IORESOURCE_MEM, 0);
180 	if (r == NULL) {
181 		dev_err(&pdev->dev, "no memory resource defined\n");
182 		ret = -ENODEV;
183 		goto err_free_clk;
184 	}
185 
186 	r = request_mem_region(r->start, resource_size(r), pdev->name);
187 	if (r == NULL) {
188 		dev_err(&pdev->dev, "failed to request memory resource\n");
189 		ret = -EBUSY;
190 		goto err_free_clk;
191 	}
192 
193 	__add_pwm(pwm);
194 	platform_set_drvdata(pdev, pwm);
195 	return pwm;
196 
197 err_free_clk:
198 	clk_put(pwm->clk);
199 err_free:
200 	kfree(pwm);
201 	return ERR_PTR(ret);
202 }
203 
puv3_pwm_probe(struct platform_device * pdev)204 static int __devinit puv3_pwm_probe(struct platform_device *pdev)
205 {
206 	struct pwm_device *pwm = pwm_probe(pdev, pdev->id, NULL);
207 
208 	if (IS_ERR(pwm))
209 		return PTR_ERR(pwm);
210 
211 	return 0;
212 }
213 
pwm_remove(struct platform_device * pdev)214 static int __devexit pwm_remove(struct platform_device *pdev)
215 {
216 	struct pwm_device *pwm;
217 	struct resource *r;
218 
219 	pwm = platform_get_drvdata(pdev);
220 	if (pwm == NULL)
221 		return -ENODEV;
222 
223 	mutex_lock(&pwm_lock);
224 	list_del(&pwm->node);
225 	mutex_unlock(&pwm_lock);
226 
227 	r = platform_get_resource(pdev, IORESOURCE_MEM, 0);
228 	release_mem_region(r->start, resource_size(r));
229 
230 	clk_put(pwm->clk);
231 	kfree(pwm);
232 	return 0;
233 }
234 
235 static struct platform_driver puv3_pwm_driver = {
236 	.driver		= {
237 		.name	= "PKUnity-v3-PWM",
238 	},
239 	.probe		= puv3_pwm_probe,
240 	.remove		= __devexit_p(pwm_remove),
241 };
242 
pwm_init(void)243 static int __init pwm_init(void)
244 {
245 	int ret = 0;
246 
247 	ret = platform_driver_register(&puv3_pwm_driver);
248 	if (ret) {
249 		printk(KERN_ERR "failed to register puv3_pwm_driver\n");
250 		return ret;
251 	}
252 
253 	return ret;
254 }
255 arch_initcall(pwm_init);
256 
pwm_exit(void)257 static void __exit pwm_exit(void)
258 {
259 	platform_driver_unregister(&puv3_pwm_driver);
260 }
261 module_exit(pwm_exit);
262 
263 MODULE_LICENSE("GPL v2");
264