drv_standby.c 2.2 KB
Newer Older
1
/*
2
 * Copyright (c) 2019 Winner Microelectronics Co., Ltd.
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35
 *
 * SPDX-License-Identifier: Apache-2.0
 *
 * Change Logs:
 * Date           Author       Notes
 * 2019-03-13     tyx          first version
 */

#include <rthw.h>
#include <rtthread.h>
#include <rtdevice.h>
#include "drv_standby.h"

#ifdef BSP_USING_STANDBY
#include "wm_type_def.h"
#include "wm_cpu.h"
#include "wm_pmu.h"
#include "wm_irq.h"
#include "wm_regs.h"

typedef volatile unsigned long vu32;
#define M32(adr)    (*((vu32*) (adr)))
typedef void (*rom_standby_func)(void);

static const rom_standby_func pm_standby = (rom_standby_func)0x499;

#ifdef __ICCARM__
extern void standby_idr(void);
#endif

#if (1 == GCC_COMPILE)
void wm_pm_standby(void)
{
mysterywolf's avatar
mysterywolf 已提交
36
    __asm volatile (
37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78
        " cpsid            i    \n"  /* disable irq*/
        " dsb                   \n"
        " ldr r0, =0X499        \n"
        " bx r0                 \n"
        " movs r0, #0x00        \n"
        " isb                   \n"
    );
}
#endif

/**
 * This function will put w60x into run/shutdown mode.
 *
 * @param timeout How many OS Ticks that MCU can sleep
 */
void sys_start_standby(int ms)
{
    rt_uint32_t val;
    int timeout = ms;

    RT_ASSERT(timeout > 0);

    tls_pmu_clk_select(0);

    if (timeout <= 65535)
    {
        /* Enter PM_TIMER_MODE */
        tls_irq_enable(PMU_TIMER1_INT);
        tls_pmu_timer1_stop();
        tls_pmu_timer1_start(timeout);
    }
    else if(timeout <= 65535000)
    {
        timeout /= 1000;
        tls_irq_enable(PMU_TIMER0_INT);
        tls_pmu_timer0_stop();
        tls_pmu_timer0_start(timeout);
    }
    else
    {
        return;
    }
mysterywolf's avatar
mysterywolf 已提交
79
    tls_irq_enable(PMU_GPIO_WAKEUP_INT);    //Open interrupt by default to clear the interrupt flag for IO wake-up
80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99
    val = tls_reg_read32(HR_PMU_PS_CR);
    val |= 0x01;
    tls_reg_write32(HR_PMU_PS_CR, val);
}

#ifdef RT_USING_FINSH
#include <finsh.h>
#include <stdlib.h>
static void standby(uint8_t argc, char **argv)
{
    if (argc != 2)
    {
        rt_kprintf("Usage: standby timeout(ms)\n");
        rt_kprintf("eg   : standby 5000\n");
    }
    else
    {
        sys_start_standby(atoi(argv[1]));
    }
}
100
MSH_CMD_EXPORT(standby, sleep system);
101 102 103
#endif /* RT_USING_FINSH */

#endif /* BSP_USING_STANDBY */