123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481 |
- #include <hal_gpio.h>
- #include <hal_dma.h>
- #include <hal_clk.h>
- #include <hal_reset.h>
- #include <hal_cache.h>
- #include <sunxi_hal_ledc.h>
- #ifdef LEDC_DEBUG
- #define ledc_info(fmt, args...) printf("%s()%d - "fmt, __func__, __LINE__, ##args)
- #else
- #define ledc_info(fmt, args...)
- #endif
- static unsigned long base_addr = LEDC_BASE;
- struct sunxi_dma_chan *dma_chan;
- static int ledc_clk_init(void)
- {
- hal_clk_type_t clk_type = HAL_SUNXI_CCU;
- hal_clk_id_t mod_clk_id = CLK_LEDC;
- hal_clk_id_t bus_clk_id = CLK_BUS_LEDC;
- hal_clk_t mod_clk;
- hal_clk_t bus_clk;
- hal_reset_type_t reset_type = HAL_SUNXI_RESET;
- hal_reset_id_t reset_id = RST_BUS_LEDC;
- struct reset_control *reset;
- reset = hal_reset_control_get(reset_type, reset_id);
- if (hal_reset_control_deassert(reset))
- {
- ledc_info("ledc reset deassert failed!");
- return -1;
- }
- hal_reset_control_put(reset);
- mod_clk = hal_clock_get(clk_type, mod_clk_id);
- if (hal_clock_enable(mod_clk))
- {
- ledc_info("ledc clk enable mclk failed!");
- return -1;
- }
- bus_clk = hal_clock_get(clk_type, bus_clk_id);
- if (hal_clock_enable(bus_clk))
- {
- ledc_info("ledc clk enable mclk failed!");
- return -1;
- }
- return 0;
- }
- static int ledc_pinctrl_init(void)
- {
- if (hal_gpio_pinmux_set_function(LEDC_PIN, LEDC_PINMUXSEL))
- {
- ledc_info("ledc pin set default function failed!");
- return -1;
- }
- return 0;
- }
- static void ledc_dump_reg(void)
- {
- ledc_info("LEDC_CTRL_REG = %0x\n", hal_readl(base_addr + LEDC_CTRL_REG));
- ledc_info("LED_T01_TIMING_CTRL_REG = %0x\n", hal_readl(base_addr + LED_T01_TIMING_CTRL_REG));
- ledc_info("LEDC_DATA_FINISH_CNT_REG = %0x\n", hal_readl(base_addr + LEDC_DATA_FINISH_CNT_REG));
- ledc_info("LED_RST_TIMING_CTRL_REG = %0x\n", hal_readl(base_addr + LED_RST_TIMING_CTRL_REG));
- ledc_info("LEDC_WAIT_TIME0_CTRL_REG = %0x\n", hal_readl(base_addr + LEDC_WAIT_TIME0_CTRL_REG));
- ledc_info("LEDC_DATA_REG = %0x\n", hal_readl(base_addr + LEDC_DATA_REG));
- ledc_info("LEDC_DMA_CTRL_REG = %0x\n", hal_readl(base_addr + LEDC_DMA_CTRL_REG));
- ledc_info("LEDC_INTC_REG = %0x\n", hal_readl(base_addr + LEDC_INTC_REG));
- ledc_info("LEDC_INTS_REG = %0x\n", hal_readl(base_addr + LEDC_INTS_REG));
- ledc_info("LEDC_WAIT_TIME1_CTRL_REG = %0x\n", hal_readl(base_addr + LEDC_WAIT_TIME1_CTRL_REG));
- ledc_info("LEDC_FIFO_DATA0_REG = %0x\n", hal_readl(base_addr + LEDC_FIFO_DATA0_REG));
- }
- static void ledc_set_reset_ns(unsigned int reset_ns)
- {
- unsigned int n, reg_val;
- unsigned int mask = 0x1FFF;
- unsigned int min = LEDC_RESET_TIME_MIN_NS;
- unsigned int max = LEDC_RESET_TIME_MAX_NS;
- if (reset_ns < min || reset_ns > max) {
- ledc_info("invalid parameter, reset_ns should be %u-%u!\n", min, max);
- return;
- }
- n = (reset_ns - 42) / 42;
- reg_val = hal_readl(base_addr + LED_RST_TIMING_CTRL_REG);
- reg_val &= ~(mask << 16);
- reg_val |= (n << 16);
- hal_writel(reg_val, base_addr + LED_RST_TIMING_CTRL_REG);
- }
- static void ledc_set_t1h_ns(unsigned int t1h_ns)
- {
- unsigned int n, reg_val;
- unsigned int mask = 0x3F;
- unsigned int shift = 21;
- unsigned int min = LEDC_T1H_MIN_NS;
- unsigned int max = LEDC_T1H_MAX_NS;
- if (t1h_ns < min || t1h_ns > max) {
- ledc_info("invalid parameter, t1h_ns should be %u-%u!\n", min, max);
- return;
- }
- n = (t1h_ns - 42) / 42;
- reg_val = hal_readl(base_addr + LED_T01_TIMING_CTRL_REG);
- reg_val &= ~(mask << shift);
- reg_val |= n << shift;
- hal_writel(reg_val, base_addr + LED_T01_TIMING_CTRL_REG);
- }
- static void ledc_set_t1l_ns(unsigned int t1l_ns)
- {
- unsigned int n, reg_val;
- unsigned int mask = 0x1F;
- unsigned int shift = 16;
- unsigned int min = LEDC_T1L_MIN_NS;
- unsigned int max = LEDC_T1L_MAX_NS;
- if (t1l_ns < min || t1l_ns > max) {
- ledc_info("invalid parameter, t1l_ns should be %u-%u!\n", min, max);
- return;
- }
- n = (t1l_ns - 42) / 42;
- reg_val = hal_readl(base_addr + LED_T01_TIMING_CTRL_REG);
- reg_val &= ~(mask << shift);
- reg_val |= n << shift;
- hal_writel(reg_val, base_addr + LED_T01_TIMING_CTRL_REG);
- }
- static void ledc_set_t0h_ns(unsigned int t0h_ns)
- {
- unsigned int n, reg_val;
- unsigned int mask = 0x1F;
- unsigned int shift = 6;
- unsigned int min = LEDC_T0H_MIN_NS;
- unsigned int max = LEDC_T0H_MAX_NS;
- if (t0h_ns < min || t0h_ns > max) {
- ledc_info("invalid parameter, t0h_ns should be %u-%u!\n", min, max);
- return;
- }
- n = (t0h_ns - 42) / 42;
- reg_val = hal_readl(base_addr + LED_T01_TIMING_CTRL_REG);
- reg_val &= ~(mask << shift);
- reg_val |= n << shift;
- hal_writel(reg_val, base_addr + LED_T01_TIMING_CTRL_REG);
- }
- static void ledc_set_t0l_ns(unsigned int t0l_ns)
- {
- unsigned int n, reg_val;
- unsigned int min = LEDC_T0L_MIN_NS;
- unsigned int max = LEDC_T0L_MAX_NS;
- if (t0l_ns < min || t0l_ns > max) {
- ledc_info("invalid parameter, t0l_ns should be %u-%u!\n", min, max);
- return;
- }
- n = (t0l_ns - 42) / 42;
- reg_val = hal_readl(base_addr + LED_T01_TIMING_CTRL_REG);
- reg_val &= ~0x3F;
- reg_val |= n;
- hal_writel(reg_val, base_addr + LED_T01_TIMING_CTRL_REG);
- }
- static void ledc_set_wait_time0_ns(unsigned int wait_time0_ns)
- {
- unsigned int n, reg_val;
- unsigned int min = LEDC_WAIT_TIME0_MIN_NS;
- unsigned int max = LEDC_WAIT_TIME0_MAX_NS;
- if (wait_time0_ns < min || wait_time0_ns > max) {
- ledc_info("invalid parameter, wait_time0_ns should be %u-%u!\n", min, max);
- return;
- }
- n = (wait_time0_ns - 42) / 42;
- reg_val = (1 << 8) | n;
- hal_writel(reg_val, base_addr + LEDC_WAIT_TIME0_CTRL_REG);
- }
- static void ledc_set_wait_time1_ns(unsigned long long wait_time1_ns)
- {
- unsigned long long tmp, max = LEDC_WAIT_TIME1_MAX_NS;
- unsigned int min = LEDC_WAIT_TIME1_MIN_NS;
- unsigned int n, reg_val;
- if (wait_time1_ns < min || wait_time1_ns > max) {
- ledc_info("invalid parameter, wait_time1_ns should be %u-%llu!\n", min, max);
- return;
- }
- n = wait_time1_ns / 42;
- //tmp = wait_time1_ns;
- //n = div_u64(tmp, 42);
- n -= 1;
- reg_val = (1 << 31) | n;
- hal_writel(reg_val, base_addr + LEDC_WAIT_TIME1_CTRL_REG);
- }
- static void ledc_set_wait_data_time_ns(unsigned int wait_data_time_ns)
- {
- unsigned int mask = 0x1FFF;
- unsigned int shift = 16;
- unsigned int reg_val = 0;
- unsigned int n, min, max;
- min = LEDC_WAIT_DATA_TIME_MIN_NS;
- max = LEDC_WAIT_DATA_TIME_MAX_NS_IC;
- if (wait_data_time_ns < min || wait_data_time_ns > max) {
- ledc_info("invalid parameter, wait_data_time_ns should be %u-%u!\n",
- min, max);
- return;
- }
- n = (wait_data_time_ns - 42) / 42;
- reg_val &= ~(mask << shift);
- reg_val |= (n << shift);
- hal_writel(reg_val, base_addr + LEDC_DATA_FINISH_CNT_REG);
- }
- static void ledc_set_length(unsigned int length)
- {
- unsigned int reg_val;
- if (length == 0)
- return;
- reg_val = hal_readl(base_addr + LEDC_CTRL_REG);
- reg_val &= ~(0x1FFF << 16);
- reg_val |= length << 16;
- hal_writel(reg_val, base_addr + LEDC_CTRL_REG);
- reg_val = hal_readl(base_addr + LED_RST_TIMING_CTRL_REG);
- reg_val &= ~0x3FF;
- reg_val |= length - 1;
- hal_writel(reg_val, base_addr + LED_RST_TIMING_CTRL_REG);
- }
- static void ledc_set_output_mode(const char *str)
- {
- unsigned int val = 0;
- unsigned int mask = 0x7;
- unsigned int shift = 6;
- unsigned int reg_val ;
- if (str != NULL) {
- if (!strncmp(str, "GRB", 3))
- val = LEDC_OUTPUT_GRB;
- else if (!strncmp(str, "GBR", 3))
- val = LEDC_OUTPUT_GBR;
- else if (!strncmp(str, "RGB", 3))
- val = LEDC_OUTPUT_RGB;
- else if (!strncmp(str, "RBG", 3))
- val = LEDC_OUTPUT_RBG;
- else if (!strncmp(str, "BGR", 3))
- val = LEDC_OUTPUT_BGR;
- else if (!strncmp(str, "BRG", 3))
- val = LEDC_OUTPUT_BRG;
- else
- return;
- } else {
- }
- reg_val = hal_readl(base_addr + LEDC_CTRL_REG);
- reg_val &= ~(mask << shift);
- reg_val |= val;
- hal_writel(reg_val, base_addr + LEDC_CTRL_REG);
- }
- static void ledc_disable_irq(unsigned int mask)
- {
- unsigned int reg_val = 0;
- reg_val = hal_readl(base_addr + LEDC_INTC_REG);
- reg_val &= ~mask;
- hal_writel(reg_val, base_addr + LEDC_INTC_REG);
- }
- static void ledc_enable_irq(unsigned int mask)
- {
- unsigned int reg_val = 0;
- reg_val = hal_readl(base_addr + LEDC_INTC_REG);
- reg_val |= mask;
- hal_writel(reg_val, base_addr + LEDC_INTC_REG);
- }
- static void ledc_set_dma_mode(void)
- {
- unsigned int reg_val = 0;
- reg_val |= 1 << 5;
- hal_writel(reg_val, base_addr + LEDC_DMA_CTRL_REG);
- }
- static void ledc_set_cpu_mode(void)
- {
- unsigned int reg_val = 0;
- reg_val &= ~(1 << 5);
- hal_writel(reg_val, base_addr + LEDC_DMA_CTRL_REG);
- }
- static void ledc_clear_all_irq(void)
- {
- unsigned int reg_val;
- reg_val = hal_readl(base_addr + LEDC_INTS_REG);
- reg_val |= 0x1F;
- hal_writel(reg_val, base_addr + LEDC_INTS_REG);
- }
- static unsigned int ledc_get_irq_status(void)
- {
- return hal_readl(base_addr + LEDC_INTS_REG);
- }
- static void ledc_soft_reset(void)
- {
- unsigned int reg_val;
- reg_val = hal_readl(base_addr + LEDC_CTRL_REG);
- reg_val |= 1 << 1;
- hal_writel(reg_val, base_addr + LEDC_CTRL_REG);
- }
- static void ledc_reset_en(void)
- {
- unsigned int reg_val;
- reg_val = hal_readl(base_addr + LEDC_CTRL_REG);
- reg_val |= 1 << 10;
- hal_writel(reg_val, base_addr + LEDC_CTRL_REG);
- }
- static void ledc_set_data(unsigned int data)
- {
- hal_writel(data, base_addr + LEDC_DATA_REG);
- }
- static void ledc_enable(void)
- {
- unsigned int reg_val;
- reg_val = hal_readl(base_addr + LEDC_CTRL_REG);
- reg_val |= 1;
- hal_writel(reg_val, base_addr + LEDC_CTRL_REG);
- }
- static void hal_ledc_set_time(struct ledc_config *ledc)
- {
- ledc_set_reset_ns(ledc->reset_ns);
- ledc_set_t1h_ns(ledc->t1h_ns);
- ledc_set_t1l_ns(ledc->t1l_ns);
- ledc_set_t0h_ns(ledc->t0h_ns);
- ledc_set_t0l_ns(ledc->t0l_ns);
- ledc_set_wait_time0_ns(ledc->wait_time0_ns);
- ledc_set_wait_time1_ns(ledc->wait_time1_ns);
- ledc_set_wait_data_time_ns(ledc->wait_data_time_ns);
- }
- void hal_ledc_dma_callback(void *para)
- {
- printf("dma callback\n");
- }
- void hal_ledc_trans_data(struct ledc_config *ledc)
- {
- int i;
- unsigned long int size;
- unsigned int mask = 0;
- struct dma_slave_config slave_config;
- mask = LEDC_TRANS_FINISH_INT_EN | LEDC_WAITDATA_TIMEOUT_INT_EN
- | LEDC_FIFO_OVERFLOW_INT_EN | LEDC_GLOBAL_INT_EN;
- if (ledc->length <= SUNXI_LEDC_FIFO_DEPTH) {
- ledc_info("trans data by CPU mode\n");
- mask |= LEDC_FIFO_CPUREQ_INT_EN;
- ledc_reset_en();
- hal_ledc_set_time(ledc);
- ledc_set_output_mode(ledc->output_mode);
- ledc_set_cpu_mode();
- ledc_set_length(ledc->length);
- ledc_enable_irq(mask);
- for(i = 0; i < ledc->length; i++)
- ledc_set_data(ledc->data[i]);
- ledc_enable();
- } else {
- ledc_info("trans data by DMA mode\n");
- mask &= ~LEDC_FIFO_CPUREQ_INT_EN;
- ledc_reset_en();
- size = ledc->length * 4;
- hal_dcache_clean((unsigned long)ledc->data, sizeof(ledc->data));
- slave_config.direction = DMA_MEM_TO_DEV;
- slave_config.src_addr = (unsigned long)(ledc->data);
- slave_config.dst_addr = (uint32_t)(base_addr + LEDC_DATA_REG);
- slave_config.src_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
- slave_config.dst_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
- slave_config.src_maxburst = DMA_SLAVE_BURST_16;
- slave_config.dst_maxburst = DMA_SLAVE_BURST_16;
- slave_config.slave_id = sunxi_slave_id(DRQDST_LEDC, DRQSRC_SDRAM);
- hal_dma_slave_config(dma_chan, &slave_config);
- hal_dma_prep_device(dma_chan, slave_config.dst_addr, slave_config.src_addr, size, DMA_MEM_TO_DEV);
- //dma_chan->callback = ledc_dma_callback;
- hal_dma_start(dma_chan);
- hal_ledc_set_time(ledc);
- ledc_set_output_mode(ledc->output_mode);
- ledc_set_length(ledc->length);
- ledc_set_dma_mode();
- ledc_enable_irq(mask);
- ledc_enable();
- }
- }
- void hal_ledc_clear_all_irq(void)
- {
- ledc_clear_all_irq();
- }
- unsigned int hal_ledc_get_irq_status(void)
- {
- return ledc_get_irq_status();
- }
- void hal_ledc_reset(void)
- {
- ledc_disable_irq(LEDC_TRANS_FINISH_INT_EN | LEDC_WAITDATA_TIMEOUT_INT_EN
- | LEDC_FIFO_OVERFLOW_INT_EN | LEDC_GLOBAL_INT_EN | LEDC_GLOBAL_INT_EN);
- if (dma_chan)
- {
- hal_dma_stop(dma_chan);
- }
- ledc_soft_reset();
- }
- void hal_ledc_deinit(void)
- {
- hal_dma_chan_free(dma_chan);
- hal_gpio_pinmux_set_function(GPIOE(2), 2);
- //clk_deinit
- }
- void hal_ledc_init(void)
- {
- int i;
- unsigned int reg_val = 0;
- if (ledc_clk_init())
- {
- ledc_info("ledc clk init failed \n");
- }
- if (ledc_pinctrl_init())
- {
- ledc_info("ledc pinctrl init failed \n");
- }
- hal_dma_chan_request(&dma_chan);
- }
|