# Rockchip 时钟配置详细说明

发布版本: 1.0

作者邮箱: zhangqing@rock-chips.com

日期: 2018.6

文件密级:公开资料

前言

概述

产品版本

| 芯片名称   | 内核版本 |
|--------|------|
| RK3399 | 4.4  |

读者对象

本文档(本指南)主要适用于以下工程师:

技术支持工程师

软件开发工程师

# 修订记录

| 日期         | 版本   | 作者     | 修改说明      |
|------------|------|--------|-----------|
| 2018-06-08 | V1.0 | Elaine | 第一次临时版本发布 |

# Rockchip 时钟配置详细说明

- 1时钟配置
  - 1.1 CRU时钟配置
    - 1.1.1 CRU时钟树
    - 1.1.2 配置一些时钟常开
    - 1.1.3 CLK ID获取
    - 1.1.4 PLL时钟配置
    - 1.1.5 CLK\_TIMER时钟配置
    - 1.1.6 总线时钟配置
    - 1.1.7 FCLK CMOS时钟配置
    - 1.1.8 CLK\_I2C时钟配置
    - 1.1.9 CLK\_SPI时钟配置
    - 1.1.10 CLK\_UART时钟配置
    - 1.1.11 CLK\_EMMC、CLK\_SDIO、CLK\_SDMMC时钟配置
    - 1.1.12 显示相关VOP、HDCP、EDP跟ISP时钟配置
    - 1.1.13 视频编解码VDU、RGA、CODEC、IEP相关时钟配置

- 1.1.14 USB相关时钟配置
- 1.1.15 CIF相关时钟配置
- 1.2 PMUCRU时钟配置
  - 1.2.1 PMUCRU时钟树
  - 1.2.2 配置一些时钟常开
  - 1.2.3 PCLK PMU总线时钟配置
  - 1.2.4 PMU\_M0时钟配置
  - 1.2.5 PMU总线时钟配置
  - 1.2.6 PMU I2C时钟配置
  - 1.2.7 PMU\_SPI时钟配置
  - 1.2.8 PMU\_WIFI时钟配置
  - 1.2.9 PMU UART4时钟配置
- 2 时钟间依赖关系
  - 2.1 普通的父子关系
  - 2.2 不同模块间NOC复用
  - 2.3 不同模块间GRF复用
- 3时钟频率值
  - 3.1 可设置的时钟频率

# 1时钟配置

# 1.1 CRU时钟配置

# 1.1.1 CRU时钟树

时钟树太长,不做说明,详细cat /sys/kernel/debug/clk/clk summary

# 1.1.2 配置一些时钟常开

对于调试过程中,想把某些时钟设置成常开的,可以修改 rk3399 cru critical clocks这个结构体,按照现有增加时钟名字即可:

```
drivers/clk/rockchip/clk-rk3399.c

static const char *const
rk3399_cru_critical_clocks[] __initconst = {
    "aclk_usb3_noc",
};
```

这个结构中的clk在系统开机,clk初始化的时候会默认调用clk set enable接口。

注意:如果时钟不是常开的,驱动设备也没有引用这个时钟并开启,在时钟初始 化完成之后,会调用clk\_disable\_unused\_subtree(drivers/clk/clk.c)关闭没有 用的时钟。如果没有用的时钟不想被关闭,可以在dts中的增加属性 clk\_ignore\_unused:

```
chosen {
  bootargs = "earlyprintk=uart8250-32bit,0xff690000
  clk_ignore_unused";
};
```

# 1.1.3 CLK ID获取

4.4的内核dts引用时钟,是根据clk id,不像3.10通过clk name索引。 CLK ID获取,详细见文档《Rockchip Clock 开发指南》中2.3.2章节。

# 1.1.4 PLL时钟配置

PLL锁相环详细介绍见文档《Rockchip Clock 开发指南》中1.3和2.2.1章节中。一般PLL不需要修改,尤其是下面挂了显示相关时钟的,PLL最好不要重现设置否则会有抖动问题。PLL的设置可以在UBOOT中,也可以直接在cru节点里面设置。

#### 1. dts中设置

但是只在节点初始化的时候调用一次。

```
cru: clock-controller@ff760000 {
            assigned-clocks =
3
                <&cru ARMCLKL>, <&cru ARMCLKB>,
                <&cru PLL_NPLL>, <&cru PLL_CPLL>,
                <&cru PLL_GPLL>;
            assigned-clock-rates =
6
7
                <816000000>, <816000000>,
                <600000000>, <500000000>,
8
9
                <800000000>;
10
        };
```

| ARMB<br>PLL | ARML<br>PLL | DDR<br>PLL | GPLL | CPLL | VPLL | NPLL | USBPHYPLL | PMUPLL |
|-------------|-------------|------------|------|------|------|------|-----------|--------|
| 1200        | 900         | 900        | 600  | 800  | 1200 | 1000 | 480       | 700    |

# 2. PLL计算公式

```
RK平台目前有两种类型PLL,一种是NR\NF\NO(RK3066、RK3188、
RK3288), 一种REF\POSTDIV1\POSTDIV2(RK3036、RK312X、RK322X、
RK332X、RK336X、RK3399)
                           1)NR、NF、NO类型,只有整数分频
    FREF = FIN / NR
                     FOUTVCO = FREF * NF
                                            FOUTPOSTDIV =
               FREF范围: 269MHZ - 2200MHZ, VCO 范围: 440MHz
FOUTVCO / NO
- 2200MHz, 输出频率范围: 27.5MHz - 2200MHz。
                                       NF_MAX: 4096,
NR MAX: 64, NO MAX: 16 (只能偶数)
2)REF\POSTDIV1\POSTDIV2类型,支持整数分频和小数分频
                                                  整数分
        If DSMPD = 1 (DSM is disabled, "integer mode")
                                                 FOUTVCO =
FREF / REFDIV * FBDIV
                   FOUTPOSTDIV = FOUTVCO / POSTDIV1 /
POSTDIV2
             小数分频:
                           If DSMPD = 0 (DSM is enabled, "fractional
mode")
          FOUTVCO = FREF / REFDIV * (FBDIV + FRAC / 2^24)
```

FOUTPOSTDIV = FOUTVCO / POSTDIV1 / POSTDIV2 VCO 范围: 440MHz - 3200MHz,输出频率范围: 27.5MHz - 2200MHz。REFDIV\_MAX: 63,FBDIV\_MAX: 4095,POSTDIV1\_MAX: 7,POSTDIV1\_MAX: 7。

备注: 一般PLL是不需要做修改的,默认配置就可以,但是显示的时钟会对 Jitter有要求,所以一般就是显示dclk独占的PLL有这种特殊的要求。可能会做一 些微调。

PLL设置原则: PLL一般要求频率尽量在600-1200M,这样VCO比较合适(jitter 会小)。目前的时钟频率基本都是支持自动计算的,但是自动计算只能保证VCO在上述范围内,不能保证VCO是最好的。所以如果对VCO有严格要求可以按照下面方式处理: PLL频率尽量设置大一些,然后通过后端分频。如: DCLK 50M,可以把CPLL设置成1000M,然后20分频。(这样Jitter肯定比把CPLL设置成50M或者100M的时候要好)这种修改方式可以放在cru节点或者设备节点中使用assigned-clock-rates的方式(本章节(1)),或者在驱动中,先设置PLL然后再设置dclk的方式。可以在PLL频率表中增加所需要的频率,并且指定VCO。

#### 以RK3399 4.4内核为例:

