2 * Support for Medifield PNW Camera Imaging ISP subsystem.
4 * Copyright (c) 2010 Intel Corporation. All Rights Reserved.
6 * Copyright (c) 2010 Silicon Hive www.siliconhive.com.
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License version
10 * 2 as published by the Free Software Foundation.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
24 * This file contains functions for reserved memory pool management
26 #include <linux/kernel.h>
27 #include <linux/types.h>
30 #include <asm/set_memory.h>
32 #include "atomisp_internal.h"
33 #include "hmm/hmm_pool.h"
36 * reserved memory pool ops.
38 static unsigned int get_pages_from_reserved_pool(void *pool,
39 struct hmm_page_object *page_obj,
40 unsigned int size, bool cached)
44 unsigned int repool_pgnr;
46 struct hmm_reserved_pool_info *repool_info = pool;
51 spin_lock_irqsave(&repool_info->list_lock, flags);
52 if (repool_info->initialized) {
53 repool_pgnr = repool_info->index;
55 for (j = repool_pgnr-1; j >= 0; j--) {
56 page_obj[i].page = repool_info->pages[j];
57 page_obj[i].type = HMM_PAGE_TYPE_RESERVED;
64 spin_unlock_irqrestore(&repool_info->list_lock, flags);
68 static void free_pages_to_reserved_pool(void *pool,
69 struct hmm_page_object *page_obj)
72 struct hmm_reserved_pool_info *repool_info = pool;
77 spin_lock_irqsave(&repool_info->list_lock, flags);
79 if (repool_info->initialized &&
80 repool_info->index < repool_info->pgnr &&
81 page_obj->type == HMM_PAGE_TYPE_RESERVED) {
82 repool_info->pages[repool_info->index++] = page_obj->page;
85 spin_unlock_irqrestore(&repool_info->list_lock, flags);
88 static int hmm_reserved_pool_setup(struct hmm_reserved_pool_info **repool_info,
89 unsigned int pool_size)
91 struct hmm_reserved_pool_info *pool_info;
93 pool_info = kmalloc(sizeof(struct hmm_reserved_pool_info),
95 if (unlikely(!pool_info)) {
96 dev_err(atomisp_dev, "out of memory for repool_info.\n");
100 pool_info->pages = kmalloc(sizeof(struct page *) * pool_size,
102 if (unlikely(!pool_info->pages)) {
103 dev_err(atomisp_dev, "out of memory for repool_info->pages.\n");
108 pool_info->index = 0;
110 spin_lock_init(&pool_info->list_lock);
111 pool_info->initialized = true;
113 *repool_info = pool_info;
118 static int hmm_reserved_pool_init(void **pool, unsigned int pool_size)
121 unsigned int blk_pgnr;
122 unsigned int pgnr = pool_size;
123 unsigned int order = 0;
128 struct hmm_reserved_pool_info *repool_info;
132 ret = hmm_reserved_pool_setup(&repool_info, pool_size);
134 dev_err(atomisp_dev, "hmm_reserved_pool_setup failed.\n");
144 blk_pgnr = 1U << order;
145 while (blk_pgnr > pgnr) {
149 BUG_ON(order > MAX_ORDER);
151 pages = alloc_pages(GFP_KERNEL | __GFP_NOWARN, order);
152 if (unlikely(!pages)) {
155 dev_err(atomisp_dev, "%s: alloc_pages failed: %d\n",
156 __func__, fail_number);
157 /* if fail five times, will goto end */
159 /* FIXME: whether is the mechanism is ok? */
160 if (fail_number == ALLOC_PAGE_FAIL_NUM)
166 blk_pgnr = 1U << order;
168 ret = set_pages_uc(pages, blk_pgnr);
171 "set pages uncached failed\n");
172 __free_pages(pages, order);
176 for (j = 0; j < blk_pgnr; j++)
177 repool_info->pages[i++] = pages + j;
179 repool_info->index += blk_pgnr;
180 repool_info->pgnr += blk_pgnr;
189 repool_info->initialized = true;
193 dev_info(atomisp_dev,
194 "hmm_reserved_pool init successfully,"
195 "hmm_reserved_pool is with %d pages.\n",
200 static void hmm_reserved_pool_exit(void **pool)
205 struct hmm_reserved_pool_info *repool_info = *pool;
210 spin_lock_irqsave(&repool_info->list_lock, flags);
211 if (!repool_info->initialized) {
212 spin_unlock_irqrestore(&repool_info->list_lock, flags);
215 pgnr = repool_info->pgnr;
216 repool_info->index = 0;
217 repool_info->pgnr = 0;
218 repool_info->initialized = false;
219 spin_unlock_irqrestore(&repool_info->list_lock, flags);
221 for (i = 0; i < pgnr; i++) {
222 ret = set_pages_wb(repool_info->pages[i], 1);
225 "set page to WB err...ret=%d\n", ret);
227 W/A: set_pages_wb seldom return value = -EFAULT
228 indicate that address of page is not in valid
229 range(0xffff880000000000~0xffffc7ffffffffff)
230 then, _free_pages would panic; Do not know why
231 page address be valid, it maybe memory corruption by lowmemory
234 __free_pages(repool_info->pages[i], 0);
237 kfree(repool_info->pages);
243 static int hmm_reserved_pool_inited(void *pool)
245 struct hmm_reserved_pool_info *repool_info = pool;
250 return repool_info->initialized;
253 struct hmm_pool_ops reserved_pops = {
254 .pool_init = hmm_reserved_pool_init,
255 .pool_exit = hmm_reserved_pool_exit,
256 .pool_alloc_pages = get_pages_from_reserved_pool,
257 .pool_free_pages = free_pages_to_reserved_pool,
258 .pool_inited = hmm_reserved_pool_inited,