diff options
Diffstat (limited to 'drivers/char/drm/radeon_mem.c')
-rw-r--r-- | drivers/char/drm/radeon_mem.c | 148 |
1 files changed, 70 insertions, 78 deletions
diff --git a/drivers/char/drm/radeon_mem.c b/drivers/char/drm/radeon_mem.c index 134f894e6e4b..030a6fad0d86 100644 --- a/drivers/char/drm/radeon_mem.c +++ b/drivers/char/drm/radeon_mem.c | |||
@@ -1,7 +1,7 @@ | |||
1 | /* radeon_mem.c -- Simple GART/fb memory manager for radeon -*- linux-c -*- | 1 | /* radeon_mem.c -- Simple GART/fb memory manager for radeon -*- linux-c -*- |
2 | * | 2 | * |
3 | * Copyright (C) The Weather Channel, Inc. 2002. All Rights Reserved. | 3 | * Copyright (C) The Weather Channel, Inc. 2002. All Rights Reserved. |
4 | * | 4 | * |
5 | * The Weather Channel (TM) funded Tungsten Graphics to develop the | 5 | * The Weather Channel (TM) funded Tungsten Graphics to develop the |
6 | * initial release of the Radeon 8500 driver under the XFree86 license. | 6 | * initial release of the Radeon 8500 driver under the XFree86 license. |
7 | * This notice must be preserved. | 7 | * This notice must be preserved. |
@@ -35,16 +35,17 @@ | |||
35 | #include "radeon_drv.h" | 35 | #include "radeon_drv.h" |
36 | 36 | ||
37 | /* Very simple allocator for GART memory, working on a static range | 37 | /* Very simple allocator for GART memory, working on a static range |
38 | * already mapped into each client's address space. | 38 | * already mapped into each client's address space. |
39 | */ | 39 | */ |
40 | 40 | ||
41 | static struct mem_block *split_block(struct mem_block *p, int start, int size, | 41 | static struct mem_block *split_block(struct mem_block *p, int start, int size, |
42 | DRMFILE filp ) | 42 | DRMFILE filp) |
43 | { | 43 | { |
44 | /* Maybe cut off the start of an existing block */ | 44 | /* Maybe cut off the start of an existing block */ |
45 | if (start > p->start) { | 45 | if (start > p->start) { |
46 | struct mem_block *newblock = drm_alloc(sizeof(*newblock), DRM_MEM_BUFS ); | 46 | struct mem_block *newblock = |
47 | if (!newblock) | 47 | drm_alloc(sizeof(*newblock), DRM_MEM_BUFS); |
48 | if (!newblock) | ||
48 | goto out; | 49 | goto out; |
49 | newblock->start = start; | 50 | newblock->start = start; |
50 | newblock->size = p->size - (start - p->start); | 51 | newblock->size = p->size - (start - p->start); |
@@ -56,10 +57,11 @@ static struct mem_block *split_block(struct mem_block *p, int start, int size, | |||
56 | p->size -= newblock->size; | 57 | p->size -= newblock->size; |
57 | p = newblock; | 58 | p = newblock; |
58 | } | 59 | } |
59 | 60 | ||
60 | /* Maybe cut off the end of an existing block */ | 61 | /* Maybe cut off the end of an existing block */ |
61 | if (size < p->size) { | 62 | if (size < p->size) { |
62 | struct mem_block *newblock = drm_alloc(sizeof(*newblock), DRM_MEM_BUFS ); | 63 | struct mem_block *newblock = |
64 | drm_alloc(sizeof(*newblock), DRM_MEM_BUFS); | ||
63 | if (!newblock) | 65 | if (!newblock) |
64 | goto out; | 66 | goto out; |
65 | newblock->start = start + size; | 67 | newblock->start = start + size; |
@@ -72,40 +74,39 @@ static struct mem_block *split_block(struct mem_block *p, int start, int size, | |||
72 | p->size = size; | 74 | p->size = size; |
73 | } | 75 | } |
74 | 76 | ||
75 | out: | 77 | out: |
76 | /* Our block is in the middle */ | 78 | /* Our block is in the middle */ |
77 | p->filp = filp; | 79 | p->filp = filp; |
78 | return p; | 80 | return p; |
79 | } | 81 | } |
80 | 82 | ||
81 | static struct mem_block *alloc_block( struct mem_block *heap, int size, | 83 | static struct mem_block *alloc_block(struct mem_block *heap, int size, |
82 | int align2, DRMFILE filp ) | 84 | int align2, DRMFILE filp) |
83 | { | 85 | { |
84 | struct mem_block *p; | 86 | struct mem_block *p; |
85 | int mask = (1 << align2)-1; | 87 | int mask = (1 << align2) - 1; |
86 | 88 | ||
87 | list_for_each(p, heap) { | 89 | list_for_each(p, heap) { |
88 | int start = (p->start + mask) & ~mask; | 90 | int start = (p->start + mask) & ~mask; |
89 | if (p->filp == 0 && start + size <= p->start + p->size) | 91 | if (p->filp == 0 && start + size <= p->start + p->size) |
90 | return split_block( p, start, size, filp ); | 92 | return split_block(p, start, size, filp); |
91 | } | 93 | } |
92 | 94 | ||
93 | return NULL; | 95 | return NULL; |
94 | } | 96 | } |
95 | 97 | ||
96 | static struct mem_block *find_block( struct mem_block *heap, int start ) | 98 | static struct mem_block *find_block(struct mem_block *heap, int start) |
97 | { | 99 | { |
98 | struct mem_block *p; | 100 | struct mem_block *p; |
99 | 101 | ||
100 | list_for_each(p, heap) | 102 | list_for_each(p, heap) |
101 | if (p->start == start) | 103 | if (p->start == start) |
102 | return p; | 104 | return p; |
103 | 105 | ||
104 | return NULL; | 106 | return NULL; |
105 | } | 107 | } |
106 | 108 | ||
107 | 109 | static void free_block(struct mem_block *p) | |
108 | static void free_block( struct mem_block *p ) | ||
109 | { | 110 | { |
110 | p->filp = NULL; | 111 | p->filp = NULL; |
111 | 112 | ||
@@ -117,7 +118,7 @@ static void free_block( struct mem_block *p ) | |||
117 | p->size += q->size; | 118 | p->size += q->size; |
118 | p->next = q->next; | 119 | p->next = q->next; |
119 | p->next->prev = p; | 120 | p->next->prev = p; |
120 | drm_free(q, sizeof(*q), DRM_MEM_BUFS ); | 121 | drm_free(q, sizeof(*q), DRM_MEM_BUFS); |
121 | } | 122 | } |
122 | 123 | ||
123 | if (p->prev->filp == 0) { | 124 | if (p->prev->filp == 0) { |
@@ -125,7 +126,7 @@ static void free_block( struct mem_block *p ) | |||
125 | q->size += p->size; | 126 | q->size += p->size; |
126 | q->next = p->next; | 127 | q->next = p->next; |
127 | q->next->prev = q; | 128 | q->next->prev = q; |
128 | drm_free(p, sizeof(*q), DRM_MEM_BUFS ); | 129 | drm_free(p, sizeof(*q), DRM_MEM_BUFS); |
129 | } | 130 | } |
130 | } | 131 | } |
131 | 132 | ||
@@ -133,14 +134,14 @@ static void free_block( struct mem_block *p ) | |||
133 | */ | 134 | */ |
134 | static int init_heap(struct mem_block **heap, int start, int size) | 135 | static int init_heap(struct mem_block **heap, int start, int size) |
135 | { | 136 | { |
136 | struct mem_block *blocks = drm_alloc(sizeof(*blocks), DRM_MEM_BUFS ); | 137 | struct mem_block *blocks = drm_alloc(sizeof(*blocks), DRM_MEM_BUFS); |
137 | 138 | ||
138 | if (!blocks) | 139 | if (!blocks) |
139 | return DRM_ERR(ENOMEM); | 140 | return DRM_ERR(ENOMEM); |
140 | 141 | ||
141 | *heap = drm_alloc(sizeof(**heap), DRM_MEM_BUFS ); | 142 | *heap = drm_alloc(sizeof(**heap), DRM_MEM_BUFS); |
142 | if (!*heap) { | 143 | if (!*heap) { |
143 | drm_free( blocks, sizeof(*blocks), DRM_MEM_BUFS ); | 144 | drm_free(blocks, sizeof(*blocks), DRM_MEM_BUFS); |
144 | return DRM_ERR(ENOMEM); | 145 | return DRM_ERR(ENOMEM); |
145 | } | 146 | } |
146 | 147 | ||
@@ -149,16 +150,15 @@ static int init_heap(struct mem_block **heap, int start, int size) | |||
149 | blocks->filp = NULL; | 150 | blocks->filp = NULL; |
150 | blocks->next = blocks->prev = *heap; | 151 | blocks->next = blocks->prev = *heap; |
151 | 152 | ||
152 | memset( *heap, 0, sizeof(**heap) ); | 153 | memset(*heap, 0, sizeof(**heap)); |
153 | (*heap)->filp = (DRMFILE) -1; | 154 | (*heap)->filp = (DRMFILE) - 1; |
154 | (*heap)->next = (*heap)->prev = blocks; | 155 | (*heap)->next = (*heap)->prev = blocks; |
155 | return 0; | 156 | return 0; |
156 | } | 157 | } |
157 | 158 | ||
158 | |||
159 | /* Free all blocks associated with the releasing file. | 159 | /* Free all blocks associated with the releasing file. |
160 | */ | 160 | */ |
161 | void radeon_mem_release( DRMFILE filp, struct mem_block *heap ) | 161 | void radeon_mem_release(DRMFILE filp, struct mem_block *heap) |
162 | { | 162 | { |
163 | struct mem_block *p; | 163 | struct mem_block *p; |
164 | 164 | ||
@@ -166,7 +166,7 @@ void radeon_mem_release( DRMFILE filp, struct mem_block *heap ) | |||
166 | return; | 166 | return; |
167 | 167 | ||
168 | list_for_each(p, heap) { | 168 | list_for_each(p, heap) { |
169 | if (p->filp == filp) | 169 | if (p->filp == filp) |
170 | p->filp = NULL; | 170 | p->filp = NULL; |
171 | } | 171 | } |
172 | 172 | ||
@@ -179,40 +179,37 @@ void radeon_mem_release( DRMFILE filp, struct mem_block *heap ) | |||
179 | p->size += q->size; | 179 | p->size += q->size; |
180 | p->next = q->next; | 180 | p->next = q->next; |
181 | p->next->prev = p; | 181 | p->next->prev = p; |
182 | drm_free(q, sizeof(*q),DRM_MEM_DRIVER); | 182 | drm_free(q, sizeof(*q), DRM_MEM_DRIVER); |
183 | } | 183 | } |
184 | } | 184 | } |
185 | } | 185 | } |
186 | 186 | ||
187 | /* Shutdown. | 187 | /* Shutdown. |
188 | */ | 188 | */ |
189 | void radeon_mem_takedown( struct mem_block **heap ) | 189 | void radeon_mem_takedown(struct mem_block **heap) |
190 | { | 190 | { |
191 | struct mem_block *p; | 191 | struct mem_block *p; |
192 | 192 | ||
193 | if (!*heap) | 193 | if (!*heap) |
194 | return; | 194 | return; |
195 | 195 | ||
196 | for (p = (*heap)->next ; p != *heap ; ) { | 196 | for (p = (*heap)->next; p != *heap;) { |
197 | struct mem_block *q = p; | 197 | struct mem_block *q = p; |
198 | p = p->next; | 198 | p = p->next; |
199 | drm_free(q, sizeof(*q),DRM_MEM_DRIVER); | 199 | drm_free(q, sizeof(*q), DRM_MEM_DRIVER); |
200 | } | 200 | } |
201 | 201 | ||
202 | drm_free( *heap, sizeof(**heap),DRM_MEM_DRIVER ); | 202 | drm_free(*heap, sizeof(**heap), DRM_MEM_DRIVER); |
203 | *heap = NULL; | 203 | *heap = NULL; |
204 | } | 204 | } |
205 | 205 | ||
206 | |||
207 | |||
208 | /* IOCTL HANDLERS */ | 206 | /* IOCTL HANDLERS */ |
209 | 207 | ||
210 | static struct mem_block **get_heap( drm_radeon_private_t *dev_priv, | 208 | static struct mem_block **get_heap(drm_radeon_private_t * dev_priv, int region) |
211 | int region ) | ||
212 | { | 209 | { |
213 | switch( region ) { | 210 | switch (region) { |
214 | case RADEON_MEM_REGION_GART: | 211 | case RADEON_MEM_REGION_GART: |
215 | return &dev_priv->gart_heap; | 212 | return &dev_priv->gart_heap; |
216 | case RADEON_MEM_REGION_FB: | 213 | case RADEON_MEM_REGION_FB: |
217 | return &dev_priv->fb_heap; | 214 | return &dev_priv->fb_heap; |
218 | default: | 215 | default: |
@@ -220,103 +217,98 @@ static struct mem_block **get_heap( drm_radeon_private_t *dev_priv, | |||
220 | } | 217 | } |
221 | } | 218 | } |
222 | 219 | ||
223 | int radeon_mem_alloc( DRM_IOCTL_ARGS ) | 220 | int radeon_mem_alloc(DRM_IOCTL_ARGS) |
224 | { | 221 | { |
225 | DRM_DEVICE; | 222 | DRM_DEVICE; |
226 | drm_radeon_private_t *dev_priv = dev->dev_private; | 223 | drm_radeon_private_t *dev_priv = dev->dev_private; |
227 | drm_radeon_mem_alloc_t alloc; | 224 | drm_radeon_mem_alloc_t alloc; |
228 | struct mem_block *block, **heap; | 225 | struct mem_block *block, **heap; |
229 | 226 | ||
230 | if ( !dev_priv ) { | 227 | if (!dev_priv) { |
231 | DRM_ERROR( "%s called with no initialization\n", __FUNCTION__ ); | 228 | DRM_ERROR("%s called with no initialization\n", __FUNCTION__); |
232 | return DRM_ERR(EINVAL); | 229 | return DRM_ERR(EINVAL); |
233 | } | 230 | } |
234 | 231 | ||
235 | DRM_COPY_FROM_USER_IOCTL( alloc, (drm_radeon_mem_alloc_t __user *)data, | 232 | DRM_COPY_FROM_USER_IOCTL(alloc, (drm_radeon_mem_alloc_t __user *) data, |
236 | sizeof(alloc) ); | 233 | sizeof(alloc)); |
237 | 234 | ||
238 | heap = get_heap( dev_priv, alloc.region ); | 235 | heap = get_heap(dev_priv, alloc.region); |
239 | if (!heap || !*heap) | 236 | if (!heap || !*heap) |
240 | return DRM_ERR(EFAULT); | 237 | return DRM_ERR(EFAULT); |
241 | 238 | ||
242 | /* Make things easier on ourselves: all allocations at least | 239 | /* Make things easier on ourselves: all allocations at least |
243 | * 4k aligned. | 240 | * 4k aligned. |
244 | */ | 241 | */ |
245 | if (alloc.alignment < 12) | 242 | if (alloc.alignment < 12) |
246 | alloc.alignment = 12; | 243 | alloc.alignment = 12; |
247 | 244 | ||
248 | block = alloc_block( *heap, alloc.size, alloc.alignment, | 245 | block = alloc_block(*heap, alloc.size, alloc.alignment, filp); |
249 | filp ); | ||
250 | 246 | ||
251 | if (!block) | 247 | if (!block) |
252 | return DRM_ERR(ENOMEM); | 248 | return DRM_ERR(ENOMEM); |
253 | 249 | ||
254 | if ( DRM_COPY_TO_USER( alloc.region_offset, &block->start, | 250 | if (DRM_COPY_TO_USER(alloc.region_offset, &block->start, sizeof(int))) { |
255 | sizeof(int) ) ) { | 251 | DRM_ERROR("copy_to_user\n"); |
256 | DRM_ERROR( "copy_to_user\n" ); | ||
257 | return DRM_ERR(EFAULT); | 252 | return DRM_ERR(EFAULT); |
258 | } | 253 | } |
259 | 254 | ||
260 | return 0; | 255 | return 0; |
261 | } | 256 | } |
262 | 257 | ||
263 | 258 | int radeon_mem_free(DRM_IOCTL_ARGS) | |
264 | |||
265 | int radeon_mem_free( DRM_IOCTL_ARGS ) | ||
266 | { | 259 | { |
267 | DRM_DEVICE; | 260 | DRM_DEVICE; |
268 | drm_radeon_private_t *dev_priv = dev->dev_private; | 261 | drm_radeon_private_t *dev_priv = dev->dev_private; |
269 | drm_radeon_mem_free_t memfree; | 262 | drm_radeon_mem_free_t memfree; |
270 | struct mem_block *block, **heap; | 263 | struct mem_block *block, **heap; |
271 | 264 | ||
272 | if ( !dev_priv ) { | 265 | if (!dev_priv) { |
273 | DRM_ERROR( "%s called with no initialization\n", __FUNCTION__ ); | 266 | DRM_ERROR("%s called with no initialization\n", __FUNCTION__); |
274 | return DRM_ERR(EINVAL); | 267 | return DRM_ERR(EINVAL); |
275 | } | 268 | } |
276 | 269 | ||
277 | DRM_COPY_FROM_USER_IOCTL( memfree, (drm_radeon_mem_free_t __user *)data, | 270 | DRM_COPY_FROM_USER_IOCTL(memfree, (drm_radeon_mem_free_t __user *) data, |
278 | sizeof(memfree) ); | 271 | sizeof(memfree)); |
279 | 272 | ||
280 | heap = get_heap( dev_priv, memfree.region ); | 273 | heap = get_heap(dev_priv, memfree.region); |
281 | if (!heap || !*heap) | 274 | if (!heap || !*heap) |
282 | return DRM_ERR(EFAULT); | 275 | return DRM_ERR(EFAULT); |
283 | 276 | ||
284 | block = find_block( *heap, memfree.region_offset ); | 277 | block = find_block(*heap, memfree.region_offset); |
285 | if (!block) | 278 | if (!block) |
286 | return DRM_ERR(EFAULT); | 279 | return DRM_ERR(EFAULT); |
287 | 280 | ||
288 | if (block->filp != filp) | 281 | if (block->filp != filp) |
289 | return DRM_ERR(EPERM); | 282 | return DRM_ERR(EPERM); |
290 | 283 | ||
291 | free_block( block ); | 284 | free_block(block); |
292 | return 0; | 285 | return 0; |
293 | } | 286 | } |
294 | 287 | ||
295 | int radeon_mem_init_heap( DRM_IOCTL_ARGS ) | 288 | int radeon_mem_init_heap(DRM_IOCTL_ARGS) |
296 | { | 289 | { |
297 | DRM_DEVICE; | 290 | DRM_DEVICE; |
298 | drm_radeon_private_t *dev_priv = dev->dev_private; | 291 | drm_radeon_private_t *dev_priv = dev->dev_private; |
299 | drm_radeon_mem_init_heap_t initheap; | 292 | drm_radeon_mem_init_heap_t initheap; |
300 | struct mem_block **heap; | 293 | struct mem_block **heap; |
301 | 294 | ||
302 | if ( !dev_priv ) { | 295 | if (!dev_priv) { |
303 | DRM_ERROR( "%s called with no initialization\n", __FUNCTION__ ); | 296 | DRM_ERROR("%s called with no initialization\n", __FUNCTION__); |
304 | return DRM_ERR(EINVAL); | 297 | return DRM_ERR(EINVAL); |
305 | } | 298 | } |
306 | 299 | ||
307 | DRM_COPY_FROM_USER_IOCTL( initheap, (drm_radeon_mem_init_heap_t __user *)data, | 300 | DRM_COPY_FROM_USER_IOCTL(initheap, |
308 | sizeof(initheap) ); | 301 | (drm_radeon_mem_init_heap_t __user *) data, |
302 | sizeof(initheap)); | ||
309 | 303 | ||
310 | heap = get_heap( dev_priv, initheap.region ); | 304 | heap = get_heap(dev_priv, initheap.region); |
311 | if (!heap) | 305 | if (!heap) |
312 | return DRM_ERR(EFAULT); | 306 | return DRM_ERR(EFAULT); |
313 | 307 | ||
314 | if (*heap) { | 308 | if (*heap) { |
315 | DRM_ERROR("heap already initialized?"); | 309 | DRM_ERROR("heap already initialized?"); |
316 | return DRM_ERR(EFAULT); | 310 | return DRM_ERR(EFAULT); |
317 | } | 311 | } |
318 | |||
319 | return init_heap( heap, initheap.start, initheap.size ); | ||
320 | } | ||
321 | |||
322 | 312 | ||
313 | return init_heap(heap, initheap.start, initheap.size); | ||
314 | } | ||