```
static struct rockchip_pll_rate_table
    rk3399_vpl1_rates[] = {
            /* _mhz, _refdiv, _fbdiv, _postdiv1,
    _postdiv2, _dsmpd, _frac */
            RK3036_PLL_RATE( 594000000, 1, 123, 5, 1, 0,
    12582912), /* vco = 29700000000 */
            RK3036_PLL_RATE( 593406593, 1, 123, 5, 1, 0,
    10508804), /* vco = 2967032965 */
            RK3036_PLL_RATE( 297000000, 1, 123, 5, 2, 0,
    12582912), /* vco = 29700000000 */
            RK3036_PLL_RATE( 296703297, 1, 123, 5, 2, 0,
    10508807), /* vco = 2967032970 */
            RK3036_PLL_RATE( 148500000, 1, 129, 7, 3, 0,
    15728640), /* vco = 3118500000 */
            RK3036_PLL_RATE( 148351648, 1, 123, 5, 4, 0,
    10508800), /* vco = 2967032960 */
9
            RK3036_PLL_RATE( 106500000, 1, 124, 7, 4, 0,
     4194304), /* vco = 2982000000 */
10
            RK3036_PLL_RATE( 74250000, 1, 129, 7, 6, 0,
    15728640), /* vco = 3118500000 */
            RK3036_PLL_RATE( 74175824, 1, 129, 7, 6, 0,
    13550823), /* vco = 3115384608 */
            RK3036_PLL_RATE( 65000000, 1, 113, 7, 6, 0,
    12582912), /* vco = 27300000000 */
13
            RK3036_PLL_RATE( 59340659, 1, 121, 7, 7, 0,
     2581098), /* vco = 2907692291 */
14
            RK3036_PLL_RATE( 54000000, 1, 110, 7, 7, 0,
     4194304), /* vco = 2646000000 */
15
            RK3036_PLL_RATE( 27000000, 1, 55, 7, 7, 0,
     2097152), /* vco = 1323000000 */
16
            RK3036_PLL_RATE( 26973027, 1, 55, 7, 7, 0,
     1173232), /* vco = 1321678323 */
17
            { /* sentinel */ },
```

可以修改表中的频率对应的\_refdiv,\_fbdiv,\_postdiv1,\_postdiv2,以达到比较合适的VCO。(对应3.10内核的处理方式请参考文档Rockchip Clock 开发指南中3.1章节)

# 1.1.5 CLK\_TIMER时钟配置

Timer的时钟都是从24M直接经过gating bypass过来的。所以没有频率设置的概念,只有开关时钟的说法。如果想把clk配置成常开,请参考本文1.1.2. 如果驱动自己控制可以如下操作:

```
struct clk *clk;
clk = clk_get(NULL, "clk_timer00");
clk_prepare_enable(clk);
```

# 1.1.6 总线时钟配置

总线时钟分为高速跟低速的,高速时钟aclk\_perihp、hclk\_perihp、pclk\_perihp,低速时钟是aclk\_perilp0、hclk\_perilp0、pclk\_perilp0和hclk\_perilp1是可以配置时钟频率的,但是时钟树上这些时钟其下面的子时钟都是gating,只能开关,不能设置频率,如果希望修改频率,只能修改父时钟的频率。CCI做核间通信的总线时钟ACLK\_CCI。DDR总线ACLK\_CENTER。

# 频率设置方法

dts中设置(但是只在节点初始化的时候调用一次)

```
cru: clock-controller@ff760000 {
 2
            assigned-clocks =
 3
                <&cru ACLK_PERIHP>, <&cru HCLK_PERIHP>,
4
                <&cru PCLK_PERIHP>,
 5
                <&cru ACLK_PERILPO>, <&cru HCLK_PERILPO>,
                <&cru PCLK_PERILPO>,
6
                <&cru HCLK_PERILP1>, <&cru PCLK_PERILP1>,
 7
8
                <&cru ACLK_CCI>, <&cru ACLK_CENTER>;
9
            assigned-clock-rates =
                <150000000>, <75000000>,
11
                <37500000>,
                <100000000>, <100000000>,
12
13
                <50000000>.
14
                <100000000>,
                               <50000000>,
                <40000000>, <40000000>;
15
16
        };
```

总线没有单独的驱动,也没有单独的节点,所以频率设置都是在cru节点中通过 assigned的方式处理。

# 时钟频率范围

总线中aclk一般是用于数据传输,hclk跟pclk用于寄存器读写等。设计的频率:

| CLK          | SIZEOFF | CLK          | SIZEOFF |
|--------------|---------|--------------|---------|
| ACLK_PERIHP  | 300M    | ACLK_PERILP0 | 300M    |
| HCLK_PERIHP  | 150M    | HCLK_PERILP0 | 150M    |
| PCLK_PERIHP  | 150M    | PCLK_PERILP0 | 150M    |
| HCLK_PERILP1 | 150M    | PCLK_PERILP1 | 150M    |
| CCI          | 600M    | ACLK_CENTER  | 300M    |

如果超频需要考虑加压 (logic路加压)

# 1.1.7 FCLK\_CMOS时钟配置

fclk\_cm0s是在cru而fclk\_cm0s\_src\_pmu是在pmucru的,两个时钟设置是有差异的。Pmucru请看本章节中1.2.4.fclk\_cm0s是可以配置时钟的,其下面的时钟都是

gating(hclk\_m0\_perilp\_noc\clk\_m0\_perilp\_dec\dclk\_m0\_perilp\hclk\_m0\_perilp\ sclk\_m0\_perilp),只能开关,不能设置频率,如果希望修改频率,只能修改fclk cm0s的频率。而且频率只能从GPLL或者CPLL分频下来。

# 频率设置方法

dts中设置,但是只在节点初始化的时候调用一次。

```
cru: clock-controller@ff760000 {
    assigned-clocks = <&cru FCLK_CMOS>;
    assigned-clock-rates = <100000000>;
};
```

可以放在cru节点,也可以放在设备的节点里面。 驱动中设置频率 总线一般没有单独的驱动,所以一般是在dts设置,如果部分驱动内部想调整这个频率也是可以的。

```
struct clk *clk;
clk = clk_get(NULL, "fclk_cm0s");
clk_set_rate(clk, rate);/* rate单位hz */
```

#### 时钟频率范围

M0设计的频率是100M,如果超频需要考虑加压(logic路加压)

# 1.1.8 CLK I2C时钟配置

需要注意I2c1、2、3、5、6、7在cru模块中(I2C0、4、8在pmucru设置 频率参考本章节1.2.5),3399的芯片i2c有两个时钟,一个控制时钟clk\_i2c1一个配置时钟pclk\_i2c1。控制时钟的频率只能从CPLL、GPLL分频,配置时钟频率 是从pclk\_perilp1来,自身只是gating不能修改频率,如果修改频率就要修改 pclk perilp1(见1.1.5)。

# 频率设置方法(以i2c1为例)

1. dts中设置,但是只在节点初始化的时候调用一次。

可以放在cru节点,也可以放在设备的节点里面。

#### 2. 驱动中设置频率

#### 时钟频率范围

一般使用控制时钟频率不超过100M,配置时钟不超过100M。如果超频需要考虑加压(logic路加压)

# 1.1.9 CLK SPI时钟配置

需要注意spi0、1、2、4、5在cru模块中(spi3在pmucru设置频率参考本章节1.2.6),3399的芯片spi有两个时钟,一个控制时钟clk\_spi0一个配置时钟pclk\_spi0。控制时钟的频率只能从CPLL、GPLL分频,配置时钟频率是spi0、1、2、4是从pclk\_perilp1,spi5从hclk\_perilp1,自身只是gating不能修改频率,如果修改频率就要修改pclk\_perilp1和hclk\_perilp1(见1.1.5)。

