source: mainline/kernel/generic/src/mm/reserve.c@ c46bfbc

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since c46bfbc was 63e27ef, checked in by Jiri Svoboda <jiri@…>, 8 years ago

ASSERT → assert

  • Property mode set to 100644
File size: 4.0 KB
RevLine 
[630a8ef]1/*
2 * Copyright (c) 2011 Jakub Jermar
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 *
9 * - Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * - Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * - The name of the author may not be used to endorse or promote products
15 * derived from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 */
28
29/** @addtogroup genericmm
30 * @{
31 */
32
33/**
34 * @file
35 * @brief Memory reservations.
36 */
37
[63e27ef]38#include <assert.h>
[630a8ef]39#include <mm/reserve.h>
40#include <mm/frame.h>
[010be476]41#include <mm/slab.h>
[e17f703]42#include <synch/spinlock.h>
43#include <typedefs.h>
44#include <arch/types.h>
[19c8030]45
46static bool reserve_initialized = false;
[e17f703]47
48IRQ_SPINLOCK_STATIC_INITIALIZE_NAME(reserve_lock, "reserve_lock");
49static ssize_t reserve = 0;
[630a8ef]50
[8d308b9]51/** Initialize memory reservations tracking.
52 *
53 * This function must be called after frame zones are created and merged
54 * and before any address space area is created.
55 */
56void reserve_init(void)
57{
58 reserve = frame_total_free_get();
[19c8030]59 reserve_initialized = true;
[8d308b9]60}
61
[010be476]62/** Try to reserve memory.
63 *
64 * This function may not be called from contexts that do not allow memory
65 * reclaiming, such as some invocations of frame_alloc_generic().
66 *
67 * @param size Number of frames to reserve.
68 * @return True on success or false otherwise.
69 */
[630a8ef]70bool reserve_try_alloc(size_t size)
71{
[e17f703]72 bool reserved = false;
73
[63e27ef]74 assert(reserve_initialized);
[19c8030]75
[e17f703]76 irq_spinlock_lock(&reserve_lock, true);
77 if (reserve >= 0 && (size_t) reserve >= size) {
78 reserve -= size;
79 reserved = true;
[010be476]80 } else {
81 /*
82 * Some reservable frames may be cached by the slab allocator.
83 * Try to reclaim some reservable memory. Try to be gentle for
84 * the first time. If it does not help, try to reclaim
85 * everything.
86 */
87 irq_spinlock_unlock(&reserve_lock, true);
88 slab_reclaim(0);
89 irq_spinlock_lock(&reserve_lock, true);
90 if (reserve >= 0 && (size_t) reserve >= size) {
91 reserve -= size;
92 reserved = true;
93 } else {
94 irq_spinlock_unlock(&reserve_lock, true);
95 slab_reclaim(SLAB_RECLAIM_ALL);
96 irq_spinlock_lock(&reserve_lock, true);
97 if (reserve >= 0 && (size_t) reserve >= size) {
98 reserve -= size;
99 reserved = true;
100 }
101 }
[e17f703]102 }
103 irq_spinlock_unlock(&reserve_lock, true);
104
105 return reserved;
[630a8ef]106}
107
[010be476]108/** Reserve memory.
109 *
110 * This function simply marks the respective amount of memory frames reserved.
111 * It does not implement any sort of blocking for the case there is not enough
112 * reservable memory. It will simply take the reserve into negative numbers and
113 * leave the blocking up to the allocation phase.
114 *
115 * @param size Number of frames to reserve.
116 */
[630a8ef]117void reserve_force_alloc(size_t size)
118{
[19c8030]119 if (!reserve_initialized)
120 return;
121
[e17f703]122 irq_spinlock_lock(&reserve_lock, true);
123 reserve -= size;
124 irq_spinlock_unlock(&reserve_lock, true);
[630a8ef]125}
126
[010be476]127/** Unreserve memory.
128 *
129 * @param size Number of frames to unreserve.
130 */
[630a8ef]131void reserve_free(size_t size)
132{
[19c8030]133 if (!reserve_initialized)
134 return;
135
[e17f703]136 irq_spinlock_lock(&reserve_lock, true);
137 reserve += size;
138 irq_spinlock_unlock(&reserve_lock, true);
[630a8ef]139}
140
141/** @}
142 */
Note: See TracBrowser for help on using the repository browser.