aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/gpu/drm/nouveau/nvkm/subdev/gpio/base.c
diff options
context:
space:
mode:
authorBen Skeggs <bskeggs@redhat.com>2015-01-13 07:13:14 -0500
committerBen Skeggs <bskeggs@redhat.com>2015-01-21 21:15:10 -0500
commitc39f472e9f14e49a9bc091977ced0ec45fc00c57 (patch)
tree75af3291cccda2482913cc0044888a8a86f4841b /drivers/gpu/drm/nouveau/nvkm/subdev/gpio/base.c
parent055a65d5987a7f246c3fc2297158286882dbdbcf (diff)
drm/nouveau: remove symlinks, move core/ to nvkm/ (no code changes)
The symlinks were annoying some people, and they're not used anywhere else in the kernel tree. The include directory structure has been changed so that symlinks aren't needed anymore. NVKM has been moved from core/ to nvkm/ to make it more obvious as to what the directory is for, and as some minor prep for when NVKM gets split out into its own module (virt) at a later date. Signed-off-by: Ben Skeggs <bskeggs@redhat.com>
Diffstat (limited to 'drivers/gpu/drm/nouveau/nvkm/subdev/gpio/base.c')
-rw-r--r--drivers/gpu/drm/nouveau/nvkm/subdev/gpio/base.c255
1 files changed, 255 insertions, 0 deletions
diff --git a/drivers/gpu/drm/nouveau/nvkm/subdev/gpio/base.c b/drivers/gpu/drm/nouveau/nvkm/subdev/gpio/base.c
new file mode 100644
index 000000000000..7ad99b763f4c
--- /dev/null
+++ b/drivers/gpu/drm/nouveau/nvkm/subdev/gpio/base.c
@@ -0,0 +1,255 @@
1/*
2 * Copyright 2011 Red Hat Inc.
3 *
4 * Permission is hereby granted, free of charge, to any person obtaining a
5 * copy of this software and associated documentation files (the "Software"),
6 * to deal in the Software without restriction, including without limitation
7 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8 * and/or sell copies of the Software, and to permit persons to whom the
9 * Software is furnished to do so, subject to the following conditions:
10 *
11 * The above copyright notice and this permission notice shall be included in
12 * all copies or substantial portions of the Software.
13 *
14 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
17 * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
18 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20 * OTHER DEALINGS IN THE SOFTWARE.
21 *
22 * Authors: Ben Skeggs
23 */
24
25#include <subdev/bios.h>
26#include <subdev/bios/gpio.h>
27
28#include "priv.h"
29
30static int
31nouveau_gpio_drive(struct nouveau_gpio *gpio,
32 int idx, int line, int dir, int out)
33{
34 const struct nouveau_gpio_impl *impl = (void *)nv_object(gpio)->oclass;
35 return impl->drive ? impl->drive(gpio, line, dir, out) : -ENODEV;
36}
37
38static int
39nouveau_gpio_sense(struct nouveau_gpio *gpio, int idx, int line)
40{
41 const struct nouveau_gpio_impl *impl = (void *)nv_object(gpio)->oclass;
42 return impl->sense ? impl->sense(gpio, line) : -ENODEV;
43}
44
45static int
46nouveau_gpio_find(struct nouveau_gpio *gpio, int idx, u8 tag, u8 line,
47 struct dcb_gpio_func *func)
48{
49 struct nouveau_bios *bios = nouveau_bios(gpio);
50 u8 ver, len;
51 u16 data;
52
53 if (line == 0xff && tag == 0xff)
54 return -EINVAL;
55
56 data = dcb_gpio_match(bios, idx, tag, line, &ver, &len, func);
57 if (data)
58 return 0;
59
60 /* Apple iMac G4 NV18 */
61 if (nv_device_match(nv_object(gpio), 0x0189, 0x10de, 0x0010)) {
62 if (tag == DCB_GPIO_TVDAC0) {
63 *func = (struct dcb_gpio_func) {
64 .func = DCB_GPIO_TVDAC0,
65 .line = 4,
66 .log[0] = 0,
67 .log[1] = 1,
68 };
69 return 0;
70 }
71 }
72
73 return -ENOENT;
74}
75
76static int
77nouveau_gpio_set(struct nouveau_gpio *gpio, int idx, u8 tag, u8 line, int state)
78{
79 struct dcb_gpio_func func;
80 int ret;
81
82 ret = nouveau_gpio_find(gpio, idx, tag, line, &func);
83 if (ret == 0) {
84 int dir = !!(func.log[state] & 0x02);
85 int out = !!(func.log[state] & 0x01);
86 ret = nouveau_gpio_drive(gpio, idx, func.line, dir, out);
87 }
88
89 return ret;
90}
91
92static int
93nouveau_gpio_get(struct nouveau_gpio *gpio, int idx, u8 tag, u8 line)
94{
95 struct dcb_gpio_func func;
96 int ret;
97
98 ret = nouveau_gpio_find(gpio, idx, tag, line, &func);
99 if (ret == 0) {
100 ret = nouveau_gpio_sense(gpio, idx, func.line);
101 if (ret >= 0)
102 ret = (ret == (func.log[1] & 1));
103 }
104
105 return ret;
106}
107
108static void
109nouveau_gpio_intr_fini(struct nvkm_event *event, int type, int index)
110{
111 struct nouveau_gpio *gpio = container_of(event, typeof(*gpio), event);
112 const struct nouveau_gpio_impl *impl = (void *)nv_object(gpio)->oclass;
113 impl->intr_mask(gpio, type, 1 << index, 0);
114}
115
116static void
117nouveau_gpio_intr_init(struct nvkm_event *event, int type, int index)
118{
119 struct nouveau_gpio *gpio = container_of(event, typeof(*gpio), event);
120 const struct nouveau_gpio_impl *impl = (void *)nv_object(gpio)->oclass;
121 impl->intr_mask(gpio, type, 1 << index, 1 << index);
122}
123
124static int
125nouveau_gpio_intr_ctor(struct nouveau_object *object, void *data, u32 size,
126 struct nvkm_notify *notify)
127{
128 struct nvkm_gpio_ntfy_req *req = data;
129 if (!WARN_ON(size != sizeof(*req))) {
130 notify->size = sizeof(struct nvkm_gpio_ntfy_rep);
131 notify->types = req->mask;
132 notify->index = req->line;
133 return 0;
134 }
135 return -EINVAL;
136}
137
138static void
139nouveau_gpio_intr(struct nouveau_subdev *subdev)
140{
141 struct nouveau_gpio *gpio = nouveau_gpio(subdev);
142 const struct nouveau_gpio_impl *impl = (void *)nv_object(gpio)->oclass;
143 u32 hi, lo, i;
144
145 impl->intr_stat(gpio, &hi, &lo);
146
147 for (i = 0; (hi | lo) && i < impl->lines; i++) {
148 struct nvkm_gpio_ntfy_rep rep = {
149 .mask = (NVKM_GPIO_HI * !!(hi & (1 << i))) |
150 (NVKM_GPIO_LO * !!(lo & (1 << i))),
151 };
152 nvkm_event_send(&gpio->event, rep.mask, i, &rep, sizeof(rep));
153 }
154}
155
156static const struct nvkm_event_func
157nouveau_gpio_intr_func = {
158 .ctor = nouveau_gpio_intr_ctor,
159 .init = nouveau_gpio_intr_init,
160 .fini = nouveau_gpio_intr_fini,
161};
162
163int
164_nouveau_gpio_fini(struct nouveau_object *object, bool suspend)
165{
166 const struct nouveau_gpio_impl *impl = (void *)object->oclass;
167 struct nouveau_gpio *gpio = nouveau_gpio(object);
168 u32 mask = (1 << impl->lines) - 1;
169
170 impl->intr_mask(gpio, NVKM_GPIO_TOGGLED, mask, 0);
171 impl->intr_stat(gpio, &mask, &mask);
172
173 return nouveau_subdev_fini(&gpio->base, suspend);
174}
175
176static struct dmi_system_id gpio_reset_ids[] = {
177 {
178 .ident = "Apple Macbook 10,1",
179 .matches = {
180 DMI_MATCH(DMI_SYS_VENDOR, "Apple Inc."),
181 DMI_MATCH(DMI_PRODUCT_NAME, "MacBookPro10,1"),
182 }
183 },
184 { }
185};
186
187int
188_nouveau_gpio_init(struct nouveau_object *object)
189{
190 struct nouveau_gpio *gpio = nouveau_gpio(object);
191 int ret;
192
193 ret = nouveau_subdev_init(&gpio->base);
194 if (ret)
195 return ret;
196
197 if (gpio->reset && dmi_check_system(gpio_reset_ids))
198 gpio->reset(gpio, DCB_GPIO_UNUSED);
199
200 return ret;
201}
202
203void
204_nouveau_gpio_dtor(struct nouveau_object *object)
205{
206 struct nouveau_gpio *gpio = (void *)object;
207 nvkm_event_fini(&gpio->event);
208 nouveau_subdev_destroy(&gpio->base);
209}
210
211int
212nouveau_gpio_create_(struct nouveau_object *parent,
213 struct nouveau_object *engine,
214 struct nouveau_oclass *oclass,
215 int length, void **pobject)
216{
217 const struct nouveau_gpio_impl *impl = (void *)oclass;
218 struct nouveau_gpio *gpio;
219 int ret;
220
221 ret = nouveau_subdev_create_(parent, engine, oclass, 0, "GPIO", "gpio",
222 length, pobject);
223 gpio = *pobject;
224 if (ret)
225 return ret;
226
227 gpio->find = nouveau_gpio_find;
228 gpio->set = nouveau_gpio_set;
229 gpio->get = nouveau_gpio_get;
230 gpio->reset = impl->reset;
231
232 ret = nvkm_event_init(&nouveau_gpio_intr_func, 2, impl->lines,
233 &gpio->event);
234 if (ret)
235 return ret;
236
237 nv_subdev(gpio)->intr = nouveau_gpio_intr;
238 return 0;
239}
240
241int
242_nouveau_gpio_ctor(struct nouveau_object *parent, struct nouveau_object *engine,
243 struct nouveau_oclass *oclass, void *data, u32 size,
244 struct nouveau_object **pobject)
245{
246 struct nouveau_gpio *gpio;
247 int ret;
248
249 ret = nouveau_gpio_create(parent, engine, oclass, &gpio);
250 *pobject = nv_object(gpio);
251 if (ret)
252 return ret;
253
254 return 0;
255}