# 频率设置方法

1. dts中设置,但是只在节点初始化的时候调用一次。

```
spi0: spi@ff1c0000 {

clocks = <&cru SCLK_SPI0>, <&cru PCLK_SPI0>;

clock-names = "spiclk", "apb_pclk";

assigned-clocks = <&cru SCLK_SPI0>;

assigned-clock-rates = <500000000>;

};
```

可以放在cru节点,也可以放在设备的节点里面。

2. 驱动中设置频率

SPI的驱动文件drivers/spi/spi-rockchip.c中:

```
1    rs->apb_pclk = devm_clk_get(&pdev->dev,
    "apb_pclk");
2    rs->spiclk = devm_clk_get(&pdev->dev, "spiclk");
3    clk_set_rate(rs->spiclk, rate);/* rate单位hz */
```

# 时钟频率范围

一般使用控制时钟频率不超过50M,配置时钟不超过50M。如果超频需要考虑加压(logic路加压)

# 1.1.10 CLK\_UART时钟配置

需要注意uart0、1、2、3在cru模块中(uart4在pmucru设置频率参考本章节1.2.8),3399的芯片uart有两个时钟,一个控制时钟clk\_uart0一个配置时钟pclk\_uart0。控制时钟支持小数分频和整数分频。clk\_uart0\_div由CPLL、GPLL、USB480M整数分频得到,clk\_uart0\_frac是由clk\_uart0\_div做输入时钟然后使用小数分频器分频的(小数分频需要注意输入时钟要是输出时钟的20倍以上,否则时钟Jitter很差)。具体使用时整数分频能满足走整数分频,整数分频不能满足走小数分频。配置时钟频率是从pclk\_perilp1来,自身只是gating不能修改频率,如果修改频率就要修改pclk perilp1(见1.1.5)。

#### 频率设置方法

1. dts中设置,但是只在节点初始化的时候调用一次。

```
uart0: serial@ff180000 {
clocks = <&cru SCLK_UARTO>, <&cru PCLK_UARTO>;
clock-names = "baudclk", "apb_pclk";
assigned-clocks = <&cru SCLK_UARTO>;
assigned-clock-rates = <24000000>;
};
```

可以放在cru节点,也可以放在设备的节点里面。

2. 驱动中设置频率

UART的驱动文件drivers/tty/serial/8250/8250 dw.c

```
1 data->clk = devm_clk_get(&pdev->dev, "baudclk");
2 data->pclk = devm_clk_get(&pdev->dev, "apb_pclk");
3 clk_set_rate(data->clk, rate);/* rate单位hz */
```

这个主要看uart要求的波特率是多少,一般uart的频率是波特率\*16(HZ),一般我们平台默认支持115200、1500000两种,其他波特率要具体看PLL的频率是否可以分到。

# 1.1.11 CLK EMMC、CLK SDIO、CLK SDMMC时钟配置

这几个比较特殊,由于内部是双边沿采集数据,所以要求时钟的占空比是50%,也就要求必须是偶数分频。Emmc有两个时钟,clk\_emmc是控制器时钟,要求偶数分频的。Aclk\_emmc是数据传输和配置时钟。SDIO有有两个时钟,clk\_sdio是控制器时钟,要求偶数分频的。hclk\_sdio是配置时钟。SDMMC有有两个时钟,clk\_sdmmc是控制器时钟,要求偶数分频的。hclk\_sdmmc是配置时钟。控制时钟都是可以配置频率的,aclk\_emmc、hclk\_sdmmc也是可以单独配置频率,hclk\_sdio是一个gating是从hclk\_perilp1来,需要修改Hclk\_sdio只能修改hclk\_perilp1(见1.1.5).

#### 频率设置方法

1. dts中设置,但是只在节点初始化的时候调用一次。

```
sdhci: sdhci@fe330000 {
 2
            assigned-clocks = <&cru SCLK_EMMC>;
            assigned-clock-rates = <200000000>;
            clocks = <&cru SCLK_EMMC>, <&cru ACLK_EMMC>;
 4
            clock-names = "clk_xin", "clk_ahb";
 5
        };
 7
        sdio0: dwmmc@fe310000 {
             assigned-clocks = <&cru SCLK_SDIO>;
9
            assigned-clock-rates = <100000000>;
            clocks = <&cru HCLK_SDIO>, <&cru SCLK_SDIO>,
10
11
                 <&cru SCLK_SDIO_DRV>, <&cru</pre>
    SCLK_SDIO_SAMPLE>;
             clock-names = "biu", "ciu", "ciu-drive",
12
    "ciu-sample":
13
        };
        sdmmc: dwmmc@fe320000 {
14
            assigned-clocks = <&cru SCLK_SDMMC>;
15
            assigned-clock-rates = <100000000>;
16
            clocks = <&cru HCLK_SDMMC>, <&cru</pre>
    SCLK_SDMMC>,
18
                 <&cru SCLK_SDMMC_DRV>, <&cru
    SCLK_SDMMC_SAMPLE>;
            clock-names = "biu", "ciu", "ciu-drive",
19
    "ciu-sample";
        }:
```

可以放在cru节点,也可以放在设备的节点里面。

2. 驱动中设置频率

EMMC的驱动文件drivers/mmc/host/Sdhci-of-arasan.c

```
1 sdhci_arasan->clk_ahb = devm_clk_get(&pdev->dev,
    "clk_ahb");
2 clk_xin = devm_clk_get(&pdev->dev, "clk_xin");
3 clk_set_rate(clk_xin, rate);/* rate单位hz */
```

#### 时钟频率范围

时钟的频率范围如下,这个是输入频率,模块内部还有二分频,所以模块输出的 实际频率是时钟树上看到频率的2分频。

| CLK名称      | SIZEOFF频率           |
|------------|---------------------|
| EMMC       | 200M                |
| ACLK_EMMC  | 100M (300M/b 64Bit) |
| SDMMC/SDIO | 300/240M (<=300M)   |

注意 对于频率设置需要说明,EMMC、SDIO、SDMMC的控制时钟的 parent一般有CPLL、GPLL、NPLL、PPLL、UPLL。一般这些PLL中,CPLL被显示独占,如果EMMC需要200M频率,那么要求PLL频率是400M\800M\1200M, 所以控制时钟能分到的频率要看PLL的频率是多少?一定是偶数分频得到的频率 才可以(如果PLL只有600M和800M,那么只能分出150\200\300\400M,实际在 控制器输出频率只能那个有75、100、150、200M)。

# 1.1.12 显示相关VOP、HDCP、EDP跟ISP时钟配置

显示相关的时钟需求比较多,dclk一般要求任意频率,因为显示的分辨率不同dclk频率不同。而aclk跟Hclk做为数据传输和寄存器配置时钟一般是固定在一个值上,不会变化,一旦显示情况下修改aclk很hclk可能会造成显示抖动等。

DCLK 如果要支持任意分辨率也就是要求DCLK可以出任意频率,一般这种情况DCLK要独占一个PLL,如果是双显也就是要两个PLL独立给DCLK使用(是否支持一般在芯片设计的时候就已经确定了,或者说支持双显后可以不需要支持其他特殊的功能)。RK3399的平台如果打开宏

