beginning to fill in the lower levels of the heap allocation code
This commit is contained in:
162
include/comrogue/internals/dlist.h
Normal file
162
include/comrogue/internals/dlist.h
Normal file
@@ -0,0 +1,162 @@
|
||||
/*
|
||||
* This file is part of the COMROGUE Operating System for Raspberry Pi
|
||||
*
|
||||
* Copyright (c) 2013, Eric J. Bowersox / Erbosoft Enterprises
|
||||
* All rights reserved.
|
||||
*
|
||||
* This program is free for commercial and non-commercial use as long as the following conditions are
|
||||
* adhered to.
|
||||
*
|
||||
* Copyright in this file remains Eric J. Bowersox and/or Erbosoft, and as such any copyright notices
|
||||
* in the code are not to be removed.
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without modification, are permitted
|
||||
* provided that the following conditions are met:
|
||||
*
|
||||
* * Redistributions of source code must retain the above copyright notice, this list of conditions and
|
||||
* the following disclaimer.
|
||||
* * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and
|
||||
* the following disclaimer in the documentation and/or other materials provided with the distribution.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED
|
||||
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
|
||||
* PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
|
||||
* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
|
||||
* TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
||||
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
||||
* POSSIBILITY OF SUCH DAMAGE.
|
||||
*
|
||||
* "Raspberry Pi" is a trademark of the Raspberry Pi Foundation.
|
||||
*/
|
||||
#ifndef __DLIST_H_INCLUDED
|
||||
#define __DLIST_H_INCLUDED
|
||||
|
||||
#ifndef __ASM__
|
||||
|
||||
#include <comrogue/types.h>
|
||||
|
||||
/*----------------------------------------
|
||||
* Doubly-linked circular list operations
|
||||
*----------------------------------------
|
||||
*/
|
||||
|
||||
/* Declare a field in a structure to make it part of a doubly-linked list. */
|
||||
#define DLIST_FIELD_DECLARE(type, fieldname) \
|
||||
struct { type *pDLNext; type *pDLPrev; } fieldname
|
||||
|
||||
/* Access "next" and "previous" elements. */
|
||||
#define dlistNext(ptr, fieldname) ((ptr)->fieldname.pDLNext)
|
||||
#define dlistPrev(ptr, fieldname) ((ptr)->fieldname.pDLPrev)
|
||||
|
||||
/* Initialize a node to be a list of only itself. */
|
||||
#define dlistNodeInit(ptr, fieldname) \
|
||||
do { dlistNext(ptr, fieldname) = dlistPrev(ptr, fieldname) = (ptr); } while (0)
|
||||
|
||||
/* Insert a node before another in a list. */
|
||||
#define dlistInsertBefore(ptrInsBefore, ptrNew, fieldname) \
|
||||
do { dlistPrev(ptrNew, fieldname) = dlistPrev(ptrInsBefore, fieldname); \
|
||||
dlistNext(ptrNew, fieldname) = (ptrInsBefore); \
|
||||
dlistNext(dlistPrev(ptrInsBefore, fieldname), fieldname) = (ptrNew); \
|
||||
dlistPrev(ptrInsBefore) = (ptrNew); } while (0)
|
||||
|
||||
/* Insert a node after another in a list. */
|
||||
#define dlistInsertAfter(ptrInsAfter, ptrNew, fieldname) \
|
||||
do { dlistNext(ptrNew, fieldname) = dlistNext(ptrInsAfter, fieldname); \
|
||||
dlistPrev(ptrNew, fieldname) = (ptrInsAfter); \
|
||||
dlistPrev(dlistNext(ptrInsAfter, fieldname), fieldname) = (ptrNew); \
|
||||
dlistNext(ptrInsAfter, fieldname) = (ptrNew); } while (0)
|
||||
|
||||
/* Join two lists together. */
|
||||
#define dlistJoin(ptrA, ptrB, fieldname) \
|
||||
do { void *tmp; \
|
||||
dlistNext(dlistPrev(ptrA, fieldname), fieldname) = (ptrB); \
|
||||
dlistNext(dlistPrev(ptrB, fieldname), fieldname) = (ptrA); \
|
||||
tmp = dlistPrev(ptrA, fieldname); dlistPrev(ptrA, fieldname) = dlistPrev(ptrB, fieldname); \
|
||||
dlistPrev(ptrB, fieldname) = tmp; } while (0)
|
||||
|
||||
/* Split a list into two lists. (Same code as joining) */
|
||||
#define dlistSplit(ptrA, ptrB, fieldname) dlistJoin(ptrA, ptrB, fieldname)
|
||||
|
||||
/* Remove an element from a list. */
|
||||
#define dlistRemove(ptr, fieldname) \
|
||||
do { dlistPrev(dlistNext(ptr, fieldname), fieldname) = dlistPrev(ptr, fieldname); \
|
||||
dlistNext(dlistPrev(ptr, fieldname), fieldname) = dlistNext(ptr, fieldname); \
|
||||
dlistNodeInit(ptr, fieldname); } while (0)
|
||||
|
||||
/* Declare a loop to iterate through a list. */
|
||||
#define dlistForEach(var, ptr, fieldname) \
|
||||
for ((var) = (ptr); (var); (var) = ((dlistNext(var, fieldname) != ptr) ? dlistNext(var, fieldname) : NULL))
|
||||
|
||||
/* Declare a loop to iterate through a list in reverse order. */
|
||||
#define dlistForEachReverse(var, ptr, fieldname) \
|
||||
for ((var) = ((ptr) ? dlistPrev(ptr, fieldname) : NULL); (var); \
|
||||
(var) = (((var) != (ptr)) ? dlistPrev(var, fieldname) : NULL))
|
||||
|
||||
/*------------------------------------
|
||||
* List header and operations thereon
|
||||
*------------------------------------
|
||||
*/
|
||||
|
||||
/* Declaration and static initializers for a list header field. */
|
||||
#define DLIST_HEAD_DECLARE(type, headfieldname) \
|
||||
struct { type *pDLHFirst; } headfieldname
|
||||
#define DLIST_HEAD_INIT { NULL }
|
||||
#define DLIST_HEAD_NAMED_INIT(headfieldname) \
|
||||
.headfieldname = DLIST_HEAD_INIT
|
||||
|
||||
/* Pointers to the first and last elements of a list. */
|
||||
#define dlistFirst(hptr, fieldname) ((hptr)->pDLHFirst)
|
||||
#define dlistLast(hptr, fieldname) (dlistFirst(hptr) ? dlistPrev(dlistFirst(hptr), fieldname) : NULL)
|
||||
|
||||
/* Reinitializer for a list. */
|
||||
#define dlistListInit(hptr, fieldname) do { dlistFirst(hptr, fieldname) = NULL; } while (0)
|
||||
|
||||
/* Move forward and back in a list, returning NULL if we go past the "end" in either case. */
|
||||
#define dlistListNext(hptr, ptr, fieldname) \
|
||||
((dlistLast(hptr, fieldname) != (ptr)) ? dlistNext(ptr, fieldname) : NULL)
|
||||
#define dlistListPrev(hptr, ptr, fieldname) \
|
||||
((dlistFirst(hptr, fieldname) != (ptr)) ? dlistPrev(ptr, fieldname) : NULL)
|
||||
|
||||
/* Insert an element before another one in a list. */
|
||||
#define dlistListInsertBefore(hptr, ptrInsBefore, ptrNew, fieldname) \
|
||||
do { dlistInsertBefore(ptrInsBefore, ptrNew, fieldname); \
|
||||
if (dlistFirst(hptr, fieldname) == (ptrInsBefore)) { dlistFirst(hptr, fieldname) = (ptrNew); } } while (0)
|
||||
|
||||
/* Insert an element after another one in a list. */
|
||||
#define dlistListInsertAfter(hptr, ptrInsAfter, ptrNew, fieldname) \
|
||||
dlistInsertAfter(ptrInsAfter, ptrNew, fieldname)
|
||||
|
||||
/* Insert an element as the first one in a list. */
|
||||
#define dlistListInsertFirst(hptr, ptrNew, fieldname) \
|
||||
do { if (dlistFirst(hptr, fieldname)) { dlistInsertBefore(dlistFirst(hptr, fieldname), ptrNew, fieldname); } \
|
||||
dlistFirst(hptr, fieldname) = (ptrNew); } while (0)
|
||||
|
||||
/* Insert an element as the last one in a list. */
|
||||
#define dlistListInsertLast(hptr, ptrNew, fieldname) \
|
||||
do { if (dlistFirst(hptr, fieldname)) { dlistInsertBefore(dlistFirst(hptr, fieldname), ptrNew, fieldname); } \
|
||||
dlistFirst(hptr, fieldname) = dlistNext(ptrNew, fieldname); } while (0)
|
||||
|
||||
/* Remove an element from a list. */
|
||||
#define dlistListRemove(hptr, ptr, fieldname) \
|
||||
do { if (dlistFirst(hptr, fieldname) == (ptr)) { dlistFirst(hptr, fieldname) = dlistNext(ptr, fieldname); } \
|
||||
if (dlistFirst(hptr, fieldname) != (ptr)) { dlistRemove(ptr, fieldname); } \
|
||||
else { dlistListInit(hptr, fieldname); } } while (0)
|
||||
|
||||
/* Remove the first element from a list. */
|
||||
#define dlistListRemoveFirst(hptr, type, fieldname) \
|
||||
do { type *tmp = dlistFirst(hptr, fieldname); dlistListRemove(hptr, tmp, fieldname); } while (0)
|
||||
|
||||
/* Remove the last element from a list. */
|
||||
#define dlistListRemoveLast(hptr, type, fieldname) \
|
||||
do { type *tmp = dlistLast(hptr, fieldname); dlistListRemove(hptr, tmp, fieldname); } while (0)
|
||||
|
||||
/* Declare a loop to iterate through a list. */
|
||||
#define dlistListForEach(var, hptr, fieldname) dlistForEach(var, dlistFirst(hptr, fieldname), fieldname)
|
||||
|
||||
/* Declare a loop to iterate through a list in reverse order. */
|
||||
#define dlistListForEachReverse(var, hptr, fieldname) dlistForEachReverse(var, dlistFirst(hptr, fieldname), fieldname)
|
||||
|
||||
#endif /* __ASM__ */
|
||||
|
||||
#endif /* __DLIST_H_INCLUDED */
|
||||
@@ -53,6 +53,13 @@
|
||||
#define SYS_PGTBL_SIZE 1024 /* page tables must be located on this boundary and are this size */
|
||||
#define SYS_PGTBL_BITS 8 /* log2(SYS_PGTBL_SIZE/4), number of bits in a page table address */
|
||||
#define SYS_PGTBL_ENTRIES 256 /* SYS_PGTBL_SIZE/4, number of entries in a page table */
|
||||
#define SYS_CACHELINE_SIZE 64 /* width of a cache line in bytes */
|
||||
#define SYS_CACHELINE_BITS 6 /* log2(SYS_CACHELINE_SIZE), number of bits in a cache line */
|
||||
|
||||
#define SYS_PAGE_MASK (SYS_PAGE_SIZE - 1)
|
||||
#define SYS_CACHELINE_MASK (SYS_CACHELINE_SIZE - 1)
|
||||
|
||||
#define SYS_CACHELINE_CEILING(sz) (((sz) + SYS_CACHELINE_MASK) & ~SYS_CACHELINE_MASK)
|
||||
|
||||
/* Section descriptor bits */
|
||||
#define TTBSEC_PXN 0x00000001 /* Privileged Execute-Never */
|
||||
|
||||
Reference in New Issue
Block a user