mirror of
				https://github.com/markqvist/OpenModem.git
				synced 2025-10-30 22:19:14 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			233 lines
		
	
	
	
		
			5.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			233 lines
		
	
	
	
		
			5.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /**
 | |
|  * \file
 | |
|  * <!--
 | |
|  * This file is part of BeRTOS.
 | |
|  *
 | |
|  * Bertos is free software; you can redistribute it and/or modify
 | |
|  * it under the terms of the GNU General Public License as published by
 | |
|  * the Free Software Foundation; either version 2 of the License, or
 | |
|  * (at your option) any later version.
 | |
|  *
 | |
|  * This program is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License
 | |
|  * along with this program; if not, write to the Free Software
 | |
|  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
 | |
|  *
 | |
|  * As a special exception, you may use this file as part of a free software
 | |
|  * library without restriction.  Specifically, if other files instantiate
 | |
|  * templates or use macros or inline functions from this file, or you compile
 | |
|  * this file and link it with other files to produce an executable, this
 | |
|  * file does not by itself cause the resulting executable to be covered by
 | |
|  * the GNU General Public License.  This exception does not however
 | |
|  * invalidate any other reasons why the executable file might be covered by
 | |
|  * the GNU General Public License.
 | |
|  *
 | |
|  * Copyright 2005, 2006, 2007, 2008 Develer S.r.l. (http://www.develer.com/)
 | |
|  * -->
 | |
|  *
 | |
|  * \brief Support for reading program memory on Harvard architectures.
 | |
|  *
 | |
|  * Support is currently provided for AVR microcontrollers only.
 | |
|  *
 | |
|  * These macros allow building code twice, with and without
 | |
|  * pgm support (e.g.: strcpy() and strcpy_P()).
 | |
|  *
 | |
|  * Set the _PROGMEM predefine to compile in conditional
 | |
|  * program-memory support.
 | |
|  *
 | |
|  *
 | |
|  * \note This module contains code ripped out from avr-libc,
 | |
|  *       which is distributed under a 3-clause BSD license.
 | |
|  *
 | |
|  * \author Bernie Innocenti <bernie@codewiz.org>
 | |
|  */
 | |
| #ifndef MWARE_PGM_H
 | |
| #define MWARE_PGM_H
 | |
| 
 | |
| #include <cfg/compiler.h> /* For intXX_t */
 | |
| #include <cpu/detect.h>
 | |
| #include <cpu/attr.h>     /* For CPU_HARVARD */
 | |
| #include <cpu/types.h>    /* For SIZEOF_INT */
 | |
| 
 | |
| #if CPU_AVR
 | |
| 
 | |
| 	#ifdef __AVR_ENHANCED__
 | |
| 		#define pgm_read8(addr) \
 | |
| 		({ \
 | |
| 			uint16_t __addr16 = (uint16_t)(addr); \
 | |
| 			uint8_t __result; \
 | |
| 			__asm__ \
 | |
| 			( \
 | |
| 				"lpm %0, Z" "\n\t" \
 | |
| 				: "=r" (__result) \
 | |
| 				: "z" (__addr16) \
 | |
| 			); \
 | |
| 			__result; \
 | |
| 		})
 | |
| 		#define pgm_read16(addr) \
 | |
| 		({ \
 | |
| 			uint16_t __addr16 = (uint16_t)(addr); \
 | |
| 			uint16_t __result; \
 | |
| 			__asm__ \
 | |
| 			( \
 | |
| 				"lpm %A0, Z+"   "\n\t" \
 | |
| 				"lpm %B0, Z"    "\n\t" \
 | |
| 				: "=r" (__result), "=z" (__addr16) \
 | |
| 				: "1" (__addr16) \
 | |
| 			); \
 | |
| 			__result; \
 | |
| 		})
 | |
| 
 | |
| 
 | |
| 	#else /* !__AVR_ENHANCED__ */
 | |
| 
 | |
| 		#define pgm_read8(addr) \
 | |
| 		({ \
 | |
| 			uint16_t __addr16 = (uint16_t)(addr); \
 | |
| 			uint8_t __result; \
 | |
| 			__asm__ \
 | |
| 			( \
 | |
| 				"lpm" "\n\t" \
 | |
| 				"mov %0, r0" "\n\t" \
 | |
| 				: "=r" (__result) \
 | |
| 				: "z" (__addr16) \
 | |
| 				: "r0" \
 | |
| 			); \
 | |
| 			__result; \
 | |
| 		})
 | |
| 		#define pgm_read16(addr) \
 | |
| 		({ \
 | |
| 			uint16_t __addr16 = (uint16_t)(addr); \
 | |
| 			uint16_t __result; \
 | |
| 			__asm__ \
 | |
| 			( \
 | |
| 				"lpm"           "\n\t" \
 | |
| 				"mov %A0, r0"   "\n\t" \
 | |
| 				"adiw r30, 1"   "\n\t" \
 | |
| 				"lpm"           "\n\t" \
 | |
| 				"mov %B0, r0"   "\n\t" \
 | |
| 				: "=r" (__result), "=z" (__addr16) \
 | |
| 				: "1" (__addr16) \
 | |
| 				: "r0" \
 | |
| 			); \
 | |
| 			__result; \
 | |
| 		})
 | |
| 
 | |
| 	#endif /* !__AVR_ENHANCED__ */
 | |
