2013-03-06 10:29:49 +01:00
|
|
|
/**
|
|
|
|
* Native CPU lpm.h implementation
|
|
|
|
*
|
2013-03-13 21:56:56 +01:00
|
|
|
* Uses system calls to emulate CPU power modes.
|
|
|
|
*
|
2014-05-15 18:07:02 +02:00
|
|
|
* Copyright (C) 2013 Ludwig Ortmann <ludwig.ortmann@fu-berlin.de>
|
2013-03-06 10:29:49 +01:00
|
|
|
*
|
2013-11-22 20:47:05 +01:00
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser General
|
2013-06-18 17:21:38 +02:00
|
|
|
* Public License. See the file LICENSE in the top level directory for more
|
|
|
|
* details.
|
2013-03-06 10:29:49 +01:00
|
|
|
*
|
2013-03-13 21:56:56 +01:00
|
|
|
* @ingroup lpm
|
|
|
|
* @ingroup native_cpu
|
2013-03-06 10:29:49 +01:00
|
|
|
* @{
|
|
|
|
* @file
|
|
|
|
* @author Ludwig Ortmann <ludwig.ortmann@fu-berlin.de>
|
|
|
|
* @}
|
|
|
|
*/
|
2013-03-06 01:08:15 +01:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2013-03-07 13:53:14 +01:00
|
|
|
#include <unistd.h>
|
2013-05-14 18:31:47 +02:00
|
|
|
#ifdef MODULE_UART0
|
|
|
|
#include <sys/select.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#endif
|
2013-06-13 18:14:21 +02:00
|
|
|
#include <err.h>
|
2013-03-06 01:08:15 +01:00
|
|
|
|
|
|
|
#include "lpm.h"
|
|
|
|
#include "debug.h"
|
2013-05-14 18:31:47 +02:00
|
|
|
#include "cpu.h"
|
2013-11-07 17:23:08 +01:00
|
|
|
|
|
|
|
#include "native_internal.h"
|
2013-05-14 18:31:47 +02:00
|
|
|
#ifdef MODULE_UART0
|
2013-07-15 20:52:53 +02:00
|
|
|
#include "board_internal.h"
|
2013-05-14 18:31:47 +02:00
|
|
|
#endif
|
2013-03-06 01:08:15 +01:00
|
|
|
|
|
|
|
static enum lpm_mode native_lpm;
|
|
|
|
|
|
|
|
void lpm_init(void)
|
|
|
|
{
|
|
|
|
DEBUG("lpm_init()\n");
|
|
|
|
native_lpm = LPM_ON;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-05-07 12:36:32 +02:00
|
|
|
void _native_lpm_sleep(void)
|
2013-05-14 18:31:47 +02:00
|
|
|
{
|
|
|
|
#ifdef MODULE_UART0
|
2013-08-30 17:03:34 +02:00
|
|
|
int nfds;
|
2013-06-26 23:29:09 +02:00
|
|
|
|
|
|
|
/* set fds */
|
|
|
|
FD_ZERO(&_native_rfds);
|
|
|
|
nfds = _native_set_uart_fds();
|
|
|
|
nfds++;
|
|
|
|
|
2013-11-07 17:23:08 +01:00
|
|
|
_native_in_syscall++; // no switching here
|
2013-08-30 17:03:34 +02:00
|
|
|
nfds = select(nfds, &_native_rfds, NULL, NULL, NULL);
|
2013-11-07 17:23:08 +01:00
|
|
|
_native_in_syscall--;
|
|
|
|
|
2013-08-30 17:03:34 +02:00
|
|
|
DEBUG("_native_lpm_sleep: returned: %i\n", nfds);
|
2013-06-21 03:52:57 +02:00
|
|
|
|
2013-08-30 17:03:34 +02:00
|
|
|
if (nfds != -1) {
|
2013-06-13 18:14:21 +02:00
|
|
|
/* uart ready, handle input */
|
|
|
|
/* TODO: switch to ISR context */
|
|
|
|
_native_handle_uart0_input();
|
2013-06-12 16:18:32 +02:00
|
|
|
}
|
2014-01-29 10:53:09 +01:00
|
|
|
else if ((errno == EAGAIN) || (errno == EWOULDBLOCK)) {
|
|
|
|
/* would block / resource unavailable .. it appears a
|
|
|
|
* contended socket can show this behavior sometimes */
|
|
|
|
_native_in_syscall++;
|
|
|
|
warn("_native_lpm_sleep: select()");
|
|
|
|
_native_in_syscall--;
|
|
|
|
return;
|
|
|
|
}
|
2013-06-24 22:37:35 +02:00
|
|
|
else if (errno != EINTR) {
|
2013-06-13 18:14:21 +02:00
|
|
|
/* select failed for reason other than signal */
|
2013-11-20 15:25:11 +01:00
|
|
|
err(EXIT_FAILURE, "lpm_set(): select()");
|
2013-05-14 18:31:47 +02:00
|
|
|
}
|
2013-06-21 03:52:57 +02:00
|
|
|
|
2013-06-13 18:14:21 +02:00
|
|
|
/* otherwise select was interrupted because of a signal, continue below */
|
2013-05-14 18:31:47 +02:00
|
|
|
#else
|
2013-11-13 23:43:26 +01:00
|
|
|
_native_in_syscall++; // no switching here
|
2014-06-20 20:00:20 +02:00
|
|
|
real_pause();
|
2013-11-13 23:43:26 +01:00
|
|
|
_native_in_syscall--;
|
2013-06-21 03:52:57 +02:00
|
|
|
#endif
|
2013-05-14 18:31:47 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (_native_sigpend > 0) {
|
2013-05-14 18:31:47 +02:00
|
|
|
DEBUG("\n\n\t\treturn from syscall, calling native_irq_handler\n\n");
|
2013-11-07 17:23:08 +01:00
|
|
|
_native_in_syscall++;
|
|
|
|
_native_syscall_leave();
|
2013-05-14 18:31:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-13 21:56:56 +01:00
|
|
|
/**
|
|
|
|
* LPM_IDLE uses sleep() to wait for interrupts
|
|
|
|
* LPM_OFF exits process
|
|
|
|
* other modes not supported at the moment
|
|
|
|
*/
|
2013-03-06 01:08:15 +01:00
|
|
|
enum lpm_mode lpm_set(enum lpm_mode target)
|
|
|
|
{
|
|
|
|
enum lpm_mode last_lpm;
|
|
|
|
|
2013-05-14 18:31:47 +02:00
|
|
|
//DEBUG("lpm_set(%i)\n", target);
|
2013-03-06 01:08:15 +01:00
|
|
|
|
|
|
|
last_lpm = native_lpm;
|
|
|
|
native_lpm = target;
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
switch(native_lpm) { /* @contiki :-p */
|
2013-03-06 01:08:15 +01:00
|
|
|
case LPM_ON:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LPM_IDLE:
|
2013-05-14 18:31:47 +02:00
|
|
|
//DEBUG("lpm_set(): pause()\n");
|
|
|
|
|
|
|
|
//pause();
|
|
|
|
_native_lpm_sleep();
|
2013-03-06 01:08:15 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
/* XXX: unfinished modes: */
|
|
|
|
case LPM_SLEEP:
|
|
|
|
/*TODO: implement*/
|
|
|
|
printf("XXX: lpm_set(): LPM_SLEEP not implemented\n");
|
|
|
|
//sigsuspend();
|
|
|
|
|
|
|
|
case LPM_POWERDOWN:
|
|
|
|
/*TODO: implement*/
|
|
|
|
printf("XXX: lpm_set(): LPM_POWERDOWN not implemented\n");
|
|
|
|
//sigsuspend();
|
|
|
|
|
|
|
|
case LPM_OFF:
|
|
|
|
printf("lpm_set(): exit()\n");
|
|
|
|
exit(0);
|
|
|
|
|
|
|
|
default:
|
|
|
|
DEBUG("XXX: unsupported power mode: %i\n", native_lpm);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
return last_lpm;
|
|
|
|
}
|
|
|
|
|
|
|
|
void lpm_awake(void)
|
|
|
|
{
|
|
|
|
DEBUG("XXX: lpm_awake()\n");
|
|
|
|
native_lpm = LPM_ON;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
void lpm_begin_awake(void)
|
|
|
|
{
|
|
|
|
DEBUG("XXX: lpm_begin_awake()\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
void lpm_end_awake(void)
|
|
|
|
{
|
|
|
|
DEBUG("XXX: lpm_end_awake()\n");
|
|
|
|
native_lpm = LPM_ON;
|
|
|
|
return;
|
|
|
|
}
|
2013-03-13 21:56:56 +01:00
|
|
|
|
2013-03-06 01:08:15 +01:00
|
|
|
enum lpm_mode lpm_get(void)
|
|
|
|
{
|
|
|
|
return native_lpm;
|
|
|
|
}
|