RK3399\_TWO\_PLL\_FOR\_VOP是支持双显任意频率,dclk\_vop0独占CPLL,dclk\_vop1独占VPLL。并且dclk\_vop0和dclk\_vop1都CLK\_SET\_RATE\_PARENT的属性,dclk需要多少就会将对于的parent的Pll设置成多少。如果没有开启宏。只有dclk\_vop0独占CPLL可以支持任意频率,dclk\_vop1就是在当前parent可以就近分频。

```
#ifdef RK3399_TWO_PLL_FOR_VOP
COMPOSITE(DCLK_VOP0_DIV, "dclk_vop0_div",
mux_pll_src_vpll_cpll_gpll_p,

CLK_SET_RATE_PARENT |
CLK_SET_RATE_NO_REPARENT,

RK3399_CLKSEL_CON(49), 8, 2, MFLAGS, 0,

8, DFLAGS,

RK3399_CLKGATE_CON(10), 12, GFLAGS),

#else
COMPOSITE(DCLK_VOP0_DIV, "dclk_vop0_div",
mux_pll_src_vpll_cpll_gpll_p,

CLK_SET_RATE_PARENT,
```

```
9
                RK3399_CLKSEL_CON(49), 8, 2, MFLAGS, 0,
    8, DFLAGS,
                RK3399\_CLKGATE\_CON(10), 12, GFLAGS),
    #endif
12
13
    #ifdef RK3399_TWO_PLL_FOR_VOP
14
        COMPOSITE(DCLK_VOP1_DIV, "dclk_vop1_div",
    mux_pll_src_vpll_cpll_gpll_p,
15
                CLK_SET_RATE_PARENT |
    CLK_SET_RATE_NO_REPARENT,
16
                RK3399_CLKSEL_CON(50), 8, 2, MFLAGS, 0,
    8, DFLAGS,
17
                RK3399\_CLKGATE\_CON(10), 13, GFLAGS),
18
    #else
19
        COMPOSITE(DCLK_VOP1_DIV, "dclk_vop1_div",
    mux_pll_src_dmyvpll_cpll_gpll_p, 0,
                RK3399_CLKSEL_CON(50), 8, 2, MFLAGS, 0,
    8, DFLAGS,
21
                RK3399\_CLKGATE\_CON(10), 13, GFLAGS),
22 #endif
```

ACLK、HCLK 支持频率设置,但是如果有Uboot显示的时候,要求Uboot 到kernel,频率不能发生变化,否则会闪屏。也就是要求uboot跟kernel的aclk hclk频率一致,其parent一致,parent频率一致。

# 频率设置方法

#### 1. dts中设置

指定vop dclk的parent,因为rk3399硬件设计原因,要求给hdmi使用的那个vop一定要在vpll上(跟hdmi是同源的),另一个vop的parent是cpll。在各自的dts中,显示的节点中增加。

```
1  &vopb_rk_fb {
2          assigned-clocks = <&cru DCLK_VOP0_DIV>;
3          assigned-clock-parents = <&cru PLL_VPLL>;
4     };
5     &vopl_rk_fb {
6          assigned-clocks = <&cru DCLK_VOP1_DIV>;
7          assigned-clock-parents = <&cru PLL_CPLL>;
8     };
```

而ACLK跟HCLK只需要初始化一次:

```
1
       &vopb {
           clocks = <&cru ACLK_VOPO>, <&cru DCLK_VOPO>,
   <&cru HCLK_VOP0>, <&cru</pre>
3
           DCLK_VOP0_DIV>;
           clock-names = "aclk_vop", "dclk_vop",
4
   "hclk_vop", "dclk_source";
           assigned-clocks = <&cru ACLK_VOP0>, <&cru
   HCLK_VOP0>;
6
           assigned-clock-rates = <400000000>,
   <100000000>;
7
       };
```

可以放在cru节点,也可以放在设备的节点里面。HDCP、EDP跟ISP设置类似处理。

HDCP ACLK\_HDCP可以从CPLL、GPLL、PPLL分频得到、HCLK\_HDCP和PCLK\_HDCP是从ACLK\_HDCP分频得到。都有独立的gating。

**EDP** 只有PCLK\_EDP,可以从CPLL、GPLL分频得到。其他的EDP时钟都是挂在这个下面的独立gating。

ISP ACLK\_ISP0、ACLK\_ISP1用于总线数据传输的可以从CPLL、GPLL、PPLL分频得到、HCLK\_ISP0、HCLK\_ISP1是寄存器配置时钟直接从ACLK分频、SCLK\_ISP0、SCLK\_ISP1是控制器时钟可以从CPLL、GPLL、NPLL分频得到。引用时钟的时候ACLK引用ACLK\_ISP0\_WRAPPER,HCLK引用HCLK\_ISP0\_WRAPPER,由CLK引用HCLK。ISP0\_WRAPPER,由CLK则用的可以从CPLL、GPLL、NPLL分频得到。引用时钟的时候ACLK引用ACLK\_ISP0\_WRAPPER,由CLK引用HCLK则的parent时钟保证ACLK跟HCLK时钟通路开启。

#### 2. 驱动中设置频率

vop的驱动文件drivers/gpu/drm/rockchip/rockchip drm vop.c

```
vop->hclk = devm_clk_get(vop->dev, "hclk_vop");
 2
        if (IS_ERR(vop->hclk)) {
            dev_err(vop->dev, "failed to get hclk
    source\n");
 4
            return PTR_ERR(vop->hclk);
6
        vop->aclk = devm_clk_get(vop->dev, "aclk_vop");
        if (IS_ERR(vop->aclk)) {
 7
            dev_err(vop->dev, "failed to get aclk
    source\n");
9
            return PTR_ERR(vop->aclk);
10
        vop->dclk = devm_clk_get(vop->dev, "dclk_vop");
11
        if (IS_ERR(vop->dclk)) {
12
            dev_err(vop->dev, "failed to get dclk
13
    source\n");
14
            return PTR_ERR(vop->dclk);
15
        }
```

Dclk主要是看屏的分辨率是多少。因为PLL使用自动计算的,jitter不是最优,如果需要调整,可以在PLL表格中自己计算合适的VCO填进去(VCO的计算方式详细见RK3399 datasheet的2.6.2)。

```
Drivers/clk/rockchip/clk-rk3399.c
 3
4
        static struct rockchip_pll_rate_table
    rk3399_vpl1_rates[] = {
            /* _mhz, _refdiv, _fbdiv, _postdiv1,
    _postdiv2, _dsmpd, _frac */
            RK3036_PLL_RATE( 594000000, 1, 123, 5, 1, 0,
    12582912), /* vco = 2970000000 */
            RK3036_PLL_RATE( 593406593, 1, 123, 5, 1, 0,
    10508804), /* vco = 2967032965 */
            RK3036_PLL_RATE( 297000000, 1, 123, 5, 2, 0,
    12582912), /* vco = 2970000000 */
            RK3036_PLL_RATE( 296703297, 1, 123, 5, 2, 0,
    10508807), /* vco = 2967032970 */
            RK3036_PLL_RATE( 148500000, 1, 129, 7, 3, 0,
10
    15728640), /* vco = 3118500000 */
            RK3036_PLL_RATE( 148351648, 1, 123, 5, 4, 0,
11
    10508800), /* vco = 2967032960 */
            RK3036_PLL_RATE( 106500000, 1, 124, 7, 4, 0,
    4194304), /* vco = 2982000000 */
            RK3036_PLL_RATE( 74250000, 1, 129, 7, 6, 0,
13
    15728640), /* vco = 3118500000 */
            RK3036_PLL_RATE( 74175824, 1, 129, 7, 6, 0,
14
    13550823), /* vco = 3115384608 */
            RK3036_PLL_RATE( 65000000, 1, 113, 7, 6, 0,
    12582912), /* vco = 27300000000 */
            RK3036_PLL_RATE( 59340659, 1, 121, 7, 7, 0,
16
     2581098), /* vco = 2907692291 */
            RK3036_PLL_RATE( 54000000, 1, 110, 7, 7, 0,
17
     4194304), /* vco = 2646000000 */
18
            RK3036_PLL_RATE( 27000000, 1, 55, 7, 7, 0,
     2097152), /* vco = 1323000000 */
19
            RK3036_PLL_RATE( 26973027, 1, 55, 7, 7, 0,
     1173232), /* vco = 1321678323 */
            { /* sentinel */ },
        };
```

