coreboot-libre-fam15h-rdimm/3rdparty/chromeec/chip/lm4/system.c

758 lines
19 KiB
C

/* Copyright 2012 The Chromium OS Authors. All rights reserved.
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
/* System module for Chrome EC : LM4 hardware specific implementation */
#include "clock.h"
#include "common.h"
#include "console.h"
#include "cpu.h"
#include "host_command.h"
#include "panic.h"
#include "registers.h"
#include "system.h"
#include "task.h"
#include "timer.h"
#include "util.h"
/* Indices for hibernate data registers */
enum hibdata_index {
HIBDATA_INDEX_SCRATCHPAD, /* General-purpose scratchpad */
HIBDATA_INDEX_WAKE, /* Wake reasons for hibernate */
HIBDATA_INDEX_SAVED_RESET_FLAGS, /* Saved reset flags */
#ifdef CONFIG_SOFTWARE_PANIC
HIBDATA_INDEX_SAVED_PANIC_REASON, /* Saved panic reason */
HIBDATA_INDEX_SAVED_PANIC_INFO, /* Saved panic data */
HIBDATA_INDEX_SAVED_PANIC_EXCEPTION /* Saved panic exception code */
#endif
};
/* Flags for HIBDATA_INDEX_WAKE */
#define HIBDATA_WAKE_RTC BIT(0) /* RTC alarm */
#define HIBDATA_WAKE_HARD_RESET BIT(1) /* Hard reset via short RTC alarm */
#define HIBDATA_WAKE_PIN BIT(2) /* Wake pin */
/*
* Time to hibernate to trigger a power-on reset. 50 ms is sufficient for the
* EC itself, but we need a longer delay to ensure the rest of the components
* on the same power rail are reset and 5VALW has dropped.
*/
#define HIB_RESET_USEC 1000000
/*
* Convert between microseconds and the hibernation module RTC subsecond
* register which has 15-bit resolution. Divide down both numerator and
* denominator to avoid integer overflow while keeping the math accurate.
*/
#define HIB_RTC_USEC_TO_SUBSEC(us) ((us) * (32768/64) / (1000000/64))
#define HIB_RTC_SUBSEC_TO_USEC(ss) ((ss) * (1000000/64) / (32768/64))
/**
* Wait for a write to commit to a hibernate register.
*
* @return EC_SUCCESS if successful, non-zero if error.
*/
static int wait_for_hibctl_wc(void)
{
int i;
/* Wait for write-capable */
for (i = 0; i < 1000000; i++) {
if (LM4_HIBERNATE_HIBCTL & LM4_HIBCTL_WRC)
return EC_SUCCESS;
}
return EC_ERROR_TIMEOUT;
}
/**
* Read hibernate register at specified index.
*
* @return The value of the register or 0 if invalid index.
*/
static uint32_t hibdata_read(enum hibdata_index index)
{
if (index < 0 || index >= LM4_HIBERNATE_HIBDATA_ENTRIES)
return 0;
return LM4_HIBERNATE_HIBDATA[index];
}
/**
* Write hibernate register at specified index.
*
* @return nonzero if error.
*/
static int hibdata_write(enum hibdata_index index, uint32_t value)
{
int rv;
if (index < 0 || index >= LM4_HIBERNATE_HIBDATA_ENTRIES)
return EC_ERROR_INVAL;
/* Wait for ok-to-write */
rv = wait_for_hibctl_wc();
if (rv != EC_SUCCESS)
return rv;
/* Write register */
LM4_HIBERNATE_HIBDATA[index] = value;
/* Wait for write-complete */
return wait_for_hibctl_wc();
}
static void check_reset_cause(void)
{
uint32_t hib_status = LM4_HIBERNATE_HIBRIS;
uint32_t raw_reset_cause = LM4_SYSTEM_RESC;
uint32_t hib_wake_flags = hibdata_read(HIBDATA_INDEX_WAKE);
uint32_t flags = 0;
/* Clear the reset causes now that we've read them */
LM4_SYSTEM_RESC = 0;
wait_for_hibctl_wc();
LM4_HIBERNATE_HIBIC = hib_status;
hibdata_write(HIBDATA_INDEX_WAKE, 0);
if (raw_reset_cause & 0x02) {
/*
* Full power-on reset of chip. This resets the flash
* protection registers to their permanently-stored values.
* Note that this is also triggered by hibernation, because
* that de-powers the chip.
*/
flags |= EC_RESET_FLAG_POWER_ON;
} else if (!flags && (raw_reset_cause & 0x01)) {
/*
* LM4 signals the reset pin in RESC for all power-on resets,
* even though the external pin wasn't asserted. Make setting
* this flag mutually-exclusive with power on flag, so we can
* use it to indicate a keyboard-triggered reset.
*/
flags |= EC_RESET_FLAG_RESET_PIN;
}
if (raw_reset_cause & 0x04)
flags |= EC_RESET_FLAG_BROWNOUT;
if (raw_reset_cause & 0x10)
flags |= EC_RESET_FLAG_SOFT;
if (raw_reset_cause & 0x28) {
/* Watchdog timer 0 or 1 */
flags |= EC_RESET_FLAG_WATCHDOG;
}
/* Handle other raw reset causes */
if (raw_reset_cause && !flags)
flags |= EC_RESET_FLAG_OTHER;
if ((hib_status & 0x09) &&
(hib_wake_flags & HIBDATA_WAKE_HARD_RESET)) {
/* Hibernation caused by software-triggered hard reset */
flags |= EC_RESET_FLAG_HARD;
/* Consume the hibernate reasons so we don't see them below */
hib_status &= ~0x09;
}
if ((hib_status & 0x01) && (hib_wake_flags & HIBDATA_WAKE_RTC))
flags |= EC_RESET_FLAG_RTC_ALARM;
if ((hib_status & 0x08) && (hib_wake_flags & HIBDATA_WAKE_PIN))
flags |= EC_RESET_FLAG_WAKE_PIN;
if (hib_status & 0x04)
flags |= EC_RESET_FLAG_LOW_BATTERY;
/* Restore then clear saved reset flags */
flags |= hibdata_read(HIBDATA_INDEX_SAVED_RESET_FLAGS);
hibdata_write(HIBDATA_INDEX_SAVED_RESET_FLAGS, 0);
system_set_reset_flags(flags);
}
/*
* A3 and earlier chip stepping has a problem accessing flash during shutdown.
* To work around that, we jump to RAM before hibernating. This function must
* live in RAM. It must be called with interrupts disabled, cannot call other
* functions, and can't be declared static (or else the compiler optimizes it
* into the main hibernate function.
*/
void __attribute__((noinline)) __attribute__((section(".iram.text")))
__enter_hibernate(int hibctl)
{
LM4_HIBERNATE_HIBCTL = hibctl;
while (1)
;
}
/**
* Read the real-time clock.
*
* @param ss_ptr Destination for sub-seconds value, if not null.
*
* @return the real-time clock seconds value.
*/
uint32_t system_get_rtc_sec_subsec(uint32_t *ss_ptr)
{
uint32_t rtc, rtc2;
uint32_t rtcss, rtcss2;
/*
* The hibernate module isn't synchronized, so need to read repeatedly
* to guarantee a valid read.
*/
do {
rtc = LM4_HIBERNATE_HIBRTCC;
rtcss = LM4_HIBERNATE_HIBRTCSS & 0x7fff;
rtcss2 = LM4_HIBERNATE_HIBRTCSS & 0x7fff;
rtc2 = LM4_HIBERNATE_HIBRTCC;
} while (rtc != rtc2 || rtcss != rtcss2);
if (ss_ptr)
*ss_ptr = rtcss;
return rtc;
}
timestamp_t system_get_rtc(void)
{
uint32_t rtc, rtc_ss;
timestamp_t time;
rtc = system_get_rtc_sec_subsec(&rtc_ss);
time.val = ((uint64_t)rtc) * SECOND + HIB_RTC_SUBSEC_TO_USEC(rtc_ss);
return time;
}
/**
* Set the real-time clock.
*
* @param seconds New clock value.
*/
void system_set_rtc(uint32_t seconds)
{
wait_for_hibctl_wc();
LM4_HIBERNATE_HIBRTCLD = seconds;
wait_for_hibctl_wc();
}
/**
* Set the hibernate RTC match time at a given time from now
*
* @param seconds Number of seconds from now for RTC match
* @param microseconds Number of microseconds from now for RTC match
*/
static void set_hibernate_rtc_match_time(uint32_t seconds,
uint32_t microseconds)
{
uint32_t rtc, rtcss;
/*
* Make sure that the requested delay is not less then the
* amount of time it takes to set the RTC match registers,
* otherwise, the match event could be missed.
*/
if (seconds == 0 && microseconds < HIB_SET_RTC_MATCH_DELAY_USEC)
microseconds = HIB_SET_RTC_MATCH_DELAY_USEC;
/* Calculate the wake match */
rtc = system_get_rtc_sec_subsec(&rtcss) + seconds;
rtcss += HIB_RTC_USEC_TO_SUBSEC(microseconds);
if (rtcss > 0x7fff) {
rtc += rtcss >> 15;
rtcss &= 0x7fff;
}
/* Set RTC alarm match */
wait_for_hibctl_wc();
LM4_HIBERNATE_HIBRTCM0 = rtc;
wait_for_hibctl_wc();
LM4_HIBERNATE_HIBRTCSS = rtcss << 16;
wait_for_hibctl_wc();
}
/**
* Use hibernate module to set up an RTC interrupt at a given
* time from now
*
* @param seconds Number of seconds before RTC interrupt
* @param microseconds Number of microseconds before RTC interrupt
*/
void system_set_rtc_alarm(uint32_t seconds, uint32_t microseconds)
{
/* Clear pending interrupt */
wait_for_hibctl_wc();
LM4_HIBERNATE_HIBIC = LM4_HIBERNATE_HIBRIS;
/* Set match time */
set_hibernate_rtc_match_time(seconds, microseconds);
/* Enable RTC interrupt on match */
wait_for_hibctl_wc();
LM4_HIBERNATE_HIBIM = 1;
/*
* Wait for the write to commit. This ensures that the RTC interrupt
* actually gets enabled. This is important if we're about to switch
* the system to the 30 kHz oscillator, which might prevent the write
* from committing.
*/
wait_for_hibctl_wc();
}
/**
* Disable and clear the RTC interrupt.
*/
void system_reset_rtc_alarm(void)
{
/* Disable hibernate interrupts */
wait_for_hibctl_wc();
LM4_HIBERNATE_HIBIM = 0;
/* Clear interrupts */
wait_for_hibctl_wc();
LM4_HIBERNATE_HIBIC = LM4_HIBERNATE_HIBRIS;
}
/**
* Hibernate module interrupt
*/
void __hibernate_irq(void)
{
system_reset_rtc_alarm();
}
DECLARE_IRQ(LM4_IRQ_HIBERNATE, __hibernate_irq, 1);
/**
* Enable hibernate interrupt
*/
void system_enable_hib_interrupt(void)
{
task_enable_irq(LM4_IRQ_HIBERNATE);
}
/**
* Internal hibernate function.
*
* @param seconds Number of seconds to sleep before RTC alarm
* @param microseconds Number of microseconds to sleep before RTC alarm
* @param flags Additional hibernate wake flags
*/
static void hibernate(uint32_t seconds, uint32_t microseconds, uint32_t flags)
{
uint32_t hibctl;
/* Set up wake reasons and hibernate flags */
hibctl = LM4_HIBERNATE_HIBCTL | LM4_HIBCTL_PINWEN;
if (flags & HIBDATA_WAKE_PIN)
hibctl |= LM4_HIBCTL_PINWEN;
else
hibctl &= ~LM4_HIBCTL_PINWEN;
if (seconds || microseconds) {
hibctl |= LM4_HIBCTL_RTCWEN;
flags |= HIBDATA_WAKE_RTC;
set_hibernate_rtc_match_time(seconds, microseconds);
/* Enable RTC interrupt on match */
wait_for_hibctl_wc();
LM4_HIBERNATE_HIBIM = 1;
} else {
hibctl &= ~LM4_HIBCTL_RTCWEN;
}
wait_for_hibctl_wc();
LM4_HIBERNATE_HIBCTL = hibctl;
/* Clear pending interrupt */
wait_for_hibctl_wc();
LM4_HIBERNATE_HIBIC = LM4_HIBERNATE_HIBRIS;
/* Store hibernate flags */
hibdata_write(HIBDATA_INDEX_WAKE, flags);
__enter_hibernate(hibctl | LM4_HIBCTL_HIBREQ);
}
void system_hibernate(uint32_t seconds, uint32_t microseconds)
{
/* Flush console before hibernating */
cflush();
hibernate(seconds, microseconds, HIBDATA_WAKE_PIN);
}
void chip_pre_init(void)
{
/* Enable clocks to GPIO block C in run and sleep modes. */
clock_enable_peripheral(CGC_OFFSET_GPIO, 0x0004, CGC_MODE_ALL);
/*
* Ensure PC0:3 are set to JTAG function. They should be set this way
* on a cold boot, but on a warm reboot a previous misbehaving image
* could have set them differently.
*/
if (((LM4_GPIO_PCTL(LM4_GPIO_C) & 0x0000ffff) == 0x00001111) &&
((LM4_GPIO_AFSEL(LM4_GPIO_C) & 0x0f) == 0x0f) &&
((LM4_GPIO_DEN(LM4_GPIO_C) & 0x0f) == 0x0f) &&
((LM4_GPIO_PUR(LM4_GPIO_C) & 0x0f) == 0x0f))
return; /* Already properly configured */
/* Unlock commit register for JTAG pins */
LM4_GPIO_LOCK(LM4_GPIO_C) = LM4_GPIO_LOCK_UNLOCK;
LM4_GPIO_CR(LM4_GPIO_C) |= 0x0f;
/* Reset JTAG pins */
LM4_GPIO_PCTL(LM4_GPIO_C) =
(LM4_GPIO_PCTL(LM4_GPIO_C) & 0xffff0000) | 0x00001111;
LM4_GPIO_AFSEL(LM4_GPIO_C) |= 0x0f;
LM4_GPIO_DEN(LM4_GPIO_C) |= 0x0f;
LM4_GPIO_PUR(LM4_GPIO_C) |= 0x0f;
/* Set interrupt on either edge of the JTAG signals */
LM4_GPIO_IS(LM4_GPIO_C) &= ~0x0f;
LM4_GPIO_IBE(LM4_GPIO_C) |= 0x0f;
/* Re-lock commit register */
LM4_GPIO_CR(LM4_GPIO_C) &= ~0x0f;
LM4_GPIO_LOCK(LM4_GPIO_C) = 0;
}
void system_pre_init(void)
{
uint32_t hibctl;
#ifdef CONFIG_SOFTWARE_PANIC
uint32_t reason, info;
uint8_t exception;
#endif
/*
* Enable clocks to the hibernation module in run, sleep,
* and deep sleep modes.
*/
clock_enable_peripheral(CGC_OFFSET_HIB, 0x1, CGC_MODE_ALL);
/*
* Enable the hibernation oscillator, if it's not already enabled.
* This should only need setting if the EC completely lost power (for
* example, the battery was pulled).
*/
if (!(LM4_HIBERNATE_HIBCTL & LM4_HIBCTL_CLK32EN)) {
int i;
/* Enable clock to hibernate module */
wait_for_hibctl_wc();
LM4_HIBERNATE_HIBCTL |= LM4_HIBCTL_CLK32EN;
/* Wait for write-complete */
for (i = 0; i < 1000000; i++) {
if (LM4_HIBERNATE_HIBRIS & 0x10)
break;
}
/* Enable and reset RTC */
wait_for_hibctl_wc();
LM4_HIBERNATE_HIBCTL |= LM4_HIBCTL_RTCEN;
system_set_rtc(0);
/* Clear all hibernate data entries */
for (i = 0; i < LM4_HIBERNATE_HIBDATA_ENTRIES; i++)
hibdata_write(i, 0);
}
/*
* Set wake reasons to RTC match and WAKE pin by default.
* Before going in to hibernate, these may change.
*/
hibctl = LM4_HIBERNATE_HIBCTL;
hibctl |= LM4_HIBCTL_RTCWEN;
hibctl |= LM4_HIBCTL_PINWEN;
wait_for_hibctl_wc();
LM4_HIBERNATE_HIBCTL = hibctl;
/*
* Initialize registers after reset to work around LM4 chip errata
* (still present in A3 chip stepping).
*/
wait_for_hibctl_wc();
LM4_HIBERNATE_HIBRTCT = 0x7fff;
wait_for_hibctl_wc();
LM4_HIBERNATE_HIBIM = 0;
check_reset_cause();
#ifdef CONFIG_SOFTWARE_PANIC
/* Restore then clear saved panic reason */
reason = hibdata_read(HIBDATA_INDEX_SAVED_PANIC_REASON);
info = hibdata_read(HIBDATA_INDEX_SAVED_PANIC_INFO);
exception = hibdata_read(HIBDATA_INDEX_SAVED_PANIC_EXCEPTION);
if (reason || info || exception) {
panic_set_reason(reason, info, exception);
hibdata_write(HIBDATA_INDEX_SAVED_PANIC_REASON, 0);
hibdata_write(HIBDATA_INDEX_SAVED_PANIC_INFO, 0);
hibdata_write(HIBDATA_INDEX_SAVED_PANIC_EXCEPTION, 0);
}
#endif
/* Initialize bootcfg if needed */
if (LM4_SYSTEM_BOOTCFG != CONFIG_BOOTCFG_VALUE) {
/* read-modify-write */
LM4_FLASH_FMD = (LM4_SYSTEM_BOOTCFG_MASK & LM4_SYSTEM_BOOTCFG)
| (~LM4_SYSTEM_BOOTCFG_MASK & CONFIG_BOOTCFG_VALUE);
LM4_FLASH_FMA = 0x75100000;
LM4_FLASH_FMC = 0xa4420008; /* WRKEY | COMT */
while (LM4_FLASH_FMC & 0x08)
;
}
/* Brown-outs should trigger a reset */
LM4_SYSTEM_PBORCTL |= 0x02;
}
void system_reset(int flags)
{
uint32_t save_flags = 0;
/* Disable interrupts to avoid task swaps during reboot */
interrupt_disable();
/* Save current reset reasons if necessary */
if (flags & SYSTEM_RESET_PRESERVE_FLAGS)
save_flags = system_get_reset_flags() | EC_RESET_FLAG_PRESERVED;
if (flags & SYSTEM_RESET_LEAVE_AP_OFF)
save_flags |= EC_RESET_FLAG_AP_OFF;
hibdata_write(HIBDATA_INDEX_SAVED_RESET_FLAGS, save_flags);
if (flags & SYSTEM_RESET_HARD) {
#ifdef CONFIG_SOFTWARE_PANIC
uint32_t reason, info;
uint8_t exception;
/* Panic data will be wiped by hard reset, so save it */
panic_get_reason(&reason, &info, &exception);
hibdata_write(HIBDATA_INDEX_SAVED_PANIC_REASON, reason);
hibdata_write(HIBDATA_INDEX_SAVED_PANIC_INFO, info);
hibdata_write(HIBDATA_INDEX_SAVED_PANIC_EXCEPTION, exception);
#endif
/*
* Bounce through hibernate to trigger a hard reboot. Do
* not wake on wake pin, since we need the full duration.
*/
hibernate(0, HIB_RESET_USEC, HIBDATA_WAKE_HARD_RESET);
} else
CPU_NVIC_APINT = 0x05fa0004;
/* Spin and wait for reboot; should never return */
while (1)
;
}
int system_set_scratchpad(uint32_t value)
{
return hibdata_write(HIBDATA_INDEX_SCRATCHPAD, value);
}
uint32_t system_get_scratchpad(void)
{
return hibdata_read(HIBDATA_INDEX_SCRATCHPAD);
}
const char *system_get_chip_vendor(void)
{
return "ti";
}
static char to_hex(int x)
{
if (x >= 0 && x <= 9)
return '0' + x;
return 'a' + x - 10;
}
const char *system_get_chip_id_string(void)
{
static char str[15] = "Unknown-";
char *p = str + 8;
uint32_t did = LM4_SYSTEM_DID1 >> 16;
if (*p)
return (const char *)str;
*p = to_hex(did >> 12);
*(p + 1) = to_hex((did >> 8) & 0xf);
*(p + 2) = to_hex((did >> 4) & 0xf);
*(p + 3) = to_hex(did & 0xf);
*(p + 4) = '\0';
return (const char *)str;
}
const char *system_get_raw_chip_name(void)
{
switch ((LM4_SYSTEM_DID1 & 0xffff0000) >> 16) {
case 0x10de:
return "tm4e1g31h6zrb";
case 0x10e2:
return "lm4fsxhh5bb";
case 0x10e3:
return "lm4fs232h5bb";
case 0x10e4:
return "lm4fs99h5bb";
case 0x10e6:
return "lm4fs1ah5bb";
case 0x10ea:
return "lm4fs1gh5bb";
default:
return system_get_chip_id_string();
}
}
const char *system_get_chip_name(void)
{
const char *postfix = "-tm"; /* test mode */
static char str[20];
const char *raw_chip_name = system_get_raw_chip_name();
char *p = str;
if (LM4_TEST_MODE_ENABLED) {
/* Debug mode is enabled. Postfix chip name. */
while (*raw_chip_name)
*(p++) = *(raw_chip_name++);
while (*postfix)
*(p++) = *(postfix++);
*p = '\0';
return (const char *)str;
} else {
return raw_chip_name;
}
}
int system_get_bbram(enum system_bbram_idx idx, uint8_t *value)
{
return EC_ERROR_UNIMPLEMENTED;
}
int system_set_bbram(enum system_bbram_idx idx, uint8_t value)
{
return EC_ERROR_UNIMPLEMENTED;
}
const char *system_get_chip_revision(void)
{
static char rev[3];
/* Extract the major[15:8] and minor[7:0] revisions. */
rev[0] = 'A' + ((LM4_SYSTEM_DID0 >> 8) & 0xff);
rev[1] = '0' + (LM4_SYSTEM_DID0 & 0xff);
rev[2] = 0;
return rev;
}
/*****************************************************************************/
/* Console commands */
void print_system_rtc(enum console_channel ch)
{
uint32_t rtc;
uint32_t rtcss;
rtc = system_get_rtc_sec_subsec(&rtcss);
cprintf(ch, "RTC: 0x%08x.%04x (%d.%06d s)\n",
rtc, rtcss, rtc, HIB_RTC_SUBSEC_TO_USEC(rtcss));
}
#ifdef CONFIG_CMD_RTC
static int command_system_rtc(int argc, char **argv)
{
if (argc == 3 && !strcasecmp(argv[1], "set")) {
char *e;
uint32_t t = strtoi(argv[2], &e, 0);
if (*e)
return EC_ERROR_PARAM2;
system_set_rtc(t);
} else if (argc > 1) {
return EC_ERROR_INVAL;
}
print_system_rtc(CC_COMMAND);
return EC_SUCCESS;
}
DECLARE_CONSOLE_COMMAND(rtc, command_system_rtc,
"[set <seconds>]",
"Get/set real-time clock");
#ifdef CONFIG_CMD_RTC_ALARM
/**
* Test the RTC alarm by setting an interrupt on RTC match.
*/
static int command_rtc_alarm_test(int argc, char **argv)
{
int s = 1, us = 0;
char *e;
ccprintf("Setting RTC alarm\n");
system_enable_hib_interrupt();
if (argc > 1) {
s = strtoi(argv[1], &e, 10);
if (*e)
return EC_ERROR_PARAM1;
}
if (argc > 2) {
us = strtoi(argv[2], &e, 10);
if (*e)
return EC_ERROR_PARAM2;
}
system_set_rtc_alarm(s, us);
return EC_SUCCESS;
}
DECLARE_CONSOLE_COMMAND(rtc_alarm, command_rtc_alarm_test,
"[seconds [microseconds]]",
"Test alarm");
#endif /* CONFIG_CMD_RTC_ALARM */
#endif /* CONFIG_CMD_RTC */
/*****************************************************************************/
/* Host commands */
#ifdef CONFIG_HOSTCMD_RTC
static enum ec_status system_rtc_get_value(struct host_cmd_handler_args *args)
{
struct ec_response_rtc *r = args->response;
r->time = system_get_rtc_sec_subsec(NULL);
args->response_size = sizeof(*r);
return EC_RES_SUCCESS;
}
DECLARE_HOST_COMMAND(EC_CMD_RTC_GET_VALUE,
system_rtc_get_value,
EC_VER_MASK(0));
static enum ec_status system_rtc_set_value(struct host_cmd_handler_args *args)
{
const struct ec_params_rtc *p = args->params;
system_set_rtc(p->time);
return EC_RES_SUCCESS;
}
DECLARE_HOST_COMMAND(EC_CMD_RTC_SET_VALUE,
system_rtc_set_value,
EC_VER_MASK(0));
#endif /* CONFIG_HOSTCMD_RTC */