/************************************************************************** * * Copyright (C) 2009 Steve Karg * * Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files (the * "Software"), to deal in the Software without restriction, including * without limitation the rights to use, copy, modify, merge, publish, * distribute, sublicense, and/or sell copies of the Software, and to * permit persons to whom the Software is furnished to do so, subject to * the following conditions: * * The above copyright notice and this permission notice shall be included * in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. * *********************************************************************/ #include #include "hardware.h" /* stack checking */ #if defined(__GNUC__) extern uint8_t _end; extern uint8_t __stack; #endif #if defined(__GNUC__) #define STACK_CANARY (0xC5) void stack_init( void) __attribute__ ((naked)) __attribute__ ((section(".init1"))); #endif void stack_init( void) { #if defined(__GNUC__) #if 0 uint8_t *p = &_end; while (p <= &__stack) { *p = STACK_CANARY; p++; } #else __asm volatile ( " ldi r30,lo8(_end)\n" " ldi r31,hi8(_end)\n" " ldi r24,lo8(0xc5)\n" /* STACK_CANARY = 0xc5 */ " ldi r25,hi8(__stack)\n" " rjmp .cmp\n" ".loop:\n" " st Z+,r24\n" ".cmp:\n" " cpi r30,lo8(__stack)\n" " cpc r31,r25\n" " brlo .loop\n" " breq .loop"::); #endif #endif } unsigned stack_size( void) { #if defined(__GNUC__) return (&__stack) - (&_end); #else return 0; #endif } uint8_t stack_byte( unsigned offset) { #if defined(__GNUC__) return *(&_end + offset); #else offset = offset; return 0; #endif } unsigned stack_unused( void) { unsigned count = 0; #if defined(__GNUC__) uint8_t *p = &_end; while (p <= &__stack) { if ((*p) != STACK_CANARY) { count = p - (&_end); break; } p++; } #endif return count; }