而对于对应的频率范围(如果做4K显示带宽不够可以相应提高ACLK的频率,但是要注意电压是不是够是否需要提压):

| CLK         | SIZEOFF | CLK         | SIZEOFF |
|-------------|---------|-------------|---------|
| ACLK_VOP0/1 | 400M    | ACLK_VIO    | 400M    |
| DCLK_VOP0   | 600M    | ACLK_ISP0/1 | 400M    |
| DCLK_VOP1   | 300M    | CLK_ISP0/1  | 500M    |

| CLK_VOP0/1_PWM | <b>SMOR</b> OFF | <b>CLK</b> _EDP | <b>SOZEOFF</b> |
|----------------|-----------------|-----------------|----------------|
| ACLK HDCP      | 400M            |                 |                |

# 1.1.13 视频编解码VDU、RGA、CODEC、IEP相关时钟配置

主要是VDU、RGA、CODEC、IEP相关的时钟配置。

**VDU** ACLK\_VDU可以从CPLL、GPLL、NPLL、PPLL分频得到,HCLK\_VDU从ACLK分频得到。控制器时钟SCLK\_VDU\_CORE和SCLK\_VDU\_CA从CPLL、GPLL、NPLL分频得到。

RGA ACLK\_RGA可以从CPLL、GPLL、NPLL、PPLL分频得到,HCLK RGA从ACLK分频得到。

CODEC ACLK\_VCODEC可以从CPLL、GPLL、NPLL、PPLL分频得到,HCLK\_VCODEC从ACLK分频得到。控制器时钟SCLK\_RGA\_CORE从CPLL、GPLL、NPLL、PPLL分频得到。

**IEP** ACLK\_IEP可以从CPLL、GPLL、NPLL、PPLL分频得到,HCLK\_IEP 从ACLK分频得到。

#### 频率设置方法

1. dts中设置,但是只在节点初始化的时候调用一次。

可以放在cru节点,也可以放在设备的节点里面(其他模块类似处理)。

2. 驱动中设置频率

VCODEC的驱动文件drivers/video/rockchip/vcodec/vcodec service.c

```
pservice->aclk_vcodec = devm_clk_get(dev,
    "aclk_vcodec");
pservice->hclk_vcodec = devm_clk_get(dev,
    "hclk_vcodec");
pservice->clk_cabac = devm_clk_get(dev,
    "clk_cabac");
pservice->clk_core = devm_clk_get(dev, "clk_core");
```

#### 时钟频率范围

而对于对应的频率范围(如果做4K视频编解码带宽不够可以相应提高ACLK的频率,但是要注意电压是不是够是否需要提压):

| CLK          | SIZEOFF | CLK          | SIZEOFF |
|--------------|---------|--------------|---------|
| ACLK_VCODEC  | 400M    | ACLK_IEP     | 400M    |
| ACLK_VDU     | 400M    | ACLK_RGA     | 400M    |
| CLK_VDU_CORE | 300M    | CLK_RGA_CORE | 400M    |
| CLK_VDU_CA   | 300M    |              |         |

# 1.1.14 USB相关时钟配置

USB主要包括aclk、Host、otg还有就是usb内部phy。

USB3 ACLK USB3可以从CPLL、GPLL、NPLL分频得到。

**HOST** HCLK\_HOST0跟HCLK\_HOST1都是挂在HCLK\_PERI下面,只有GATING属性,如果要修改频率只能修改HCLK PERI(详细见本文1.1.5)。

OTG ACLK\_USB3OTG0、ACLK\_USB3OTG1是挂在从ACLK\_USB3下面,只有GATING属性,如果要修改频率只能修改ACLK\_USB3。
SCLK\_USB3OTG0\_REF直接从24M来的,只有GATING属性,
SCLK\_USB3OTG0\_SUSPEND可以从24M或者32K上获取可以设置频率。

PHY SCLK\_USBPHY1\_480M是USB PHY内部PLL锁相环出的,一般是固定 频率480M,可以选择内部锁相环输出可以直接选择24M输出。

#### 频率设置方法

1. dts中设置,但是只在节点初始化的时候调用一次。

```
usb_host0_ohci: usb@fe3a0000 {
            clocks = <&cru HCLK_HOST0>, <&cru</pre>
    HCLK_HOSTO_ARB>,
                  <&cru SCLK_USBPHY0_480M_SRC>;
            clock-names = "hclk_host0", "hclk_host0_arb",
    "usbphy0_480m";
 5
       };
 7
        usb_host1_ehci: usb@fe3c0000 {
            clocks = <&cru HCLK_HOST1>, <&cru</pre>
    HCLK_HOST1_ARB>,
9
                  <&cru SCLK_USBPHY1_480M_SRC>;
            clock-names = "hclk_host1", "hclk_host1_arb",
    "usbphy1_480m";
11
       };
12
        usbdrd3_0: usb@fe800000 {
            clocks = <&cru SCLK_USB30TG0_REF>, <&cru</pre>
    SCLK_USB3OTG0_SUSPEND>,
```

可以放在cru节点,也可以放在设备的节点里面(其他模块类似处理)。

2. 驱动中设置频率

VCODEC的驱动文件drivers/usb/dwc3/dwc3-rockchip.c

```
1     clk = of_clk_get(np, i);
2     if (IS_ERR(clk)) {
3         ret = PTR_ERR(clk);
4         goto err0;
5     }
6     ret = clk_prepare_enable(clk);
7     if (ret < 0) {
8           clk_put(clk);
9         goto err0;
10     }</pre>
```

# 时钟频率范围

而对于对应的频率范围(如果USB有大数据拷贝等可以相应提高ACLK的频率,但是要注意电压是不是够是否需要提压),ACLK USB的SIZEOFF频率400M。

# 1.1.15 CIF相关时钟配置

Cif主要是SCLK\_CIF\_OUT,可能有24M或者27M这样的时钟要求。这个时钟源可以直接选择24M进行分频也可以选择CPLL、GPLL、NPLL然后再分频。

# 频率设置方法

1. dts中设置,但是只在节点初始化的时候调用一次。

可以放在cru节点,也可以放在设备的节点里面(其他模块类似处理)。

# 1.2 PMUCRU时钟配置

# 1.2.1 PMUCRU时钟树