| 
 | |
| 	#define pgm_read32(addr)	((uint32_t)(pgm_read16(addr) | (((uint32_t)pgm_read16(((const uint8_t *)(addr)) + 2)) << 16)))
 | |
| 	#ifndef PROGMEM
 | |
| 	#define PROGMEM  __attribute__((__progmem__))
 | |
| 	#endif
 | |
| 	#ifndef PSTR
 | |
| 	#define PSTR(s) ({ static const char __c[] PROGMEM = (s); &__c[0]; })
 | |
| 	#endif
 | |
| 	#ifndef PFUNC
 | |
| 	#define PFUNC(x)      x ## _P
 | |
| 	#endif
 | |
| 
 | |
| #elif CPU_HARVARD
 | |
| 	#error Missing CPU support
 | |
| #endif
 | |
| 
 | |
| 
 | |
| #if !CPU_HARVARD
 | |
| 	#define pgm_read8(a)     (*(const uint8_t  *)(a))
 | |
| 	#define pgm_read16(a)    (*(const uint16_t *)(a))
 | |
| 	#define pgm_read32(a)    (*(const uint32_t *)(a))
 | |
| #endif
 | |
| 
 | |
| #define pgm_read_char(a)        pgm_read8(a)
 | |
| #define pgm_read_uint16_t(addr) pgm_read16(addr)
 | |
| 
 | |
| 
 | |
| #if SIZEOF_INT == 2
 | |
| 	#define pgm_read_int(addr) ((int)pgm_read16(addr))
 | |
| #elif SIZEOF_INT == 4
 | |
| 	#define pgm_read_int(addr) ((int)pgm_read32(addr))
 | |
| #else
 | |
| 	#error Missing support for CPU word size!
 | |
| #endif
 | |
| 
 | |
| #ifndef PSTR
 | |
| #define PSTR            /* nothing */
 | |
| #endif
 | |
| 
 | |
| #ifndef PFUNC
 | |
| #define PFUNC(x) x
 | |
| #endif
 | |
| 
 | |
| #ifndef PROGMEM
 | |
| #define PROGMEM         /* nothing */
 | |
| #endif
 | |
| 
 | |
| /**
 | |
|  * \name Types for variables stored in program memory (harvard processors).
 | |
|  * \{
 | |
|  */
 | |
| typedef PROGMEM char pgm_char;
 | |
| typedef PROGMEM int8_t pgm_int8_t;
 | |
| typedef PROGMEM uint8_t pgm_uint8_t;
 | |
| typedef PROGMEM int16_t pgm_int16_t;
 | |
| typedef PROGMEM uint16_t pgm_uint16_t;
 | |
| typedef PROGMEM int32_t pgm_int32_t;
 | |
| typedef PROGMEM uint32_t pgm_uint32_t;
 | |
| /*\}*/
 | |
| 
 | |
| /**
 | |
|  * \name PGM support macros.
 | |
|  *
 | |
|  * These macros enable dual compilation of code for both program
 | |
|  * and data memory.
 | |
|  *
 | |
|  * Such a function may be defined like this:
 | |
|  *
 | |
|  * \code
 | |
|  *	void PGM_FUNC(lcd_puts)(PGM_ATTR const char *str)
 | |
|  *	{
 | |
|  *		char c;
 | |
|  *		while ((c = PGM_READ_CHAR(str++))
 | |
|  *			lcd_putchar(c);
 | |
|  *	}
 | |
|  * \endcode
 | |
|  *
 | |
|  * The above code can be compiled twice: once with the _PROGMEM preprocessor
 | |
|  * symbol defined, and once without.  The two object modules can then be
 | |
|  * linked in the same application for use by client code:
 | |
|  *
 | |
|  * \code
 | |
|  *	lcd_puts("Hello, world!");
 | |
|  *	lcd_puts_P(PSTR("Hello, world!"));
 | |
|  *
 | |
|  *	// To be used when invoking inside other PGM_FUNC functions:
 | |
|  *	PGM_FUNC(lcd_puts)(some_string);
 | |
|  * \endcode
 | |
|  *
 | |
|  * \{
 | |
|  */
 | |
| #ifdef _PROGMEM
 | |
| 	#define PGM_READ8(a)     pgm_read8(a)
 | |
| 	#define PGM_READ16(a)    pgm_read16(a)
 | |
| 	#define PGM_READ32(a)    pgm_read32(a)
 | |
| 	#define PGM_FUNC(x)      PFUNC(x)
 | |
| 	#define PGM_STR(x)       PSTR(x)
 | |
| 	#define PGM_ATTR         PROGMEM
 | |
| #else
 | |
| 	#define PGM_READ8(a)     (*(const uint8_t  *)(a))
 | |
| 	#define PGM_READ16(a)    (*(const uint16_t *)(a))
 | |
| 	#define PGM_READ32(a)    (*(const uint32_t *)(a))
 | |
| 	#define PGM_FUNC(x)      x
 | |
| 	#define PGM_STR(x)       x
 | |
| 	#define PGM_ATTR         /* nothing */
 | |
| #endif
 | |
| 
 | |
| #define PGM_READ_CHAR(addr)      PGM_READ8(addr)
 | |
| 
 | |
| /* \} */
 | |
| 
 | |
| 
 | |
| #endif /* MWARE_PGM_H */
 | 
