导航

    全志在线开发者论坛

    • 注册
    • 登录
    • 搜索
    • 版块
    • 话题
    • 在线文档
    • 社区主页
    1. 主页
    2. awwwwa
    3. 帖子
    A
    • 资料
    • 关注 0
    • 粉丝 18
    • 我的积分 13359
    • 主题 11
    • 帖子 528
    • 最佳 136
    • 群组 1

    awwwwa 发布的帖子

    • 回复: 编译镜像时报错

      @lozenyin 编译环境安装mkimage android-tools-mkbootimg

      发布在 V Series
      A
      awwwwa
    • 回复: ISP_ERR

      @nimadibaj
      [ISP_ERR]isp_ctx_save_exit, line: 109,open /mnt/isp0_1920_1088_30_0_ctx_saved.bin failed, err:Read-only file system.
      这个是暂存isp参数,由于没有文件所以打开失败。可以无视。

      真正的问题是Got signal 11, exiting ...,程序出现段错误,请检查指针使用

      发布在 V Series
      A
      awwwwa
    • 回复: 程序运行之后告诉我有错误

      刷的固件对不上芯片,例如V853烧录V851s,V853烧录V853s

      发布在 V Series
      A
      awwwwa
    • 回复: 搭建开发环境,出现问了了,总是报fatal: cannot obtain manifest https://sdk.aw-ol.com/git_repo/V853Tina_Open/manifest.git
      1. 执行命令设置全局保存密码
      git config --global credential.helper store
      
      1. 执行命令输入密码
      git clone https://sdk.aw-ol.com/git_repo/V853Tina_Open/manifest.git
      
      1. 使用repo拉取sdk
      repo init -u https://sdk.aw-ol.com/git_repo/V853Tina_Open/manifest.git -b master -m tina-v853-open.xml
      

      由于repo更新,目前不支持通过repo输入密码,请先使用git命令输入保存密码

      发布在 V Series
      A
      awwwwa
    • 回复: 弱弱的问一句,T527是正式量产了吗?

      @hqembed 有Tina Linux AIoT,支持Buildroot,Debian

      发布在 T Series
      A
      awwwwa
    • 回复: 【T113-I】 请问该产品支持spi传输32bit数据吗

      @wayneyao IP版本不同,驱动有修改,以手册为准,控制器支持32bit,请提供设备树和驱动修改

      发布在 T Series
      A
      awwwwa
    • 回复: GT911 触摸滑动不流畅

      可以尝试中断绑定其他CPU核心,例如CPU1

      发布在 Linux
      A
      awwwwa
    • 回复: T113S3双路 dual lvds驱动不起来

      uboot和dtsi里面添加dual link IO的配置

      demo:
      
      lvds2link_pins_a: lvds2link@0 {
           allwinner,pins = "PD0", "PD1", "PD2", "PD3", "PD4", "PD5", "PD8", "PD9", "PD6", "PD7", \
           "PD10", "PD11", "PD12", "PD13", "PD14", "PD15", "PD18", "PD19", "PD16", "PD17";
           allwinner,pname = "PD0", "PD1", "PD2", "PD3", "PD4", "PD5", "PD8", "PD9", "PD6", "PD7", \
           "PD10", "PD11", "PD12", "PD13", "PD14", "PD15", "PD18", "PD19", "PD16", "PD17";
           allwinner,function = "lvds1";
           allwinner,muxsel = <3>;
           allwinner,drive = <3>;
           allwinner,pull = <0>;
       };
      
      lvds2link_pins_b: lvds2link@1 {
           allwinner,pins = "PD0", "PD1", "PD2", "PD3", "PD4", "PD5", "PD8", "PD9", "PD6", "PD7", \
           "PD10", "PD11", "PD12", "PD13", "PD14", "PD15", "PD18", "PD19", "PD16", "PD17";
           allwinner,pname = "PD0", "PD1", "PD2", "PD3", "PD4", "PD5", "PD8", "PD9", "PD6", "PD7", \
           "PD10", "PD11", "PD12", "PD13", "PD14", "PD15", "PD18", "PD19", "PD16", "PD17";
           allwinner,function = "lvds1_suspend";
           allwinner,muxsel = <7>;
           allwinner,drive = <3>;
           allwinner,pull = <0>;
       };
      

      dts中 修改为lvds dual link模式,并引用dtsi里配好的dual link IO

      &lcd0{
      ...
       lcd_lvds_if          = <1>;
      ...
      ...
      pinctrl-0 = <&lvds2link_pins_a>;
      pinctrl-1 = <&lvds2link_pins_b>;
      ...
      }
      

      其他和single link的配置方法无异,如果点不亮请检查时序

      53899931-a632-431e-ac3a-5826278b35e9-LCDTiming.jpg

      这里看到dclk配置是80

      ac38a30a-5d44-4530-a265-33d443cddda7-image.png

      但是手册需求的是54

      1b0f929a-658a-4d7a-b097-ea05513595e5-image.png

      这里提供一套1920x720的时序作为参考

      &lcd0 {
      	lcd_used            = <1>;
      
      	lcd_driver_name     = "default_lcd";
      	lcd_backlight       = <50>;
      	lcd_if              = <3>;
      
      	lcd_x               = <1920>;
      	lcd_y               = <720>;
      	lcd_width           = <150>;
      	lcd_height          = <94>;
      	lcd_dclk_freq       = <94>;
      
      	lcd_pwm_used        = <1>;
      	lcd_pwm_ch          = <3>;
      	lcd_pwm_freq        = <50000>;
      	lcd_pwm_pol         = <1>;
      	lcd_pwm_max_limit   = <255>;
      
      	lcd_hbp             = <64>;
      	lcd_ht              = <2064>;
      	lcd_hspw            = <20>;
      	lcd_vbp             = <30>;
      	lcd_vt              = <760>;
      	lcd_vspw            = <10>;
      
      	lcd_lvds_if         = <1>;
      	lcd_lvds_colordepth = <0>;
      	lcd_lvds_mode       = <0>;
      	lcd_frm             = <1>;
      	lcd_hv_clk_phase    = <0>;
      	lcd_hv_sync_polarity= <0>;
      	lcd_gamma_en        = <0>;
      	lcd_bright_curve_en = <0>;
      	lcd_cmap_en         = <0>;
      
      	deu_mode            = <0>;
      	lcdgamma4iep        = <22>;
      	smart_color         = <90>;
      
      	pinctrl-0 = <&lvds2link_pins_a>;
      	pinctrl-1 = <&lvds2link_pins_b>;
      };
      
      发布在 T Series
      A
      awwwwa
    • 回复: V853 和 V853S NPU算力差了0.2,这个0.2在具体应用上会有明显的性能差距吗?主要用来做目标检测,静态场景。

      根据具体的模型和需求的规格来看,实际感觉差不多

      发布在 V Series
      A
      awwwwa
    • 回复: 全志v536 4k编解码

      @zhangerhu 硬件只是同编同解可以试一下,瓶颈估计在带宽上

      发布在 V Series
      A
      awwwwa
    • 回复: V851s使用spi nand启动不了 VFS: Cannot open root device "ubi0_4" or unknown-block(0,0): error -2

      @dnvwf

      重点:rootfstype=squashfs

      v851s/configs/xxx/env.cfg

      
      #kernel command arguments
      earlyprintk=sunxi-uart,0x02500000
      initcall_debug=0
      console=ttyS0,115200
      nand_root=/dev/ubiblock0_4
      mmc_root=/dev/mmcblk0p4
      nor_root=/dev/mtdblock3
      init=/init
      rdinit=/rdinit
      loglevel=8
      coherent_pool=32K
      #reserve_list=30M@64M,78M@128M,200M@512M
      mac=
      wifi_mac=
      bt_mac=
      specialstr=
      root_partition=rootfs
      mtd_name=sys
      rootfstype=squashfs
      #set kernel cmdline if boot.img or recovery.img has no cmdline we will use this
      setargs_nor=setenv bootargs  earlyprintk=${earlyprintk} clk_ignore_unused initcall_debug=${initcall_debug} console=${console} loglevel=${loglevel} root=${nor_root} rootwait init=${init} rdinit=${rdinit} partitions=${partitions} cma=${cma} coherent_pool=${coherent_pool} ion_carveout_list=${reserve_list}
      setargs_nand=setenv bootargs earlyprintk=${earlyprintk} clk_ignore_unused initcall_debug=${initcall_debug} console=${console} loglevel=${loglevel}  ubi.mtd=${mtd_name} root=${nand_root} rootfstype=${rootfstype} rootwait init=${init} rdinit=${rdinit} partitions=${partitions} cma=${cma} mac_addr=${mac} wifi_mac=${wifi_mac} bt_mac=${bt_mac} selinux=${selinux} specialstr=${specialstr} coherent_pool=${coherent_pool} ion_carveout_list=${reserve_list}
      setargs_nand_ubi=setenv bootargs ubi.mtd=${mtd_name} ubi.block=0,${root_partition} earlyprintk=${earlyprintk} clk_ignore_unused initcall_debug=${initcall_debug} console=${console} loglevel=${loglevel} root=${nand_root} rootfstype=${rootfstype} init=${init} partitions=${partitions} cma=${cma} snum=${snum} mac_addr=${mac} wifi_mac=${wifi_mac} bt_mac=${bt_mac} specialstr=${specialstr} gpt=1
      setargs_mmc=setenv  bootargs earlyprintk=${earlyprintk} clk_ignore_unused initcall_debug=${initcall_debug} console=${console} loglevel=${loglevel} root=${mmc_root}  rootwait init=${init} partitions=${partitions} cma=${cma} mac_addr=${mac} wifi_mac=${wifi_mac} bt_mac=${bt_mac} selinux=${selinux} specialstr=${specialstr} coherent_pool=${coherent_pool} ion_carveout_list=${reserve_list}
      #nand command syntax: sunxi_flash read address partition_name read_bytes
      #0x4007f800 = 0x40080000(kernel entry) - 0x800(boot.img header 2k)
      boot_partition=boot
      boot_normal=sunxi_flash read 44800000 ${boot_partition};bootm 44800000
      boot_recovery=sunxi_flash read 44800000 extend;bootm 44800000
      boot_fastboot=fastboot
      #recovery key
      recovery_key_value_max=0x13
      recovery_key_value_min=0x10
      #fastboot key
      fastboot_key_value_max=0x8
      fastboot_key_value_min=0x2
      
      #uboot system env config
      bootdelay=1
      #default bootcmd, will change at runtime according to key press
      #default nand boot
      bootcmd=run setargs_nand boot_normal
      #verify the kernel
      verify=N
      
      发布在 V Series
      A
      awwwwa
    • 回复: t113如何开启双frambuffer?

      需要真实存在两个屏幕,也可以一个lcd屏幕一个spi屏幕

      发布在 其它全志芯片讨论区
      A
      awwwwa
    • 回复: V851se的u-boot引导

      会,启动介质优先级描述了每个介质被选择为启动介质的可能性。BROM 首先读取具有最高优先级的介质的 boot0。如果该介质不存在或存在任何问题,BROM 将尝试下一个介质。否则,该介质将被选择为启动介质。

      具体可以查看手册GPIO Boot Select表格

      发布在 V Series
      A
      awwwwa
    • 回复: R128 Vector支持

      @layzerlee 需要修改编译参数加上v扩展支持

      CONFIG_TOOLCHAIN_MACH_FLAGS="-mcmodel=medany -mabi=lp64dv -march=rv64gcxvthead"
      CONFIG_TOOLCHAIN_FPU_ABI="-march=rv64gcxvthead"
      
      发布在 A Series
      A
      awwwwa
    • 回复: V853S mipi LCD显示驱动调试,colorbar显示异常

      dclk过高,屏幕分屏了?
      这里提供一个py脚本计算分频系数,这里对应的是HV屏,DSI也可以参考

      def find_closest_clock(target_clock, clock_list):
          clock_list = sorted(clock_list)
          low, high = 0, len(clock_list) - 1
          closest = clock_list[low]
          while low <= high:
              mid = (low + high) // 2
              if clock_list[mid] < target_clock:
                  low = mid + 1
              elif clock_list[mid] > target_clock:
                  high = mid - 1
              else:
                  return clock_list[mid]
              if abs(clock_list[mid] - target_clock) < abs(closest - target_clock):
                  closest = clock_list[mid]
      
          return closest
      
      def calculate_divisor(clock_need, clock_list, min_divisor=6):
          is_perfect = True
          for i in clock_list:
              for j in range(0, 255):
                  if (clock_need * j) == i:
                      closest_clock = i
                      divisor = j
                      return closest_clock, divisor, is_perfect
      
          is_perfect = False
      
          closest_clock = find_closest_clock(clock_need, clock_list)
          
          if closest_clock == 0:
              return None, None, None
          
          divisor = closest_clock // clock_need
          if divisor < min_divisor:
              min_diff = float('inf')
              best_clock = None
              for clock in clock_list:
                  if clock >= clock_need * min_divisor:
                      current_divisor = clock // clock_need
                      if current_divisor < min_divisor:
                          continue
                      diff = abs(clock - clock_need * current_divisor)
                      if diff < min_diff:
                          min_diff = diff
                          best_clock = clock
              if best_clock is not None:
                  return best_clock, best_clock // clock_need, is_perfect
          
          return closest_clock, divisor, is_perfect
      
      clock_list = [
          408, 420, 432, 444, 456, 468, 480, 492, 504, 516, 528, 540, 552, 564,
          576, 588, 600, 612, 624, 636, 648, 660, 672, 684, 696, 708, 720, 732,
          744, 756, 768, 780, 792, 804, 816, 828, 840, 852, 864, 876, 888, 900,
          912, 924, 936, 948, 960, 972, 984, 996, 1008, 1020, 1032, 1044, 1056,
          1068, 1080, 1092, 1104, 1116, 1128, 1140, 1152, 1164, 1176, 1188, 1200,
          1212, 1224, 1236, 1248, 1260, 1272, 1284, 1296, 1308, 1320, 1332, 1344,
          1356, 1368, 1380, 1392, 1404, 1416, 1428, 1440, 1452, 1464, 1476, 1488,
          1500, 1512, 1524, 1536, 1548, 1560, 1572, 1584, 1596, 1608, 1620, 1632,
          1644, 1656, 1668, 1680, 1692
      ]
      
      
      clock_need = int(input("请输入需要的时钟(MHz): "))
      closest_clock, divisor, is_perfect = calculate_divisor(clock_need, clock_list)
      if is_perfect:
          print(f"父时钟: {closest_clock}MHz, 分频系数: {divisor}, 分频后的频率: {closest_clock / divisor}MHz")
      else:
          print(f"无法找到完美,最近的父时钟: {closest_clock}, 分频系数: {divisor}, 分频后的频率: {closest_clock / divisor}MHz")
      print("请修改分频系数表 clk_tbl 中 HV 分频系数为: {LCD_IF_HV, " + hex(divisor) + ", 1, 1, 0}")
      
      发布在 V Series
      A
      awwwwa
    • 回复: 求助D1的隐藏分区分布讲解

      可以查看《Tina_Linux_OTA_开发指南.pdf》 文档

      发布在 MR Series
      A
      awwwwa
    • 回复: 全志t113tina如何在内核阶段显示进度条!!!

      需要在内核中实现这个功能的UI,可以采用手绘UI方法,操作内核DISP的接口实现。或者使用开机动画轮播图片实现

      发布在 其它全志芯片讨论区
      A
      awwwwa
    • 回复: R128 Vector支持

      请贴出编译日志。

      发布在 A Series
      A
      awwwwa
    • 回复: 打补丁后,编译失败

      第一次编译时候请取消勾选 mpp sample smartIPC_demo 编译完成一次之后使用 cleanmpp && mkmpp 来编译mpp相关demo。

      a8dd20c7-6698-4ab4-800d-348903622b78-image.png

      发布在 V Series
      A
      awwwwa
    • 回复: t113i + openwrt 21.02 RCU报错,严重时导致板子卡死

      看一下编写的驱动是否出现死锁

      发布在 Linux
      A
      awwwwa
    • 回复: R329代码里找不到BOOT0启动打印“HELLO! BOOT0 is starting!”

      @yimu163com 在 R329代码里找不到BOOT0启动打印“HELLO! BOOT0 is starting!” 中说:

      Binary file ./device/config/chips/r329/bin/boot0_nand_sun50iw11p1.bin matches
      Binary file ./device/config/chips/r329/bin/sboot_sun50iw11p1.bin matches
      Binary file ./device/config/chips/r329/bin/boot0_spinor_sun50iw11p1.bin matches
      Binary file ./device/config/chips/r329/bin/boot0_sdcard_sun50iw11p1.bin matches

      在这几个bin里,BOOT0不开源

      发布在 其它全志芯片讨论区
      A
      awwwwa
    • 回复: R128S2 SDK 编译错误

      @yilan7805 看上去解压工具链压缩文件有损坏,重新下载一下?

      发布在 MR Series
      A
      awwwwa
    • 回复: T113如何移植RT补丁包

      联系板卡提供商,目前几家都已经提供rt补丁

      发布在 Special
      A
      awwwwa
    • 回复: 哪吒D1-H 通过bootargs来静态设置ip

      OpenWRT会重写kernel部分的配置,具体请参考OpenWRT文档

      发布在 MR Series
      A
      awwwwa
    • 回复: t113 适配ICNL9707 480x1280mipi屏 屏幕只显示一半 且显示还有阴影

      看一下屏幕dclk,是否由于dclk过高导致错误

      发布在 其它全志芯片讨论区
      A
      awwwwa
    • 回复: 触摸零点偏移

      如果是CTP,GT911驱动需要设置校准数组,FT系列需要配置零点
      如果是RTP,需要tslib单独校准

      发布在 其它全志芯片讨论区
      A
      awwwwa
    • 回复: 全志T系列HDMI显示较暗

      @shuaige HDMI只输出数字信号,没有背光数据,看看是不是图像本来就比较暗

      发布在 T Series
      A
      awwwwa
    • 回复: ubuntu使用scp传文件给Tina 2.0 SDK

      默认没有密码,加密码需要配置下

      发布在 MR Series
      A
      awwwwa
    • 回复: T113 uboot 识别不到u盘设备

      uboot下是否开启usb储存设备的驱动?默认没有进行配置

      发布在 其它全志芯片讨论区
      A
      awwwwa
    • 回复: V851编译错误

      opencv编译错误,编译opencv需要虚拟机内存 >16G,CPU > 4核,否则编译任务会被Linux强制结束

      发布在 编译和烧写问题专区
      A
      awwwwa
    • 回复: T527下载SDK出错

      @chaogai 请使用python2进行下载

      发布在 T Series
      A
      awwwwa
    • 回复: T527 SDK下载

      请提供完整的报错截图

      发布在 T Series
      A
      awwwwa
    • 回复: hello world 编写执行文件没出来,文件我都有,为什么没找到

      这个是交叉编译工具链没找到,确认下有这个文件吗
      `

      发布在 V Series
      A
      awwwwa
    • 回复: 关于NPU开发中onnx中op Split的疑似bug问题反馈

      请提供下转换工具的版本,类似于acuity-toolkit-binary-6.21.1

      发布在 V Series
      A
      awwwwa
    • 回复: T527 使用 DRM 驱动 edp 屏幕

      DSI

      	dsi1_backlight: backlight0 {
      		compatible = "pwm-backlight";
      		status = "okay";
      		brightness-levels = <
      			  0   1   2   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  36  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  79
      			 80  81  82  83  84  85  86  87
      			 88  89  90  91  92  93  94  95
      			 96  97  98  99 100 101 102 103
      			104 105 106 107 108 109 110 111
      			112 113 114 115 116 117 118 119
      			120 121 122 123 124 125 126 127
      			128 129 130 131 132 133 134 135
      			136 137 138 139 140 141 142 143
      			144 145 146 147 148 149 150 151
      			152 153 154 155 156 157 158 159
      			160 161 162 163 164 165 166 167
      			168 169 170 171 172 173 174 175
      			176 177 178 179 180 181 182 183
      			184 185 186 187 188 189 190 191
      			192 193 194 195 196 197 198 199
      			200 201 202 203 204 205 206 207
      			208 209 210 211 212 213 214 215
      			216 217 218 219 220 221 222 223
      			224 225 226 227 228 229 230 231
      			232 233 234 235 236 237 238 239
      			240 241 242 243 244 245 246 247
      			248 249 250 251 252 253 254 255>;
      		default-brightness-level = <200>;
      		enable-gpios = <&pio PH 18 GPIO_ACTIVE_HIGH>;
      		pwms = <&pwm0 0 50000 0>;
      	};
      
      &de {
      	chn_cfg_mode = <3>;
      	status = "okay";
      };
      
      &vo0 {
      	status = "okay";
      };
      
      &dlcd1 {
      	status = "okay";
      };
      
      &dsi1 {
      	status = "okay";
      	pinctrl-0 = <&dsi1_4lane_pins_a>;
      	pinctrl-1 = <&dsi1_4lane_pins_b>;
      	pinctrl-names = "active","sleep";
      	ports {
      		dsi1_out: port@1{
      			reg = <1>;
      			dsi_out_panel: endpoint {
      				remote-endpoint = <&panel_in>;
      			};
      		};
              };
      	panel: panel@0 {
      		compatible = "panel-dsi";
      		status = "okay";
      		reg = <0>;
      		power0-supply = <&reg_cldo4>;
      		power1-supply = <&reg_cldo1>;
      
      		reset-gpios = <&pio PD 22 GPIO_ACTIVE_HIGH>; //reset
      		backlight = <&dsi1_backlight>;
      		dsi,flags = <MIPI_DSI_MODE_VIDEO>;
      		dsi,lanes = <4>;
      		dsi,format = <0>;
      		panel-init-sequence = [
      			15 00 02 E0 00
      			15 00 02 E1 93
      			15 00 02 E2 65
      			15 00 02 E3 F8
      			15 00 02 80 03
      			15 00 02 E0 01
      			15 00 02 00 00
      			15 00 02 01 25
      			15 00 02 03 00
      			15 00 02 04 30
      			15 00 02 0C 74
      			15 00 02 17 00
      			15 00 02 18 C7
      			15 00 02 19 01
      			15 00 02 1A 00
      			15 00 02 1B C7
      			15 00 02 1C 01
      			15 00 02 24 FE
      			15 00 02 37 19
      			15 00 02 35 28
      			15 00 02 38 05
      			15 00 02 39 08
      			15 00 02 3A 12
      			15 00 02 3C 7E
      			15 00 02 3D FF
      			15 00 02 3E FF
      			15 00 02 3F 7F
      			15 00 02 40 06
      			15 00 02 41 A0
      			15 00 02 43 1E
      			15 00 02 44 0B
      			15 00 02 55 02
      			15 00 02 57 6A
      			15 00 02 59 0A
      			15 00 02 5A 2E
      			15 00 02 5B 1A
      			15 00 02 5C 15
      			15 00 02 5D 7F
      			15 00 02 5E 61
      			15 00 02 5F 50
      			15 00 02 60 43
      			15 00 02 61 3F
      			15 00 02 62 32
      			15 00 02 63 35
      			15 00 02 64 1F
      			15 00 02 65 38
      			15 00 02 66 36
      			15 00 02 67 36
      			15 00 02 68 54
      			15 00 02 69 42
      			15 00 02 6A 48
      			15 00 02 6B 39
      			15 00 02 6C 34
      			15 00 02 6D 26
      			15 00 02 6E 14
      			15 00 02 6F 02
      			15 00 02 70 7F
      			15 00 02 71 61
      			15 00 02 72 50
      			15 00 02 73 43
      			15 00 02 74 3F
      			15 00 02 75 32
      			15 00 02 76 35
      			15 00 02 77 1F
      			15 00 02 78 38
      			15 00 02 79 36
      			15 00 02 7A 36
      			15 00 02 7B 54
      			15 00 02 7C 42
      			15 00 02 7D 48
      			15 00 02 7E 39
      			15 00 02 7F 34
      			15 00 02 80 26
      			15 00 02 81 14
      			15 00 02 82 02
      			15 00 02 E0 02
      			15 00 02 00 52
      			15 00 02 01 5F
      			15 00 02 02 5F
      			15 00 02 03 50
      			15 00 02 04 77
      			15 00 02 05 57
      			15 00 02 06 5F
      			15 00 02 07 4E
      			15 00 02 08 4C
      			15 00 02 09 5F
      			15 00 02 0A 4A
      			15 00 02 0B 48
      			15 00 02 0C 5F
      			15 00 02 0D 46
      			15 00 02 0E 44
      			15 00 02 0F 40
      			15 00 02 10 5F
      			15 00 02 11 5F
      			15 00 02 12 5F
      			15 00 02 13 5F
      			15 00 02 14 5F
      			15 00 02 15 5F
      			15 00 02 16 53
      			15 00 02 17 5F
      			15 00 02 18 5F
      			15 00 02 19 51
      			15 00 02 1A 77
      			15 00 02 1B 57
      			15 00 02 1C 5F
      			15 00 02 1D 4F
      			15 00 02 1E 4D
      			15 00 02 1F 5F
      			15 00 02 20 4B
      			15 00 02 21 49
      			15 00 02 22 5F
      			15 00 02 23 47
      			15 00 02 24 45
      			15 00 02 25 41
      			15 00 02 26 5F
      			15 00 02 27 5F
      			15 00 02 28 5F
      			15 00 02 29 5F
      			15 00 02 2A 5F
      			15 00 02 2B 5F
      			15 00 02 2C 13
      			15 00 02 2D 1F
      			15 00 02 2E 1F
      			15 00 02 2F 01
      			15 00 02 30 17
      			15 00 02 31 17
      			15 00 02 32 1F
      			15 00 02 33 0D
      			15 00 02 34 0F
      			15 00 02 35 1F
      			15 00 02 36 05
      			15 00 02 37 07
      			15 00 02 38 1F
      			15 00 02 39 09
      			15 00 02 3A 0B
      			15 00 02 3B 11
      			15 00 02 3C 1F
      			15 00 02 3D 1F
      			15 00 02 3E 1F
      			15 00 02 3F 1F
      			15 00 02 40 1F
      			15 00 02 41 1F
      			15 00 02 42 12
      			15 00 02 43 1F
      			15 00 02 44 1F
      			15 00 02 45 00
      			15 00 02 46 17
      			15 00 02 47 17
      			15 00 02 48 1F
      			15 00 02 49 0C
      			15 00 02 4A 0E
      			15 00 02 4B 1F
      			15 00 02 4C 04
      			15 00 02 4D 06
      			15 00 02 4E 1F
      			15 00 02 4F 08
      			15 00 02 50 0A
      			15 00 02 51 10
      			15 00 02 52 1F
      			15 00 02 53 1F
      			15 00 02 54 1F
      			15 00 02 55 1F
      			15 00 02 56 1F
      			15 00 02 57 1F
      			15 00 02 58 40
      			15 00 02 5B 10
      			15 00 02 5C 06
      			15 00 02 5D 40
      			15 00 02 5E 00
      			15 00 02 5F 00
      			15 00 02 60 40
      			15 00 02 61 03
      			15 00 02 62 04
      			15 00 02 63 6C
      			15 00 02 64 6C
      			15 00 02 65 75
      			15 00 02 66 08
      			15 00 02 67 B4
      			15 00 02 68 08
      			15 00 02 69 6C
      			15 00 02 6A 6C
      			15 00 02 6B 0C
      			15 00 02 6D 00
      			15 00 02 6E 00
      			15 00 02 6F 88
      			15 00 02 75 BB
      			15 00 02 76 00
      			15 00 02 77 05
      			15 00 02 78 2A
      			15 00 02 E0 04
      			15 00 02 37 58
      			15 00 02 00 0E
      			15 00 02 02 B3
      			15 00 02 09 61
      			15 00 02 0E 48
      			15 00 02 E0 00
      			05 32 01 11
      			05 05 01 29
      			15 32 02 35 00
      		];
      
      		panel-exit-sequence = [
      			05 14 01 28
      			05 50 01 10
      		];
      
      		display-timings {
      			native-mode = <&dsi1_timing0>;
      
      			dsi1_timing0: timing0 {
      				clock-frequency = <68215200>;
      				hback-porch = <20>;
      				hactive = <800>;
      				hfront-porch = <20>;
      				hsync-len = <20>;
      				vback-porch = <8>;
      				vactive = <1280>;
      				vfront-porch = <30>;
      				vsync-len = <4>;
      
      			};
      		};
      		port {
      			panel_in: endpoint {
                                      remote-endpoint = <&dsi_out_panel>;
                              };
      		};
      	};
      };
      
      &dsi0combophy {
      	status = "disabled";
      };
      
      &dsi1combophy {
      	status = "okay";
      };
      发布在 T Series
      A
      awwwwa
    • 回复: 只用的全志的板子,怎么才能往rootfs中添加文件

      target/allwinner/t113-nezha/base-files
      target/allwinner/t113-nezha/busybox-init-base-files

      根据选择的overlay方式而定

      发布在 Linux
      A
      awwwwa
    • 回复: 将系统文件设置ext4格式,启动提示no filesystem could mount root

      @zifeiyu 内核也需要勾选上相应的驱动
      2f5d0e59-d2d1-4bf9-a78c-53b173efbd5d-image.png

      发布在 Linux
      A
      awwwwa
    • 回复: 关于T113的SDK下载问题

      需要安装Python2.7 f613d86a-b91b-4738-a0b2-710303c9c987-image.png

      发布在 代码下载问题专区
      A
      awwwwa
    • 回复: TLT113-MiniEVM 启动引导失败:E/TC:0 0 check_hardware_info:90 hardware check error1

      @scanli 如果刷写过,需要创龙官方来提供支持

      发布在 创龙科技专区
      A
      awwwwa
    • 回复: TLT113-MiniEVM 启动引导失败:E/TC:0 0 check_hardware_info:90 hardware check error1

      @scanli 请问设备是否刷写过安全启动?如果没有刷写过但是出现这个情况可以做以下临时修改

      在 device/config/chips/t113_i/configs/default/ 修改 boot_package.cfg 删除 optee.fex 行

      在 lichee/linux-5.4/arch/arm/boot/dts/sun8iw20p1.dtsi 中注释

      psci {
      	compatible = "arm,psci-1.0";
      	method = "smc";
      };
      

      前往 lichee/linux-5.4/arch/arm/mach-sunxi/platsmp.c 加入

      static int sun8i_t113_smp_boot_secondary(unsigned int cpu,
      				    struct task_struct *idle)
      {
          u32 reg;
          void __iomem *cpucfg_membase = ioremap(0x09010000, 0x10);
          void __iomem *cpuexec_membase[] = {ioremap(0x070005C4, 0x10),ioremap(0x070005C8, 0x10)};
      	
      	if (cpu != 1)
      	    return 0;
      
      	spin_lock(&cpu_lock);
      
      	writel(__pa_symbol(secondary_startup),	cpuexec_membase[cpu]);
      
      	reg = readl(cpucfg_membase);
      	writel(reg | BIT(cpu), cpucfg_membase);
      
      	spin_unlock(&cpu_lock);
      
      	return 0;
      }
      
      static const struct smp_operations sun8i_t113_smp_ops __initconst = {
      	.smp_boot_secondary	= sun8i_t113_smp_boot_secondary,
      };
      CPU_METHOD_OF_DECLARE(sun8i_t113_smp, "allwinner,sun8iw20p1", &sun8i_t113_smp_ops);
      

      注意这只是临时修改跳过检查,之后最好使用创龙提供的虚拟机复现看看会不会也出现这个问题

      发布在 创龙科技专区
      A
      awwwwa
    • 回复: mboot和mkernel可以代替make命令吗

      @zifeiyu 对

      发布在 编译和烧写问题专区
      A
      awwwwa
    • 回复: rs485的设备树怎么修改

      @zifeiyu

      DE和RO为使能管脚。DE为低电平、RE为低电平时为接收;DE为高电平、RE为高电平时为发送;RO和DI为数据管脚。RO为接收,DI为发送;因此我们经常将DE和RE直接连接,用一个IO口控制。

      需要添加以下三个成员:

      • sunxi,uart-rs485
        • 0:485模式关闭
        • 1:485模式使能
      • sunxi,uart-485fl
        • 0:485 gpio管脚数值为0时表示发送状态
        • 1:485 gpio管脚数值为1时表示发送状态
      • sunxi,uart-485oe-gpios
        • 用于外部转换芯片使能信号的gpio引脚,GPIO_ACTIVE_HIGH含义为默认为高电平

      示例:

      uart1: uart@2500400 {
      			...
      			status = "okay";
      			//添加以下三行
      			sunxi,uart-rs485 = <1>;
      			sunxi,uart-485fl = <1>;
      			sunxi,uart-485oe-gpios = <&pio PG 8 GPIO_ACTIVE_HIGH>;
      };
      

      A40I 5.10 内核:

      uart5_pins_a: uart5_pins@0 {
      	pins = "PH6", "PH7";
      	function = "uart5";
      };
      
      uart5_pins_b: uart5_pins@1 {
      	pins = "PH6", "PH7";
      	function = "gpio_in";
      };
      
      &uart5 {
      	pinctrl-names = "default", "sleep";
      	pinctrl-0 = <&uart5_pins_a>;
      	pinctrl-1 = <&uart5_pins_b>;
      	rs485-enable = <0>;
      	rs485-mode = <0>;   /* defalut 0:rx 1:tx */
      	rs485-mode-pin = <&pio PH 8 GPIO_ACTIVE_LOW>;
      	status = "disabled";
      };
      
      发布在 其它全志芯片讨论区
      A
      awwwwa
    • 回复: mboot和mkernel可以代替make命令吗

      @zifeiyu rootfs需要用make命令编译,不过只需要编译一次

      发布在 编译和烧写问题专区
      A
      awwwwa
    • 回复: mboot和mkernel可以代替make命令吗

      mboot,mkernel是编译boot和内核的,并没有编译rootfs,如果已经编译好了rootfs则可以代替,如果没有编译rootfs则需要make来编译rootfs

      发布在 编译和烧写问题专区
      A
      awwwwa
    • 回复: TLT113-MiniEVM 启动引导失败:E/TC:0 0 check_hardware_info:90 hardware check error1

      make distclean后删除out文件夹重新编译看看

      发布在 创龙科技专区
      A
      awwwwa
    • 回复: R128-S2 使用外部1.2VDC的情况下, 如何关闭 APP_LDO

      @maplerian 开启外部DCDC, POWERTCTL会自动关闭APP_LDO,这个是一个硬件行为

      发布在 MR Series
      A
      awwwwa
    • 回复: R128-S2 使用外部1.2VDC的情况下, 如何关闭 APP_LDO

      @maplerian 在深度休眠下,内部APP_LDO由PWRCTRL硬件关闭,且深度休眠下GPIO控制器断电不会启用DCDC,唤醒时由PWRCTRL硬件逻辑开启内部APP_LDO,然后启用GPIO控制器拉高启用外部DCDC,启用后自动切换到外部DCDC供电

      发布在 MR Series
      A
      awwwwa
    • 回复: R128-S2 使用外部1.2VDC的情况下, 如何关闭 APP_LDO

      APP_LDO的关闭是PWRCTRL的硬件行为,由配置处理器deep sleep mode后,CPU进入WFI触发PWRCTRL的该行为,无寄存器配置。PWRCTRL仅支持手动关闭DSP。外挂DCDC可以通过读取LDO_EN寄存器查看目前是否开启内部LDO

      发布在 MR Series
      A
      awwwwa
    • 回复: XR829在T113-i芯片+原SDK基础上蓝牙有成功过的伙伴吗?

      @chrisvista 如果配置UART适配8250规范则是AS0,如果是sunxi-uart规范则是S0

      发布在 Wireless & Analog Series
      A
      awwwwa
    • 回复: T113 ledc 驱动bug 长时间运行以后出错

      贴一下完整的log看看

      发布在 MR Series
      A
      awwwwa
    • 回复: XR829在T113-i芯片+原SDK基础上蓝牙有成功过的伙伴吗?

      测试 AWOL 开源的 Tina SDK + T113-S3 板子,hciattach -n ttyS1 xradio 正常

      发布在 Wireless & Analog Series
      A
      awwwwa
    • 回复: T113 ledc 驱动bug 长时间运行以后出错

      “sunxi_ledc_irq_handler()1313 - wait time is more than 600000 ns,going to reset ledc and drop this operation!”

      看代码,此打印的意思是ledc传输数据后超过600ms才进入中断处理,看上去是被别的高优先级中断抢占

      发布在 MR Series
      A
      awwwwa
    • 回复: XR829 蓝牙模块串口同步错误

      测试使用串口 ttyS1 正常,关闭 CONFIG_SERIAL_8250

      发布在 创龙科技专区
      A
      awwwwa
    • 回复: T113 tina能不能通过u盘做动态修改屏参

      @ethan_00 可以通过U盘来写入设备树dtb overlays覆盖原屏参

      发布在 MR Series
      A
      awwwwa
    • 回复: t113-s3因为硬件原因导致内核无法启动

      正常的,内核没有输出检查一下设备树配置

      发布在 Linux
      A
      awwwwa
    • 回复: Tina-T113 openWrt 新增软件包,选中menuconfig后,编译报错,makefile完全就是按照sdk里QT对应的makefile来写的

      Makefile:68: *** missing separator. Stop.

      这个错误通常表示在 Makefile 的第 68 行或附近存在语法错误。Makefile 中使用的规则必须遵循严格的缩进规范,通常是使用 Tab 键进行缩进,而不是空格。

      请检查 Makefile 第 68 行附近的代码,确保每个规则的命令部分都以 Tab 键开头,并且规则名称和命令之间使用冒号(:)分隔。此外,还需要确认 Makefile 的每一行都是使用相同的缩进方式,要么都是 Tab,要么都是空格,不要混用。

      发布在 MR Series
      A
      awwwwa
    • 回复: D1-H哪吒开发板HDMI默认可以使用吗

      默认HDMI不开,需要命令开启

      cd /sys/kernel/debug/dispdbg
      echo disp0 > name; echo switch1 > command; echo 4 10 0 0 0x4 0x101 0 0 0 8 > param; echo 1 > start;
      
      发布在 MR Series
      A
      awwwwa
    • 回复: Tina-linux f133-evb1 打包提示 ERROR: booto files are all invalid files, now exit

      boot0丢失了

      发布在 MR Series
      A
      awwwwa
    • 回复: 求助 r128 dsp

      RI-2020.5-linux 编译器需要向购买开发板的商家/代理提交申请,需要提供自己的HIFI5 DSP Xtensa Xplorer的证明

      发布在 A Series
      A
      awwwwa
    • 回复: 个人开发者如何获取D1、D1s的 SDK?需要签NDA吗?

      @damiaa 现在是新版本平台了,直接拿awol账号下载,免去再注册一个平台的麻烦:https://d1.docs.aw-ol.com/study/study_3getsdktoc/

      发布在 代码下载问题专区
      A
      awwwwa
    • 回复: 修改了linux-5.4/board.dts的uart0部分,但是烧写到板子上面没有生效

      @zifeiyu 那为什么pack打包到固件的有,刷进去就没了,是不是刷写到emmc但是TF卡启动了?

      发布在 编译和烧写问题专区
      A
      awwwwa
    • 回复: 修改了linux-5.4/board.dts的uart0部分,但是烧写到板子上面没有生效

      @zifeiyu 看看刷的固件对不对

      发布在 编译和烧写问题专区
      A
      awwwwa
    • 回复: 修改了linux-5.4/board.dts的uart0部分,但是烧写到板子上面没有生效

      @zifeiyu 那你得检查下是不是刷错固件了,或者没有运行pack命令刷的还是老版本

      发布在 编译和烧写问题专区
      A
      awwwwa
    • 回复: 修改了linux-5.4/board.dts的uart0部分,但是烧写到板子上面没有生效

      @zifeiyu 那看一下合成的dts的uart情况,out/t113-100ask/image/.sunxi.dts

      发布在 编译和烧写问题专区
      A
      awwwwa
    • 回复: 修改了linux-5.4/board.dts的uart0部分,但是烧写到板子上面没有生效

      也可以看一下 out/t113-100ask/image/.sunxi.dts 文件的

      df000f5d-de21-41a2-a3a9-742684a6b856-image.png

      发布在 编译和烧写问题专区
      A
      awwwwa
    • 回复: 修改了linux-5.4/board.dts的uart0部分,但是烧写到板子上面没有生效

      @zifeiyu uboot阶段fdt print看看有没有okay,另外图没发上来

      发布在 编译和烧写问题专区
      A
      awwwwa
    • 回复: 修改了linux-5.4/board.dts的uart0部分,但是烧写到板子上面没有生效

      使用mkernel强制重编译内核

      发布在 编译和烧写问题专区
      A
      awwwwa
    • 回复: 请教t113 uboot cpu时钟怎么修改

      1b38310c-c745-429b-9b00-b5bbf1153833-image.png

      没想到吧,uboot-board.dts里还有一个

      发布在 编译和烧写问题专区
      A
      awwwwa
    • 回复: 全志在线开源芯片 新 SDK 平台下载方法汇总

      @spade8

      检查下你的repo安装是不是正常的

      d13de3d0-af6e-415f-9a2f-62e74c463ad3-image.png

      发布在 代码下载问题专区
      A
      awwwwa
    • 回复: v851 使用sample_rtsp 报ISP_ERR

      检查使用的SDK是哪一个版本,目前awol的Tina 5.0不支持sample_rtsp,这个demo适配的是Tina 4.0,需要向代理获取支持

      发布在 V Series
      A
      awwwwa
    • 回复: 无法烧录固件

      请贴出LOG,截图

      发布在 MR Series
      A
      awwwwa
    • 回复: D1s TVIN使用trecorderdemo问题

      @meihao 先camerademo看看能不能抓图

      发布在 MR Series
      A
      awwwwa
    • 回复: tina linux SDL1.2.15

      @wj8331585 支持,出现段错误可以编译一个gdb进去看看哪里错了

      测试代码:

      w = 1920;
      h = 1080;
      bpp = 32;
      
      SDL_Init( SDL_INIT_EVERYTHING );
      SDL_SetVideoMode( w,h, bpp, SDL_HWSURFACE|SDL_DOUBLEBUF );
      发布在 爱搞机专区
      A
      awwwwa
    • 回复: XR829的LPCLK

      只要LPCLK引脚接地,就可以使用内部RC时钟32K,这个时钟不准确。如果不需要使用低功耗可以不外挂

      发布在 Wireless & Analog Series
      A
      awwwwa
    • 回复: H133的内核交叉编译工具链在哪看用的是哪个啊

      prebuilt/gcc/linux-x86/arm/toolchain-sunxi-musl

      发布在 其它全志芯片讨论区
      A
      awwwwa
    • 回复: V853SDK:奇怪的帧缓冲区操作

      @alb702 add your layer and cat that file see what layer you configured

      发布在 V Series
      A
      awwwwa
    • 回复: yolov5部署,onnx-npu模型转换问题

      需要修改这两个出错的算子

      发布在 V Series
      A
      awwwwa
    • 回复: 关于V851S的opencv问题

      提供了Patch:https://github.com/YuzukiHD/TinyVision/blob/main/tina/openwrt/package/thirdparty/vision/opencv/patches/0004-support-sunxi-vin-camera.patch

      发布在 V Series
      A
      awwwwa
    • 回复: R128 WIFI AP模式获取连接设备的IP地址的问题

      AP的数据是在 lwip 的 static struct etharp_entry arp_table[ARP_TABLE_SIZE]; 中维护

      #if LWIP_XR_IMPL
      ip4_addr_t* etharp_get_ip_from_mac(struct eth_addr *ethaddr) {
        int i;
        ip4_addr_t* ip_ret = NULL;
      
        LWIP_ASSERT("ethaddr != NULL", ethaddr != NULL);
      
        for (i = 0; i < ARP_TABLE_SIZE; i++) {
          if (memcmp(ethaddr, &arp_table[i].ethaddr, sizeof(struct eth_addr)) == 0) {
            ip_ret = &arp_table[i].ipaddr;
            break;
          }
        }
      
        return ip_ret;
      }
      #endif
      
      发布在 A Series
      A
      awwwwa
    • 回复: 请问谁有AIC600E3的DRAM初始化代码?

      @dingdong AIC600E3 is a different chip with t113-s4

      发布在 其它全志芯片讨论区
      A
      awwwwa
    • 回复: V853SDK:奇怪的帧缓冲区操作

      @alb702 try to cat /sys/class/disp/disp/attr/sys

      empty:

      root@TinaLinux:/# cat /sys/class/disp/disp/attr/sys
      screen 0:
      de_rate 300000000 hz, ref_fps:62
      mgr0: 800x1280 fmt[rgb] cs[0x204] range[full] eotf[0x4] bits[8bits] err[0] force_sync[0] unblank direct_show[false] iommu[1]
      dmabuf: cache[0] cache max[0] umap skip[0] umap skip max[52]
              lcd output      backlight(100)  fps:62.1         800x1280
              err:0   skip:228        irq:2786        vsync:0 vsync_skip:0
         BUF    enable ch[1] lyr[0] z[16] prem[N] a[pixel 255] fmt[  0] fb[ 800,1280; 800,1280; 800,1280] crop[   0,   0, 800,1280] frame[   0,   0, 800,1280] addr[ff800000,       0,       0] flags[0x       0] trd[0,0] depth[ 0] 
      

      add layer

      cat /sys/class/disp/disp/attr/sys
      screen 0:
      de_rate 300000000 hz, ref_fps:62
      mgr0: 800x1280 fmt[rgb] cs[0x204] range[full] eotf[0x4] bits[8bits] err[0] force_sync[0] unblank direct_show[false] iommu[1]
      dmabuf: cache[0] cache max[0] umap skip[0] umap skip max[52]
              lcd output      backlight(100)  fps:62.1         800x1280
              err:933 skip:228        irq:6815        vsync:0 vsync_skip:0
       COLOR    enable ch[0] lyr[0] z[16] prem[N] a[pixel   0] fmt[  0] fb[   0,   0;   0,   0;   0,   0] crop[   0,   0, 200,1280] frame[   0,   0, 200,1280] addr[ffff0000,       0,       0] flags[0x       0] trd[0,0]
      depth[ 0]  COLOR    enable ch[0] lyr[1] z[16] prem[N] a[pixel   0] fmt[  0] fb[   0,   0;   0,   0;   0,   0] crop[ 200,   0, 200,1280] frame[ 200,   0, 200,1280] addr[ff00ff00,       0,       0] flags[0x       0] trd[0,0]
      depth[ 0]  COLOR    enable ch[0] lyr[2] z[16] prem[N] a[pixel   0] fmt[  0] fb[   0,   0;   0,   0;   0,   0] crop[ 400,   0, 200,1280] frame[ 400,   0, 200,1280] addr[ff0000ff,       0,       0] flags[0x       0] trd[0,0]
      depth[ 0]  COLOR    enable ch[0] lyr[3] z[16] prem[N] a[pixel   0] fmt[  0] fb[   0,   0;   0,   0;   0,   0] crop[ 600,   0, 200,1280] frame[ 600,   0, 200,1280] addr[ffffff00,       0,       0] flags[0x       0] trd[0,0]
      depth[ 0]    BUF    enable ch[1] lyr[0] z[16] prem[N] a[pixel 255] fmt[  0] fb[ 800,1280; 800,1280; 800,1280] crop[   0,   0, 800,1280] frame[   0,   0, 800,1280] addr[ff800000,       0,       0] flags[0x       0] trd[0,0]
      depth[ 0] 
      
      发布在 V Series
      A
      awwwwa
    • 回复: 全H616启动,GPIO和eFuse启动选择,是在哪里设置的

      @fleieng 如果烧录了eFuse会写eFuse标志位,就是eFuse选择,如果没烧eFuse,则是GPIO选择,芯片出厂默认是GPIO选择

      发布在 H/F/TV Series
      A
      awwwwa
    • 回复: V853SDK:奇怪的帧缓冲区操作

      try to use english to describe, the translate I can't understand. 😞

      发布在 V Series
      A
      awwwwa
    • 回复: 按照嘉立创开源上面做的H616开发板,可以正常烧写系统,但是启动不了,usb烧写串口输出信息,但是不知道从那个方面考虑。

      @fleieng 检查 BOOT_SEL 配置

      364f6fd5-1143-4cf2-bc01-a3b0e15796f6-image.png

      发布在 H/F/TV Series
      A
      awwwwa
    • 回复: T113-S3解码1080P视频花屏,set scaledown视频画面不动

      package/allwinner/tina_multimedia/tplayer/configs/t113_linux_cedarx.conf

      检查是否使用 video lbc mode, is_lossy and rc_en

      如果配置了设置 0 0 0

      ########### paramter ############
      [paramter]
      start_play_cache_video_frame_num = 4
      start_play_cache_size = 8            # KB
      cache_buffer_size = 10240               # KB
      cache_buffer_size_live = 10240          # KB
      start_play_cache_time = 5              # Second
      max_start_play_chache_size = 512      # KB
      max_cache_buffer_size = 10240           # KB
      
      max_http_stream_buf_size = 2048 #KB
      
      # the number of audio stream cached before decoding.
      max_audio_stream_frame_num = 128
      
      # see comment in player.cpp
      av_sync_duration = 0              # millisecond
      
      # picture num for modules
      pic_4list_num = 2
      pic_4di_num = 0
      pic_4rotate_num = 0
      pic_4smooth_num = 2
      # picture format: mb32/nv21/nv12/nv
      # deinterlace format
      deinterlace_fmt = nv12
      
      # video decoder output picture format
      vd_output_fmt = nv21
      
      # video lbc mode, is_lossy and rc_en
      vd_lbc_mode = 2
      vd_lbc_is_lossy = 1
      vd_lbc_rc_en = 0
      
      # gpu align stride values: 16/32
      # mali 400mp2 32bit
      gpu_align_bitwidth = 32
      
      # if value 1 : send 3-frames black-pic to GPU
      # SP: switch Program
      #black_pic_4_SP = 0
      
      # compensate for av vsync
      compensate_vsync = 4
      
      #use transform module to rotate the video. notice:only r18/r58/r40 support transform module
      tr_rotate_flag = 0         #1 means rotate,0 means not rotate
      tr_rotate_degree = 0    #0 means the degree is 0, 90 means the degree is 90,180 means the degree is 180, 270 means the degree is 270
      
      #use g2d module to rotate the video. notice:F133/R528
      g2d_rotate_degree = 0
      
      #the max width and height of video in this product,which need align 64
      hold_last_picture_with = 1920
      hold_last_picture_height = 1088
      
      #The first frame decoded will be used for quick display.
      show_1th_frame_quick = 1
      
      # log will output if level >= log_level
      #VERBOSE = 2,
      #DEBUG = 3,
      #INFO = 4,
      #WARNING = 5,
      #ERROR = 6,
      log_level = 6
      
      # define platform for ve phy addr offset
      platform = t113
      
      # scaledown large video(>=1080p) or not,0:means not scaledown,1:means scaledown
      scaledown_large_video_flag = 0
      
      #use ion buf in streamManager.c , 0:means not use ion buf,1:means use ion buf
      use_ion_buf_flag = 0
      
      ########### plugin ##############
      # 1. audio decoder plugin
      
      [adecoder-0]
      comment = aac_adecoder
      id = adecoder.aac
      lib = libaw_aacdec.so
      
      [adecoder-1]
      comment = alac_adecoder
      id = adecoder.alac
      lib = libaw_alacdec.so
      
      [adecoder-2]
      comment = amr_adecoder
      id = adecoder.amr
      lib = libaw_amrdec.so
      
      [adecoder-3]
      comment = ape_adecoder
      id = adecoder.ape
      lib = libaw_apedec.so
      
      [adecoder-4]
      comment = flac_adecoder
      id = adecoder.flac
      lib = libaw_flacdec.so
      
      [adecoder-5]
      comment = mp3_adecoder
      id = adecoder.mp3
      lib = libaw_mp3dec.so
      
      [adecoder-6]
      comment = ogg_adecoder
      id = adecoder.ogg
      lib = libaw_oggdec.so
      
      [adecoder-7]
      comment = wav_adecoder
      id = adecoder.wav
      lib = libaw_wavdec.so
      
      [adecoder-8]
      comment = atrc_adecoder
      id = adecoder.atrc
      lib = libaw_atrcdec.so
      
      [adecoder-9]
      comment = ra_adecoder
      id = adecoder.ra
      lib = libaw_radec.so
      
      [adecoder-10]
      comment = dsd_adecoder
      id = adecoder.dsd
      lib = libaw_dsddec.so
      
      [adecoder-11]
      comment = g729_adecoder
      id = adecoder.g729
      lib = libaw_g729dec.so
      
      [adecoder-12]
      comment = opus_adecoder
      id = adecoder.opus
      lib = libaw_opusdec.so
      
      # 2. video decoder plugin
      
      [vdecoder-0]
      comment = h264_vdecoder
      id = vdecoder.h264
      lib = libawh264.so
      init = CedarPluginVDInit
      
      [vdecoder-1]
      comment = mjpeg_vdecoder
      id = vdecoder.mjpeg
      lib = libawmjpeg.so
      init = CedarPluginVDInit
      
      [vdecoder-2]
      comment = mpeg2_vdecoder
      id = vdecoder.mpeg2
      lib = libawmpeg2.so
      init = CedarPluginVDInit
      
      [vdecoder-3]
      comment = mpeg4base_vdecoder
      id = vdecoder.mpeg4base
      lib = libawmpeg4base.so
      init = CedarPluginVDInit
      
      [vdecoder-4]
      comment = mpeg4dx_vdecoder
      id = vdecoder.mpeg4dx
      lib = libawmpeg4dx.so
      init = CedarPluginVDInit
      reference = vdecoder.mpeg4base
      
      [vdecoder-5]
      comment = mpeg4h263_vdecoder
      id = vdecoder.mpeg4h263
      lib = libawmpeg4h263.so
      init = CedarPluginVDInit
      reference = vdecoder.mpeg4base
      
      [vdecoder-6]
      comment = mpeg4normal_vdecoder
      id = vdecoder.mpeg4normal
      lib = libawmpeg4normal.so
      init = CedarPluginVDInit
      reference = vdecoder.mpeg4base
      
      [vdecoder-7]
      comment = wmv3_vdecoder
      id = vdecoder.wmv3
      lib = libawwmv3.so
      init = CedarPluginVDInit
      
      [vdecoder-8]
      comment = h265_vdecoder
      id = vdecoder.h265
      lib = libawh265.so
      init = CedarPluginVDInit
      
      # 3. external plugin
      
      #[plugin-0]
      #comment = rtp_plugin
      #id = rtp
      #lib = librtp.so
      
      发布在 MR Series
      A
      awwwwa
    • 回复: T113pro使用tina sdk无法将模块自动拷进rootfs

      检查一下modules的配置

      define KernelPackage/net-broadcom
        SUBMENU:=$(WIRELESS_MENU)
        DEPENDS:= +ap6256-firmware +@PACKAGE_broadcom-rftest
        TITLE:=broadcom(ap6256...) support
        FILES:=$(LINUX_DIR)/drivers/net/wireless/bcmdhd/bcmdhd.ko
        KCONFIG:=\
      	  CONFIG_BCMDHD=m \
      	  CONFIG_BCMDHD_SDIO=y \
      	  CONFIG_BCMDHD_OOB=y \
      	  CONFIG_BCMDHD_FW_PATH="/etc/firmware/fw_bcmdhd.bin" \
      	  CONFIG_BCMDHD_NVRAM_PATH="/etc/firmware/nvram.txt" \
      	  CONFIG_BCMDHD_SDIO=y \
      	  CONFIG_SUNXI_RFKILL=y \
      	  CONFIG_MMC=y \
      	  CONFIG_PWRSEQ_EMMC=y \
      	  CONFIG_PWRSEQ_SIMPLE=y \
      	  CONFIG_MMC_SUNXI=y \
      	  CONFIG_MMC_SUNXI_V4P1X=y \
      	  CONFIG_MMC_SUNXI_V4P00X=y \
      	  CONFIG_MMC_SUNXI_V4P10X=y \
      	  CONFIG_MMC_SUNXI_V4P5X=y \
      	  CONFIG_MMC_SUNXI_V5P3X=y \
      
        AUTOLOAD:=$(call AutoProbe,bcmdhd,1)
      endef
      
      define KernelPackage/net-broadcom/description
       Kernel modules for Broadcom AP6256...  support
      endef
      
      $(eval $(call KernelPackage,net-broadcom))
      
      发布在 编译和烧写问题专区
      A
      awwwwa
    • 回复: 为什么t527的设备树uart0没有配置引脚

      为了兼容卡打印的问题,UART的引脚在boot0&Uboot中初始化,如不需要卡打印可以配置上需要的引脚

      发布在 T Series
      A
      awwwwa
    • 回复: 如何申请下载Tina SDK权限?

      AWOL 支持这些芯片,提供SDK下载

      508a68b1-39ed-43e3-93ae-979221e8f2ab-image.png

      参考 https://www.aw-ol.com/docs 中的文档下载

      其他芯片需要联系你购买商家的客服或者代理获取

      发布在 Linux
      A
      awwwwa
    • 回复: D1s DMA驱动Ledc 问题

      使用dma模式搬运数据,需要打上如下补丁:

      ledc在dma模式的代码存在问题,已修复该问题,ledc的dma通道之前也没有支持,需要添加上

      diff --git a/hal/source/dma/platform/dma-sun8iw20.h b/hal/source/dma/platform/dma-sun8iw20.h
      index 2140014..fc5c72e 100644
      --- a/hal/source/dma/platform/dma-sun8iw20.h
      +++ b/hal/source/dma/platform/dma-sun8iw20.h
      @@ -136,6 +136,7 @@
       #define DRQDST_OTG_EP3          32
       #define DRQDST_OTG_EP4          33
       #define DRQDST_OTG_EP5          34
      +#define DRQDST_LEDC		42
       #define DRQDST_TWI0_TX		43
       #define DRQDST_TWI1_TX          44
       #define DRQDST_TWI2_TX          45
      
      diff --git a/hal/source/ledc/hal_ledc.c b/hal/source/ledc/hal_ledc.c
      index 5499452..ddf471f 100755
      --- a/hal/source/ledc/hal_ledc.c
      +++ b/hal/source/ledc/hal_ledc.c
      @@ -20,9 +20,10 @@
       #define led_err(fmt, args...)  printf("%s()%d - "fmt, __func__, __LINE__, ##args)
       
       #define LEDC_PIN_SLEEP 0
      +#define LEDC_DMA_BUF_SIZE 4096
       
       struct ledc_config ledc_config = {
      -	.led_count = 3,
      +	.led_count = 1024,
       	.reset_ns = 84,
       	.t1h_ns = 800,
       	.t1l_ns = 450,
      @@ -37,6 +38,7 @@
       static unsigned long base_addr = LEDC_BASE;
       struct sunxi_dma_chan *dma_chan;
       struct sunxi_led *led;
      +static uint8_t already_init;
       
       static hal_irqreturn_t sunxi_ledc_irq_handler(void *dummy)
       {
      @@ -448,15 +450,16 @@
       
       void hal_ledc_dma_callback(void *para)
       {
      -	printf("dma callback\n");
      +	ledc_info("dma transfer end\n");
       }
       
      -void hal_ledc_trans_data(struct ledc_config *ledc)
      +int hal_ledc_trans_data(struct ledc_config *ledc)
       {
      -	int i;
      +	int i, ret;
       	unsigned long int size;
       	unsigned int mask = 0;
       	struct dma_slave_config slave_config;
      +	unsigned int const *buf = ledc->data;
       
       	mask = LEDC_TRANS_FINISH_INT_EN | LEDC_WAITDATA_TIMEOUT_INT_EN
       		| LEDC_FIFO_OVERFLOW_INT_EN | LEDC_GLOBAL_INT_EN;
      @@ -480,23 +483,32 @@
       
       		ledc_reset_en();
       		size = ledc->length * 4;
      +		if (size <= LEDC_DMA_BUF_SIZE) {
      +			memcpy(ledc->align_dma_buf, buf, ledc->length);
      +			buf = ledc->align_dma_buf;
      +		}
       
      -		hal_dcache_clean((unsigned long)ledc->data, sizeof(ledc->data));
      +		hal_dcache_clean((unsigned long)buf, size);
       
       		slave_config.direction = DMA_MEM_TO_DEV;
      -		slave_config.src_addr = (unsigned long)(ledc->data);
      +		slave_config.src_addr = (unsigned long)buf;
       		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.src_maxburst = DMA_SLAVE_BURST_4;
      +		slave_config.dst_maxburst = DMA_SLAVE_BURST_4;
       		slave_config.slave_id = sunxi_slave_id(DRQDST_LEDC, DRQSRC_SDRAM);
      -		hal_dma_slave_config(dma_chan, &slave_config);
      +		ret = hal_dma_slave_config(dma_chan, &slave_config);
      +		if (ret) {
      +			led_err("dma slave config failed\n");
      +			return -1;
      +		}
       
      -		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);
      +		ret = hal_dma_prep_device(dma_chan, slave_config.dst_addr, slave_config.src_addr, size, DMA_MEM_TO_DEV);
      +		if (ret) {
      +			led_err("dma prep device failed\n");
      +			return -1;
      +		}
       
       		hal_ledc_set_time(ledc);
       		ledc_set_output_mode(ledc->output_mode);
      @@ -504,7 +516,15 @@
       		ledc_set_dma_mode();
       		ledc_enable_irq(mask);
       		ledc_enable();
      +
      +		dma_chan->callback = hal_ledc_dma_callback;
      +		ret = hal_dma_start(dma_chan);
      +		if (ret) {
      +			led_err("dma start trans failed\n");
      +			return -1;
      +		}
       	}
      +	return 0;
       }
       
       void hal_ledc_clear_all_irq(void)
      @@ -598,6 +618,14 @@
       
       int hal_ledc_init(void)
       {
      +	if (already_init) {
      +		already_init++;
      +		ledc_info("ledc has been inited, return ok\n");
      +		return 0;
      +	}
      +
      +	int ret, i;
      +
       	ledc_info("hal_led_init\n");
       
       	led = malloc(sizeof(struct sunxi_led));
      @@ -611,25 +639,40 @@
       	led->config.data = malloc(sizeof(unsigned int) * led->config.led_count);
       	if (NULL == led->config.data) {
       		led_err("sunxi led config data malloc err\n");
      -		goto err1;
      +		goto err0;
       	}
      +	for(i = 0;i < led->config.led_count;i++)
      +		led->config.data[i] = 0;
       
       	if (ledc_clk_init())
       	{
       		led_err("ledc clk init failed \n");
      +		goto err1;
       	}
       
       	if (ledc_pinctrl_init())
       	{
       		led_err("ledc pinctrl init failed \n");
      +		goto err2;
       	}
       
      -	hal_dma_chan_request(&dma_chan);
      +	ret = hal_dma_chan_request(&dma_chan);
      +	if (ret == HAL_DMA_CHAN_STATUS_BUSY)
      +	{
      +		led_err("request dma_chan failed\n");
      +		goto err3;
      +	}
      +	led->config.align_dma_buf = dma_alloc_coherent(LEDC_DMA_BUF_SIZE);
      +	if (!led->config.align_dma_buf)
      +	{
      +		led_err("alloc dma memory failed\n");
      +		goto err4;
      +	}
       
       	if (hal_request_irq(SUNXI_IRQ_LEDC, sunxi_ledc_irq_handler, "ledc", led) < 0)
       	{
       		led_err("ledc request irq failed \n");
      -		goto errirq;
      +		goto err5;
       	}
       
       	hal_enable_irq(SUNXI_IRQ_LEDC);
      @@ -638,13 +681,22 @@
       	pm_devops_register(&pm_ledc);
       #endif
       
      +	already_init++;
       	ledc_info("hal_led_init success\n");
       
       	return 0;
       
      -errirq:
      -	free(led->config.data);
      +err5:
      +	dma_free_coherent(led->config.align_dma_buf);
      +err4:
      +	hal_dma_chan_free(dma_chan);
      +err3:
      +	ledc_pinctrl_exit();
      +err2:
      +	ledc_clk_exit();
       err1:
      +	free(led->config.data);
      +err0:
       	free(led);
       
       	return -1;
      @@ -652,27 +704,35 @@
       
       void hal_ledc_deinit(void)
       {
      +	if (already_init > 0) {
      +		if (--already_init == 0) {
       #ifdef CONFIG_COMPONENTS_PM
      -	pm_devops_unregister(&pm_ledc);
      +			pm_devops_unregister(&pm_ledc);
       #endif
      -	hal_disable_irq(SUNXI_IRQ_LEDC);
      -	hal_free_irq(SUNXI_IRQ_LEDC);
      -	hal_dma_chan_free(dma_chan);
      -	ledc_pinctrl_exit();
      -	ledc_clk_exit();
      -	free(led->config.data);
      -	free(led);
      +			hal_disable_irq(SUNXI_IRQ_LEDC);
      +			hal_free_irq(SUNXI_IRQ_LEDC);
      +			dma_free_coherent(led->config.align_dma_buf);
      +			hal_dma_chan_free(dma_chan);
      +			ledc_pinctrl_exit();
      +			ledc_clk_exit();
      +			free(led->config.data);
      +			free(led);
      +		}
      +	}
       }
       
       int sunxi_set_all_led(unsigned int brightness)
       {
      -	int i;
      +	int i, ret;
       
       	led->config.length = led->config.led_count;
       	for(i = 0;i < led->config.led_count;i++)
       		led->config.data[i] = brightness;
       
      -	hal_ledc_trans_data(&led->config);
      +	ret = hal_ledc_trans_data(&led->config);
      +	if (ret) {
      +		led_err("ledc trans data error\n");
      +	}
       
       	return 0;
       }
      @@ -680,6 +740,7 @@
       int sunxi_set_led_brightness(int led_num, unsigned int brightness)
       {
       	u32 reg_val;
      +	int i, ret;
       
       	if (NULL == led) {
       		led_err("err : ledc is not init\n");
      @@ -691,10 +752,16 @@
       		return -1;
       	}
       
      -	led->config.length = 1;
      +	led->config.length = led_num;
       	led->config.data[led_num-1] = brightness;
       
      -	hal_ledc_trans_data(&led->config);
      +	for (i = 0; i < led_num; i++)
      +		ledc_info("the %d led light is %u\n", i + 1, led->config.data[i]);
      +
      +	ret = hal_ledc_trans_data(&led->config);
      +	if (ret) {
      +		led_err("ledc trans data error\n");
      +	}
       
       	reg_val = hal_ledc_get_irq_status();
       	ledc_info("ledc interrupt status reg is %x", reg_val);
      diff --git a/hal/source/ledc/platform_ledc.h b/hal/source/ledc/platform_ledc.h
      index 39f6933..2fa9c38 100644
      --- a/hal/source/ledc/platform_ledc.h
      +++ b/hal/source/ledc/platform_ledc.h
      @@ -33,8 +33,6 @@
       #ifndef __PLATFORM_LEDC_H__
       #define __PLATFORM_LEDC_H__
       
      -#define DRQDST_LEDC  43
      -
       #if defined(CONFIG_ARCH_SUN8IW18P1)
       #include "platform/ledc_sun8iw18.h"
       #endif
      diff --git a/hal/test/ledc/test_ledc.c b/hal/test/ledc/test_ledc.c
      index 1ade796..79694c0 100755
      --- a/hal/test/ledc/test_ledc.c
      +++ b/hal/test/ledc/test_ledc.c
      @@ -20,6 +20,7 @@
       int ledc_test(int argc, char **argv)
       {
       	int brightness = 0;
      +	int led_num;
       
       	printf("========LEDC TEST========\n");
       
      @@ -32,9 +33,14 @@
       		return 0;
       	}
       
      -	brightness = atoi(argv[2]);
      +	led_num = atoi(argv[1]);
      +	if (led_num < 1 || led_num > 1024)
      +	{
      +		printf("The led_num you entered should be between 1 and 1024\n");
      +	}
      +	brightness = atoi(argv[3]);
       
      -	switch(argv[1][0])
      +	switch(argv[2][0])
       	{
       		case 'R' : brightness <<= 8; break;
       		case 'G' : brightness <<= 16; break;
      @@ -43,7 +49,8 @@
       			   return -1;
       	}
       
      -	sunxi_set_led_brightness(1, brightness);
      +	sunxi_set_led_brightness(led_num, brightness);
      +	printf("led is %d\n", led_num);
       	printf("brightness is %d\n", brightness);
       
       	return 0;
      diff --git a/include/hal/sunxi_hal_ledc.h b/include/hal/sunxi_hal_ledc.h
      index a386338..e5a70d1 100644
      --- a/include/hal/sunxi_hal_ledc.h
      +++ b/include/hal/sunxi_hal_ledc.h
      @@ -43,6 +43,7 @@
       	unsigned long long wait_time1_ns;
       	unsigned int wait_data_time_ns;
       	char *output_mode;
      +	unsigned int *align_dma_buf;
       	unsigned int *data;
       	unsigned int length;
       };
      @@ -86,7 +87,7 @@
       
       int hal_ledc_init(void);
       void hal_ledc_deinit(void);
      -void hal_ledc_trans_data(struct ledc_config *ledc);
      +int hal_ledc_trans_data(struct ledc_config *ledc);
       void hal_ledc_clear_all_irq(void);
       unsigned int hal_ledc_get_irq_status(void);
       void hal_ledc_dma_callback(void *para);
      

      DMA下 LED 颜色异常

      diff --git a/hal/source/ledc/hal_ledc.c b/hal/source/ledc/hal_ledc.c
      index ddf471f4..8d818662 100755
      --- a/hal/source/ledc/hal_ledc.c
      +++ b/hal/source/ledc/hal_ledc.c
      @@ -308,11 +308,15 @@ static void ledc_set_wait_data_time_ns(unsigned int wait_data_time_ns)
       	hal_writel(reg_val, base_addr + LEDC_DATA_FINISH_CNT_REG);
       }
       
      +/*
      + * set the num of leds on the led-strip
      + * max support up to 1024 leds
      + */
       static void ledc_set_length(unsigned int length)
       {
       	unsigned int reg_val;
       
      -	if (length == 0)
      +	if (length == 0 || length > 1024)
       		return;
       
       	reg_val = hal_readl(base_addr + LEDC_CTRL_REG);
      @@ -721,12 +725,22 @@ void hal_ledc_deinit(void)
       	}
       }
       
      -int sunxi_set_all_led(unsigned int brightness)
      +/*
      + * set the brightness of all the leds in led-strip to a uniform value
      + * @length: the num of leds on led-strip
      + * @brightness: the brightness data
      + */
      +int sunxi_set_all_led(unsigned int length, unsigned int brightness)
       {
       	int i, ret;
       
      -	led->config.length = led->config.led_count;
      -	for(i = 0;i < led->config.led_count;i++)
      +	if (length > led->config.led_count) {
      +		led_err("%d: max support 1024 leds\n", length);
      +		return -1;
      +	}
      +
      +	led->config.length = length;
      +	for(i = 0;i < led->length;i++)
       		led->config.data[i] = brightness;
       
       	ret = hal_ledc_trans_data(&led->config);
      @@ -737,7 +751,13 @@ int sunxi_set_all_led(unsigned int brightness)
       	return 0;
       }
       
      -int sunxi_set_led_brightness(int led_num, unsigned int brightness)
      +/*
      + * set the brightness of each led on the led strip
      + * @length: all the num of leds on the led strip
      + * @led_num: the specified led that you want to set
      + * @brightness: the led brightness data
      + */
      +int sunxi_set_led_brightness(unsigned int length, unsigned int led_num, unsigned int brightness)
       {
       	u32 reg_val;
       	int i, ret;
      @@ -747,12 +767,13 @@ int sunxi_set_led_brightness(int led_num, unsigned int brightness)
       		return -1;
       	}
       
      -	if (led_num > led->config.led_count) {
      +	if (length > led->config.led_count || len_num > length) {
       		led_err("has not the %d led\n", led_num);
       		return -1;
       	}
       
      -	led->config.length = led_num;
      +	led->config.length = length;
      +	/* set the specified led brightness, others set default brightness: 0x0*/
       	led->config.data[led_num-1] = brightness;
       
       	for (i = 0; i < led_num; i++)
      
      发布在 MR Series
      A
      awwwwa
    • 回复: D1s ledc驱动代码bug,DMA模式无法使用

      @leomini5 我这边没有D1s的板卡,在T113上测试通过,测试1024颗灯,这个可能是RV才有的bug?

      发布在 MR Series
      A
      awwwwa
    • 回复: Tina Ledc 无法驱动超过32个灯

      重复帖:D1s ledc驱动代码bug,DMA模式无法使用

      https://bbs.aw-ol.com/topic/5071/share/1

      发布在 MR Series
      A
      awwwwa
    • 回复: D1s ledc驱动代码bug,DMA模式无法使用

      之前在其他帖发过DMA模式的patch,这里同步下

      lichee/linux-5.4/drivers/leds/leds-sunxi.c

      // SPDX-License-Identifier: GPL-2.0-only
      /*
       * drivers/leds/leds-sunxi.c - Allwinner RGB LED Driver
       *
       * Copyright (C) 2018 Allwinner Technology Limited. All rights reserved.
       *      http://www.allwinnertech.com
       *
       *Author : Albert Yu <yuxyun@allwinnertech.com>
       *	   Lewis <liuyu@allwinnertech.com>
       * This program is free software; you can redistribute it and/or modify
       * it under the terms of the GNU General Public License version 2 as
       * published by the Free Software Foundation.
       *
       */
      
      #include <linux/module.h>
      #include <linux/delay.h>
      #include <linux/leds.h>
      #include <linux/io.h>
      #include <linux/of.h>
      #include <linux/slab.h>
      #include <linux/clk.h>
      #include <linux/dmaengine.h>
      #include <linux/interrupt.h>
      #include <linux/platform_device.h>
      #include <linux/pinctrl/consumer.h>
      #include <linux/dma-mapping.h>
      #include <linux/debugfs.h>
      #include <linux/uaccess.h>
      #include <linux/delay.h>
      #include <linux/regulator/consumer.h>
      #include <linux/reset.h>
      
      #if IS_ENABLED(CONFIG_PM)
      #include <linux/pm.h>
      #endif
      #include "leds-sunxi.h"
      
      /* For debug */
      #define LED_ERR(fmt, arg...) pr_err("%s()%d - "fmt, __func__, __LINE__, ##arg)
      
      #define dprintk(level_mask, fmt, arg...)				\
      do {									\
      	if (unlikely(debug_mask & level_mask))				\
      		pr_warn("%s()%d - "fmt, __func__, __LINE__, ##arg);	\
      } while (0)
      
      static u32 debug_mask = 1;
      static struct sunxi_led *sunxi_led_global;
      static struct class *led_class;
      
      #define sunxi_slave_id(d, s) (((d)<<16) | (s))
      
      /*For Driver */
      static void led_dump_reg(struct sunxi_led *led, u32 offset, u32 len)
      {
      	u32 i;
      	u8 buf[64], cnt = 0;
      
      	for (i = 0; i < len; i = i + REG_INTERVAL) {
      		if (i%HEXADECIMAL == 0)
      			cnt += sprintf(buf + cnt, "0x%08x: ",
      					(u32)(led->res->start + offset + i));
      
      		cnt += sprintf(buf + cnt, "%08x ",
      				readl(led->iomem_reg_base + offset + i));
      
      		if (i%HEXADECIMAL == REG_CL) {
      			pr_warn("%s\n", buf);
      			cnt = 0;
      		}
      	}
      }
      
      static void sunxi_clk_get(struct sunxi_led *led)
      {
      	struct device *dev = led->dev;
      	struct device_node *np = dev->of_node;
      
      	led->clk_ledc = of_clk_get(np, 0);
      	if (IS_ERR(led->clk_ledc))
      		LED_ERR("failed to get clk_ledc!\n");
      
      	led->clk_cpuapb = of_clk_get(np, 1);
      	if (IS_ERR(led->clk_cpuapb))
      		LED_ERR("failed to get clk_cpuapb!\n");
      }
      
      static void sunxi_clk_put(struct sunxi_led *led)
      {
      	clk_put(led->clk_ledc);
      	clk_put(led->clk_cpuapb);
      	led->clk_ledc = NULL;
      	led->clk_cpuapb = NULL;
      }
      
      static void sunxi_clk_enable(struct sunxi_led *led)
      {
      	clk_prepare_enable(led->clk_ledc);
      	clk_prepare_enable(led->clk_cpuapb);
      }
      
      static void sunxi_clk_disable(struct sunxi_led *led)
      {
      	clk_disable_unprepare(led->clk_ledc);
      }
      
      static void sunxi_clk_init(struct sunxi_led *led)
      {
      	sunxi_clk_get(led);
      	sunxi_clk_enable(led);
      }
      
      static void sunxi_clk_deinit(struct sunxi_led *led)
      {
      	sunxi_clk_disable(led);
      	sunxi_clk_put(led);
      }
      
      static u32 sunxi_get_reg(int offset)
      {
      	struct sunxi_led *led = sunxi_led_global;
      	u32 value = ioread32(((u8 *)led->iomem_reg_base) + offset);
      
      	return value;
      }
      
      static void sunxi_set_reg(int offset, u32 value)
      {
      	struct sunxi_led *led = sunxi_led_global;
      
      	iowrite32(value, ((u8 *)led->iomem_reg_base) + offset);
      }
      
      static inline void sunxi_set_reset_ns(struct sunxi_led *led)
      {
      	u32 n, reg_val;
      	u32 mask = 0x1FFF;
      	u32 min = SUNXI_RESET_TIME_MIN_NS;
      	u32 max = SUNXI_RESET_TIME_MAX_NS;
      
      	if (led->reset_ns < min || led->reset_ns > max) {
      		LED_ERR("invalid parameter, reset_ns should be %u-%u!\n",
      				min, max);
      		return;
      	}
      
      	n = (led->reset_ns - 42) / 42;
      	reg_val = sunxi_get_reg(LED_RESET_TIMING_CTRL_REG_OFFSET);
      	reg_val &= ~(mask << 16);
      	reg_val |= (n << 16);
      	sunxi_set_reg(LED_RESET_TIMING_CTRL_REG_OFFSET, reg_val);
      }
      
      static inline void sunxi_set_t1h_ns(struct sunxi_led *led)
      {
      	u32 n, reg_val;
      	u32 mask = 0x3F;
      	u32 shift = 21;
      	u32 min = SUNXI_T1H_MIN_NS;
      	u32 max = SUNXI_T1H_MAX_NS;
      
      	if (led->t1h_ns < min || led->t1h_ns > max) {
      		LED_ERR("invalid parameter, t1h_ns should be %u-%u!\n",
      				min, max);
      		return;
      	}
      
      	n = (led->t1h_ns - 42) / 42;
      	reg_val = sunxi_get_reg(LED_T01_TIMING_CTRL_REG_OFFSET);
      	reg_val &= ~(mask << shift);
      	reg_val |= n << shift;
      	sunxi_set_reg(LED_T01_TIMING_CTRL_REG_OFFSET, reg_val);
      }
      
      static inline void sunxi_set_t1l_ns(struct sunxi_led *led)
      {
      	u32 n, reg_val;
      	u32 mask = 0x1F;
      	u32 shift = 16;
      	u32 min = SUNXI_T1L_MIN_NS;
      	u32 max = SUNXI_T1L_MAX_NS;
      
      	if (led->t1l_ns < min || led->t1l_ns > max) {
      		LED_ERR("invalid parameter, t1l_ns should be %u-%u!\n",
      				min, max);
      		return;
      	}
      
      	n = (led->t1l_ns - 42) / 42;
      	reg_val = sunxi_get_reg(LED_T01_TIMING_CTRL_REG_OFFSET);
      	reg_val &= ~(mask << shift);
      	reg_val |= n << shift;
      	sunxi_set_reg(LED_T01_TIMING_CTRL_REG_OFFSET, reg_val);
      }
      
      static inline void sunxi_set_t0h_ns(struct sunxi_led *led)
      {
      	u32 n, reg_val;
      	u32 mask = 0x1F;
      	u32 shift = 6;
      	u32 min = SUNXI_T0H_MIN_NS;
      	u32 max = SUNXI_T0H_MAX_NS;
      
      	if (led->t0h_ns < min || led->t0h_ns > max) {
      		LED_ERR("invalid parameter, t0h_ns should be %u-%u!\n",
      			min, max);
      		return;
      	}
      
      	n = (led->t0h_ns - 42) / 42;
      	reg_val = sunxi_get_reg(LED_T01_TIMING_CTRL_REG_OFFSET);
      	reg_val &= ~(mask << shift);
      	reg_val |= n << shift;
      	sunxi_set_reg(LED_T01_TIMING_CTRL_REG_OFFSET, reg_val);
      }
      
      static inline void sunxi_set_t0l_ns(struct sunxi_led *led)
      {
      	u32 n, reg_val;
      	u32 min = SUNXI_T0L_MIN_NS;
      	u32 max = SUNXI_T0L_MAX_NS;
      
      	if (led->t0l_ns < min || led->t0l_ns > max) {
      		LED_ERR("invalid parameter, t0l_ns should be %u-%u!\n",
      				min, max);
      		return;
      	}
      
      	n = (led->t0l_ns - 42) / 42;
      	reg_val = sunxi_get_reg(LED_T01_TIMING_CTRL_REG_OFFSET);
      	reg_val &= ~0x3F;
      	reg_val |= n;
      	sunxi_set_reg(LED_T01_TIMING_CTRL_REG_OFFSET, reg_val);
      }
      
      static inline void sunxi_set_wait_time0_ns(struct sunxi_led *led)
      {
      	u32 n, reg_val;
      	u32 min = SUNXI_WAIT_TIME0_MIN_NS;
      	u32 max = SUNXI_WAIT_TIME0_MAX_NS;
      
      	if (led->wait_time0_ns < min || led->wait_time0_ns > max) {
      		LED_ERR("invalid parameter, wait_time0_ns should be %u-%u!\n",
      				min, max);
      		return;
      	}
      
      	n = (led->wait_time0_ns - 42) / 42;
      	reg_val = (1 << 8) | n;
      	sunxi_set_reg(LEDC_WAIT_TIME0_CTRL_REG, reg_val);
      }
      
      static inline void sunxi_set_wait_time1_ns(struct sunxi_led *led)
      {
      	unsigned long long tmp, max = SUNXI_WAIT_TIME1_MAX_NS;
      	u32 min = SUNXI_WAIT_TIME1_MIN_NS;
      	u32 n, reg_val;
      
      	if (led->wait_time1_ns < min || led->wait_time1_ns > max) {
      		LED_ERR("invalid parameter, wait_time1_ns should be %u-%llu!\n",
      			min, max);
      		return;
      	}
      
      	tmp = led->wait_time1_ns;
      	n = div_u64(tmp, 42);
      	n -= 1;
      	reg_val = (1 << 31) | n;
      	sunxi_set_reg(LEDC_WAIT_TIME1_CTRL_REG, reg_val);
      }
      
      static inline void sunxi_set_wait_data_time_ns(struct sunxi_led *led)
      {
      	u32 min, max;
      #ifndef SUNXI_FPGA_LEDC
      	u32 mask = 0x1FFF, shift = 16, reg_val = 0, n;
      #endif
      	min = SUNXI_WAIT_DATA_TIME_MIN_NS;
      #ifdef SUNXI_FPGA_LEDC
      	/*
      	 * For FPGA platforms, it is easy to meet wait data timeout for
      	 * the obvious latency of task which is because of less cpu cores
      	 * and lower cpu frequency compared with IC platforms, so here we
      	 * permit long enough time latency.
      	 */
      	max = SUNXI_WAIT_DATA_TIME_MAX_NS_FPGA;
      #else /* SUNXI_FPGA_LEDC */
      	max = SUNXI_WAIT_DATA_TIME_MAX_NS_IC;
      #endif /* SUNXI_FPGA_LEDC */
      
      	if (led->wait_data_time_ns < min || led->wait_data_time_ns > max) {
      		LED_ERR("invalid parameter, wait_data_time_ns should be %u-%u!\n",
      			min, max);
      		return;
      	}
      
      #ifndef SUNXI_FPGA_LEDC
      	n = (led->wait_data_time_ns - 42) / 42;
      	reg_val &= ~(mask << shift);
      	reg_val |= (n << shift);
      	sunxi_set_reg(LEDC_DATA_FINISH_CNT_REG_OFFSET, reg_val);
      #endif /* SUNXI_FPGA_LEDC */
      }
      
      static void sunxi_ledc_set_time(struct sunxi_led *led)
      {
      	sunxi_set_reset_ns(led);
      	sunxi_set_t1h_ns(led);
      	sunxi_set_t1l_ns(led);
      	sunxi_set_t0h_ns(led);
      	sunxi_set_t0l_ns(led);
      	sunxi_set_wait_time0_ns(led);
      	sunxi_set_wait_time1_ns(led);
      	sunxi_set_wait_data_time_ns(led);
      }
      
      static void sunxi_ledc_set_length(struct sunxi_led *led)
      {
      	u32 reg_val;
      	u32 length = led->length;
      
      	if (length == 0)
      		return;
      
      	if (length > led->led_count)
      		return;
      
      	reg_val = sunxi_get_reg(LEDC_CTRL_REG_OFFSET);
      	reg_val &= ~(0x1FFF << 16);
      	reg_val |=  length << 16;
      	sunxi_set_reg(LEDC_CTRL_REG_OFFSET, reg_val);
      
      	reg_val = sunxi_get_reg(LED_RESET_TIMING_CTRL_REG_OFFSET);
      	reg_val &= ~0x3FF;
      	reg_val |= length - 1;
      	sunxi_set_reg(LED_RESET_TIMING_CTRL_REG_OFFSET, reg_val);
      }
      
      static void sunxi_ledc_set_output_mode(struct sunxi_led *led, const char *str)
      {
      	u32 val;
      	u32 mask = 0x7;
      	u32 shift = 6;
      	u32 reg_val = sunxi_get_reg(LEDC_CTRL_REG_OFFSET);
      
      	if (str != NULL) {
      		if (!strncmp(str, "GRB", 3))
      			val = SUNXI_OUTPUT_GRB;
      		else if (!strncmp(str, "GBR", 3))
      			val = SUNXI_OUTPUT_GBR;
      		else if (!strncmp(str, "RGB", 3))
      			val = SUNXI_OUTPUT_RGB;
      		else if (!strncmp(str, "RBG", 3))
      			val = SUNXI_OUTPUT_RBG;
      		else if (!strncmp(str, "BGR", 3))
      			val = SUNXI_OUTPUT_BGR;
      		else if (!strncmp(str, "BRG", 3))
      			val = SUNXI_OUTPUT_BRG;
      		else
      			return;
      	} else {
      		val = led->output_mode.val;
      	}
      
      	reg_val &= ~(mask << shift);
      	reg_val |= val;
      
      	sunxi_set_reg(LEDC_CTRL_REG_OFFSET, reg_val);
      
      	if (str != NULL) {
      		if (strncmp(str, led->output_mode.str, 3))
      			memcpy(led->output_mode.str, str, 3);
      	}
      
      	if (val != led->output_mode.val)
      		led->output_mode.val = val;
      }
      
      static void sunxi_ledc_enable_irq(u32 mask)
      {
      	u32 reg_val = 0;
      
      	reg_val |= mask;
      	sunxi_set_reg(LEDC_INT_CTRL_REG_OFFSET, reg_val);
      }
      
      static void sunxi_ledc_disable_irq(u32 mask)
      {
      	u32 reg_val = 0;
      
      	reg_val = sunxi_get_reg(LEDC_INT_CTRL_REG_OFFSET);
      	reg_val &= ~mask;
      	sunxi_set_reg(LEDC_INT_CTRL_REG_OFFSET, reg_val);
      }
      
      static inline void sunxi_ledc_enable(struct sunxi_led *led)
      {
      	u32 reg_val;
      
      	reg_val = sunxi_get_reg(LEDC_CTRL_REG_OFFSET);
      	reg_val |=  1;
      	sunxi_set_reg(LEDC_CTRL_REG_OFFSET, reg_val);
      }
      
      static inline void sunxi_ledc_reset(struct sunxi_led *led)
      {
      	u32 reg_val = sunxi_get_reg(LEDC_CTRL_REG_OFFSET);
      
      	sunxi_ledc_disable_irq(LEDC_TRANS_FINISH_INT_EN | LEDC_FIFO_CPUREQ_INT_EN
      			| LEDC_WAITDATA_TIMEOUT_INT_EN | LEDC_FIFO_OVERFLOW_INT_EN
      			| LEDC_GLOBAL_INT_EN);
      
      	if (debug_mask & DEBUG_INFO2) {
      		dprintk(DEBUG_INFO2, "dump reg:\n");
      		led_dump_reg(led, 0, 0x30);
      	}
      
      	reg_val |= 1 << 1;
      	sunxi_set_reg(LEDC_CTRL_REG_OFFSET, reg_val);
      }
      
      #ifdef CONFIG_DEBUG_FS
      static ssize_t reset_ns_write(struct file *filp, const char __user *buf,
      			size_t count, loff_t *offp)
      {
      	int err;
      	char buffer[64];
      	u32 min, max;
      	unsigned long val;
      	struct sunxi_led *led = sunxi_led_global;
      
      	min = SUNXI_RESET_TIME_MIN_NS;
      	max = SUNXI_RESET_TIME_MAX_NS;
      
      	if (count >= sizeof(buffer))
      		goto err_out;
      
      	if (copy_from_user(buffer, buf, count))
      		goto err_out;
      
      	buffer[count] = '\0';
      
      	err = kstrtoul(buffer, 10, &val);
      	if (err)
      		goto err_out;
      
      	if (val < min || val > max)
      		goto err_out;
      
      	led->reset_ns = val;
      	sunxi_set_reset_ns(led);
      
      	*offp += count;
      
      	return count;
      
      err_out:
      	LED_ERR("invalid parameter, reset_ns should be %u-%u!\n",
      		min, max);
      
      	return -EINVAL;
      }
      
      static ssize_t reset_ns_read(struct file *filp, char __user *buf,
      			size_t count, loff_t *offp)
      {
      	int r;
      	char buffer[64];
      	struct sunxi_led *led = sunxi_led_global;
      
      	r = snprintf(buffer, 64, "%u\n", led->reset_ns);
      
      	return simple_read_from_buffer(buf, count, offp, buffer, r);
      }
      
      static const struct file_operations reset_ns_fops = {
      	.owner = THIS_MODULE,
      	.write = reset_ns_write,
      	.read  = reset_ns_read,
      };
      
      static ssize_t t1h_ns_write(struct file *filp, const char __user *buf,
      			size_t count, loff_t *offp)
      {
      	int err;
      	char buffer[64];
      	u32 min, max;
      	unsigned long val;
      	struct sunxi_led *led = sunxi_led_global;
      
      	min = SUNXI_T1H_MIN_NS;
      	max = SUNXI_T1H_MAX_NS;
      
      	if (count >= sizeof(buffer))
      		return -EINVAL;
      
      	if (copy_from_user(buffer, buf, count))
      		return -EFAULT;
      
      	buffer[count] = '\0';
      
      	err = kstrtoul(buffer, 10, &val);
      	if (err)
      		return -EINVAL;
      
      	if (val < min || val > max)
      		goto err_out;
      
      	led->t1h_ns = val;
      
      	sunxi_set_t1h_ns(led);
      
      	*offp += count;
      
      	return count;
      
      err_out:
      	LED_ERR("invalid parameter, t1h_ns should be %u-%u!\n",
      		min, max);
      
      	return -EINVAL;
      }
      
      static ssize_t t1h_ns_read(struct file *filp, char __user *buf,
      			size_t count, loff_t *offp)
      {
      	int r;
      	char buffer[64];
      	struct sunxi_led *led = sunxi_led_global;
      
      	r = snprintf(buffer, 64, "%u\n", led->t1h_ns);
      
      	return simple_read_from_buffer(buf, count, offp, buffer, r);
      }
      
      static const struct file_operations t1h_ns_fops = {
      	.owner = THIS_MODULE,
      	.write = t1h_ns_write,
      	.read  = t1h_ns_read,
      };
      
      static ssize_t t1l_ns_write(struct file *filp, const char __user *buf,
      			size_t count, loff_t *offp)
      {
      	int err;
      	char buffer[64];
      	u32 min, max;
      	unsigned long val;
      	struct sunxi_led *led = sunxi_led_global;
      
      	min = SUNXI_T1L_MIN_NS;
      	max = SUNXI_T1L_MAX_NS;
      
      	if (count >= sizeof(buffer))
      		goto err_out;
      
      	if (copy_from_user(buffer, buf, count))
      		goto err_out;
      
      	buffer[count] = '\0';
      
      	err = kstrtoul(buffer, 10, &val);
      	if (err)
      		goto err_out;
      
      	if (val < min || val > max)
      		goto err_out;
      
      	led->t1l_ns = val;
      	sunxi_set_t1l_ns(led);
      
      	*offp += count;
      
      	return count;
      
      err_out:
      	LED_ERR("invalid parameter, t1l_ns should be %u-%u!\n",
      		min, max);
      
      	return -EINVAL;
      }
      
      static ssize_t t1l_ns_read(struct file *filp, char __user *buf,
      			size_t count, loff_t *offp)
      {
      	int r;
      	char buffer[64];
      	struct sunxi_led *led = sunxi_led_global;
      
      	r = snprintf(buffer, 64, "%u\n", led->t1l_ns);
      
      	return simple_read_from_buffer(buf, count, offp, buffer, r);
      }
      
      static const struct file_operations t1l_ns_fops = {
      	.owner = THIS_MODULE,
      	.write = t1l_ns_write,
      	.read  = t1l_ns_read,
      };
      
      static ssize_t t0h_ns_write(struct file *filp, const char __user *buf,
      			size_t count, loff_t *offp)
      {
      	int err;
      	char buffer[64];
      	u32 min, max;
      	unsigned long val;
      	struct sunxi_led *led = sunxi_led_global;
      
      	min = SUNXI_T0H_MIN_NS;
      	max = SUNXI_T0H_MAX_NS;
      
      	if (count >= sizeof(buffer))
      		goto err_out;
      
      	if (copy_from_user(buffer, buf, count))
      		goto err_out;
      
      	buffer[count] = '\0';
      
      	err = kstrtoul(buffer, 10, &val);
      	if (err)
      		goto err_out;
      
      	if (val < min || val > max)
      		goto err_out;
      
      	led->t0h_ns = val;
      	sunxi_set_t0h_ns(led);
      
      	*offp += count;
      
      	return count;
      
      err_out:
      	LED_ERR("invalid parameter, t0h_ns should be %u-%u!\n",
      		min, max);
      
      	return -EINVAL;
      }
      
      static ssize_t t0h_ns_read(struct file *filp, char __user *buf,
      			size_t count, loff_t *offp)
      {
      	int r;
      	char buffer[64];
      	struct sunxi_led *led = sunxi_led_global;
      
      	r = snprintf(buffer, 64, "%u\n", led->t0h_ns);
      
      	return simple_read_from_buffer(buf, count, offp, buffer, r);
      }
      
      static const struct file_operations t0h_ns_fops = {
      	.owner = THIS_MODULE,
      	.write = t0h_ns_write,
      	.read  = t0h_ns_read,
      };
      
      static ssize_t t0l_ns_write(struct file *filp, const char __user *buf,
      			size_t count, loff_t *offp)
      {
      	int err;
      	char buffer[64];
      	u32 min, max;
      	unsigned long val;
      	struct sunxi_led *led = sunxi_led_global;
      
      	min = SUNXI_T0L_MIN_NS;
      	max = SUNXI_T0L_MAX_NS;
      
      	if (count >= sizeof(buffer))
      		goto err_out;
      
      	if (copy_from_user(buffer, buf, count))
      		goto err_out;
      
      	buffer[count] = '\0';
      
      	err = kstrtoul(buffer, 10, &val);
      	if (err)
      		goto err_out;
      
      	if (val < min || val > max)
      		goto err_out;
      
      	led->t0l_ns = val;
      	sunxi_set_t0l_ns(led);
      
      	*offp += count;
      
      	return count;
      
      err_out:
      	LED_ERR("invalid parameter, t0l_ns should be %u-%u!\n",
      		min, max);
      
      	return -EINVAL;
      }
      
      static ssize_t t0l_ns_read(struct file *filp, char __user *buf,
      			size_t count, loff_t *offp)
      {
      	int r;
      	char buffer[64];
      	struct sunxi_led *led = sunxi_led_global;
      
      	r = snprintf(buffer, 64, "%u\n", led->t0l_ns);
      
      	return simple_read_from_buffer(buf, count, offp, buffer, r);
      }
      
      static const struct file_operations t0l_ns_fops = {
      	.owner = THIS_MODULE,
      	.write = t0l_ns_write,
      	.read  = t0l_ns_read,
      };
      
      static ssize_t wait_time0_ns_write(struct file *filp, const char __user *buf,
      				size_t count, loff_t *offp)
      {
      	int err;
      	char buffer[64];
      	u32 min, max;
      	unsigned long val;
      	struct sunxi_led *led = sunxi_led_global;
      
      	min = SUNXI_WAIT_TIME0_MIN_NS;
      	max = SUNXI_WAIT_TIME0_MAX_NS;
      
      	if (count >= sizeof(buffer))
      		goto err_out;
      
      	if (copy_from_user(buffer, buf, count))
      		goto err_out;
      
      	buffer[count] = '\0';
      
      	err = kstrtoul(buffer, 10, &val);
      	if (err)
      		goto err_out;
      
      	if (val < min || val > max)
      		goto err_out;
      
      	led->wait_time0_ns = val;
      	sunxi_set_wait_time0_ns(led);
      
      	*offp += count;
      
      	return count;
      
      err_out:
      	LED_ERR("invalid parameter, wait_time0_ns should be %u-%u!\n",
      		min, max);
      
      	return -EINVAL;
      }
      
      static ssize_t wait_time0_ns_read(struct file *filp, char __user *buf,
      				size_t count, loff_t *offp)
      {
      	int r;
      	char buffer[64];
      	struct sunxi_led *led = sunxi_led_global;
      
      	r = snprintf(buffer, 64, "%u\n", led->wait_time0_ns);
      
      	return simple_read_from_buffer(buf, count, offp, buffer, r);
      }
      
      static const struct file_operations wait_time0_ns_fops = {
      	.owner = THIS_MODULE,
      	.write = wait_time0_ns_write,
      	.read  = wait_time0_ns_read,
      };
      
      static ssize_t wait_time1_ns_write(struct file *filp, const char __user *buf,
      				size_t count, loff_t *offp)
      {
      	int err;
      	char buffer[64];
      	u32 min;
      	unsigned long long max;
      	unsigned long long val;
      	struct sunxi_led *led = sunxi_led_global;
      
      	min = SUNXI_WAIT_TIME1_MIN_NS;
      	max = SUNXI_WAIT_TIME1_MAX_NS;
      
      	if (count >= sizeof(buffer))
      		goto err_out;
      
      	if (copy_from_user(buffer, buf, count))
      		goto err_out;
      
      	buffer[count] = '\0';
      
      	err = kstrtoull(buffer, 10, &val);
      	if (err)
      		goto err_out;
      
      	if (val < min || val > max)
      		goto err_out;
      
      	led->wait_time1_ns = val;
      	sunxi_set_wait_time1_ns(led);
      
      	*offp += count;
      
      	return count;
      
      err_out:
      	LED_ERR("invalid parameter, wait_time1_ns should be %u-%lld!\n",
      		min, max);
      
      	return -EINVAL;
      }
      
      static ssize_t wait_time1_ns_read(struct file *filp, char __user *buf,
      				size_t count, loff_t *offp)
      {
      	int r;
      	char buffer[64];
      	struct sunxi_led *led = sunxi_led_global;
      
      	r = snprintf(buffer, 64, "%lld\n", led->wait_time1_ns);
      
      	return simple_read_from_buffer(buf, count, offp, buffer, r);
      }
      
      static const struct file_operations wait_time1_ns_fops = {
      	.owner = THIS_MODULE,
      	.write = wait_time1_ns_write,
      	.read  = wait_time1_ns_read,
      };
      
      static ssize_t wait_data_time_ns_write(struct file *filp,
      				const char __user *buf,
      				size_t count, loff_t *offp)
      {
      	int err;
      	char buffer[64];
      	u32 min, max;
      	unsigned long val;
      	struct sunxi_led *led = sunxi_led_global;
      
      	min = SUNXI_WAIT_DATA_TIME_MIN_NS;
      #ifdef SUNXI_FPGA_LEDC
      	max = SUNXI_WAIT_DATA_TIME_MAX_NS_FPGA;
      #else
      	max = SUNXI_WAIT_DATA_TIME_MAX_NS_IC;
      #endif
      
      	if (count >= sizeof(buffer))
      		goto err_out;
      
      	if (copy_from_user(buffer, buf, count))
      		goto err_out;
      
      	buffer[count] = '\0';
      
      	err = kstrtoul(buffer, 10, &val);
      	if (err)
      		goto err_out;
      
      	if (val < min || val > max)
      		goto err_out;
      
      	led->wait_data_time_ns = val;
      	sunxi_set_wait_data_time_ns(led);
      
      	*offp += count;
      
      	return count;
      
      err_out:
      	LED_ERR("invalid parameter, wait_data_time_ns should be %u-%u!\n",
      		min, max);
      
      	return -EINVAL;
      }
      
      static ssize_t wait_data_time_ns_read(struct file *filp, char __user *buf,
      				size_t count, loff_t *offp)
      {
      	int r;
      	char buffer[64];
      	struct sunxi_led *led = sunxi_led_global;
      
      	r = snprintf(buffer, 64, "%u\n", led->wait_data_time_ns);
      
      	return simple_read_from_buffer(buf, count, offp, buffer, r);
      }
      
      static const struct file_operations wait_data_time_ns_fops = {
      	.owner = THIS_MODULE,
      	.write = wait_data_time_ns_write,
      	.read  = wait_data_time_ns_read,
      };
      
      static int data_show(struct seq_file *s, void *data)
      {
      	int i;
      	struct sunxi_led *led = sunxi_led_global;
      
      	for (i = 0; i < led->led_count; i++) {
      		if (!(i % 4)) {
      			if (i + 4 <= led->led_count)
      				seq_printf(s, "%04d-%04d", i, i + 4);
      			else
      				seq_printf(s, "%04d-%04d", i, led->led_count);
      		}
      		seq_printf(s, " 0x%08x", led->data[i]);
      		if (((i % 4) == 3) || (i == led->led_count - 1))
      			seq_puts(s, "\n");
      	}
      
      	return 0;
      }
      
      static int data_open(struct inode *inode, struct file *file)
      {
      	return single_open(file, data_show, inode->i_private);
      }
      
      static const struct file_operations data_fops = {
      	.owner = THIS_MODULE,
      	.open  = data_open,
      	.read = seq_read,
      	.llseek = seq_lseek,
      	.release = single_release,
      };
      
      static ssize_t output_mode_write(struct file *filp, const char __user *buf,
      			size_t count, loff_t *offp)
      {
      	char buffer[64];
      	struct sunxi_led *led = sunxi_led_global;
      
      	if (count >= sizeof(buffer))
      		goto err_out;
      
      	if (copy_from_user(buffer, buf, count))
      		goto err_out;
      
      	buffer[count] = '\0';
      
      	sunxi_ledc_set_output_mode(led, buffer);
      
      	*offp += count;
      
      	return count;
      
      err_out:
      	LED_ERR("invalid parameter!\n");
      
      	return -EINVAL;
      }
      
      static ssize_t output_mode_read(struct file *filp, char __user *buf,
      			size_t count, loff_t *offp)
      {
      	int r;
      	char buffer[64];
      	struct sunxi_led *led = sunxi_led_global;
      
      	r = snprintf(buffer, 64, "%s\n", led->output_mode.str);
      
      	return simple_read_from_buffer(buf, count, offp, buffer, r);
      }
      
      static const struct file_operations output_mode_fops = {
      	.owner = THIS_MODULE,
      	.write = output_mode_write,
      	.read  = output_mode_read,
      };
      
      static ssize_t hwversion_read(struct file *filp, char __user *buf,
      			size_t count, loff_t *offp)
      {
      	int r;
      	char buffer[64];
      	u32 reg_val, major_ver, minor_ver;
      
      	reg_val = sunxi_get_reg(LEDC_VER_NUM_REG);
      	major_ver = reg_val >> 16;
      	minor_ver = reg_val & 0xF;
      
      	r = snprintf(buffer, 64, "r%up%u\n", major_ver, minor_ver);
      
      	return simple_read_from_buffer(buf, count, offp, buffer, r);
      }
      
      static const struct file_operations hwversion_fops = {
      	.owner = THIS_MODULE,
      	.read  = hwversion_read,
      };
      
      static void sunxi_led_create_debugfs(struct sunxi_led *led)
      {
      	struct dentry *debugfs_dir, *debugfs_file;
      
      	debugfs_dir = debugfs_create_dir("sunxi_leds", NULL);
      	if (IS_ERR_OR_NULL(debugfs_dir)) {
      		LED_ERR("debugfs_create_dir failed!\n");
      		return;
      	}
      
      	led->debugfs_dir = debugfs_dir;
      
      	debugfs_file = debugfs_create_file("reset_ns", 0660,
      				debugfs_dir, NULL, &reset_ns_fops);
      	if (!debugfs_file)
      		LED_ERR("debugfs_create_file for reset_ns failed!\n");
      
      	debugfs_file = debugfs_create_file("t1h_ns", 0660,
      				debugfs_dir, NULL, &t1h_ns_fops);
      	if (!debugfs_file)
      		LED_ERR("debugfs_create_file for t1h_ns failed!\n");
      
      	debugfs_file = debugfs_create_file("t1l_ns", 0660,
      				debugfs_dir, NULL, &t1l_ns_fops);
      	if (!debugfs_file)
      		LED_ERR("debugfs_create_file for t1l_ns failed!\n");
      
      	debugfs_file = debugfs_create_file("t0h_ns", 0660,
      				debugfs_dir, NULL, &t0h_ns_fops);
      	if (!debugfs_file)
      		LED_ERR("debugfs_create_file for t0h_ns failed!\n");
      
      	debugfs_file = debugfs_create_file("t0l_ns", 0660,
      				debugfs_dir, NULL, &t0l_ns_fops);
      	if (!debugfs_file)
      		LED_ERR("debugfs_create_file for t0l_ns failed!\n");
      
      	debugfs_file = debugfs_create_file("wait_time0_ns", 0660,
      				debugfs_dir, NULL, &wait_time0_ns_fops);
      	if (!debugfs_file)
      		LED_ERR("debugfs_create_file for wait_time0_ns failed!\n");
      
      	debugfs_file = debugfs_create_file("wait_time1_ns", 0660,
      				debugfs_dir, NULL, &wait_time1_ns_fops);
      	if (!debugfs_file)
      		LED_ERR("debugfs_create_file for wait_time1_ns failed!\n");
      
      	debugfs_file = debugfs_create_file("wait_data_time_ns", 0660,
      				debugfs_dir, NULL, &wait_data_time_ns_fops);
      	if (!debugfs_file)
      		LED_ERR("debugfs_create_file for wait_data_time_ns failed!\n");
      
      	debugfs_file = debugfs_create_file("data", 0440,
      				debugfs_dir, NULL, &data_fops);
      	if (!debugfs_file)
      		LED_ERR("debugfs_create_file for data failed!\n");
      
      	debugfs_file = debugfs_create_file("output_mode", 0660,
      				debugfs_dir, NULL, &output_mode_fops);
      	if (!debugfs_file)
      		LED_ERR("debugfs_create_file for output_mode failed!\n");
      
      	if (!debugfs_file)
      		LED_ERR("debugfs_create_file for trans_mode failed!\n");
      
      	debugfs_file = debugfs_create_file("hwversion", 0440,
      				debugfs_dir, NULL, &hwversion_fops);
      	if (!debugfs_file)
      		LED_ERR("debugfs_create_file for hwversion failed!\n");
      }
      
      static void sunxi_led_remove_debugfs(struct sunxi_led *led)
      {
      	debugfs_remove_recursive(led->debugfs_dir);
      }
      #endif /* CONFIG_DEBUG_FS */
      
      static void sunxi_ledc_set_dma_mode(struct sunxi_led *led)
      {
      	u32 reg_val = 0;
      
      	reg_val |= 1 << 5;
      	sunxi_set_reg(LEDC_DMA_CTRL_REG, reg_val);
      
      	sunxi_ledc_disable_irq(LEDC_FIFO_CPUREQ_INT_EN);
      }
      
      static void sunxi_ledc_set_cpu_mode(struct sunxi_led *led)
      {
      	u32 reg_val = 0;
      
      	reg_val &= ~(1 << 5);
      	sunxi_set_reg(LEDC_DMA_CTRL_REG, reg_val);
      
      	sunxi_ledc_enable_irq(LEDC_FIFO_CPUREQ_INT_EN);
      }
      
      static void sunxi_ledc_dma_callback(void *param)
      {
      	dprintk(DEBUG_INFO, "finish\n");
      }
      
      static void sunxi_ledc_trans_data(struct sunxi_led *led)
      {
      	int i, err;
      	size_t size;
      	unsigned long flags;
      	phys_addr_t dst_addr;
      	struct dma_slave_config slave_config;
      	struct device *dev = led->dev;
      	struct dma_async_tx_descriptor *dma_desc;
      
      	/* less than 32 lights use cpu transmission. */
      	/* more than 32 lights use dma transmission. */
      	if (led->length <= SUNXI_LEDC_FIFO_DEPTH) {
      		dprintk(DEBUG_INFO, "cpu xfer\n");
      		ktime_get_coarse_real_ts64(&(led->start_time));
      		sunxi_ledc_set_time(led);
      		sunxi_ledc_set_output_mode(led, led->output_mode.str);
      		sunxi_ledc_set_cpu_mode(led);
      		sunxi_ledc_set_length(led);
      
      		sunxi_ledc_enable_irq(LEDC_TRANS_FINISH_INT_EN | LEDC_WAITDATA_TIMEOUT_INT_EN
      				| LEDC_FIFO_OVERFLOW_INT_EN | LEDC_GLOBAL_INT_EN);
      
      		sunxi_ledc_enable(led);
      
      		for (i = 0; i < led->length; i++)
      			sunxi_set_reg(LEDC_DATA_REG_OFFSET, led->data[i]);
      
      	} else {
      		dprintk(DEBUG_INFO, "dma xfer\n");
      
      		size = led->length * 4;
      		led->src_dma = dma_map_single(dev, led->data,
      					size, DMA_TO_DEVICE);
      		dst_addr = led->res->start + LEDC_DATA_REG_OFFSET;
      
      		flags = DMA_PREP_INTERRUPT | DMA_CTRL_ACK;
      
      		slave_config.direction = DMA_MEM_TO_DEV;
      		slave_config.src_addr = led->src_dma;
      		slave_config.dst_addr = dst_addr;
      		slave_config.src_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
      		slave_config.dst_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
      		slave_config.src_maxburst = 4;
      		slave_config.dst_maxburst = 4;
      
      		err = dmaengine_slave_config(led->dma_chan, &slave_config);
      		if (err < 0) {
      			LED_ERR("dmaengine_slave_config failed!\n");
      			return;
      		}
      
      		dma_desc = dmaengine_prep_slave_single(led->dma_chan,
      							led->src_dma,
      							size,
      							DMA_MEM_TO_DEV,
      							flags);
      		if (!dma_desc) {
      			LED_ERR("dmaengine_prep_slave_single failed!\n");
      			return;
      		}
      
      		dma_desc->callback = sunxi_ledc_dma_callback;
      
      		dmaengine_submit(dma_desc);
      		dma_async_issue_pending(led->dma_chan);
      
      		ktime_get_coarse_real_ts64(&(led->start_time));
      		sunxi_ledc_set_time(led);
      		sunxi_ledc_set_output_mode(led, led->output_mode.str);
      		sunxi_ledc_set_dma_mode(led);
      		sunxi_ledc_set_length(led);
      		sunxi_ledc_enable_irq(LEDC_TRANS_FINISH_INT_EN | LEDC_WAITDATA_TIMEOUT_INT_EN
      				| LEDC_FIFO_OVERFLOW_INT_EN | LEDC_GLOBAL_INT_EN);
      		sunxi_ledc_enable(led);
      	}
      }
      
      static inline void sunxi_ledc_clear_all_irq(void)
      {
      	u32 reg_val = sunxi_get_reg(LEDC_INT_STS_REG_OFFSET);
      
      	reg_val &= ~0x1F;
      	sunxi_set_reg(LEDC_INT_STS_REG_OFFSET, reg_val);
      }
      
      static inline void sunxi_ledc_clear_irq(enum sunxi_ledc_irq_status_reg irq)
      {
      	u32 reg_val = sunxi_get_reg(LEDC_INT_STS_REG_OFFSET);
      
      	reg_val &= ~irq;
      	sunxi_set_reg(LEDC_INT_STS_REG_OFFSET, reg_val);
      }
      
      static int sunxi_ledc_complete(struct sunxi_led *led)
      {
      	unsigned long flags = 0;
      	unsigned long timeout = 0;
      	u32 reg_val;
      
      	/*wait_event_timeout return 0   : timeout
      	 *wait_event_timeout return > 0 : thr left time
      	 * */
      	timeout = wait_event_timeout(led->wait, led->result, 5*HZ);
      	if (timeout == 0) {
      		reg_val = sunxi_get_reg(LEDC_INT_STS_REG_OFFSET);
      		pr_err("LEDC INTERRUPT STATUS REG IS %x", reg_val);
      		LED_ERR("led xfer timeout\n");
      		reg_val = sunxi_get_reg(LEDC_INT_STS_REG_OFFSET);
      		pr_err("LEDC INTERRUPT STATUS REG IS %x", reg_val);
      		return -ETIME;
      	} else if (led->result == RESULT_ERR) {
      		return -ECOMM;
      	}
      
      	dprintk(DEBUG_INFO, "xfer complete\n");
      
      	spin_lock_irqsave(&led->lock, flags);
      	led->result = 0;
      	spin_unlock_irqrestore(&led->lock, flags);
      
      	return 0;
      }
      
      static irqreturn_t sunxi_ledc_irq_handler(int irq, void *dev_id)
      {
      	long delta_time_ns;
      	u32 irq_status, max_ns;
      	struct sunxi_led *led = sunxi_led_global;
      	struct device *dev = led->dev;
      	struct timespec64 current_time;
      
      	spin_lock(&led->lock);
      
      	irq_status = sunxi_get_reg(LEDC_INT_STS_REG_OFFSET);
      
      	sunxi_ledc_clear_all_irq();
      
      	if (irq_status & LEDC_TRANS_FINISH_INT) {
      		sunxi_ledc_reset(led);
      		led->result = RESULT_COMPLETE;
      		goto out;
      	}
      
      	if (irq_status & LEDC_WAITDATA_TIMEOUT_INT) {
      		ktime_get_coarse_real_ts64(&current_time);
      		delta_time_ns = current_time.tv_sec - led->start_time.tv_sec;
      		delta_time_ns *= 1000 * 1000 * 1000;
      		delta_time_ns += current_time.tv_nsec - led->start_time.tv_nsec;
      
      		max_ns = led->wait_data_time_ns;
      
      		if (delta_time_ns <= max_ns) {
      			spin_unlock(&led->lock);
      			return IRQ_HANDLED;
      		}
      
      		sunxi_ledc_reset(led);
      
      		if (delta_time_ns <= max_ns * 2) {
      			sunxi_ledc_trans_data(led);
      		} else {
      			LED_ERR("wait time is more than %d ns,"
      				"going to reset ledc and drop this operation!\n",
      				max_ns);
      			led->result = RESULT_ERR;
      		}
      
      		goto out;
      	}
      
      	if (irq_status & LEDC_FIFO_OVERFLOW_INT) {
      		LED_ERR("there exists fifo overflow issue, irq_status=0x%x!\n",
      				irq_status);
      		sunxi_ledc_reset(led);
      		led->result = RESULT_ERR;
      		goto out;
      	}
      
      out:
      	if (led->dma_chan)
      		dma_unmap_single(dev, led->src_dma, led->length * 4, DMA_TO_DEVICE);
      	wake_up(&led->wait);
      	led->length = 0;
      	spin_unlock(&led->lock);
      	return IRQ_HANDLED;
      }
      
      static int sunxi_ledc_irq_init(struct sunxi_led *led)
      {
      	int err;
      	struct device *dev = led->dev;
      	unsigned long flags = 0;
      	const char *name = "ledcirq";
      	struct platform_device *pdev;
      
      	pdev = container_of(dev, struct platform_device, dev);
      
      	spin_lock_init(&led->lock);
      
      	led->irqnum = platform_get_irq(pdev, 0);
      	if (led->irqnum < 0)
      		LED_ERR("failed to get ledc irq!\n");
      
      	err = request_irq(led->irqnum, sunxi_ledc_irq_handler,
      				flags, name, dev);
      	if (err) {
      		LED_ERR("failed to install IRQ handler for irqnum %d\n",
      			led->irqnum);
      		return -EPERM;
      	}
      
      	return 0;
      }
      
      static void sunxi_ledc_irq_deinit(struct sunxi_led *led)
      {
      	free_irq(led->irqnum, led->dev);
      	sunxi_ledc_disable_irq(LEDC_TRANS_FINISH_INT_EN | LEDC_FIFO_CPUREQ_INT_EN
      			| LEDC_WAITDATA_TIMEOUT_INT_EN | LEDC_FIFO_OVERFLOW_INT_EN
      			| LEDC_GLOBAL_INT_EN);
      }
      
      static void sunxi_ledc_pinctrl_init(struct sunxi_led *led)
      {
      	struct device *dev = led->dev;
      	struct pinctrl *pinctrl = devm_pinctrl_get_select_default(dev);
      
      	led->pctrl = pinctrl;
      	if (IS_ERR(pinctrl))
      		LED_ERR("devm_pinctrl_get_select_default failed!\n");
      }
      
      static int led_regulator_request(struct sunxi_led *led)
      {
      	struct regulator *regu = NULL;
      
      	/* Consider "n*" as nocare. Support "none", "nocare", "null", "" etc. */
      	if ((led->regulator_id[0] == 'n') || (led->regulator_id[0] == 0))
      		return 0;
      
      	regu = regulator_get(NULL, led->regulator_id);
      	if (IS_ERR(regu)) {
      		LED_ERR("get regulator %s failed!\n", led->regulator_id);
      		return -1;
      	}
      	led->regulator = regu;
      
      	return 0;
      }
      
      static int led_regulator_release(struct sunxi_led *led)
      {
      	if (led->regulator == NULL)
      		return 0;
      
      	regulator_put(led->regulator);
      	led->regulator = NULL;
      
      	return 1;
      }
      
      static int sunxi_ledc_dma_get(struct sunxi_led *led)
      {
      	if (led->dma_chan == NULL) {
      		led->dma_chan = dma_request_chan(led->dev, "tx");
      		if (IS_ERR(led->dma_chan)) {
      			LED_ERR("failed to get the DMA channel!\n");
      			return -EFAULT;
      		}
      	}
      	return 0;
      }
      
      static int sunxi_set_led_brightness(struct led_classdev *led_cdev,
      			enum led_brightness value)
      {
      	unsigned long flags;
      	u32 r, g, b, shift, old_data, new_data, length;
      	struct sunxi_led_info *pinfo;
      	struct sunxi_led_classdev_group *pcdev_group;
      	struct sunxi_led *led = sunxi_led_global;
      	int err;
      
      	pinfo = container_of(led_cdev, struct sunxi_led_info, cdev);
      
      	switch (pinfo->type) {
      	case LED_TYPE_G:
      		pcdev_group = container_of(pinfo,
      			struct sunxi_led_classdev_group, g);
      		g = value;
      		shift = 16;
      		break;
      	case LED_TYPE_R:
      		pcdev_group = container_of(pinfo,
      			struct sunxi_led_classdev_group, r);
      		r = value;
      		shift = 8;
      		break;
      
      	case LED_TYPE_B:
      		pcdev_group = container_of(pinfo,
      			struct sunxi_led_classdev_group, b);
      		b = value;
      		shift = 0;
      		break;
      	}
      
      	old_data = led->data[pcdev_group->led_num];
      	if (((old_data >> shift) & 0xFF) == value)
      		return 0;
      
      	if (pinfo->type != LED_TYPE_R)
      		r = pcdev_group->r.cdev.brightness;
      	if (pinfo->type != LED_TYPE_G)
      		g = pcdev_group->g.cdev.brightness;
      	if (pinfo->type != LED_TYPE_B)
      		b = pcdev_group->b.cdev.brightness;
      
      	/* LEDC treats input data as GRB by default */
      	new_data = (g << 16) | (r << 8) | b;
      	length = pcdev_group->led_num + 1;
      
      	spin_lock_irqsave(&led->lock, flags);
      	led->data[pcdev_group->led_num] = new_data;
      	led->length = length;
      	spin_unlock_irqrestore(&led->lock, flags);
      
      	/* prepare for dma xfer, dynamic apply dma channel */
      	if (led->length > SUNXI_LEDC_FIFO_DEPTH) {
      		err = sunxi_ledc_dma_get(led);
      		if (err)
      			return err;
      	}
      
      	sunxi_ledc_trans_data(led);
      	if (debug_mask & DEBUG_INFO2) {
      		dprintk(DEBUG_INFO2, "dump reg:\n");
      		led_dump_reg(led, 0, 0x30);
      	}
      
      	sunxi_ledc_complete(led);
      
      	if (debug_mask & DEBUG_INFO1)
      		pr_warn("num = %03u\n", length);
      
      	return 0;
      }
      
      static int sunxi_register_led_classdev(struct sunxi_led *led)
      {
      	int i, err;
      	size_t size;
      	struct device *dev = led->dev;
      	struct led_classdev *pcdev_RGB;
      
      	dprintk(DEBUG_INIT, "led_classdev start\n");
      	if (!led->led_count)
      		led->led_count = SUNXI_DEFAULT_LED_COUNT;
      
      	size = sizeof(struct sunxi_led_classdev_group) * led->led_count;
      	led->pcdev_group = devm_kzalloc(dev, size, GFP_KERNEL);
      	if (!led->pcdev_group)
      		return -ENOMEM;
      
      	for (i = 0; i < led->led_count; i++) {
      		led->pcdev_group[i].r.type = LED_TYPE_R;
      		pcdev_RGB = &led->pcdev_group[i].r.cdev;
      		pcdev_RGB->name = devm_kzalloc(dev, 16, GFP_KERNEL);
      		if (!pcdev_RGB->name)
      			return -ENOMEM;
      		sprintf((char *)pcdev_RGB->name, "sunxi_led%dr", i);
      		pcdev_RGB->brightness = LED_OFF;
      		pcdev_RGB->brightness_set_blocking = sunxi_set_led_brightness;
      		pcdev_RGB->dev = dev;
      		err = led_classdev_register(dev, pcdev_RGB);
      		if (err < 0) {
      			LED_ERR("led_classdev_register %s failed!\n",
      				pcdev_RGB->name);
      			return err;
      		}
      
      		led->pcdev_group[i].g.type = LED_TYPE_G;
      		pcdev_RGB = &led->pcdev_group[i].g.cdev;
      		pcdev_RGB->name = devm_kzalloc(dev, 16, GFP_KERNEL);
      		if (!pcdev_RGB->name)
      			return -ENOMEM;
      		sprintf((char *)pcdev_RGB->name, "sunxi_led%dg", i);
      		pcdev_RGB->brightness = LED_OFF;
      		pcdev_RGB->brightness_set_blocking = sunxi_set_led_brightness;
      		pcdev_RGB->dev = dev;
      		err = led_classdev_register(dev, pcdev_RGB);
      		if (err < 0) {
      			LED_ERR("led_classdev_register %s failed!\n",
      			pcdev_RGB->name);
      			return err;
      		}
      
      		led->pcdev_group[i].b.type = LED_TYPE_B;
      		pcdev_RGB = &led->pcdev_group[i].b.cdev;
      		pcdev_RGB->name = devm_kzalloc(dev, 16, GFP_KERNEL);
      		if (!pcdev_RGB->name)
      			return -ENOMEM;
      		sprintf((char *)pcdev_RGB->name, "sunxi_led%db", i);
      		pcdev_RGB->brightness = LED_OFF;
      		pcdev_RGB->brightness_set_blocking = sunxi_set_led_brightness;
      		pcdev_RGB->dev = dev;
      		err = led_classdev_register(dev, pcdev_RGB);
      		if (err < 0) {
      			LED_ERR("led_classdev_register %s failed!\n",
      					pcdev_RGB->name);
      			return err;
      		}
      
      		led->pcdev_group[i].led_num = i;
      	}
      
      	size = sizeof(u32) * led->led_count;
      	led->data = devm_kzalloc(dev, size, GFP_KERNEL);
      	if (!led->data)
      		return -ENOMEM;
      
      	return 0;
      }
      
      static void sunxi_unregister_led_classdev(struct sunxi_led *led)
      {
      	int i;
      
      	for (i = 0; i < led->led_count; i++) {
      		kfree(led->pcdev_group[i].b.cdev.name);
      		led->pcdev_group[i].b.cdev.name = NULL;
      		kfree(led->pcdev_group[i].g.cdev.name);
      		led->pcdev_group[i].g.cdev.name = NULL;
      		kfree(led->pcdev_group[i].r.cdev.name);
      		led->pcdev_group[i].r.cdev.name = NULL;
      		led_classdev_unregister(&led->pcdev_group[i].b.cdev);
      		led_classdev_unregister(&led->pcdev_group[i].g.cdev);
      		led_classdev_unregister(&led->pcdev_group[i].r.cdev);
      	}
      	kfree(led->data);
      	led->data = NULL;
      
      
      	kfree(led->pcdev_group);
      	led->pcdev_group = NULL;
      }
      
      static inline int sunxi_get_u32_of_property(const char *propname, int *val)
      {
      	int err;
      	struct sunxi_led *led = sunxi_led_global;
      	struct device *dev = led->dev;
      	struct device_node *np = dev->of_node;
      
      	err = of_property_read_u32(np, propname, val);
      	if (err < 0)
      		LED_ERR("failed to get the value of propname %s!\n", propname);
      
      	return err;
      }
      
      static inline int sunxi_get_str_of_property(const char *propname,
      					const char **out_string)
      {
      	int err;
      	struct sunxi_led *led = sunxi_led_global;
      	struct device *dev = led->dev;
      	struct device_node *np = dev->of_node;
      
      	err = of_property_read_string(np, propname, out_string);
      	if (err < 0)
      		LED_ERR("failed to get the string of propname %s!\n", propname);
      
      	return err;
      }
      
      static void sunxi_get_para_of_property(struct sunxi_led *led)
      {
      	int err;
      	u32 val;
      	const char *str;
      
      	err = sunxi_get_u32_of_property("led_count", &val);
      	if (!err)
      		led->led_count = val;
      
      	memcpy(led->output_mode.str, "GRB", 3);
      	led->output_mode.val = SUNXI_OUTPUT_GRB;
      	err = sunxi_get_str_of_property("output_mode", &str);
      	if (!err)
      		if (!strncmp(str, "BRG", 3) ||
      			!strncmp(str, "GBR", 3) ||
      			!strncmp(str, "RGB", 3) ||
      			!strncmp(str, "RBG", 3) ||
      			!strncmp(str, "BGR", 3))
      			memcpy(led->output_mode.str, str, 3);
      
      	err =  sunxi_get_str_of_property("led_regulator", &str);
      	if (!err) {
      		if (strlen(str) >= sizeof(led->regulator_id))
      			LED_ERR("illegal regulator id\n");
      		else {
      			strcpy(led->regulator_id, str);
      			pr_info("led_regulator: %s\n", led->regulator_id);
      		}
      	}
      
      	err = sunxi_get_u32_of_property("reset_ns", &val);
      	if (!err)
      		led->reset_ns = val;
      
      	err = sunxi_get_u32_of_property("t1h_ns", &val);
      	if (!err)
      		led->t1h_ns = val;
      
      	err = sunxi_get_u32_of_property("t1l_ns", &val);
      	if (!err)
      		led->t1l_ns = val;
      
      	err = sunxi_get_u32_of_property("t0h_ns", &val);
      	if (!err)
      		led->t0h_ns = val;
      
      	err = sunxi_get_u32_of_property("t0l_ns", &val);
      	if (!err)
      		led->t0l_ns = val;
      
      	err = sunxi_get_u32_of_property("wait_time0_ns", &val);
      	if (!err)
      		led->wait_time0_ns = val;
      
      	err = sunxi_get_u32_of_property("wait_time1_ns", &val);
      	if (!err)
      		led->wait_time1_ns = val;
      
      	err = sunxi_get_u32_of_property("wait_data_time_ns", &val);
      	if (!err)
      		led->wait_data_time_ns = val;
      }
      static void sunxi_led_set_all(struct sunxi_led *led, u8 channel,
      		enum led_brightness value)
      {
      	u32 i;
      	struct led_classdev *led_cdev;
      
      	if (channel%3 == 0) {
      		for (i = 0; i < led->led_count; i++) {
      			led_cdev = &led->pcdev_group[i].r.cdev;
      			mutex_lock(&led_cdev->led_access);
      			sunxi_set_led_brightness(led_cdev, value);
      			mutex_unlock(&led_cdev->led_access);
      		}
      	} else if (channel%3 == 1) {
      		for (i = 0; i < led->led_count; i++) {
      			led_cdev = &led->pcdev_group[i].g.cdev;
      			mutex_lock(&led_cdev->led_access);
      			sunxi_set_led_brightness(led_cdev, value);
      			mutex_unlock(&led_cdev->led_access);
      		}
      	} else {
      		for (i = 0; i < led->led_count; i++) {
      			led_cdev = &led->pcdev_group[i].b.cdev;
      			mutex_lock(&led_cdev->led_access);
      			sunxi_set_led_brightness(led_cdev, value);
      			mutex_unlock(&led_cdev->led_access);
      		}
      	}
      }
      
      static ssize_t led_show(struct class *class,
      			struct class_attribute *attr,
      			char *buf)
      {
      	struct sunxi_led *led = sunxi_led_global;
      
      	sunxi_led_set_all(led, 0, 0);
      	sunxi_led_set_all(led, 1, 0);
      	sunxi_led_set_all(led, 2, 0);
      
      	sunxi_led_set_all(led, 0, 20);
      	msleep(500);
      	sunxi_led_set_all(led, 1, 20);
      	msleep(500);
      	sunxi_led_set_all(led, 2, 20);
      	msleep(500);
      
      	sunxi_led_set_all(led, 0, 0);
      	sunxi_led_set_all(led, 1, 0);
      	sunxi_led_set_all(led, 2, 0);
      
      	return 0;
      }
      
      static struct class_attribute led_class_attrs[] = {
      	__ATTR(light, 0644, led_show, NULL),
      	//__ATTR_NULL,
      };
      
      static void led_node_init(void)
      {
      	int i;
      	int err;
      	/* sys/class/led/xxx */
      	for (i = 0; i < ARRAY_SIZE(led_class_attrs); i++) {
      		err = class_create_file(led_class, &led_class_attrs[i]);
      		if (err) {
      			LED_ERR("class_create_file() failed!\n");
      			while (i--)
      				class_remove_file(led_class, &led_class_attrs[i]);
      			class_destroy(led_class);
      			led_class = NULL;
      		}
      	}
      }
      
      static int sunxi_led_probe(struct platform_device *pdev)
      {
      	int err;
      	struct sunxi_led *led;
      	struct device *dev = &pdev->dev;
      	struct resource *mem_res = NULL;
      	int ret;
      
      	dprintk(DEBUG_INIT, "start\n");
      
      	led = devm_kzalloc(dev, sizeof(struct sunxi_led), GFP_KERNEL);
      	if (!led)
      		return -ENOMEM;
      
      	sunxi_led_global = led;
      
      	platform_set_drvdata(pdev, led);
      	led->dev = dev;
      
      	mem_res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
      	if (mem_res == NULL) {
      		LED_ERR("failed to get MEM res\n");
      		ret = -ENXIO;
      		goto emem;
      	}
      
      	if (!request_mem_region(mem_res->start, resource_size(mem_res),
      				mem_res->name)) {
      		LED_ERR("failed to request mem region\n");
      		ret = -EINVAL;
      		goto emem;
      	}
      
      	led->iomem_reg_base = ioremap(mem_res->start, resource_size(mem_res));
      	if (!led->iomem_reg_base) {
      		ret = -EIO;
      		goto eiomap;
      	}
      	led->res = mem_res;
      
      	led->output_mode.str = devm_kzalloc(dev, 3, GFP_KERNEL);
      	if (!led->output_mode.str) {
      		ret = -ENOMEM;
      		goto ezalloc_str;
      	}
      
      	sunxi_get_para_of_property(led);
      
      	err = led_regulator_request(led);
      	if (err < 0) {
      		LED_ERR("request regulator failed!\n");
      		ret = err;
      		goto eregulator;
      	}
      
      	err = sunxi_register_led_classdev(led);
      	if (err) {
      		LED_ERR("failed to register led classdev\n");
      		ret = err;
      		goto eclassdev;
      	}
      
      	sunxi_ledc_set_time(led);
      
      	led->reset = devm_reset_control_get(&pdev->dev, NULL);
      	if (IS_ERR(led->reset)) {
      		LED_ERR("get reset clk error\n");
      		return -EINVAL;
      	}
      	ret = reset_control_deassert(led->reset);
      	if (ret) {
      		LED_ERR("deassert clk error, ret:%d\n", ret);
      		return ret;
      	}
      
      	sunxi_clk_init(led);
      
      	init_waitqueue_head(&led->wait);
      
      	err = sunxi_ledc_irq_init(led);
      	if (err) {
      		LED_ERR("failed to init irq\n");
      		ret = err;
      		goto eirq;
      	}
      
      	sunxi_ledc_pinctrl_init(led);
      
      #ifdef CONFIG_DEBUG_FS
      	sunxi_led_create_debugfs(led);
      #endif /* CONFIG_DEBUG_FS */
      
      	led_class = class_create(THIS_MODULE, "led");
      	if (IS_ERR(led_class)) {
      		LED_ERR("class_register err\n");
      		class_destroy(led_class);
      		ret = -EFAULT;
      		goto eclass;
      	}
      	led_node_init();
      	dprintk(DEBUG_INIT, "finish\n");
      	return 0;
      
      eclass:
      #ifdef CONFIG_DEBUG_FS
      	sunxi_led_remove_debugfs(led);
      #endif /* CONFIG_DEBUG_FS */
      
      	sunxi_ledc_irq_deinit(led);
      
      eirq:
      	sunxi_unregister_led_classdev(led);
      	sunxi_clk_deinit(led);
      
      eclassdev:
      	led_regulator_release(led);
      
      eregulator:
      	kfree(led->output_mode.str);
      
      ezalloc_str:
      	iounmap(led->iomem_reg_base);
      	led->iomem_reg_base = NULL;
      
      eiomap:
      	release_mem_region(mem_res->start, resource_size(mem_res));
      
      emem:
      	kfree(led);
      	return ret;
      }
      
      static int sunxi_led_remove(struct platform_device *pdev)
      {
      	struct sunxi_led *led = platform_get_drvdata(pdev);
      
      	class_destroy(led_class);
      
      #ifdef CONFIG_DEBUG_FS
      	sunxi_led_remove_debugfs(led);
      #endif /* CONFIG_DEBUG_FS */
      
      	if (led->dma_chan) {
      		dmaengine_terminate_all(led->dma_chan);
      		dma_release_channel(led->dma_chan);
      		led->dma_chan = NULL;
      	}
      
      	sunxi_ledc_irq_deinit(led);
      
      	sunxi_unregister_led_classdev(led);
      	sunxi_clk_deinit(led);
      
      	led_regulator_release(led);
      
      	kfree(led->output_mode.str);
      	led->output_mode.str = NULL;
      
      	iounmap(led->iomem_reg_base);
      	led->iomem_reg_base = NULL;
      
      	release_mem_region(led->res->start, resource_size(led->res));
      
      	kfree(led);
      	led = NULL;
      
      	dprintk(DEBUG_INIT, "finish\n");
      	return 0;
      }
      
      #if IS_ENABLED(CONFIG_PM)
      static inline void sunxi_led_save_regs(struct sunxi_led *led)
      {
      	int i;
      
      	for (i = 0; i < ARRAY_SIZE(sunxi_led_regs_offset); i++)
      		led->regs_backup[i] = readl(led->iomem_reg_base + sunxi_led_regs_offset[i]);
      }
      
      static inline void sunxi_led_restore_regs(struct sunxi_led *led)
      {
      	int i;
      
      	for (i = 0; i < ARRAY_SIZE(sunxi_led_regs_offset); i++)
      		writel(led->regs_backup[i], led->iomem_reg_base + sunxi_led_regs_offset[i]);
      }
      
      static void sunxi_led_enable_irq(struct sunxi_led *led)
      {
      	enable_irq(led->irqnum);
      }
      
      static void sunxi_led_disable_irq(struct sunxi_led *led)
      {
      	disable_irq_nosync(led->irqnum);
      }
      
      static int sunxi_led_gpio_state_select(struct sunxi_led *led, char *name)
      {
      	int err;
      	struct pinctrl_state *pctrl_state;
      
      	pctrl_state = pinctrl_lookup_state(led->pctrl, name);
      	if (IS_ERR(pctrl_state)) {
      		dev_err(led->dev, "pinctrl_lookup_state(%s) failed! return %p\n",
      				name, pctrl_state);
      		return PTR_ERR(pctrl_state);
      	}
      
      	err = pinctrl_select_state(led->pctrl, pctrl_state);
      	if (err < 0) {
      		dev_err(led->dev, "pinctrl_select_state(%s) failed! return %d\n",
      				name, err);
      		return err;
      	}
      
      	return 0;
      }
      
      static void sunxi_led_enable_clk(struct sunxi_led *led)
      {
      	clk_prepare_enable(led->clk_ledc);
      	clk_prepare_enable(led->clk_cpuapb);
      }
      
      static void sunxi_led_disable_clk(struct sunxi_led *led)
      {
      	clk_disable_unprepare(led->clk_cpuapb);
      	clk_disable_unprepare(led->clk_ledc);
      }
      
      static int sunxi_led_power_on(struct sunxi_led *led)
      {
      	int err;
      
      	if (led->regulator == NULL)
      		return 0;
      
      	err = regulator_enable(led->regulator);
      	if (err) {
      		dev_err(led->dev, "enable regulator %s failed!\n", led->regulator_id);
      		return err;
      	}
      	return 0;
      }
      
      static int sunxi_led_power_off(struct sunxi_led *led)
      {
      	int err;
      
      	if (led->regulator == NULL)
      		return 0;
      
      	err = regulator_disable(led->regulator);
      	if (err) {
      		dev_err(led->dev, "disable regulator %s failed!\n", led->regulator_id);
      		return err;
      	}
      	return 0;
      }
      
      static int sunxi_led_suspend(struct device *dev)
      {
      	struct platform_device *pdev = to_platform_device(dev);
      	struct sunxi_led *led = platform_get_drvdata(pdev);
      
      	dev_dbg(led->dev, "[%s] enter standby\n", __func__);
      
      	sunxi_led_disable_irq(led);
      
      	sunxi_led_save_regs(led);
      
      	sunxi_led_gpio_state_select(led, PINCTRL_STATE_SLEEP);
      
      	sunxi_led_disable_clk(led);
      
      	reset_control_assert(led->reset);
      
      	sunxi_led_power_off(led);
      
      	return 0;
      }
      
      static int sunxi_led_resume(struct device *dev)
      {
      	struct platform_device *pdev = to_platform_device(dev);
      	struct sunxi_led *led = platform_get_drvdata(pdev);
      
      	dev_dbg(led->dev, "[%s] return from standby\n", __func__);
      
      	sunxi_led_power_on(led);
      
      	reset_control_deassert(led->reset);
      
      	sunxi_led_enable_clk(led);
      
      	sunxi_led_gpio_state_select(led, PINCTRL_STATE_DEFAULT);
      
      	sunxi_led_restore_regs(led);
      
      	sunxi_led_enable_irq(led);
      
      	return 0;
      }
      
      static const struct dev_pm_ops sunxi_led_pm_ops = {
      	.suspend = sunxi_led_suspend,
      	.resume = sunxi_led_resume,
      };
      
      #define SUNXI_LED_PM_OPS (&sunxi_led_pm_ops)
      #endif
      
      static const struct of_device_id sunxi_led_dt_ids[] = {
      	{.compatible = "allwinner,sunxi-leds"},
      	{},
      };
      
      static struct platform_driver sunxi_led_driver = {
      	.probe		= sunxi_led_probe,
      	.remove		= sunxi_led_remove,
      	.driver		= {
      		.name	= "sunxi-leds",
      		.owner	= THIS_MODULE,
      #if IS_ENABLED(CONFIG_PM)
      		.pm	= SUNXI_LED_PM_OPS,
      #endif
      		.of_match_table = sunxi_led_dt_ids,
      	},
      };
      
      module_platform_driver(sunxi_led_driver);
      module_param_named(debug, debug_mask, int, 0664);
      
      MODULE_ALIAS("sunxi leds dirver");
      MODULE_ALIAS("platform : leds dirver");
      MODULE_LICENSE("GPL v2");
      MODULE_VERSION("1.2.3");
      MODULE_AUTHOR("Albert Yu <yuxyun@allwinnertech.com>");
      MODULE_AUTHOR("liuyu <SWCliuyus@allwinnertech.com>");
      MODULE_DESCRIPTION("Allwinner ledc-controller driver");
      

      lichee/linux-5.4/drivers/leds/leds-sunxi.h

      /*
       * Copyright (C) 2018 Allwinner Technology Limited. All rights reserved.
       * Albert Yu <yuxyun@allwinnertech.com>
       *
       * This program is free software; you can redistribute it and/or modify
       * it under the terms of the GNU General Public License version 2 as
       * published by the Free Software Foundation.
       *
       */
      #ifndef __LINUX_LEDS_SUNXI_H
      #define __LINUX_LEDS_SUNXI_H
      
      #include <linux/device.h>
      #include <linux/list.h>
      #include <linux/mutex.h>
      #include <linux/rwsem.h>
      #include <linux/spinlock.h>
      #include <linux/timer.h>
      #include <linux/workqueue.h>
      
      #define HEXADECIMAL	(0x10)
      #define REG_INTERVAL	(0x04)
      #define REG_CL		(0x0c)
      
      #define RESULT_COMPLETE	1
      #define RESULT_ERR	2
      
      #define SUNXI_LEDC_REG_BASE_ADDR 0x06700000
      
      #define SUNXI_MAX_LED_COUNT 1024
      
      #define SUNXI_DEFAULT_LED_COUNT 8
      
      #define SUNXI_RESET_TIME_MIN_NS 84
      #define SUNXI_RESET_TIME_MAX_NS 327000
      
      #define SUNXI_T1H_MIN_NS 84
      #define SUNXI_T1H_MAX_NS 2560
      
      #define SUNXI_T1L_MIN_NS 84
      #define SUNXI_T1L_MAX_NS 1280
      
      #define SUNXI_T0H_MIN_NS 84
      #define SUNXI_T0H_MAX_NS 1280
      
      #define SUNXI_T0L_MIN_NS 84
      #define SUNXI_T0L_MAX_NS 2560
      
      #define SUNXI_WAIT_TIME0_MIN_NS 84
      #define SUNXI_WAIT_TIME0_MAX_NS 10000
      
      #define SUNXI_WAIT_TIME1_MIN_NS 84
      #define SUNXI_WAIT_TIME1_MAX_NS 85000000000
      
      #define SUNXI_WAIT_DATA_TIME_MIN_NS 84
      #define SUNXI_WAIT_DATA_TIME_MAX_NS_IC 655000
      #define SUNXI_WAIT_DATA_TIME_MAX_NS_FPGA 20000000
      
      #define SUNXI_LEDC_FIFO_DEPTH 32 /* 32 * 4 bytes */
      #define SUNXI_LEDC_FIFO_TRIG_LEVEL 15
      
      #if defined(CONFIG_FPGA_V4_PLATFORM) || defined(CONFIG_FPGA_V7_PLATFORM)
      #define SUNXI_FPGA_LEDC
      #endif
      
      enum sunxi_ledc_output_mode_val {
      	SUNXI_OUTPUT_GRB = 0 << 6,
      	SUNXI_OUTPUT_GBR = 1 << 6,
      	SUNXI_OUTPUT_RGB = 2 << 6,
      	SUNXI_OUTPUT_RBG = 3 << 6,
      	SUNXI_OUTPUT_BGR = 4 << 6,
      	SUNXI_OUTPUT_BRG = 5 << 6
      };
      
      struct sunxi_ledc_output_mode {
      	char *str;
      	enum sunxi_ledc_output_mode_val val;
      };
      
      enum sunxi_ledc_trans_mode_val {
      	LEDC_TRANS_CPU_MODE,
      	LEDC_TRANS_DMA_MODE
      };
      
      enum sunxi_ledc_reg {
      	LEDC_CTRL_REG_OFFSET              = 0x00,
      	LED_T01_TIMING_CTRL_REG_OFFSET    = 0x04,
      	LEDC_DATA_FINISH_CNT_REG_OFFSET   = 0x08,
      	LED_RESET_TIMING_CTRL_REG_OFFSET  = 0x0c,
      	LEDC_WAIT_TIME0_CTRL_REG          = 0x10,
      	LEDC_DATA_REG_OFFSET              = 0x14,
      	LEDC_DMA_CTRL_REG                 = 0x18,
      	LEDC_INT_CTRL_REG_OFFSET          = 0x1c,
      	LEDC_INT_STS_REG_OFFSET           = 0x20,
      	LEDC_WAIT_TIME1_CTRL_REG          = 0x28,
      	LEDC_VER_NUM_REG                  = 0x2c,
      	LEDC_FIFO_DATA                    = 0x30,
      	LEDC_TOTAL_REG_SIZE = LEDC_FIFO_DATA + SUNXI_LEDC_FIFO_DEPTH
      };
      
      enum sunxi_ledc_irq_ctrl_reg {
      	LEDC_TRANS_FINISH_INT_EN     = (1 << 0),
      	LEDC_FIFO_CPUREQ_INT_EN      = (1 << 1),
      	LEDC_WAITDATA_TIMEOUT_INT_EN = (1 << 3),
      	LEDC_FIFO_OVERFLOW_INT_EN    = (1 << 4),
      	LEDC_GLOBAL_INT_EN           = (1 << 5),
      };
      
      enum sunxi_ledc_irq_status_reg {
      	LEDC_TRANS_FINISH_INT     = (1 << 0),
      	LEDC_FIFO_CPUREQ_INT      = (1 << 1),
      	LEDC_WAITDATA_TIMEOUT_INT = (1 << 3),
      	LEDC_FIFO_OVERFLOW_INT    = (1 << 4),
      	LEDC_FIFO_FULL            = (1 << 16),
      	LEDC_FIFO_EMPTY           = (1 << 17),
      };
      
      enum sunxi_led_type {
      	LED_TYPE_R,
      	LED_TYPE_G,
      	LED_TYPE_B
      };
      
      struct sunxi_led_info {
      	enum sunxi_led_type type;
      	struct led_classdev cdev;
      };
      
      struct sunxi_led_classdev_group {
      	u32 led_num;
      	struct sunxi_led_info r;
      	struct sunxi_led_info g;
      	struct sunxi_led_info b;
      };
      
      static u32 sunxi_led_regs_offset[] = {
      	LEDC_CTRL_REG_OFFSET,
      	LED_RESET_TIMING_CTRL_REG_OFFSET,
      	LED_T01_TIMING_CTRL_REG_OFFSET,
      	LEDC_WAIT_TIME0_CTRL_REG,
      	LEDC_WAIT_TIME1_CTRL_REG,
      	LEDC_INT_CTRL_REG_OFFSET,
      #ifndef SUNXI_FPGA_LEDC
      	LEDC_DATA_FINISH_CNT_REG_OFFSET,
      #endif
      };
      
      struct sunxi_led {
      	u32 reset_ns;
      	u32 t1h_ns;
      	u32 t1l_ns;
      	u32 t0h_ns;
      	u32 t0l_ns;
      	u32 wait_time0_ns;
      	unsigned long long wait_time1_ns;
      	u32 wait_data_time_ns;
      	u32 irqnum;
      	u32 led_count;
      	u32 *data;
      	u32 length;
      	u8 result;
      	spinlock_t lock;
      	struct device *dev;
      	dma_addr_t src_dma;
      	struct dma_chan *dma_chan;
      	wait_queue_head_t wait;
      	struct timespec64 start_time;
      	struct clk *clk_ledc;
      	struct clk *clk_cpuapb;
      	struct pinctrl *pctrl;
      	void __iomem *iomem_reg_base;
      	struct resource	*res;
      	struct sunxi_ledc_output_mode output_mode;
      	struct sunxi_led_classdev_group *pcdev_group;
      	struct dentry *debugfs_dir;
      	char regulator_id[16];
      	struct regulator *regulator;
      	struct reset_control *reset;
      	u32 regs_backup[ARRAY_SIZE(sunxi_led_regs_offset)];
      };
      
      enum {
      	DEBUG_INIT    = 1U << 0,
      	DEBUG_SUSPEND = 1U << 1,
      	DEBUG_INFO    = 1U << 2,
      	DEBUG_INFO1   = 1U << 3,
      	DEBUG_INFO2   = 1U << 4,
      };
      
      #endif /* __LINUX_LEDS_SUNXI_H */
      

      kernel/linux-5.4/arch/riscv/boot/dts/sunxi/sun20iw1p1.dtsi

      		ledc: ledc@2008000 {
      			#address-cells = <1>;
      			#size-cells = <0>;
      			compatible = "allwinner,sunxi-leds";
      			reg = <0x0 0x02008000 0x0 0x400>;
      			interrupts-extended = <&plic0 36 IRQ_TYPE_LEVEL_HIGH>;
      			interrupt-names = "ledcirq";
      			clocks = <&ccu CLK_LEDC>, <&ccu CLK_BUS_LEDC>;
      			clock-names = "clk_ledc", "clk_cpuapb";
      			dmas = <&dma 42>, <&dma 42>;
      			dma-names = "rx", "tx";
      			resets = <&ccu RST_BUS_LEDC>;
      			reset-names = "ledc_reset";
      			status = "disable";
      		};
      
      发布在 MR Series
      A
      awwwwa
    • 回复: R128-S3不支持NAND FLASH吗?还有SPI0端口不能指定到PA24-P29吗

      @chen0505 PA引导是BROM不支持,BROM是固化在芯片里的,如果需要修改需要改版芯片

      Note Description
      D/S Datasheet Support but no sample test
      S/T Support and sample test
      √ Support and mass produce
      X Not support
      (Blank) Can support but not support now
      Part Number Capacity JEDEC ID PACKAGE R128 Support Status
      MT29F2G01ABAGDSF-AAT(NW794) 256M 0x2C, 0x24 SOP-16 D/S
      MT29F1G01ABAFDSF-AAT(NW808) 128M 0x2C, 0x14 SOP-16 D/S
      MT29F2G01ABAGDWB 256M 0x2C,0x24 SOP-16 D/S
      W25N01GVZEIG 128M 0xEF 0xAA 0x21 D/S
      W25N02KVZEIR 256M 0xEF 0xAA 0x22
      MX35LF2GE4AB 256M 0xC2 0x22 16-SOP
      MX35LF1GE4AB 128M 0xC2 0x12 8-WSON √
      MX35LF2GE4AD 256M 0xC2 0x26 0x03 8-WSON D/S
      MX35LF1G24AD 128M 0xC2 0x14
      MX35LF4GE4AD 512M
      GD5F1GQ4UBYIGR 128M 0xC8 0xD1 8-pad 8X6 WSON D/S
      GD5F2GQ4UB9IGR 256M 0xC8 0xD2 8-pad 8X6 WSON D/S
      GD5F1GQ4UCYIG 128M 0xC8 0xB1 0x48 WSON8(8x6mm) D/S
      GD5F1GQ5UEYIG 128M 0xC8 0x51 D/S
      GD5F2GQ5UEYIGR 256M 0xC8 0x52
      GD5F4GQ6UEYIG 512M 0xC8 0x55
      F50L1G41LB(2M) -> F50L1G41LB-104YG2M F50L1G41LB-104YG2ME 128M 0xC8 0x01 0x7F 0x7F 0x7F 8-WSON D/S
      F50L2G41XA 256M 0x2C, 0x24
      DS35Q1GA-1B 128M 0xE5 0x71 8-WSON D/S
      DS35Q2GA-IB 256M 0xE5 0x72
      DS35Q2GBXXX 256M 0xE5 0xf2
      DS35Q4GMXXX 512M 0xE5 0xf4
      DS35X1GBXXX 1G 0xE5 0xF1 8-WSON
      TC58CVG2S0H 512M 0x98 0xCD
      TC58CVG0S3HRAIG 128M 0x98 0xC2 8-WSON D/S
      FS35ND01G-S1F1QWFI 128M 0xCD 0xB1 WSON8(8x6mm) D/S
      FS35ND01G-S1Y2 128M 0xCD 0xEA WSON8
      F35SQA001G 128M 0xCD 0x71 WSON8 √
      F35SQA002G 256M 0xCD 0x72
      EM73C044VCD-H 128M 0xD5 0x1C 8-WSON
      EM73E044SNA 128M 0xD5 0x03 8-WSON
      ZD35Q1GA-IBR 128M 0xba 0x71 8-WSON D/S
      PN26G01AWSIUG 128M 0xa1, 0xe1 WSON8(8x6mm)
      XT26G01A 128M 0x0b,0xe1 D/S
      XT26G02A 256M 0x0b,0xe2
      XT26G04A 512M 0x0b,0xe3 WSON8(8x6mm)
      XT26G01CWSIG 128M 0x0b,0x11 D/S
      XT26G02CWSIG 256M 0x0b,0x12
      XT26G04CWSIG 512M 0x0b,0x13
      XT26G02ELGIGA 256M 0x2c, 0x24
      FM25S01 128M 0xa1,0xa1 WSON8(8x6mm)
      FM25S01A 128M 0xa1,0xe4
      FM25S02A 256M 0xa1,0xe5
      SCN00SA3W1AI8A 128M 0xC8 0x21 8-WSON
      S35ML01G3 128M 0x01 0x15
      S35ML02G300WH100 256M 0x01 0x25
      S35ML04G3 512M 0x01 0x35
      HYF1GQ4UDACAE 128M 0xC9 0x21
      发布在 MR Series
      A
      awwwwa
    • 回复: R128-S3不支持NAND FLASH吗?还有SPI0端口不能指定到PA24-P29吗

      支持NAND的,下图的5个模块分别是NOR和NAND,

      cde0e619-a94f-4a48-ba6e-8009ff8333b2-180736e57c1f7d393f970adf063b818d.jpg

      启动使用的SPI是在PB组,需要挂在PB上才能启动,并且需要专门配置的SDK,可以在SDK中看到有NAND的选项

      d32faa07-3fae-4eb9-bcd4-2098d418c658-image.png

      附上两种存储介质的启动流程:

      NOR:
      168b962d-16b3-40ae-a75c-f160d05c07fb-image.png

      NAND:
      5484a4a9-3e22-4724-8a8d-2053aaf536aa-image.png

      同时也支持 eMMC、SD-NAND启动

      发布在 MR Series
      A
      awwwwa
    • 回复: Tina启动阶段能放动画吗?

      @leomini5 VE是编解码器

      发布在 MR Series
      A
      awwwwa
    • 回复: 求教PhoenixSuit无设备连接问题

      参考这里,重新安装下驱动
      https://r128.docs.aw-ol.com/r128/prepare_dev_env/#windows_1

      发布在 编译和烧写问题专区
      A
      awwwwa
    • 回复: V853 SDK:Linux内核中的MIPI DSI

      从 log中可以看到,

      [    0.293211] disp 0, clk: pll(426000000),clk(426000000),dclk(71000000) dsi_rate(71000000)
      [    0.293211]      clk real:pll(420000000),clk(420000000),dclk(105000000) dsi_rate(150000000)
      

      内核引脚正确初始化并执行送图,Uboot运行而Kernel不执行函数是因为Kernel驱动当接收到Uboot传入的显示屏信息后不会重新初始化屏幕,这样可以保证整个开机logo显示流畅。

      这里log看到的驱动是ICN6202 TO LVDS,这是一颗转换MIPI DSI 转 LVDS的芯片,其MIPI初始化数据与普通 MIPI 屏幕不同,请检查下是否使用了正确的初始化。

      MIPI 与 RGB 不同,MIPI需要配置屏幕的参数,需要针对屏幕编写驱动

      发布在 V Series
      A
      awwwwa
    • 回复: d1s 原理图请求指导问题

      此处的电容不可以省略

      dd52baa4-1ff8-472f-96ae-a9a51efdcbfb-image.png

      AVCC,VRAx 需要外挂电容

      发布在 MR Series
      A
      awwwwa
    • 回复: MQ-Quad H616 主线内核编译调试记录(u-boot、kernel、buildroot)

      @king-zhao aw859a 的 WIFI 驱动可以在 OrangePi 官方仓库里找到

      发布在 H/F/TV Series
      A
      awwwwa
    • 回复: Tina启动阶段能放动画吗?

      可以播放动画,在启动阶段调用VE进行播放

      发布在 MR Series
      A
      awwwwa
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 2 / 6