```
pll_ppll
                                                                                   0 0
                                                              676000000
                                                              676000000
   llgg
                                                               48285715
      pclk_pmu_src
                                                                                   0 0 0 0 0
          pclk_wdt_m0_pmu
                                           0
                                                               48285715
          pclk uart4 pmu
                                                               48285715
          pclk_mailbox_pmu
                                                               48285715
                                                                                   0 0 0 0 0
                                                               48285715
          pclk_timer_pmu
                                           0
          pclk_spi3_pmu
                                                         0
                                                               48285715
                                                               48285715
          pclk_rkpwm_pmu
         pclk_i2c8_pmu
pclk_i2c4_pmu
pclk_i2c0_pmu
                                                               48285715
                                                               48285715
                                                                                   0 0
                                           0
                                                         0
                                                              48285715
         pclk_noc_pmu
pclk_sgrf_pmu
                                                               48285715
                                                         Ó
                                                               48285715
                                                                                   0 0 0 0 0
                                                               48285715
          pclk_gpio1_pmu
                                           0 0
                                                         0
          pclk_gpio0_pmu
                                                               48285715
                                                               48285715
          pc|k_intmem1_pmu
                                                               48285715
          pclk pmugrf pmu
          pclk_pmu
                                                               48285715
                                           0
                                                                                   0 0
      clk_i2c8_pmu
                                                         0
                                                               48285715
      clk_i2c4_pmu
clk_i2c0_pmu
                                                               48285715
                                                               48285715
      clk_wifi_div
                                                               26000000
                                                                                   0
                                                         0
                                                              26000000
1300000
                                                                                   0 0
          clk_wifi_pmu
                                                         ŏ
          clk_wifi_frac
       clk_spi3_pmu
                                                        1
2
1
                                                         Ō
                                                               96571429
       fclk_cmOs_pmu_ppll_src
                                                               676000000
                                                                                    0
                                                                                   0 0
                                                               84500000
          fclk_cm0s_src_pmu
             hclk_noc_pmu
dclk_cm0s_pmu
                                                               84500000
                                                               84500000
                                                                                   0 0
             hclk_cm0s_pmu
                                                         0
                                                               84500000
                                                               84500000
             sclk_cm0s_pmu
                                                               84500000
             fclk_cm0s_pmu
```

注意 上述时钟控制都是在pmucru寄存器。

# 1.2.2 配置一些时钟常开

对于调试过程中,想把某些时钟设置成常开的,

可以修改rk3399\_pmucru\_critical\_clocks这个结构体中,按照现有增加时钟名字即可。

```
Drivers/clk/rockchip/clk-rk3399.c

static const char *const
   rk3399_pmucru_critical_clocks[] __initconst = {
        "pclk_noc_pmu",
    };
```

这个结构中的clk在系统开机,clk初始化的时候会默认调用clk set enable接口。

# 1.2.3 PCLK PMU总线时钟配置

总线时钟只有pclk\_pmu\_src是可以配置时钟的,其下面的时钟都是gating(pclk\_wdt\_m0\_pmu\pclk\_uart4\_pmu\pclk\_mailbox\_pmu\pclk\_timer\_pmu\pclk\_spi3\_pmu\pclk\_rkpwm\_pmu\pclk\_i2c8\_pmu\pclk\_i2c4\_pmu\pclk\_i2c0\_pmu\pclk\_noc\_pmu\pclk\_sgrf\_pmu\pclk\_gpio1\_pmu\pclk\_gpio0\_pmu\pclk\_intmem1\_pmu\pclk\_pmugrf\_pmu\pclk\_pmu),只能开关,不能设置频率,如果希望修改频率,只能修改pclk\_pmu\_src的频率。而且频率只能从PPLL分频下来(676M整除出来的频率)。

1. dts中设置,但是只在节点初始化的时候调用一次。

```
pmucru: pmu-clock-controller@ff750000 {
    assigned-clocks = <&pmucru PCLK_SRC_PMU>;
    assigned-clock-rates = <100000000>;
};
```

可以放在pmucru节点,也可以放在设备的节点里面。

2. 驱动中设置频率

总线一般没有单独的驱动,所以一般是在dts设置,如果部分驱动内部想调整这个频率也是可以的。

```
struct clk *clk;
clk = clk_get(NULL, "pclk_pmu_src");
clk_set_rate(clk, rate);/* rate单位hz */
```

# 时钟频率范围

IC设计的频率是50M,如果超频需要考虑加压(logic路加压)

# 1.2.4 PMU\_M0时钟配置

总线时钟只有PCLK\_SRC\_PMUfclk\_cm0s\_src\_pmu是可以配置时钟的,其下面的时钟都是

