1  /*
2   * Copyright 2014 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 <bskeggs@redhat.com>
23   */
24  
25  #include <nvif/object.h>
26  #include <nvif/client.h>
27  #include <nvif/driver.h>
28  #include <nvif/ioctl.h>
29  
30  int
nvif_object_ioctl(struct nvif_object * object,void * data,u32 size,void ** hack)31  nvif_object_ioctl(struct nvif_object *object, void *data, u32 size, void **hack)
32  {
33  	struct nvif_client *client = object->client;
34  	union {
35  		struct nvif_ioctl_v0 v0;
36  	} *args = data;
37  
38  	if (size >= sizeof(*args) && args->v0.version == 0) {
39  		if (object != &client->object)
40  			args->v0.object = nvif_handle(object);
41  		else
42  			args->v0.object = 0;
43  	} else
44  		return -ENOSYS;
45  
46  	return client->driver->ioctl(client->object.priv, data, size, hack);
47  }
48  
49  void
nvif_object_sclass_put(struct nvif_sclass ** psclass)50  nvif_object_sclass_put(struct nvif_sclass **psclass)
51  {
52  	kfree(*psclass);
53  	*psclass = NULL;
54  }
55  
56  int
nvif_object_sclass_get(struct nvif_object * object,struct nvif_sclass ** psclass)57  nvif_object_sclass_get(struct nvif_object *object, struct nvif_sclass **psclass)
58  {
59  	struct {
60  		struct nvif_ioctl_v0 ioctl;
61  		struct nvif_ioctl_sclass_v0 sclass;
62  	} *args = NULL;
63  	int ret, cnt = 0, i;
64  	u32 size;
65  
66  	while (1) {
67  		size = sizeof(*args) + cnt * sizeof(args->sclass.oclass[0]);
68  		if (!(args = kmalloc(size, GFP_KERNEL)))
69  			return -ENOMEM;
70  		args->ioctl.version = 0;
71  		args->ioctl.type = NVIF_IOCTL_V0_SCLASS;
72  		args->sclass.version = 0;
73  		args->sclass.count = cnt;
74  
75  		ret = nvif_object_ioctl(object, args, size, NULL);
76  		if (ret == 0 && args->sclass.count <= cnt)
77  			break;
78  		cnt = args->sclass.count;
79  		kfree(args);
80  		if (ret != 0)
81  			return ret;
82  	}
83  
84  	*psclass = kcalloc(args->sclass.count, sizeof(**psclass), GFP_KERNEL);
85  	if (*psclass) {
86  		for (i = 0; i < args->sclass.count; i++) {
87  			(*psclass)[i].oclass = args->sclass.oclass[i].oclass;
88  			(*psclass)[i].minver = args->sclass.oclass[i].minver;
89  			(*psclass)[i].maxver = args->sclass.oclass[i].maxver;
90  		}
91  		ret = args->sclass.count;
92  	} else {
93  		ret = -ENOMEM;
94  	}
95  
96  	kfree(args);
97  	return ret;
98  }
99  
100  int
nvif_object_mthd(struct nvif_object * object,u32 mthd,void * data,u32 size)101  nvif_object_mthd(struct nvif_object *object, u32 mthd, void *data, u32 size)
102  {
103  	struct {
104  		struct nvif_ioctl_v0 ioctl;
105  		struct nvif_ioctl_mthd_v0 mthd;
106  	} *args;
107  	u32 args_size;
108  	u8 stack[128];
109  	int ret;
110  
111  	if (check_add_overflow(sizeof(*args), size, &args_size))
112  		return -ENOMEM;
113  
114  	if (args_size > sizeof(stack)) {
115  		args = kmalloc(args_size, GFP_KERNEL);
116  		if (!args)
117  			return -ENOMEM;
118  	} else {
119  		args = (void *)stack;
120  	}
121  	args->ioctl.version = 0;
122  	args->ioctl.type = NVIF_IOCTL_V0_MTHD;
123  	args->mthd.version = 0;
124  	args->mthd.method = mthd;
125  
126  	memcpy(args->mthd.data, data, size);
127  	ret = nvif_object_ioctl(object, args, args_size, NULL);
128  	memcpy(data, args->mthd.data, size);
129  	if (args != (void *)stack)
130  		kfree(args);
131  	return ret;
132  }
133  
134  void
nvif_object_unmap_handle(struct nvif_object * object)135  nvif_object_unmap_handle(struct nvif_object *object)
136  {
137  	struct {
138  		struct nvif_ioctl_v0 ioctl;
139  		struct nvif_ioctl_unmap unmap;
140  	} args = {
141  		.ioctl.type = NVIF_IOCTL_V0_UNMAP,
142  	};
143  
144  	nvif_object_ioctl(object, &args, sizeof(args), NULL);
145  }
146  
147  int
nvif_object_map_handle(struct nvif_object * object,void * argv,u32 argc,u64 * handle,u64 * length)148  nvif_object_map_handle(struct nvif_object *object, void *argv, u32 argc,
149  		       u64 *handle, u64 *length)
150  {
151  	struct {
152  		struct nvif_ioctl_v0 ioctl;
153  		struct nvif_ioctl_map_v0 map;
154  	} *args;
155  	u32 argn = sizeof(*args) + argc;
156  	int ret, maptype;
157  
158  	if (!(args = kzalloc(argn, GFP_KERNEL)))
159  		return -ENOMEM;
160  	args->ioctl.type = NVIF_IOCTL_V0_MAP;
161  	memcpy(args->map.data, argv, argc);
162  
163  	ret = nvif_object_ioctl(object, args, argn, NULL);
164  	*handle = args->map.handle;
165  	*length = args->map.length;
166  	maptype = args->map.type;
167  	kfree(args);
168  	return ret ? ret : (maptype == NVIF_IOCTL_MAP_V0_IO);
169  }
170  
171  void
nvif_object_unmap(struct nvif_object * object)172  nvif_object_unmap(struct nvif_object *object)
173  {
174  	struct nvif_client *client = object->client;
175  	if (object->map.ptr) {
176  		if (object->map.size) {
177  			client->driver->unmap(client, object->map.ptr,
178  						      object->map.size);
179  			object->map.size = 0;
180  		}
181  		object->map.ptr = NULL;
182  		nvif_object_unmap_handle(object);
183  	}
184  }
185  
186  int
nvif_object_map(struct nvif_object * object,void * argv,u32 argc)187  nvif_object_map(struct nvif_object *object, void *argv, u32 argc)
188  {
189  	struct nvif_client *client = object->client;
190  	u64 handle, length;
191  	int ret = nvif_object_map_handle(object, argv, argc, &handle, &length);
192  	if (ret >= 0) {
193  		if (ret) {
194  			object->map.ptr = client->driver->map(client,
195  							      handle,
196  							      length);
197  			if (ret = -ENOMEM, object->map.ptr) {
198  				object->map.size = length;
199  				return 0;
200  			}
201  		} else {
202  			object->map.ptr = (void *)(unsigned long)handle;
203  			return 0;
204  		}
205  		nvif_object_unmap_handle(object);
206  	}
207  	return ret;
208  }
209  
210  void
nvif_object_dtor(struct nvif_object * object)211  nvif_object_dtor(struct nvif_object *object)
212  {
213  	struct {
214  		struct nvif_ioctl_v0 ioctl;
215  		struct nvif_ioctl_del del;
216  	} args = {
217  		.ioctl.type = NVIF_IOCTL_V0_DEL,
218  	};
219  
220  	if (!nvif_object_constructed(object))
221  		return;
222  
223  	nvif_object_unmap(object);
224  	nvif_object_ioctl(object, &args, sizeof(args), NULL);
225  	object->client = NULL;
226  }
227  
228  int
nvif_object_ctor(struct nvif_object * parent,const char * name,u32 handle,s32 oclass,void * data,u32 size,struct nvif_object * object)229  nvif_object_ctor(struct nvif_object *parent, const char *name, u32 handle,
230  		 s32 oclass, void *data, u32 size, struct nvif_object *object)
231  {
232  	struct {
233  		struct nvif_ioctl_v0 ioctl;
234  		struct nvif_ioctl_new_v0 new;
235  	} *args;
236  	int ret = 0;
237  
238  	object->client = NULL;
239  	object->name = name ? name : "nvifObject";
240  	object->handle = handle;
241  	object->oclass = oclass;
242  	object->map.ptr = NULL;
243  	object->map.size = 0;
244  
245  	if (parent) {
246  		u32 args_size;
247  
248  		if (check_add_overflow(sizeof(*args), size, &args_size)) {
249  			nvif_object_dtor(object);
250  			return -ENOMEM;
251  		}
252  
253  		args = kmalloc(args_size, GFP_KERNEL);
254  		if (!args) {
255  			nvif_object_dtor(object);
256  			return -ENOMEM;
257  		}
258  
259  		object->parent = parent->parent;
260  
261  		args->ioctl.version = 0;
262  		args->ioctl.type = NVIF_IOCTL_V0_NEW;
263  		args->new.version = 0;
264  		args->new.object = nvif_handle(object);
265  		args->new.handle = handle;
266  		args->new.oclass = oclass;
267  
268  		memcpy(args->new.data, data, size);
269  		ret = nvif_object_ioctl(parent, args, args_size, &object->priv);
270  		memcpy(data, args->new.data, size);
271  		kfree(args);
272  		if (ret == 0)
273  			object->client = parent->client;
274  	}
275  
276  	if (ret)
277  		nvif_object_dtor(object);
278  	return ret;
279  }
280