xref: /openbsd-src/sys/dev/pci/drm/drm_memory.c (revision 850e275390052b330d93020bf619a739a3c277ac)
1 /*-
2  *Copyright 1999 Precision Insight, Inc., Cedar Park, Texas.
3  * Copyright 2000 VA Linux Systems, Inc., Sunnyvale, California.
4  * All Rights Reserved.
5  *
6  * Permission is hereby granted, free of charge, to any person obtaining a
7  * copy of this software and associated documentation files (the "Software"),
8  * to deal in the Software without restriction, including without limitation
9  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
10  * and/or sell copies of the Software, and to permit persons to whom the
11  * Software is furnished to do so, subject to the following conditions:
12  *
13  * The above copyright notice and this permission notice (including the next
14  * paragraph) shall be included in all copies or substantial portions of the
15  * Software.
16  *
17  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
20  * VA LINUX SYSTEMS AND/OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
21  * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
22  * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
23  * OTHER DEALINGS IN THE SOFTWARE.
24  *
25  * Authors:
26  *    Rickard E. (Rik) Faith <faith@valinux.com>
27  *    Gareth Hughes <gareth@valinux.com>
28  *
29  */
30 
31 /** @file drm_memory.c
32  * Wrappers for kernel memory allocation routines, and MTRR management support.
33  *
34  * This file previously implemented a memory consumption tracking system using
35  * the "area" argument for various different types of allocations, but that
36  * has been stripped out for now.
37  */
38 
39 #include "drmP.h"
40 
41 void
42 drm_mem_init(void)
43 {
44 }
45 
46 void
47 drm_mem_uninit(void)
48 {
49 }
50 
51 void*
52 drm_alloc(size_t size, int area)
53 {
54 	return malloc(size, M_DRM, M_NOWAIT);
55 }
56 
57 void *
58 drm_calloc(size_t nmemb, size_t size, int area)
59 {
60 	if (nmemb == 0 || SIZE_MAX / nmemb < size)
61 		return (NULL);
62 	else
63 		return malloc(size * nmemb, M_DRM, M_NOWAIT | M_ZERO);
64 }
65 
66 void *
67 drm_realloc(void *oldpt, size_t oldsize, size_t size, int area)
68 {
69 	void *pt;
70 
71 	pt = malloc(size, M_DRM, M_NOWAIT);
72 	if (pt == NULL)
73 		return NULL;
74 	if (oldpt && oldsize) {
75 		memcpy(pt, oldpt, min(oldsize, size));
76 		free(oldpt, M_DRM);
77 	}
78 	return pt;
79 }
80 
81 void
82 drm_free(void *pt, size_t size, int area)
83 {
84 	if (pt != NULL)
85 		free(pt, M_DRM);
86 }
87 
88 void *
89 drm_ioremap(struct drm_device *dev, drm_local_map_t *map)
90 {
91 	struct vga_pci_bar *bar = NULL;
92 	int i;
93 
94 	DRM_DEBUG("offset: 0x%x size: 0x%x type: %d\n", map->offset, map->size,
95 	    map->type);
96 
97 	if (map->type == _DRM_AGP || map->type == _DRM_FRAME_BUFFER) {
98 	/*
99 	 * there can be multiple agp maps in the same BAR, agp also
100 	 * quite possibly isn't the same as the vga device, just try
101 	 * to map it.
102 	 */
103 		DRM_DEBUG("AGP map\n");
104 		map->bst = dev->pa.pa_memt;
105 		if (bus_space_map(map->bst, map->offset,
106 		    map->size, BUS_SPACE_MAP_LINEAR, &map->bsh)) {
107 			DRM_ERROR("ioremap fail\n");
108 			return (NULL);
109 		}
110 		goto done;
111 	} else {
112 		for (i = 0; i < DRM_MAX_PCI_RESOURCE; ++i) {
113 			bar = vga_pci_bar_info(dev->vga_softc, i);
114 			if (bar == NULL)
115 				continue;
116 
117 			if (bar->base == map->offset) {
118 				DRM_DEBUG("REGISTERS map\n");
119 				map->bsr = vga_pci_bar_map(dev->vga_softc,
120 				    bar->addr, map->size, BUS_SPACE_MAP_LINEAR);
121 				if (map->bsr == NULL) {
122 					DRM_ERROR("ioremap fail\n");
123 					return (NULL);
124 				}
125 				map->bst = map->bsr->bst;
126 				map->bsh = map->bsr->bsh;
127 				goto done;
128 			}
129 		}
130 	}
131 done:
132 	/* handles are still supposed to be kernel virtual addresses */
133 	return bus_space_vaddr(map->bst, map->bsh);
134 }
135 
136 void
137 drm_ioremapfree(drm_local_map_t *map)
138 {
139 	if (map == NULL)
140 		return;
141 
142 	if (map->bsr != NULL)
143 		vga_pci_bar_unmap(map->bsr);
144 	else
145 		bus_space_unmap(map->bst, map->bsh, map->size);
146 }
147 
148 int
149 drm_mtrr_add(unsigned long offset, size_t size, int flags)
150 {
151 #ifndef DRM_NO_MTRR
152 	int act;
153 	struct mem_range_desc mrdesc;
154 
155 	mrdesc.mr_base = offset;
156 	mrdesc.mr_len = size;
157 	mrdesc.mr_flags = flags;
158 	act = MEMRANGE_SET_UPDATE;
159 	strlcpy(mrdesc.mr_owner, "drm", sizeof(mrdesc.mr_owner));
160 	return mem_range_attr_set(&mrdesc, &act);
161 #else
162 	return 0;
163 #endif
164 }
165 
166 int
167 drm_mtrr_del(int __unused handle, unsigned long offset, size_t size, int flags)
168 {
169 #ifndef DRM_NO_MTRR
170 	int act;
171 	struct mem_range_desc mrdesc;
172 
173 	mrdesc.mr_base = offset;
174 	mrdesc.mr_len = size;
175 	mrdesc.mr_flags = flags;
176 	act = MEMRANGE_SET_REMOVE;
177 	strlcpy(mrdesc.mr_owner, "drm", sizeof(mrdesc.mr_owner));
178 	return mem_range_attr_set(&mrdesc, &act);
179 #else
180 	return 0;
181 #endif
182 }
183 
184 u_int8_t
185 drm_read8(drm_local_map_t *map, unsigned long offset)
186 {
187 	u_int8_t *ptr;
188 
189 	switch (map->type) {
190 	case _DRM_SCATTER_GATHER:
191 		ptr = map->handle + offset;
192 		return  (*ptr);
193 
194 	default:
195 		return (bus_space_read_1(map->bst, map->bsh, offset));
196 	}
197 }
198 
199 u_int16_t
200 drm_read16(drm_local_map_t *map, unsigned long offset)
201 {
202 	u_int16_t *ptr;
203 	switch (map->type) {
204 	case _DRM_SCATTER_GATHER:
205 		ptr = map->handle + offset;
206 		return  (*ptr);
207 	default:
208 		return (bus_space_read_2(map->bst, map->bsh, offset));
209 	}
210 }
211 
212 u_int32_t
213 drm_read32(drm_local_map_t *map, unsigned long offset)
214 {
215 	u_int32_t *ptr;
216 	switch (map->type) {
217 	case _DRM_SCATTER_GATHER:
218 		ptr = map->handle + offset;
219 		return  (*ptr);
220 	default:
221 		return (bus_space_read_4(map->bst, map->bsh, offset));
222 	}
223 }
224 
225 void
226 drm_write8(drm_local_map_t *map, unsigned long offset, u_int8_t val)
227 {
228 	u_int8_t *ptr;
229 	switch (map->type) {
230 	case _DRM_SCATTER_GATHER:
231 		ptr = map->handle + offset;
232 		*ptr = val;
233 		break;
234 	default:
235 		bus_space_write_1(map->bst, map->bsh, offset, val);
236 	}
237 }
238 
239 void
240 drm_write16(drm_local_map_t *map, unsigned long offset, u_int16_t val)
241 {
242 	u_int16_t *ptr;
243 	switch (map->type) {
244 	case _DRM_SCATTER_GATHER:
245 		ptr = map->handle + offset;
246 		*ptr = val;
247 		break;
248 	default:
249 		bus_space_write_2(map->bst, map->bsh, offset, val);
250 	}
251 }
252 
253 void
254 drm_write32(drm_local_map_t *map, unsigned long offset, u_int32_t val)
255 {
256 	u_int32_t *ptr;
257 	switch (map->type) {
258 	case _DRM_SCATTER_GATHER:
259 		ptr = map->handle + offset;
260 		*ptr = val;
261 		break;
262 	default:
263 		bus_space_write_4(map->bst, map->bsh, offset, val);
264 	}
265 }
266