gating(hclk\_noc\_pmu\dclk\_cm0s\_pmu\hclk\_cm0s\_pmu\sclk\_cm0s\_pmu\fclk\_cm0s\_pmu\fclk\_cm0s\_pmu\, 只能开关,不能设置频率,如果希望修改频率,只能修改fclk cm0s src pmu的频率。而且频率只能从PPLL或者24M分频下来。

# 频率设置方法

1. dts中设置,但是只在节点初始化的时候调用一次。

```
pmucru: pmu-clock-controller@ff750000 {
    assigned-clocks = <&pmucru FCLK_CMOs_SRC_PMU>;
    assigned-clock-rates = <100000000>;
};
```

可以放在pmucru节点,也可以放在设备的节点里面。

2. 驱动中设置频率

总线一般没有单独的驱动,所以一般是在dts设置,如果部分驱动内部想调整这个频率也是可以的。

```
struct clk *clk;
clk = clk_get(NULL, "fclk_cm0s_src_pmu");
clk_set_rate(clk, rate);/* rate单位hz */
```

IC设计的频率是100M,如果超频需要考虑加压(logic路加压)

# 1.2.5 PMU总线时钟配置

总线时钟只有PCLK\_SRC\_PMU是可以配置时钟的,其下面的时钟都是gating(详细见时钟树),只能开关,不能设置频率,如果希望修改频率,只能修改PCLK\_SRC\_PMU的频率。而且频率只能从PPLL分频下来。

# 频率设置方法

1. dts中设置,但是只在节点初始化的时候调用一次。

```
pmucru: pmu-clock-controller@ff750000 {
    assigned-clocks = <&pmucru PCLK_SRC_PMU>;
    assigned-clock-rates = <1000000000>;
};
```

可以放在pmucru节点,也可以放在设备的节点里面。

#### 时钟频率范围

IC设计的频率是100M,如果超频需要考虑加压(logic路加压)

# 1.2.6 PMU\_I2C时钟配置

需要注意I2c0\4\8在pmucru模块中,3399的芯片i2c有两个时钟,一个控制时钟clk\_i2c0\_pmu一个配置时钟pclk\_i2c0\_pmu。控制时钟的频率只能从PPLL(676M)分频,配置时钟频率是从pclk\_pmu\_src来,自身只是gating不能修改频率,如果修改频率就要修改pclk pmu src(见1.2.1)。

# 频率设置方法(以i2c0为例)

1.dts中设置,但是只在节点初始化的时候调用一次。

```
i2c0: i2c@ff3c0000 {
    clocks = <&pmucru SCLK_I2CO_PMU>, <&pmucru
PCLK_I2CO_PMU>;

clock-names = "i2c", "pclk";

assigned-clocks = <&pmucru SCLK_I2CO_PMU>;

assigned-clock-rates = <500000000>;

}
```

可以放在pmucru节点,也可以放在设备的节点里面。

# 2. 驱动中设置频率

I2C的驱动文件drivers/i2c/busses/i2c-rk3x.c中:

```
1    i2c->clk = devm_clk_get(&pdev->dev, "i2c");
2    i2c->pclk = devm_clk_get(&pdev->dev, "pclk");
3    clk_set_rate(i2c->clk, rate);/* rate单位hz */
```

#### 时钟频率范围

一般使用控制时钟频率不超过100M,配置时钟不超过100M。如果超频需要考虑加压(logic路加压)

# 1.2.7 PMU\_SPI时钟配置

需要注意spi3在pmucru模块中,3399的芯片spi有两个时钟,一个控制时钟clk\_spi3\_pmu一个配置时钟pclk\_spi3\_pmu。控制时钟的频率只能从PPLL(676M)分频,配置时钟频率是从pclk\_pmu\_src来,自身只是gating不能修改频率,如果修改频率就要修改pclk\_pmu\_src(见1.2.1)。

# 频率设置方法

1. dts中设置,但是只在节点初始化的时候调用一次。

```
spi3: spi@ff350000 {

clocks = <&pmucru SCLK_SPI3_PMU>, <&pmucru
PCLK_SPI3_PMU>;

clock-names = "spiclk", "apb_pclk";

assigned-clocks = <&pmucru SCLK_SPI3_PMU>;

assigned-clock-rates = <500000000>;

};
```

可以放在pmucru节点,也可以放在设备的节点里面。

2. 驱动中设置频率

SPI的驱动文件drivers/spi/spi-rockchip.c中:

```
1    rs->apb_pclk = devm_clk_get(&pdev->dev,
    "apb_pclk");
2    rs->spiclk = devm_clk_get(&pdev->dev, "spiclk");
3    clk_set_rate(rs->spiclk, rate);/* rate单位hz */
```

#### 时钟频率范围

一般使用控制时钟频率不超过50M,配置时钟不超过50M。如果超频需要考虑加压(logic路加压)

# 1.2.8 PMU\_WIFI时钟配置

需要注意WIFI在pmucru模块中,3399的芯片wifi支持小数分频和整数分频。clk\_wifi\_div由PPLL整数分频得到,clk\_wifi\_frac是由clk\_wifi\_div做输入时钟然后使用小数分频器分频的(小数分频需要注意输入时钟要是输出时钟的20倍以上,否则时钟Jitter很差)。具体使用时整数分频能满足走整数分频,整数分频不能满足走小数分频。

# 频率设置方法

1. dts中设置,但是只在节点初始化的时候调用一次。

```
pmucru: pmu-clock-controller@ff750000 {
compatible = "rockchip, rk3399-pmucru";
assigned-clocks = <&pmucru SCLK_WIFI_PMU>;
assigned-clock-rates = <26000000>;
};
```

可以放在pmucru节点,也可以放在设备的节点里面。

2. 驱动中设置频率

```
struct clk *clk;
clk = clk_get(NULL, "clk_wifi_pmu");
clk_set_rate(clk, rate);/* rate单位hz */
```

#### 时钟频率范围

这个主要看wifi模组使用是什么晶振,一般常见的24M 26M 37.4M 40M。

# 1.2.9 PMU\_UART4时钟配置

需要注意uart4在pmucru模块中,3399的芯片uarti有两个时钟,一个控制时钟clk\_uart4\_pmu一个配置时钟pclk\_uart4\_pmu。控制时钟支持小数分频和整数分频。clk\_uart4\_div由PPLL整数分频得到,clk\_uart4\_frac是由clk\_uart4\_div做输入时钟然后使用小数分频器分频的(小数分频需要注意输入时钟要是输出时钟的20倍以上,否则时钟Jitter很差)。具体使用时整数分频能满足走整数分频,整数分频不能满足走小数分频。配置时钟频率是从pclk\_uart4\_pmu来,自身只是gating不能修改频率,如果修改频率就要修改pclk\_pmu\_src(见1.2.1)。

# 频率设置方法

1. dts中设置,但是只在节点初始化的时候调用一次。

```
uart4: serial@ff370000 {
    clocks = <&pmucru SCLK_UART4_PMU>, <&pmucru
PCLK_UART4_PMU>;

clock-names = "baudclk", "apb_pclk";
    assigned-clocks = <&pmucru SCLK_UART4_PMU>;
    assigned-clock-rates = <24000000>;
};
```

可以放在pmucru节点,也可以放在设备的节点里面。

# 2. 驱动中设置频率

UART的驱动文件drivers/tty/serial/8250/8250 dw.c

```
data->clk = devm_clk_get(&pdev->dev, "baudclk");
data->pclk = devm_clk_get(&pdev->dev, "apb_pclk");
clk_set_rate(data->clk, rate);/* rate单位hz */
```

#### 时钟频率范围

这个主要看uart要求的波特率是多少,一般uart的频率是波特率\*16(HZ),一般我们平台默认支持115200、1500000两种,其他波特率要具体看PLL的频率是否可以分到。

# 2 时钟间依赖关系

# 2.1 普通的父子关系

时钟结构图及时钟树如下:



时钟树的结构就是:



普通的父子关系的依赖关系就是,子时钟开启的时候需要开启父时钟,时钟结构会保证此操作,只需要开启子时钟即可,时钟结构会自动索引其父时钟并开启。只要其子时钟有在工作,父时钟就不能关闭,正常情况时钟的开关是有引用计数,如上图中的enable\_cnt,子时钟或者本身时钟被enable后计数加一,disable的时候计数减一,直到计数减为零,时钟才会被关闭。

# 2.2 不同模块间NOC复用

在设计NOC的时候,有一些模块之间的Noc是复用,这就要求任何一个模块在使用的时候,NOC时钟都要开启,而且NOC的父时钟的整个时钟通路都要开启。有这种特殊要求的有如下时钟(目前代码中都已经处理,保证NOC时钟常开):

|               | 表 2-1 NOC 依赖乡       | 关系表 1      |          |
|---------------|---------------------|------------|----------|
|               | adk_cdi_noc0:G15[3] |            |          |
| cci_noc       | adk_cdi_noc1:G15[4] |            |          |
|               | dk_dbg_noc:G15[5]   |            |          |
|               | alive_noc(详细见下表格)   |            |          |
|               | pmu_noc             |            |          |
|               | emmc_noc            |            |          |
| peri_lp_noc-  | gmac_noc            |            |          |
| peri_ip_rioc= | usb3_noc            |            |          |
|               | gic_noc             |            |          |
|               | sdioaudio_noc       |            |          |
|               |                     | msch0_noc  |          |
|               |                     | msch1_noc  |          |
|               |                     | iep_noc    |          |
|               |                     | rga_noc    |          |
|               | contor poo          | perihp_noc | sd_noc   |
|               | center_noc          | vdu_noc    |          |
|               |                     | vcodec_noc |          |
|               |                     | gpu_noc    |          |
|               |                     | edp_noc    |          |
|               |                     |            | isp0_noc |
|               |                     |            | isp1_noc |
|               |                     | vio_noc    | hdcp_noc |
|               |                     |            | vopb_noc |
|               |                     |            | vopl_noc |

| •             | pt.                                  | 2-2 NOC 依赖天条表 2                                                                                    |                                                                                                                                                                                                                                                                                                                |
|---------------|--------------------------------------|----------------------------------------------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| alive_no<br>c |                                      | alive_noc:PMUGRF0[6]                                                                               |                                                                                                                                                                                                                                                                                                                |
| 5[7]          | k_perilp0_noc:G2                     | pclk_noc_pmu:PG1[6]<br>hclk_noc_pmu:PG1[5]                                                         |                                                                                                                                                                                                                                                                                                                |
| hclk_pe       | erilp0_noc:G25[8                     | aclk_emmc_noc:G32[9]                                                                               |                                                                                                                                                                                                                                                                                                                |
| ]             | erilp1_noc:G25[9<br>erilp1_noc:G25[1 | aclk_gmac_noc:G32[1]<br>pclk_gmac_nocG32[3]                                                        |                                                                                                                                                                                                                                                                                                                |
|               |                                      | aclk_usb3_noc:G30[0]                                                                               |                                                                                                                                                                                                                                                                                                                |
|               |                                      | aclk_gic_noc:G33[1]                                                                                |                                                                                                                                                                                                                                                                                                                |
|               |                                      | sdioaudio_noc:G34[6]                                                                               |                                                                                                                                                                                                                                                                                                                |
|               |                                      | aclk_center_main_noc:G1<br>9[0]<br>aclk_center_peri_noc:G19[1]<br>pclk_center_main_noc:G18[1<br>0] | msch0_noc:G18[4] msch1_noc:G18[9] aclk_iep_noc:G16[1] hclk_iep_noc:G16[3] aclk_rga_noc:G16[9] hclk_rga_noc:G16[11] aclk_perihp_noc:G20 [12] hclk_perihp_noc:G20[13] pclk_perihp_noc:G20[14] aclk_vdu_noc:G17[9] hclk_vdu_noc:G17[11] aclk_vcodec_noc:G1 7[1] hclk_vcodec_noc:G17[3] gpu_noc pclk_edp_noc:G32[1 |

在设计GRF的时候,有一些模块之间的GRF时钟是复用,这就要求任何一个模块在GRF寄存器读写的时候,公用的GRF时钟都要开启,而且GRF的父时钟的整个时钟通路都要开启。 有这种特殊要求的有如下时钟(目前代码中都已经处理,保证GRF时钟常开):

| GRF             | CON                   | GRF             | CON              |
|-----------------|-----------------------|-----------------|------------------|
| aclk_cci_grf    | grf_a72_perf          | pclk_grf(alive) | grf_iomux        |
| aclk_cci_grf    | grf_a53_perf          | pclk_grf(alive) | grf_soc_con[0~8] |
| aclk_cci_grf    | grf_cpu_status        | pclk_grf(alive) | grf_usb3phy      |
| aclk_cci_grf    | grf_cpu_con           | pclk_grf(alive) | grf_ddrc         |
|                 |                       | pclk_grf(alive) | grf_gpio2/3/4    |
| pclk_perihp_grf | grf_hsic              | pclk_grf(alive) | grf_io_vsel      |
| pclk_perihp_grf | grf_hsicphy           | pclk_grf(alive) | grf_saradc       |
| pclk_perihp_grf | grf_usbhost0          | pclk_grf(alive) | grf_tsadc        |
| pclk_perihp_grf | grf_usbhost1          | pclk_grf(alive) | grf_usb20_phy    |
| pclk_perihp_grf | grf_usbphy            | pclk_grf(alive) | grf_dll          |
| pclk_perihp_grf | grf_usb20_host        | pclk_grf(alive) | grf_alive_lf_ena |
| pclk_perihp_grf | grf_pcie              | pclk_grf(alive) | grf_cphy         |
|                 |                       | pclk_grf(alive) | grf_uphy         |
| pclk_vio_grf    | grf_soc_con[9, 20~26] | pclk_grf(alive) | grf_pcie         |
| pclk_vio_grf    | grf_hdcp              | pclk_grf(alive) | grf_sd           |
|                 |                       |                 |                  |
| aclk_usb3_grf   | grf_sta_usb3otg       | aclk_gpu_grf    | grf_gpu_perf     |
| aclk_usb3_grf   | grf_usb3_perf         |                 |                  |

# 3 时钟频率值

# 3.1 可设置的时钟频率

| H 1. 6.1. 64 76 | 日之后之 | マルハロ田(株子                       |
|-----------------|------|--------------------------------|
| 时钟名称            | 最高频率 | 可以设置频率                         |
| CCI             | 600M | (cpll/gpll/npll/vpll) / (1~32) |
| ACLK_CENTER     | 300M | (cpll/gpll/npll) / (1~32)      |
| DDR_PCLK        | 200M | (cpll/gpll) / (1~32)           |
| ACLK_PERIHP     | 300M | (cpll/gpll) / (1~32)           |
| ACLK_PERILP0    | 300M | (cpll/gpll) / (1~32)           |
| AHB_PERILP1     | 150M | (cpll/gpll) / (1~32)           |
| ACLK_VCODEC     | 400M | (cpll/gpll/npll/ppll) / (1~32) |
| ACLK_VDU        | 400M | (cpll/gpll/npll/ppll) / (1~32) |
| CLK_VDU_CORE    | 300M | (cpll/gpll/npll) / (1~32)      |
| CLK_VDU_CA      | 300M | (cpll/gpll/npll) / (1~32)      |
| ACLK_IEP        | 400M | (cpll/gpll/npll/ppll) / (1~32) |
| ACLK_RGA        | 400M | (cpll/gpll/npll/ppll) / (1~32) |
| CLK_RGA_CORE    | 400M | (cpll/gpll/npll/ppll) / (1~32) |

| 財<br>熱<br>程<br>称 | 最高频率         | 可以沒開煙剂/ppll/upll/24M) / (1~128)          |
|------------------|--------------|------------------------------------------|
| SDMMC/SDIO       | 400/300/240M | (cpll/gpll/npll/ppll/upll/24M) / (1~128) |
| CLK_PCIE_REF     | 24/100M      | (npll/24M) / (1~128)                     |
| CLK_PCIE_CORE    | 250M         | (cpll/gpll/npll) / (1~128)               |
| CLK_PCIE_PM      | 25M/24M      | (cpll/gpll/npll/24M) / (1~128)           |
| ACLK_GMAC        | 300M         | (cpll/gpll) / (1~32)                     |
| ACLK_EMMC        | 300M         | (cpll/gpll) / (1~32)                     |
| M0-PERILP        | 300M         | (cpll/gpll) / (1~32)                     |
| CRYPTO           | 200M         | (cpll/gpll/ppll) / (1~32)                |
| SPI              | 100M         | (cpll/gpll) / (1~128)                    |
| I2S/SPDIF        | Frac<=50M    | (cpll/gpll) / (1~128)                    |
| SARADC           | 20M          | 24M/ (1~256)                             |
| TSADC            | 10M          | 24M/ (1~1024)                            |
| ACLK_VIO         | 400M         | (cpll/gpll/ppll) / (1~32)                |
| CLK_EDP          | 200M         | (cpll/gpll) / (1~32)                     |
|                  |              |                                          |
| ACLK ISP0/1      | 400M         | (cpll/gpll/ppll) / (1~32)                |
| CLK_ISPO/1_JPE   | 500M         | (cpll/gpll/npll) / (1~32)                |
| ACLK HDCP        | 400M         | (cpll/gpll/ppll) / (1~32)                |
| CLK_DP_CORE      | 200/100/10M  | (cpll/gpll/npll) / (1~32)                |
| ACLK VOP0/1      | 400M         | (cpll/gpll/npll/vpll) / (1~32)           |
| DCLK_VOP0        | 600M         | (cpll/gpll/vpll) / (1~256)               |
| DCLK_VOP1        | 300M         | (cpll/gpll/vpll) / (1~256)               |
| CLK VOP0/1 PWM   | 200M         | (cpll/gpll/vpll/24M) / (1~32)            |
| ACLK USB3        | 300M         | (cpll/gpll/npll) / (1~32)                |
| PCLK ALIVE       | 100M         | (gpll) / (1~32)                          |
| PCLK PMU         | 100M         | (ppll) / (1~32)                          |
| GPU              | 500M         | (cpll/gpll/npll/upll/ppll) / (1~32)      |
| M0-EC            | 200M         | (cpll/gpll) / (1~32)                     |
| 10               | 200111       | (CP11/ OP11/ (1 02)                      |