mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
Merge pull request #7572 from kaspar030/zptr
core: add pointer compression header
This commit is contained in:
commit
c30df3cac9
@ -27,7 +27,8 @@ LINKER_SCRIPT = kinetis.ld
|
||||
|
||||
ROM_START_ADDR = 0x00000000
|
||||
RAM_BASE_ADDR = 0x20000000
|
||||
RAM_START_ADDR = $$(($(RAM_BASE_ADDR)-($(KINETIS_SRAM_L_SIZE) * 1024)))
|
||||
RAM_START_ADDR := $(shell printf "0x%08x" $$(($(RAM_BASE_ADDR)-($(KINETIS_SRAM_L_SIZE) * 1024))))
|
||||
|
||||
# Define ROM_LEN with a non arithmetic value as it must be
|
||||
# evaluated in `cortexm_common` without a shell context
|
||||
# The `K` is correctly handled by both the linker and `cortexm_common`.
|
||||
|
@ -94,6 +94,7 @@ PSEUDOMODULES += stdio_cdc_acm
|
||||
PSEUDOMODULES += stdio_uart_rx
|
||||
PSEUDOMODULES += suit_%
|
||||
PSEUDOMODULES += wakaama_objects_%
|
||||
PSEUDOMODULES += zptr
|
||||
|
||||
# handle suit_v4 being a distinct module
|
||||
NO_PSEUDOMODULES += suit_v4
|
||||
|
@ -111,3 +111,7 @@ endif
|
||||
ifneq (,$(filter clif, $(USEMODULE)))
|
||||
INCLUDES += -I$(RIOTBASE)/sys/clif/include
|
||||
endif
|
||||
|
||||
ifneq (,$(filter zptr,$(USEMODULE)))
|
||||
include $(RIOTBASE)/sys/zptr/Makefile.include
|
||||
endif
|
||||
|
139
sys/include/zptr.h
Normal file
139
sys/include/zptr.h
Normal file
@ -0,0 +1,139 @@
|
||||
/*
|
||||
* Copyright (C) 2017 Kaspar Schleiser <kaspar@schleiser.de>
|
||||
*
|
||||
* This file is subject to the terms and conditions of the GNU Lesser
|
||||
* General Public License v2.1. See the file LICENSE in the top level
|
||||
* directory for more details.
|
||||
*/
|
||||
|
||||
/**
|
||||
* @defgroup sys_util_zptr Pointer Compression
|
||||
* @ingroup sys
|
||||
* @brief Provides 32bit -> 16bit pointer compression
|
||||
*
|
||||
* On many platforms, some pointers may have to be aligned, e.g., to 4 byte
|
||||
* bounderies.
|
||||
* On 32bit platforms, that makes it possible to store all possible aligned
|
||||
* 32bit pointers in a 16bit value as long as the total memory is small (e.g.,
|
||||
* with 4 byte alignment, all pointers within 256kb RAM can be represented by a
|
||||
* 16bit value). This can save memory, at the cost of some instructions for
|
||||
* compression/decompression.
|
||||
*
|
||||
* In order to use pointer compression, ZPTR_BASE needs to be defined to a (4
|
||||
* byte aligned) base address.
|
||||
*
|
||||
* A printf format macro (PRIzptr) is provided.
|
||||
*
|
||||
* You can then use zptr_t instead of a pointer type, using the supplied functions to compress / decompress,
|
||||
* e.g.,
|
||||
*
|
||||
* void func(void *ptr) {
|
||||
* printf("%"PRIzptr"\n", ptr);
|
||||
* ...
|
||||
* free(ptr);
|
||||
* }
|
||||
*
|
||||
* ... would become
|
||||
*
|
||||
* void func(zptr_t zptr);
|
||||
* printf("%"PRIzptr"\n", zptr);
|
||||
* ...
|
||||
* free(zptrd(zptr));
|
||||
* }
|
||||
*
|
||||
*
|
||||
* If ZPTR_BASE is unset, @ref zptr_t / @ref zptrc() / @ref zptrd() will
|
||||
* transparently and without overhead compile to normal (uncompressed) pointer
|
||||
* operations.
|
||||
*
|
||||
* @{
|
||||
*
|
||||
* @file
|
||||
* @brief 32bit -> 16bit pointer compression implementation
|
||||
*
|
||||
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
||||
*/
|
||||
|
||||
#ifndef ZPTR_H
|
||||
#define ZPTR_H
|
||||
|
||||
#include <assert.h>
|
||||
#include <stdint.h>
|
||||
#include <inttypes.h>
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
#if ZPTR_BASE || defined(DOXYGEN)
|
||||
|
||||
/**
|
||||
* @brief zptr type definition
|
||||
*/
|
||||
typedef uint16_t zptr_t;
|
||||
|
||||
/**
|
||||
* @brief zptr printf format definition
|
||||
*/
|
||||
#define PRIzptr PRIu16
|
||||
|
||||
/**
|
||||
* @brief zptr highest compressible address
|
||||
*/
|
||||
#define ZPTR_MAX_ADDR ((uintptr_t)ZPTR_BASE + (1 << 18))
|
||||
|
||||
/**
|
||||
* @brief Determine if a pointer is compressable by zptrc()
|
||||
* @param[in] pointer pointer to check
|
||||
* @returns 1 if pointer can be compressed, 0 if not
|
||||
*/
|
||||
static inline int zptr_check(void *pointer)
|
||||
{
|
||||
uintptr_t int_ptr = (uintptr_t)pointer;
|
||||
return ((!(int_ptr & 0x3)) \
|
||||
&& (int_ptr >= (uintptr_t)ZPTR_BASE) \
|
||||
&& (int_ptr < ZPTR_MAX_ADDR));
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Compress a pointer (if possible)
|
||||
*
|
||||
* Substracts ZPTR_BASE, then right-shifts @p pointer by two.
|
||||
*
|
||||
* @param[in] pointer pointer to compress
|
||||
* @returns compressed pointer
|
||||
*/
|
||||
static inline zptr_t zptrc(void *pointer)
|
||||
{
|
||||
assert(zptr_check(pointer));
|
||||
return (uint16_t)(((uint32_t)pointer - (uint32_t)ZPTR_BASE) >> 2);
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Decompress a pointer
|
||||
*
|
||||
* Left-shifts zptr_t by two, then adds ZPTR_BASE.
|
||||
*
|
||||
* @param[in] zptr compressed pointer
|
||||
* @returns decompressed pointer
|
||||
*/
|
||||
static inline void *zptrd(zptr_t zptr)
|
||||
{
|
||||
return (void *)(ZPTR_BASE + ((uint32_t)zptr << 2));
|
||||
}
|
||||
|
||||
#else /* ZPTR_BASE */
|
||||
/* fallback implementation */
|
||||
typedef void *zptr_t;
|
||||
#define PRIzptr "p"
|
||||
static inline int zptr_check(void *pointer) { (void)pointer; return 0; }
|
||||
static inline zptr_t zptrc(void *pointer) { return (zptr_t)pointer; }
|
||||
static inline void *zptrd(zptr_t zptr) { return (void *)zptr; }
|
||||
#endif
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif /* ZPTR_H */
|
||||
/** @} */
|
14
sys/zptr/Makefile.include
Normal file
14
sys/zptr/Makefile.include
Normal file
@ -0,0 +1,14 @@
|
||||
# zptr can use pointer compression of four byte aligned pointers, by right-
|
||||
# shifting them by two and stuffing in an uint16_t.
|
||||
# That only works on 32bit platforms, if total RAM size is <=2**18 and the
|
||||
# start address is known.
|
||||
ifneq (,$(filter arch_32bit,$(FEATURES_USED)))
|
||||
ifneq (,$(RAM_START_ADDR))
|
||||
ifneq (,$(RAM_LEN))
|
||||
# this handles 0xXXX and xxxK
|
||||
ifeq (1,$(shell echo $$(($(subst K,*1024,$(RAM_LEN)) <= 262144))))
|
||||
CFLAGS += -DZPTR_BASE=$(RAM_START_ADDR)
|
||||
endif
|
||||
endif
|
||||
endif
|
||||
endif
|
1
tests/unittests/tests-zptr/Makefile
Normal file
1
tests/unittests/tests-zptr/Makefile
Normal file
@ -0,0 +1 @@
|
||||
include $(RIOTBASE)/Makefile.base
|
1
tests/unittests/tests-zptr/Makefile.include
Normal file
1
tests/unittests/tests-zptr/Makefile.include
Normal file
@ -0,0 +1 @@
|
||||
USEMODULE += zptr
|
44
tests/unittests/tests-zptr/tests-zptr.c
Normal file
44
tests/unittests/tests-zptr/tests-zptr.c
Normal file
@ -0,0 +1,44 @@
|
||||
/*
|
||||
* Copyright (C) 2017 Kaspar Schleiser <kaspar@schleiser.de>
|
||||
*
|
||||
* This file is subject to the terms and conditions of the GNU Lesser
|
||||
* General Public License v2.1. See the file LICENSE in the top level
|
||||
* directory for more details.
|
||||
*/
|
||||
|
||||
/**
|
||||
* @{
|
||||
*
|
||||
* @file
|
||||
*/
|
||||
#include <errno.h>
|
||||
#include <stdint.h>
|
||||
#include <string.h>
|
||||
|
||||
#include "embUnit/embUnit.h"
|
||||
|
||||
#include "zptr.h"
|
||||
#include "tests-zptr.h"
|
||||
|
||||
static void test_zptr_basic(void)
|
||||
{
|
||||
uint32_t val;
|
||||
TEST_ASSERT(&val == zptrd(zptrc(&val)));
|
||||
}
|
||||
|
||||
Test *tests_zptr_tests(void)
|
||||
{
|
||||
EMB_UNIT_TESTFIXTURES(fixtures) {
|
||||
new_TestFixture(test_zptr_basic),
|
||||
};
|
||||
|
||||
EMB_UNIT_TESTCALLER(zptr_tests, NULL, NULL, fixtures);
|
||||
|
||||
return (Test *)&zptr_tests;
|
||||
}
|
||||
|
||||
void tests_zptr(void)
|
||||
{
|
||||
TESTS_RUN(tests_zptr_tests());
|
||||
}
|
||||
/** @} */
|
37
tests/unittests/tests-zptr/tests-zptr.h
Normal file
37
tests/unittests/tests-zptr/tests-zptr.h
Normal file
@ -0,0 +1,37 @@
|
||||
/*
|
||||
* Copyright (C) 2017 Kaspar Schleiser <kaspar@schleiser.de>
|
||||
*
|
||||
* This file is subject to the terms and conditions of the GNU Lesser
|
||||
* General Public License v2.1. See the file LICENSE in the top level
|
||||
* directory for more details.
|
||||
*/
|
||||
|
||||
/**
|
||||
* @addtogroup unittests
|
||||
* @{
|
||||
*
|
||||
* @file
|
||||
* @brief Unittests for the ``zptr`` module
|
||||
*
|
||||
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
||||
*/
|
||||
#ifndef TESTS_ZPTR_H
|
||||
#define TESTS_ZPTR_H
|
||||
|
||||
#include "embUnit.h"
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
/**
|
||||
* @brief The entry point of this test suite.
|
||||
*/
|
||||
void tests_zptr(void);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif /* TESTS_ZPTR_H */
|
||||
/** @} */
|
Loading…
Reference in New Issue
Block a user