xref: /titanic_44/usr/src/uts/common/io/drm/drm_mm.c (revision 0035d21c77a24d02faf34c10aabc120ca692efb5)
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