1*0035d21cSmiao chen - Sun Microsystems - Beijing China /*
2*0035d21cSmiao chen - Sun Microsystems - Beijing China *
3*0035d21cSmiao chen - Sun Microsystems - Beijing China * Copyright 2006 Tungsten Graphics, Inc., Bismarck, ND., USA.
4*0035d21cSmiao chen - Sun Microsystems - Beijing China * Copyright (c) 2009, Intel Corporation.
5*0035d21cSmiao chen - Sun Microsystems - Beijing China * All Rights Reserved.
6*0035d21cSmiao chen - Sun Microsystems - Beijing China *
7*0035d21cSmiao chen - Sun Microsystems - Beijing China * Permission is hereby granted, free of charge, to any person obtaining a
8*0035d21cSmiao chen - Sun Microsystems - Beijing China * copy of this software and associated documentation files(the
9*0035d21cSmiao chen - Sun Microsystems - Beijing China * "Software"), to deal in the Software without restriction, including
10*0035d21cSmiao chen - Sun Microsystems - Beijing China * without limitation the rights to use, copy, modify, merge, publish,
11*0035d21cSmiao chen - Sun Microsystems - Beijing China * distribute, sub license, and/or sell copies of the Software, and to
12*0035d21cSmiao chen - Sun Microsystems - Beijing China * permit persons to whom the Software is furnished to do so, subject to
13*0035d21cSmiao chen - Sun Microsystems - Beijing China * the following conditions:
14*0035d21cSmiao chen - Sun Microsystems - Beijing China *
15*0035d21cSmiao chen - Sun Microsystems - Beijing China * The above copyright notice and this permission notice(including the
16*0035d21cSmiao chen - Sun Microsystems - Beijing China * next paragraph) shall be included in all copies or substantial portions
17*0035d21cSmiao chen - Sun Microsystems - Beijing China * of the Software.
18*0035d21cSmiao chen - Sun Microsystems - Beijing China *
19*0035d21cSmiao chen - Sun Microsystems - Beijing China * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
20*0035d21cSmiao chen - Sun Microsystems - Beijing China * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
21*0035d21cSmiao chen - Sun Microsystems - Beijing China * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL
22*0035d21cSmiao chen - Sun Microsystems - Beijing China * THE COPYRIGHT HOLDERS, AUTHORS AND/OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM,
23*0035d21cSmiao chen - Sun Microsystems - Beijing China * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
24*0035d21cSmiao chen - Sun Microsystems - Beijing China * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
25*0035d21cSmiao chen - Sun Microsystems - Beijing China * USE OR OTHER DEALINGS IN THE SOFTWARE.
26*0035d21cSmiao chen - Sun Microsystems - Beijing China *
27*0035d21cSmiao chen - Sun Microsystems - Beijing China *
28*0035d21cSmiao chen - Sun Microsystems - Beijing China */
29*0035d21cSmiao chen - Sun Microsystems - Beijing China
30*0035d21cSmiao chen - Sun Microsystems - Beijing China /*
31*0035d21cSmiao chen - Sun Microsystems - Beijing China * Generic simple memory manager implementation. Intended to be used as a base
32*0035d21cSmiao chen - Sun Microsystems - Beijing China * class implementation for more advanced memory managers.
33*0035d21cSmiao chen - Sun Microsystems - Beijing China *
34*0035d21cSmiao chen - Sun Microsystems - Beijing China * Note that the algorithm used is quite simple and there might be substantial
35*0035d21cSmiao chen - Sun Microsystems - Beijing China * performance gains if a smarter free list is implemented.
36*0035d21cSmiao chen - Sun Microsystems - Beijing China * Currently it is just an
37*0035d21cSmiao chen - Sun Microsystems - Beijing China * unordered stack of free regions. This could easily be improved if an RB-tree
38*0035d21cSmiao chen - Sun Microsystems - Beijing China * is used instead. At least if we expect heavy fragmentation.
39*0035d21cSmiao chen - Sun Microsystems - Beijing China *
40*0035d21cSmiao chen - Sun Microsystems - Beijing China * Aligned allocations can also see improvement.
41*0035d21cSmiao chen - Sun Microsystems - Beijing China *
42*0035d21cSmiao chen - Sun Microsystems - Beijing China * Authors:
43*0035d21cSmiao chen - Sun Microsystems - Beijing China * Thomas Hellström <thomas-at-tungstengraphics-dot-com>
44*0035d21cSmiao chen - Sun Microsystems - Beijing China */
45*0035d21cSmiao chen - Sun Microsystems - Beijing China
46*0035d21cSmiao chen - Sun Microsystems - Beijing China /*
47*0035d21cSmiao chen - Sun Microsystems - Beijing China * Copyright 2009 Sun Microsystems, Inc. All rights reserved.
48*0035d21cSmiao chen - Sun Microsystems - Beijing China * Use is subject to license terms.
49*0035d21cSmiao chen - Sun Microsystems - Beijing China */
50*0035d21cSmiao chen - Sun Microsystems - Beijing China
51*0035d21cSmiao chen - Sun Microsystems - Beijing China #include "drmP.h"
52*0035d21cSmiao chen - Sun Microsystems - Beijing China
53*0035d21cSmiao chen - Sun Microsystems - Beijing China unsigned long
drm_mm_tail_space(struct drm_mm * mm)54*0035d21cSmiao chen - Sun Microsystems - Beijing China drm_mm_tail_space(struct drm_mm *mm)
55*0035d21cSmiao chen - Sun Microsystems - Beijing China {
56*0035d21cSmiao chen - Sun Microsystems - Beijing China struct list_head *tail_node;
57*0035d21cSmiao chen - Sun Microsystems - Beijing China struct drm_mm_node *entry;
58*0035d21cSmiao chen - Sun Microsystems - Beijing China
59*0035d21cSmiao chen - Sun Microsystems - Beijing China tail_node = mm->ml_entry.prev;
60*0035d21cSmiao chen - Sun Microsystems - Beijing China entry = list_entry(tail_node, struct drm_mm_node, ml_entry);
61*0035d21cSmiao chen - Sun Microsystems - Beijing China if (!entry->free)
62*0035d21cSmiao chen - Sun Microsystems - Beijing China return (0);
63*0035d21cSmiao chen - Sun Microsystems - Beijing China
64*0035d21cSmiao chen - Sun Microsystems - Beijing China return (entry->size);
65*0035d21cSmiao chen - Sun Microsystems - Beijing China }
66*0035d21cSmiao chen - Sun Microsystems - Beijing China
67*0035d21cSmiao chen - Sun Microsystems - Beijing China int
drm_mm_remove_space_from_tail(struct drm_mm * mm,unsigned long size)68*0035d21cSmiao chen - Sun Microsystems - Beijing China drm_mm_remove_space_from_tail(struct drm_mm *mm, unsigned long size)
69*0035d21cSmiao chen - Sun Microsystems - Beijing China {
70*0035d21cSmiao chen - Sun Microsystems - Beijing China struct list_head *tail_node;
71*0035d21cSmiao chen - Sun Microsystems - Beijing China struct drm_mm_node *entry;
72*0035d21cSmiao chen - Sun Microsystems - Beijing China
73*0035d21cSmiao chen - Sun Microsystems - Beijing China tail_node = mm->ml_entry.prev;
74*0035d21cSmiao chen - Sun Microsystems - Beijing China entry = list_entry(tail_node, struct drm_mm_node, ml_entry);
75*0035d21cSmiao chen - Sun Microsystems - Beijing China if (!entry->free)
76*0035d21cSmiao chen - Sun Microsystems - Beijing China return (ENOMEM);
77*0035d21cSmiao chen - Sun Microsystems - Beijing China
78*0035d21cSmiao chen - Sun Microsystems - Beijing China if (entry->size <= size)
79*0035d21cSmiao chen - Sun Microsystems - Beijing China return (ENOMEM);
80*0035d21cSmiao chen - Sun Microsystems - Beijing China
81*0035d21cSmiao chen - Sun Microsystems - Beijing China entry->size -= size;
82*0035d21cSmiao chen - Sun Microsystems - Beijing China return (0);
83*0035d21cSmiao chen - Sun Microsystems - Beijing China }
84*0035d21cSmiao chen - Sun Microsystems - Beijing China
85*0035d21cSmiao chen - Sun Microsystems - Beijing China
86*0035d21cSmiao chen - Sun Microsystems - Beijing China static int
drm_mm_create_tail_node(struct drm_mm * mm,unsigned long start,unsigned long size)87*0035d21cSmiao chen - Sun Microsystems - Beijing China drm_mm_create_tail_node(struct drm_mm *mm,
88*0035d21cSmiao chen - Sun Microsystems - Beijing China unsigned long start,
89*0035d21cSmiao chen - Sun Microsystems - Beijing China unsigned long size)
90*0035d21cSmiao chen - Sun Microsystems - Beijing China {
91*0035d21cSmiao chen - Sun Microsystems - Beijing China struct drm_mm_node *child;
92*0035d21cSmiao chen - Sun Microsystems - Beijing China
93*0035d21cSmiao chen - Sun Microsystems - Beijing China child = (struct drm_mm_node *)
94*0035d21cSmiao chen - Sun Microsystems - Beijing China drm_alloc(sizeof (*child), DRM_MEM_MM);
95*0035d21cSmiao chen - Sun Microsystems - Beijing China if (!child)
96*0035d21cSmiao chen - Sun Microsystems - Beijing China return (ENOMEM);
97*0035d21cSmiao chen - Sun Microsystems - Beijing China
98*0035d21cSmiao chen - Sun Microsystems - Beijing China child->free = 1;
99*0035d21cSmiao chen - Sun Microsystems - Beijing China child->size = size;
100*0035d21cSmiao chen - Sun Microsystems - Beijing China child->start = start;
101*0035d21cSmiao chen - Sun Microsystems - Beijing China child->mm = mm;
102*0035d21cSmiao chen - Sun Microsystems - Beijing China
103*0035d21cSmiao chen - Sun Microsystems - Beijing China list_add_tail(&child->ml_entry, &mm->ml_entry, (caddr_t)child);
104*0035d21cSmiao chen - Sun Microsystems - Beijing China list_add_tail(&child->fl_entry, &mm->fl_entry, (caddr_t)child);
105*0035d21cSmiao chen - Sun Microsystems - Beijing China
106*0035d21cSmiao chen - Sun Microsystems - Beijing China return (0);
107*0035d21cSmiao chen - Sun Microsystems - Beijing China }
108*0035d21cSmiao chen - Sun Microsystems - Beijing China
109*0035d21cSmiao chen - Sun Microsystems - Beijing China
110*0035d21cSmiao chen - Sun Microsystems - Beijing China int
drm_mm_add_space_to_tail(struct drm_mm * mm,unsigned long size)111*0035d21cSmiao chen - Sun Microsystems - Beijing China drm_mm_add_space_to_tail(struct drm_mm *mm, unsigned long size)
112*0035d21cSmiao chen - Sun Microsystems - Beijing China {
113*0035d21cSmiao chen - Sun Microsystems - Beijing China struct list_head *tail_node;
114*0035d21cSmiao chen - Sun Microsystems - Beijing China struct drm_mm_node *entry;
115*0035d21cSmiao chen - Sun Microsystems - Beijing China
116*0035d21cSmiao chen - Sun Microsystems - Beijing China tail_node = mm->ml_entry.prev;
117*0035d21cSmiao chen - Sun Microsystems - Beijing China entry = list_entry(tail_node, struct drm_mm_node, ml_entry);
118*0035d21cSmiao chen - Sun Microsystems - Beijing China if (!entry->free) {
119*0035d21cSmiao chen - Sun Microsystems - Beijing China return (drm_mm_create_tail_node(mm,
120*0035d21cSmiao chen - Sun Microsystems - Beijing China entry->start + entry->size, size));
121*0035d21cSmiao chen - Sun Microsystems - Beijing China }
122*0035d21cSmiao chen - Sun Microsystems - Beijing China entry->size += size;
123*0035d21cSmiao chen - Sun Microsystems - Beijing China return (0);
124*0035d21cSmiao chen - Sun Microsystems - Beijing China }
125*0035d21cSmiao chen - Sun Microsystems - Beijing China
126*0035d21cSmiao chen - Sun Microsystems - Beijing China static struct drm_mm_node *
drm_mm_split_at_start(struct drm_mm_node * parent,unsigned long size)127*0035d21cSmiao chen - Sun Microsystems - Beijing China drm_mm_split_at_start(struct drm_mm_node *parent,
128*0035d21cSmiao chen - Sun Microsystems - Beijing China unsigned long size)
129*0035d21cSmiao chen - Sun Microsystems - Beijing China {
130*0035d21cSmiao chen - Sun Microsystems - Beijing China struct drm_mm_node *child;
131*0035d21cSmiao chen - Sun Microsystems - Beijing China
132*0035d21cSmiao chen - Sun Microsystems - Beijing China child = (struct drm_mm_node *)
133*0035d21cSmiao chen - Sun Microsystems - Beijing China drm_alloc(sizeof (*child), DRM_MEM_MM);
134*0035d21cSmiao chen - Sun Microsystems - Beijing China if (!child)
135*0035d21cSmiao chen - Sun Microsystems - Beijing China return (NULL);
136*0035d21cSmiao chen - Sun Microsystems - Beijing China
137*0035d21cSmiao chen - Sun Microsystems - Beijing China INIT_LIST_HEAD(&child->fl_entry);
138*0035d21cSmiao chen - Sun Microsystems - Beijing China
139*0035d21cSmiao chen - Sun Microsystems - Beijing China child->free = 0;
140*0035d21cSmiao chen - Sun Microsystems - Beijing China child->size = size;
141*0035d21cSmiao chen - Sun Microsystems - Beijing China child->start = parent->start;
142*0035d21cSmiao chen - Sun Microsystems - Beijing China child->mm = parent->mm;
143*0035d21cSmiao chen - Sun Microsystems - Beijing China
144*0035d21cSmiao chen - Sun Microsystems - Beijing China list_add_tail(&child->ml_entry, &parent->ml_entry, (caddr_t)child);
145*0035d21cSmiao chen - Sun Microsystems - Beijing China INIT_LIST_HEAD(&child->fl_entry);
146*0035d21cSmiao chen - Sun Microsystems - Beijing China
147*0035d21cSmiao chen - Sun Microsystems - Beijing China parent->size -= size;
148*0035d21cSmiao chen - Sun Microsystems - Beijing China parent->start += size;
149*0035d21cSmiao chen - Sun Microsystems - Beijing China return (child);
150*0035d21cSmiao chen - Sun Microsystems - Beijing China }
151*0035d21cSmiao chen - Sun Microsystems - Beijing China
152*0035d21cSmiao chen - Sun Microsystems - Beijing China /*
153*0035d21cSmiao chen - Sun Microsystems - Beijing China * Put a block. Merge with the previous and / or next block if they are free.
154*0035d21cSmiao chen - Sun Microsystems - Beijing China * Otherwise add to the free stack.
155*0035d21cSmiao chen - Sun Microsystems - Beijing China */
156*0035d21cSmiao chen - Sun Microsystems - Beijing China
157*0035d21cSmiao chen - Sun Microsystems - Beijing China void
drm_mm_put_block(struct drm_mm_node * cur)158*0035d21cSmiao chen - Sun Microsystems - Beijing China drm_mm_put_block(struct drm_mm_node *cur)
159*0035d21cSmiao chen - Sun Microsystems - Beijing China {
160*0035d21cSmiao chen - Sun Microsystems - Beijing China
161*0035d21cSmiao chen - Sun Microsystems - Beijing China struct drm_mm *mm = cur->mm;
162*0035d21cSmiao chen - Sun Microsystems - Beijing China struct list_head *cur_head = &cur->ml_entry;
163*0035d21cSmiao chen - Sun Microsystems - Beijing China struct list_head *root_head = &mm->ml_entry;
164*0035d21cSmiao chen - Sun Microsystems - Beijing China struct drm_mm_node *prev_node = NULL;
165*0035d21cSmiao chen - Sun Microsystems - Beijing China struct drm_mm_node *next_node;
166*0035d21cSmiao chen - Sun Microsystems - Beijing China
167*0035d21cSmiao chen - Sun Microsystems - Beijing China int merged = 0;
168*0035d21cSmiao chen - Sun Microsystems - Beijing China
169*0035d21cSmiao chen - Sun Microsystems - Beijing China if (cur_head->prev != root_head) {
170*0035d21cSmiao chen - Sun Microsystems - Beijing China prev_node = list_entry(cur_head->prev,
171*0035d21cSmiao chen - Sun Microsystems - Beijing China struct drm_mm_node, ml_entry);
172*0035d21cSmiao chen - Sun Microsystems - Beijing China if (prev_node->free) {
173*0035d21cSmiao chen - Sun Microsystems - Beijing China prev_node->size += cur->size;
174*0035d21cSmiao chen - Sun Microsystems - Beijing China merged = 1;
175*0035d21cSmiao chen - Sun Microsystems - Beijing China }
176*0035d21cSmiao chen - Sun Microsystems - Beijing China }
177*0035d21cSmiao chen - Sun Microsystems - Beijing China if (cur_head->next != root_head) {
178*0035d21cSmiao chen - Sun Microsystems - Beijing China next_node = list_entry(cur_head->next,
179*0035d21cSmiao chen - Sun Microsystems - Beijing China struct drm_mm_node, ml_entry);
180*0035d21cSmiao chen - Sun Microsystems - Beijing China if (next_node->free) {
181*0035d21cSmiao chen - Sun Microsystems - Beijing China if (merged) {
182*0035d21cSmiao chen - Sun Microsystems - Beijing China prev_node->size += next_node->size;
183*0035d21cSmiao chen - Sun Microsystems - Beijing China list_del(&next_node->ml_entry);
184*0035d21cSmiao chen - Sun Microsystems - Beijing China list_del(&next_node->fl_entry);
185*0035d21cSmiao chen - Sun Microsystems - Beijing China drm_free(next_node,
186*0035d21cSmiao chen - Sun Microsystems - Beijing China sizeof (*next_node), DRM_MEM_MM);
187*0035d21cSmiao chen - Sun Microsystems - Beijing China } else {
188*0035d21cSmiao chen - Sun Microsystems - Beijing China next_node->size += cur->size;
189*0035d21cSmiao chen - Sun Microsystems - Beijing China next_node->start = cur->start;
190*0035d21cSmiao chen - Sun Microsystems - Beijing China merged = 1;
191*0035d21cSmiao chen - Sun Microsystems - Beijing China }
192*0035d21cSmiao chen - Sun Microsystems - Beijing China }
193*0035d21cSmiao chen - Sun Microsystems - Beijing China }
194*0035d21cSmiao chen - Sun Microsystems - Beijing China if (!merged) {
195*0035d21cSmiao chen - Sun Microsystems - Beijing China cur->free = 1;
196*0035d21cSmiao chen - Sun Microsystems - Beijing China list_add(&cur->fl_entry, &mm->fl_entry, (caddr_t)cur);
197*0035d21cSmiao chen - Sun Microsystems - Beijing China } else {
198*0035d21cSmiao chen - Sun Microsystems - Beijing China list_del(&cur->ml_entry);
199*0035d21cSmiao chen - Sun Microsystems - Beijing China drm_free(cur, sizeof (*cur), DRM_MEM_MM);
200*0035d21cSmiao chen - Sun Microsystems - Beijing China }
201*0035d21cSmiao chen - Sun Microsystems - Beijing China }
202*0035d21cSmiao chen - Sun Microsystems - Beijing China
203*0035d21cSmiao chen - Sun Microsystems - Beijing China struct drm_mm_node *
drm_mm_get_block(struct drm_mm_node * parent,unsigned long size,unsigned alignment)204*0035d21cSmiao chen - Sun Microsystems - Beijing China drm_mm_get_block(struct drm_mm_node *parent,
205*0035d21cSmiao chen - Sun Microsystems - Beijing China unsigned long size,
206*0035d21cSmiao chen - Sun Microsystems - Beijing China unsigned alignment)
207*0035d21cSmiao chen - Sun Microsystems - Beijing China {
208*0035d21cSmiao chen - Sun Microsystems - Beijing China
209*0035d21cSmiao chen - Sun Microsystems - Beijing China struct drm_mm_node *align_splitoff = NULL;
210*0035d21cSmiao chen - Sun Microsystems - Beijing China struct drm_mm_node *child;
211*0035d21cSmiao chen - Sun Microsystems - Beijing China unsigned tmp = 0;
212*0035d21cSmiao chen - Sun Microsystems - Beijing China
213*0035d21cSmiao chen - Sun Microsystems - Beijing China if (alignment)
214*0035d21cSmiao chen - Sun Microsystems - Beijing China tmp = parent->start % alignment;
215*0035d21cSmiao chen - Sun Microsystems - Beijing China
216*0035d21cSmiao chen - Sun Microsystems - Beijing China if (tmp) {
217*0035d21cSmiao chen - Sun Microsystems - Beijing China align_splitoff = drm_mm_split_at_start(parent, alignment - tmp);
218*0035d21cSmiao chen - Sun Microsystems - Beijing China if (!align_splitoff)
219*0035d21cSmiao chen - Sun Microsystems - Beijing China return (NULL);
220*0035d21cSmiao chen - Sun Microsystems - Beijing China }
221*0035d21cSmiao chen - Sun Microsystems - Beijing China
222*0035d21cSmiao chen - Sun Microsystems - Beijing China if (parent->size == size) {
223*0035d21cSmiao chen - Sun Microsystems - Beijing China list_del_init(&parent->fl_entry);
224*0035d21cSmiao chen - Sun Microsystems - Beijing China parent->free = 0;
225*0035d21cSmiao chen - Sun Microsystems - Beijing China return (parent);
226*0035d21cSmiao chen - Sun Microsystems - Beijing China } else {
227*0035d21cSmiao chen - Sun Microsystems - Beijing China child = drm_mm_split_at_start(parent, size);
228*0035d21cSmiao chen - Sun Microsystems - Beijing China }
229*0035d21cSmiao chen - Sun Microsystems - Beijing China
230*0035d21cSmiao chen - Sun Microsystems - Beijing China if (align_splitoff)
231*0035d21cSmiao chen - Sun Microsystems - Beijing China drm_mm_put_block(align_splitoff);
232*0035d21cSmiao chen - Sun Microsystems - Beijing China
233*0035d21cSmiao chen - Sun Microsystems - Beijing China return (child);
234*0035d21cSmiao chen - Sun Microsystems - Beijing China }
235*0035d21cSmiao chen - Sun Microsystems - Beijing China
236*0035d21cSmiao chen - Sun Microsystems - Beijing China struct drm_mm_node *
drm_mm_search_free(const struct drm_mm * mm,unsigned long size,unsigned alignment,int best_match)237*0035d21cSmiao chen - Sun Microsystems - Beijing China drm_mm_search_free(const struct drm_mm *mm,
238*0035d21cSmiao chen - Sun Microsystems - Beijing China unsigned long size,
239*0035d21cSmiao chen - Sun Microsystems - Beijing China unsigned alignment,
240*0035d21cSmiao chen - Sun Microsystems - Beijing China int best_match)
241*0035d21cSmiao chen - Sun Microsystems - Beijing China {
242*0035d21cSmiao chen - Sun Microsystems - Beijing China struct list_head *list;
243*0035d21cSmiao chen - Sun Microsystems - Beijing China const struct list_head *free_stack = &mm->fl_entry;
244*0035d21cSmiao chen - Sun Microsystems - Beijing China struct drm_mm_node *entry;
245*0035d21cSmiao chen - Sun Microsystems - Beijing China struct drm_mm_node *best;
246*0035d21cSmiao chen - Sun Microsystems - Beijing China unsigned long best_size;
247*0035d21cSmiao chen - Sun Microsystems - Beijing China unsigned wasted;
248*0035d21cSmiao chen - Sun Microsystems - Beijing China
249*0035d21cSmiao chen - Sun Microsystems - Beijing China best = NULL;
250*0035d21cSmiao chen - Sun Microsystems - Beijing China best_size = ~0UL;
251*0035d21cSmiao chen - Sun Microsystems - Beijing China
252*0035d21cSmiao chen - Sun Microsystems - Beijing China list_for_each(list, free_stack) {
253*0035d21cSmiao chen - Sun Microsystems - Beijing China entry = list_entry(list, struct drm_mm_node, fl_entry);
254*0035d21cSmiao chen - Sun Microsystems - Beijing China wasted = 0;
255*0035d21cSmiao chen - Sun Microsystems - Beijing China
256*0035d21cSmiao chen - Sun Microsystems - Beijing China if (entry->size < size)
257*0035d21cSmiao chen - Sun Microsystems - Beijing China continue;
258*0035d21cSmiao chen - Sun Microsystems - Beijing China
259*0035d21cSmiao chen - Sun Microsystems - Beijing China if (alignment) {
260*0035d21cSmiao chen - Sun Microsystems - Beijing China register unsigned tmp = entry->start % alignment;
261*0035d21cSmiao chen - Sun Microsystems - Beijing China if (tmp)
262*0035d21cSmiao chen - Sun Microsystems - Beijing China wasted += alignment - tmp;
263*0035d21cSmiao chen - Sun Microsystems - Beijing China }
264*0035d21cSmiao chen - Sun Microsystems - Beijing China
265*0035d21cSmiao chen - Sun Microsystems - Beijing China
266*0035d21cSmiao chen - Sun Microsystems - Beijing China if (entry->size >= size + wasted) {
267*0035d21cSmiao chen - Sun Microsystems - Beijing China if (!best_match)
268*0035d21cSmiao chen - Sun Microsystems - Beijing China return (entry);
269*0035d21cSmiao chen - Sun Microsystems - Beijing China if (size < best_size) {
270*0035d21cSmiao chen - Sun Microsystems - Beijing China best = entry;
271*0035d21cSmiao chen - Sun Microsystems - Beijing China best_size = entry->size;
272*0035d21cSmiao chen - Sun Microsystems - Beijing China }
273*0035d21cSmiao chen - Sun Microsystems - Beijing China }
274*0035d21cSmiao chen - Sun Microsystems - Beijing China }
275*0035d21cSmiao chen - Sun Microsystems - Beijing China
276*0035d21cSmiao chen - Sun Microsystems - Beijing China return (best);
277*0035d21cSmiao chen - Sun Microsystems - Beijing China }
278*0035d21cSmiao chen - Sun Microsystems - Beijing China
279*0035d21cSmiao chen - Sun Microsystems - Beijing China int
drm_mm_clean(struct drm_mm * mm)280*0035d21cSmiao chen - Sun Microsystems - Beijing China drm_mm_clean(struct drm_mm *mm)
281*0035d21cSmiao chen - Sun Microsystems - Beijing China {
282*0035d21cSmiao chen - Sun Microsystems - Beijing China struct list_head *head = &mm->ml_entry;
283*0035d21cSmiao chen - Sun Microsystems - Beijing China
284*0035d21cSmiao chen - Sun Microsystems - Beijing China return (head->next->next == head);
285*0035d21cSmiao chen - Sun Microsystems - Beijing China }
286*0035d21cSmiao chen - Sun Microsystems - Beijing China
287*0035d21cSmiao chen - Sun Microsystems - Beijing China int
drm_mm_init(struct drm_mm * mm,unsigned long start,unsigned long size)288*0035d21cSmiao chen - Sun Microsystems - Beijing China drm_mm_init(struct drm_mm *mm, unsigned long start, unsigned long size)
289*0035d21cSmiao chen - Sun Microsystems - Beijing China {
290*0035d21cSmiao chen - Sun Microsystems - Beijing China INIT_LIST_HEAD(&mm->ml_entry);
291*0035d21cSmiao chen - Sun Microsystems - Beijing China INIT_LIST_HEAD(&mm->fl_entry);
292*0035d21cSmiao chen - Sun Microsystems - Beijing China
293*0035d21cSmiao chen - Sun Microsystems - Beijing China return (drm_mm_create_tail_node(mm, start, size));
294*0035d21cSmiao chen - Sun Microsystems - Beijing China }
295*0035d21cSmiao chen - Sun Microsystems - Beijing China
296*0035d21cSmiao chen - Sun Microsystems - Beijing China
297*0035d21cSmiao chen - Sun Microsystems - Beijing China void
drm_mm_takedown(struct drm_mm * mm)298*0035d21cSmiao chen - Sun Microsystems - Beijing China drm_mm_takedown(struct drm_mm *mm)
299*0035d21cSmiao chen - Sun Microsystems - Beijing China {
300*0035d21cSmiao chen - Sun Microsystems - Beijing China struct list_head *bnode = mm->fl_entry.next;
301*0035d21cSmiao chen - Sun Microsystems - Beijing China struct drm_mm_node *entry;
302*0035d21cSmiao chen - Sun Microsystems - Beijing China
303*0035d21cSmiao chen - Sun Microsystems - Beijing China entry = list_entry(bnode, struct drm_mm_node, fl_entry);
304*0035d21cSmiao chen - Sun Microsystems - Beijing China
305*0035d21cSmiao chen - Sun Microsystems - Beijing China if (entry->ml_entry.next != &mm->ml_entry ||
306*0035d21cSmiao chen - Sun Microsystems - Beijing China entry->fl_entry.next != &mm->fl_entry) {
307*0035d21cSmiao chen - Sun Microsystems - Beijing China DRM_ERROR("Memory manager not clean. Delaying takedown\n");
308*0035d21cSmiao chen - Sun Microsystems - Beijing China return;
309*0035d21cSmiao chen - Sun Microsystems - Beijing China }
310*0035d21cSmiao chen - Sun Microsystems - Beijing China
311*0035d21cSmiao chen - Sun Microsystems - Beijing China list_del(&entry->fl_entry);
312*0035d21cSmiao chen - Sun Microsystems - Beijing China list_del(&entry->ml_entry);
313*0035d21cSmiao chen - Sun Microsystems - Beijing China
314*0035d21cSmiao chen - Sun Microsystems - Beijing China drm_free(entry, sizeof (*entry), DRM_MEM_MM);
315*0035d21cSmiao chen - Sun Microsystems - Beijing China }
316*0035d21cSmiao chen - Sun Microsystems - Beijing China
317*0035d21cSmiao chen - Sun Microsystems - Beijing China void
drm_mm_clean_ml(const struct drm_mm * mm)318*0035d21cSmiao chen - Sun Microsystems - Beijing China drm_mm_clean_ml(const struct drm_mm *mm)
319*0035d21cSmiao chen - Sun Microsystems - Beijing China {
320*0035d21cSmiao chen - Sun Microsystems - Beijing China const struct list_head *mlstack = &mm->ml_entry;
321*0035d21cSmiao chen - Sun Microsystems - Beijing China struct list_head *list, *temp;
322*0035d21cSmiao chen - Sun Microsystems - Beijing China struct drm_mm_node *entry;
323*0035d21cSmiao chen - Sun Microsystems - Beijing China
324*0035d21cSmiao chen - Sun Microsystems - Beijing China if (mlstack->next == NULL)
325*0035d21cSmiao chen - Sun Microsystems - Beijing China return;
326*0035d21cSmiao chen - Sun Microsystems - Beijing China
327*0035d21cSmiao chen - Sun Microsystems - Beijing China list_for_each_safe(list, temp, mlstack) {
328*0035d21cSmiao chen - Sun Microsystems - Beijing China entry = list_entry(list, struct drm_mm_node, ml_entry);
329*0035d21cSmiao chen - Sun Microsystems - Beijing China DRM_DEBUG("ml_entry 0x%x, size 0x%x, start 0x%x",
330*0035d21cSmiao chen - Sun Microsystems - Beijing China entry, entry->size, entry->start);
331*0035d21cSmiao chen - Sun Microsystems - Beijing China
332*0035d21cSmiao chen - Sun Microsystems - Beijing China list_del(&entry->fl_entry);
333*0035d21cSmiao chen - Sun Microsystems - Beijing China list_del(&entry->ml_entry);
334*0035d21cSmiao chen - Sun Microsystems - Beijing China drm_free(entry, sizeof (*entry), DRM_MEM_MM);
335*0035d21cSmiao chen - Sun Microsystems - Beijing China }
336*0035d21cSmiao chen - Sun Microsystems - Beijing China }
337