diff options
author | David Woodhouse <David.Woodhouse@intel.com> | 2009-07-29 04:28:45 -0400 |
---|---|---|
committer | David Woodhouse <David.Woodhouse@intel.com> | 2009-08-03 04:04:57 -0400 |
commit | f692775d7e0a22477143cd884e45c955448ac7d2 (patch) | |
tree | 8a801ec8cede89e694a9f4feaf8050642609ad27 /drivers/char/agp/intel-agp.c | |
parent | 91b8e3056bf9107b688eb076c9b804171364db71 (diff) |
intel-agp: fix sglist allocation to avoid vmalloc()
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
Diffstat (limited to 'drivers/char/agp/intel-agp.c')
-rw-r--r-- | drivers/char/agp/intel-agp.c | 29 |
1 files changed, 10 insertions, 19 deletions
diff --git a/drivers/char/agp/intel-agp.c b/drivers/char/agp/intel-agp.c index b9d9886ff3c3..d8c80d8be5e2 100644 --- a/drivers/char/agp/intel-agp.c +++ b/drivers/char/agp/intel-agp.c | |||
@@ -198,39 +198,30 @@ static void intel_agp_unmap_page(struct page *page, dma_addr_t dma) | |||
198 | 198 | ||
199 | static void intel_agp_free_sglist(struct agp_memory *mem) | 199 | static void intel_agp_free_sglist(struct agp_memory *mem) |
200 | { | 200 | { |
201 | struct sg_table st; | ||
202 | |||
203 | st.sgl = mem->sg_list; | ||
204 | st.orig_nents = st.nents = mem->page_count; | ||
205 | |||
206 | sg_free_table(&st); | ||
201 | 207 | ||
202 | if (mem->sg_vmalloc_flag) | ||
203 | vfree(mem->sg_list); | ||
204 | else | ||
205 | kfree(mem->sg_list); | ||
206 | mem->sg_vmalloc_flag = 0; | ||
207 | mem->sg_list = NULL; | 208 | mem->sg_list = NULL; |
208 | mem->num_sg = 0; | 209 | mem->num_sg = 0; |
209 | } | 210 | } |
210 | 211 | ||
211 | static int intel_agp_map_memory(struct agp_memory *mem) | 212 | static int intel_agp_map_memory(struct agp_memory *mem) |
212 | { | 213 | { |
214 | struct sg_table st; | ||
213 | struct scatterlist *sg; | 215 | struct scatterlist *sg; |
214 | int i; | 216 | int i; |
215 | 217 | ||
216 | DBG("try mapping %lu pages\n", (unsigned long)mem->page_count); | 218 | DBG("try mapping %lu pages\n", (unsigned long)mem->page_count); |
217 | 219 | ||
218 | if ((mem->page_count * sizeof(*mem->sg_list)) < 2*PAGE_SIZE) | 220 | if (sg_alloc_table(&st, mem->page_count, GFP_KERNEL)) |
219 | mem->sg_list = kcalloc(mem->page_count, sizeof(*mem->sg_list), | ||
220 | GFP_KERNEL); | ||
221 | |||
222 | if (mem->sg_list == NULL) { | ||
223 | mem->sg_list = vmalloc(mem->page_count * sizeof(*mem->sg_list)); | ||
224 | mem->sg_vmalloc_flag = 1; | ||
225 | } | ||
226 | |||
227 | if (!mem->sg_list) { | ||
228 | mem->sg_vmalloc_flag = 0; | ||
229 | return -ENOMEM; | 221 | return -ENOMEM; |
230 | } | ||
231 | sg_init_table(mem->sg_list, mem->page_count); | ||
232 | 222 | ||
233 | sg = mem->sg_list; | 223 | mem->sg_list = sg = st.sgl; |
224 | |||
234 | for (i = 0 ; i < mem->page_count; i++, sg = sg_next(sg)) | 225 | for (i = 0 ; i < mem->page_count; i++, sg = sg_next(sg)) |
235 | sg_set_page(sg, mem->pages[i], PAGE_SIZE, 0); | 226 | sg_set_page(sg, mem->pages[i], PAGE_SIZE, 0); |
236 | 227 | ||