

## Linux kernel: consolidation in the ARM architecture support

Thomas Petazzoni **Free Electrons** thomas.petazzoni@free-electrons.com





- Embedded Linux engineer and trainer at Free Electrons since 2008
  - Embedded Linux development: kernel and driver development, system integration, boot time and power consumption optimization, consulting, etc.
  - Embedded Linux training, Linux driver development training and Android system development training, with materials freely available under a Creative Commons license.
  - http://www.free-electrons.com
- Major contributor to Buildroot, an open-source, simple and fast embedded Linux build system
- Working on mainlining support for the Marvell Armada 370 and Armada XP ARM SoCs
- Speaker at Embedded Linux Conference, Embedded Linux Conference Europe, FOSDEM, Libre Software Meeting, etc.
- Living in Toulouse, south west of France



- The initial complaint
- ▶ The ARM "problem(s)"
- The solutions
  - Maintainer
  - Device Tree
  - Clock framework
  - Pinctrl subsystem



# Gaah. Guys, this whole ARM thing is a f\*cking pain in the ass.



# Gaah. Guys, this whole ARM thing is a f\*cking pain in the ass.

#### Linus Torvalds - 17 March 2011

# Somebody in the ARM community really needs to step up and tell people to stop dicking around.



ARM Holdings is a British company that designs processor cores.

- They define instruction sets, and design cores implementing those instruction sets, memory management units, caches, etc.
- Examples of cores: ARM926EJ-S, Cortex-A8, Cortex-A9, Cortex-M3, etc.
- ► ARM does not produce processors that people can buy



A number of *silicon-vendors* buy those designs, and create *System-on-chip* combining a ARM core and a number of peripherals

- Peripherals are typically UARTs, bus controllers (USB, SPI, I2C, PCI, etc.), Ethernet controllers, ADCs, CAN controllers, video/audio encoding/decoding, graphics (2D, 3D), etc.
- Silicon vendors combine different sets of peripherals to address different markets (industrial, automotive, consumer, etc.)
- Examples: Texas Instruments OMAP4 (dual Cortex-A9), Atmel AT91SAM9M10 (926EJ-S), Freescale i.MX 6 (from 1 to 4 Cortex-A9), etc.

High level overview of an ARM board

•







and this is only a very, very partial list. ARM has more than 150 licensees.



- This huge variety of SoCs, each getting more and more complicated, leads to a large quantity of code to support them
- ► The historical ARM maintainer, Russell King, through which all ARM code was initially going, got **overflowed by the amount of code**
- Code from sub-architectures (SoC families) started to go directly to Linus
- Focus of sub-architecture maintainers on their sub-architecture, no vision of the other sub-architectures
- Consequence: a lot of code duplication, no common infrastructures designed for common problems, similar problems solved differently, etc.

Single kernel image

- Until now, ARM platforms mainly used in the industrial embedded space, or un-modifiable consumer products
  - A given binary kernel image was configured and built specifically for each platform
  - All the existing code makes the assumption that the kernel is built for one single platform
- ARM now used in modifiable consumer products (tablets, phones, etc.)
  - Desire for distributions to provide a system for such platforms
  - On x86, easy because one kernel image supports all platforms. On ARM, each platform currently needs one specific kernel image.
  - In the future, wish to support multiple platforms inside a single kernel image.



- Sub-architecture maintainer
- Device Tree
- Clock framework
- Pinctrl subsystem

## Sub-architecture maintainer



- The lack of a cross sub-architecture vision was filled by creating a team of ARM sub-architecture maintainers, in May 2011
- Initially with Arnd Bergmann (Linaro/IBM), Nicolas Pitre (Linaro), Marc Zyngier (ARM), later joined by Olof Johansson (Tegra maintainer)
- This team takes care of reviewing and consolidating the code of the different sub-architectures, and sending it to Linus Torvalds
- Russell King continues to be the maintainer for the ARM core part (memory management, CPU support code, etc.)
- Important role of Linaro in this new maintainer team
  - Linaro is a not-for-profit engineering organization consolidating and optimizing open source Linux software and tools for the ARM architecture.





## Device tree



- All the board-specific details and SoC-specific details require specific C code to support new platforms
- A lot of very similar C code to support each and every board, to list all the peripherals, their configuration, etc.
- Code organized in a hierarchy



List of files in arch/arm/mach-at91

· • `

| at91rm9200.c          | at91x40_time.c      | board-neocore926.c   | clock.h        |
|-----------------------|---------------------|----------------------|----------------|
| at91rm9200_devices.c  | board-1arm.c        | board-pcontrol-g20.c | cpuidle.c      |
| at91rm9200_time.c     | board-afeb-9260v1.c | board-picotux200.c   | generic.h      |
| at91sam9260.c         | board-cam60.c       | board-qil-a9260.c    | gpio.c         |
| at91sam9260_devices.c | board-carmeva.c     | board-rm9200dk.c     | include        |
| at91sam9261.c         | board-cpu9krea.c    | board-rm9200ek.c     | irq.c          |
| at91sam9261_devices.c | board-cpuat91.c     | board-rsi-ews.c      | Kconfig        |
| at91sam9263.c         | board-csb337.c      | board-sam9260ek.c    | leds.c         |
| at91sam9263_devices.c | board-csb637.c      | board-sam9261ek.c    | Makefile       |
| at91sam926x_time.c    | board-dt.c          | board-sam9263ek.c    | Makefile.boot  |
| at91sam9_alt_reset.S  | board-eb01.c        | board-sam9g20ek.c    | pm.c           |
| at91sam9g45.c         | board-eb9200.c      | board-sam9-19260.c   | pm.h           |
| at91sam9g45_devices.c | board-ecbat91.c     | board-sam9m10g45ek.c | pm_slowclock.S |
| at91sam9g45_reset.S   | board-eco920.c      | board-sam9rlek.c     | sam9_smc.c     |
| at91sam9n12.c         | board-flexibity.c   | board-snapper9260.c  | sam9_smc.h     |
| at91sam9rl.c          | board-foxg20.c      | board-stamp9g20.c    | setup.c        |
| at91sam9rl_devices.c  | board-gsia18s.c     | board-usb-a926x.c    | soc.h          |
| at91sam9x5.c          | board-kafa.c        | board-y1-9200.c      |                |
| at91x40.c             | board-kb9202.c      | clock.c              |                |

Cold probing mechanism: overview





The board file registers devices (from arch/arm/mach-at91/board-snapper9260.c)

💫 The SoC file: device definition

```
static struct macb_platform_data eth_data;
static struct resource eth resources[] = {
        [0] = {
                .start = AT91SAM9260_BASE_EMAC,
                .end
                        = AT91SAM9260 BASE EMAC + SZ 16K - 1.
                .flags = IORESOURCE_MEM,
        }.
        [1] = {
                .start = AT91SAM9260 ID EMAC.
                .end
                        = AT91SAM9260_ID_EMAC,
                .flags = IORESOURCE IRQ.
        Ъ.
};
static struct platform device at91sam9260 eth device = {
                        = "macb",
        .name
        .id
                        = -1,
        dev
                        = {
                                                        = &eth_dmamask,
                                 .dma_mask
                                 .coherent_dma_mask
                                                         = DMA_BIT_MASK(32),
                                 .platform_data
                                                         = &eth data.
        Ъ.
                        = eth_resources,
        .resource
        .num resources = ARRAY SIZE(eth resources).
};
```

The SoC file: device registration

```
void init at91 add device eth(struct macb platform data *data)
       [...]
       if (gpio is valid(data->phv irg pin)) {
               at91_set_gpio_input(data->phy_irq_pin, 0);
               at91_set_deglitch(data->phy_irq_pin, 1);
       }
       /* Pins used for MII and RMII */
       at91_set_A_periph(AT91_PIN_PA19, 0);
                                             /* ETXCK_EREFCK */
       at91 set A periph(AT91 PIN PA17, 0);
                                            /* ERXDV */
       at91_set_A_periph(AT91_PIN_PA14, 0);
                                            /* ERXO */
       at91_set_A_periph(AT91_PIN_PA15, 0);
                                            /* ERX1 */
       at91_set_A_periph(AT91_PIN_PA18, 0);
                                            /* ERXER */
       at91_set_A_periph(AT91_PIN_PA16, 0);
                                            /* ETXEN */
       at91_set_A_periph(AT91_PIN_PA12, 0);
                                            /* ETXO */
       at91 set A periph(AT91 PIN PA13, 0);
                                            /* ETX1 */
       at91_set_A_periph(AT91_PIN_PA21, 0);
                                             /* EMDIO */
       at91_set_A_periph(AT91_PIN_PA20, 0);
                                             /* EMDC */
       if (!data->is rmii) {
             [...]
       3
       eth data = *data:
       platform_device_register(&at91sam9260_eth_device);
```

}

## The driver part

```
static int __init macb_probe(struct platform_device *pdev)
ſ
    [...]
3
static int __exit macb_remove(struct platform_device *pdev)
    [...]
}
static struct platform_driver macb_driver = {
                        = __exit_p(macb_remove),
        .remove
        .driver
                        = {
                        = "macb",
                .name
                .owner = THIS_MODULE,
        },
};
static int init macb init(void)
Ł
        return platform_driver_probe(&macb_driver, macb_probe);
}
static void __exit macb_exit(void)
ſ
        platform driver unregister(&macb driver):
3
module init(macb init):
module_exit(macb_exit);
```



- The general idea of the Device Tree is to separate a large part of the hardware description from the kernel sources
- ► The Device Tree is a **tree of nodes**, describing the different hardware components of a system and their characteristics
- Written in a specialized language, the Device Tree Source is compiled into a Device Tree Blob by the Device Tree Compiler
- This mechanism takes its roots from OpenFirmware used on some PowerPC platforms, and has been used on all PowerPC platforms for a long time.
- It is now also being used in other architectures in the Linux kernel such as Microblaze, OpenRISC and C6X.
- Inheritance mechanism: .dts files for boards, .dtsi for include files



#### /include/ "skeleton.dtsi"

```
/ {
    compatible = "nvidia,tegra20";
    interrupt-parent = <&intc>;
    intc: interrupt-controller {
        compatible = "arm, cortex-a9-gic";
        reg = <0x50041000 0x1000
                0x50040100 0x0100>:
        interrupt-controller;
        #interrupt-cells = <3>;
    }:
    serial@70006000 {
        compatible = "nvidia,tegra20-uart";
        reg = \langle 0x70006000 0x40 \rangle;
        reg-shift = <2>;
        interrupts = <0 36 0x04>:
        status = "disable":
    };
    serial@70006040 {
        compatible = "nvidia,tegra20-uart";
        reg = \langle 0x70006040 \ 0x40 \rangle;
        reg-shift = <2>;
        interrupts = <0 37 0x04>:
        status = "disable";
    }:
```

```
i2c@7000c000 {
    compatible = "nvidia,tegra20-i2c";
    reg = <0x7000c000 0x100>;
    interrupts = <0 38 0x04>;
    #address-cells = <1>;
    #size-cells = \langle 0 \rangle:
    status = "disable":
}:
i2c@7000c400 {
    compatible = "nvidia.tegra20-i2c":
    reg = <0x7000c400 0x100>;
    interrupts = <0 84 0x04>;
    #address-cells = <1>;
    #size-cells = <0>;
    status = "disable";
}:
usb@c5004000 {
    compatible = "nvidia,tegra20-ehci","usb-ehci";
    reg = \langle 0xc5004000 \ 0x4000 \rangle;
    interrupts = <0 21 0x04>;
```

```
phy_type = "ulpi";
status = "disable";
};
[...]
```

Device Tree: tegra-harmony.dts

```
/dts-v1/:
/include/ "tegra20.dtsi"
/ {
    model = "NVIDIA Tegra2 Harmony evaluation board";
    compatible = "nvidia.harmony", "nvidia.tegra20";
    memory {
        reg = <0x00000000 0x40000000>;
    1:
    serial@70006300 {
                                                                };
                                                            }:
        status = "okay";
        clock-frequency = <216000000>;
    };
    i2c@7000c400 {
        status = "okay";
                                                            };
        clock-frequency = <400000>;
    }:
                                                            [...]
                                                        };
```

```
i2c@7000c000 {
    status = "okay":
    clock-frequency = <400000>;
    wm8903: wm8903@1a {
        compatible = "wlf.wm8903":
        reg = \langle 0x1a \rangle;
        interrupt-parent = <&gpio>:
        interrupts = <187 \text{ } 0x04>:
        gpio-controller;
        #gpio-cells = <2>;
         [...]
usb@c5004000 {
    status = "okay":
    nvidia,phy-reset-gpio = <&gpio 169 0>;
```



- When the .dts file is in arch/arm/boot/dts, as simple as: make ARCH=arm foobar.dtb
- Then, on ARM, two cases:
  - 1. Your bootloader has DT support.

You need to load both your kernel image and DT image, and start the kernel with both addresses. The DTB address is passed to the kernel in register  $r_2$ , instead of the *ATAG* address. With U-Boot:

bootm kerneladdr - dtbaddr

#### 2. Your bootloader does not have DT support.

ARM has a special CONFIG\_ARM\_APPENDED\_DTB option that allows to append the zImage directly with the dtb. This is provided for debugging only, bootloaders are expected to provide DT support.



- Only one "board" file is needed per SoC
- Uses DT\_MACHINE\_START instead of MACHINE\_START
- Provides a dt\_compat table to list the platforms compatible with this definition
- of\_platform\_populate will instantiate the devices
- From arch/arm/mach-tegra/board-dt-tegra20.c:

```
static void __init tegra_dt_init(void)
ſ
        [...]
        of_platform_populate(NULL, tegra_dt_match_table,
                                tegra20 auxdata lookup, NULL);
}
static const char *tegra20 dt board compat[] = {
        "nvidia.tegra20".
        NULL.
};
DT_MACHINE_START(TEGRA_DT, "nVidia Tegra20 (Flattened Device Tree)")
        .init_machine
                        = tegra_dt_init,
        .dt_compat
                        = tegra20 dt board compat.
        [...]
MACHINE_END
```



```
static const struct of device id tegra i2c of match devinitconst = {
        { .compatible = "nvidia,tegra20-i2c", },
        { .compatible = "nvidia,tegra20-i2c-dvc", },
        £.
};
MODULE_DEVICE_TABLE(of, tegra_i2c_of_match);
static struct platform driver tegra i2c driver = {
        .probe = tegra_i2c_probe,
        .remove = __devexit_p(tegra_i2c_remove),
        driver = {
                .name = "tegra-i2c",
                .owner = THIS_MODULE,
                .of match table = tegra i2c of match.
        }.
};
static int init tegra i2c init driver(void)
        return platform driver register(&tegra i2c driver);
3
static void exit tegra i2c exit driver(void)
Ł
        platform_driver_unregister(&tegra_i2c_driver);
3
subsys_initcall(tegra_i2c_init_driver);
module_exit(tegra_i2c_exit_driver);
```



- With the Device Tree, each driver defines:
  - its compatible string, which uniquely identifies the driver, and allows devices to be bound to the corresponding driver
  - the properties of each device in the device tree
- This definition is called a device tree binding
- All device tree bindings are normally documented in Documentation/devicetree/bindings.

Device Tree: summary





- The different hardware parts of an SoC are driven by different clocks, operating at different frequencies
- Most of those clocks are part of a complex clock tree, where parents clocks are inputs to children clocks
- Many of those clocks are software configurable (on/off, multiple frequencies, etc.) and must be manipulated at runtime for power management reasons
- Due to the parent/child relationship, one must ensure that
  - 1. The parent clock is enabled when a child clock needs to be enabled
  - 2. The parent clock is disabled once all children have been disabled

The old clock management infrastructure

- Clocks need to be manipulated by *device drivers*: they know when to enable/disable the needed clocks
- Clocks are listed and controlled by SoC code
- Since some time, a common clock API has been defined in <linux/clk.h>, defining
  - 1. An opaque struct clk structure, that drivers could manipulate
  - A simple clk\_get, clk\_put, clk\_enable, clk\_disable, clk\_get\_rate API
- Each ARM sub-architecture had to have its own definition of struct clk and its own implementation of the API
  - A lot of code duplication
  - No common facilities, even though most clocks are relatively similar between SoC



### The new clock framework

- A proper *clock framework* has been added in kernel 3.4, released in May 2012
- Initially from Jeremy Kerr (Canonical), finally implemented and merged by Mike Turquette (Texas Instruments)
- This framework:
  - Implements the clk\_get, clk\_put, clk\_prepare, clk\_unprepare, clk\_enable, clk\_disable, clk\_get\_rate, etc. API for usage by device drivers
  - Provides data structures (struct clk\_hw and struct clkops) for SoC code to define its clocks, and a clk\_register, clk\_unregister API to register them, and clk\_register\_clkdevs to associate clocks to device names
  - Implements some basic clock types (fixed rate, gatable, divider, fixed factor, etc.)
  - Provides a *debugfs* representation of the clock tree
  - Is implemented in drivers/clk

### Clock framework, the driver side

From drivers/serial/tty/mxs-auart.c, the UART driver for i.MX23/28 SoCs.

```
static int mxs_auart_startup(struct uart_port *u)
        [...]
        clk_prepare_enable(s->clk);
        ſ...1
3
static void mxs_auart_shutdown(struct uart_port *u)
ſ
         [...]
         clk_disable_unprepare(s->clk);
}
static int __devinit mxs_auart_probe(struct platform_device *pdev)
Ł
        [...]
        s->clk = clk_get(&pdev->dev, NULL);
        [...]
        s->port.uartclk = clk_get_rate(s->clk);
        [...]
}
static int __devexit mxs_auart_remove(struct platform_device *pdev)
        [...]
        clk_put(s->clk);
        [...]
}
```

```
Clock framework, SoC side
From drivers/clk/mxs/clk-imx28.c
static struct clk_lookup uart_lookups[] __initdata = {
       { .dev_id = "duart", },
       \{ .dev id = "mxs-auart.0", \}.
       [...]
       { .dev_id = "8006a000.serial". }.
       [...]
}:
static struct clk *clks[clk max];
int __init mx28_clocks_init(void)
       [...]
       clks[ref_xtal] = mxs_clk_fixed("ref_xtal", 24000000);
       clks[pl10] = mxs_clk_pl1("pl10", "ref_xtal", PLL0CTRL0, 17, 480000000);
       [...]
       clks[ref_cpu] = mxs_clk_ref("ref_cpu", "pll0", FRACO, 0);
       clks[ref_emi] = mxs_clk_ref("ref_emi", "pll0", FRACO, 1);
       [...]
       clks[gpmi sel] = mxs clk mux("gpmi sel", CLKSEO, 2, 1, sel gpmi, ARRAY SIZE(sel gpmi));
       clks[ssp0_sel] = mxs_clk_mux("ssp0_sel", CLKSEQ, 3, 1, sel_io0, ARRAY_SIZE(sel_io0));
       clks[ssp1 sel] = mxs clk mux("ssp1 sel", CLKSEQ, 4, 1, sel io0, ARRAY SIZE(sel io0)):
       clks[ssp2 sel] = mxs clk mux("ssp2 sel", CLKSEQ, 5, 1, sel io1, ARRAY SIZE(sel io1));
       [...]
       clks[uart] = mxs_clk_gate("uart", "ref_xtal", XTAL, 31);
       [...]
       clk_register_clkdevs(clks[uart], uart_lookups, ARRAY_SIZE(uart_lookups));
       [...]
3
```



# Clock framework, SoC side

```
mxs_clk_fixed()
```

Registers a fixed-rate clock, using the clk-fixed clock type provided by the base clock framework in drivers/clk/clk-fixed.c

```
mxs_clk_ref()
```

Registers a reference clock, using the clk-ref clock type specific to i.MX, implemented in drivers/clk/mxs/clk-ref.c

```
mxs_clk_pll()
Registers a PLL clock, using the clk-pll clock type specific
to i.MX, implemented in drivers/clk/mxs/clk-pll.c
```

```
mxs_clk_mux()
Registers a muxed clock, using the clk-mux clock type
provided by the base clock framework in
drivers/clk/clk-mux.c
```



# cd /sys/kernel/debug/clk # find ./ref xtal ./ref\_xtal/pll0 ./ref\_xtal/pll0/ref\_io1 ./ref\_xtal/pl10/ref\_io1/ssp2\_sel ./ref\_xtal/pl10/ref\_io1/ssp2\_sel/ssp2\_div ./ref\_xtal/pll0/ref\_io1/ssp2\_sel/ssp2\_div/ssp2 ./ref\_xtal/pll0/ref\_io1/ssp2\_sel/ssp2\_div/ssp2/clk\_notifier\_count ./ref\_xtal/pll0/ref\_io1/ssp2\_sel/ssp2\_div/ssp2/clk\_enable\_count ./ref\_xtal/pll0/ref\_io1/ssp2\_sel/ssp2\_div/ssp2/clk\_prepare\_count ./ref\_xtal/pl10/ref\_io1/ssp2\_sel/ssp2\_div/ssp2/clk\_flags ./ref\_xtal/pl10/ref\_io1/ssp2\_sel/ssp2\_div/ssp2/clk\_rate ./ref\_xtal/pll0/ref\_io1/ssp2\_sel/ssp2\_div/clk\_notifier\_count ./ref\_xtal/pll0/ref\_io1/ssp2\_sel/ssp2\_div/clk\_enable\_count ./ref\_xtal/pll0/ref\_io1/ssp2\_sel/ssp2\_div/clk\_prepare\_count ./ref\_xtal/pll0/ref\_io1/ssp2\_sel/ssp2\_div/clk\_flags ./ref\_xtal/pll0/ref\_io1/ssp2\_sel/ssp2\_div/clk\_rate [...]

# Clock framework: summary





Introduction to pin muxing

- SoCs integrate many more peripherals than the number of available pins allows to expose.
- Many of those pins are therefore multiplexed: they can either be used as function A, or function B, or function C, or a GPIO
- Example of functions are:
  - parallel LCD lines
  - SDA/SCL lines for I2C busses
  - MISO/MOSI/CLK lines for SPI
  - ► RX/TX/CTS/DTS lines for UARTs
- This muxing is software-configurable, and depends on how the SoC is used on each particular board





Pin muxing: example



#### 8.4.2 PIO Controller B Multiplexing

Table 8-3. Multiplexing on PIO Controller B (PIOB)

| I/O Line | Peripheral A | Peripheral B | Reset<br>State | Power<br>Supply | Function | Comments |
|----------|--------------|--------------|----------------|-----------------|----------|----------|
| PB0      | SPI0_MISO    |              | I/O            | VDDIOP0         |          |          |
| PB1      | SPI0_MOSI    |              | I/O            | VDDIOP0         |          |          |
| PB2      | SPI0_SPCK    |              | I/O            | VDDIOP0         |          |          |
| PB3      | SPI0_NPCS0   |              | I/O            | VDDIOP0         |          |          |
| PB4      | TXD1         |              | I/O            | VDDIOP0         |          |          |
| PB5      | RXD1         |              | I/O            | VDDIOP0         |          |          |
| PB6      | TXD2         |              | I/O            | VDDIOP0         |          |          |
| PB7      | RXD2         |              | I/O            | VDDIOP0         |          |          |
| PB8      | TXD3         | ISI_D8       | I/O            | VDDIOP2         |          |          |
| PB9      | RXD3         | ISI_D9       | I/O            | VDDIOP2         |          |          |
| PB10     | TWD1         | ISI_D10      | I/O            | VDDIOP2         |          |          |
| PB11     | TWCK1        | ISI_D11      | I/O            | VDDIOP2         |          |          |
| PB12     | DRXD         |              | 1/0            | VDDIOP0         |          |          |
| PB13     | DTXD         |              | I/O            | VDDIOP0         |          |          |

Free Electrons. Kernel, drivers and embedded Linux development, consulting, training and support. http://free-electrons.com



- Each ARM sub-architecture had its own pin-muxing code
- The API was specific to each sub-architecture
- Lot of similar functionality implemented in different ways
- The pin-muxing had to be done at the SoC level, and couldn't be requested by device drivers



- The new pinctrl subsystem aims at solving those problems
- Mainly developed and maintained by Linus Walleij, from Linaro/ST-Ericsson
- Implemented in drivers/pinctrl
- Provides:
  - An API to register *pinctrl driver*, i.e entities knowing the list of pins, their functions, and how to configure them. Used by SoC-specific drivers to expose pin-muxing capabilities.
  - An API for *device drivers* to request the muxing of a certain set of pins
  - An interaction with the GPIO framework

# The new pin-muxing subsystem: diagram



# Declaring pin groups in the SoC dtsi

- From arch/arm/boot/dts/imx28.dtsi
- Declares the *pinctrl* device and various pin groups

```
pinctrl@80018000 {
   #address-cells = <1>;
   #size-cells = <0>;
   compatible = "fsl,imx28-pinctrl", "simple-bus";
   reg = <0x80018000 2000>;
   duart pins a: duart@0 {
            reg = \langle 0 \rangle;
            fsl.pinmux-ids = <0x3102 0x3112>:
            fsl,drive-strength = <0>;
            fsl,voltage = <1>;
            fsl,pull-up = <0>;
   }:
   duart_pins_b: duart@1 {
            reg = <1>;
            fsl.pinmux-ids = <0x3022 0x3032>:
            fsl.drive-strength = <0>:
            fsl,voltage = <1>;
            fsl.pull-up = \langle 0 \rangle;
   }:
```

```
mmc0_8bit_pins_a: mmc0-8bit@0 {
         reg = \langle 0 \rangle;
         fsl,pinmux-ids = <0x2000 0x2010 0x2020
                  0x2030 0x2040 0x2050 0x2060
                  0x2070 0x2080 0x2090 0x20a0>:
         fsl,drive-strength = <1>;
         fsl,voltage = <1>;
         fsl.pull-up = \langle 1 \rangle:
}:
mmc0_4bit_pins_a: mmc0-4bit@0 {
         reg = \langle 0 \rangle;
         fsl,pinmux-ids = <0x2000 0x2010 0x2020</pre>
                  0x2030 0x2080 0x2090 0x20a0>;
         fsl,drive-strength = <1>;
         fsl.voltage = <1>;
         fsl,pull-up = \langle 1 \rangle;
};
mmcO_cd_cfg: mmcO-cd-cfg {
         fsl,pinmux-ids = <0x2090>;
         fsl,pull-up = \langle 0 \rangle;
}:
mmc0 sck cfg: mmc0-sck-cfg {
         fsl,pinmux-ids = <0x20a0>;
         fsl.drive-strength = <2>:
         fsl.pull-up = \langle 0 \rangle:
};
```

**}**:

Associating devices with pin groups, board dts

From arch/arm/boot/dts/cfa10036.dts

```
apb@80000000 {
  apbh@80000000 {
     ssp0: ssp@80010000 {
         compatible = "fsl,imx28-mmc";
         pinctrl-names = "default";
         pinctrl-0 = <&mmc0_4bit_pins_a</pre>
                        &mmc0_cd_cfg &mmc0_sck_cfg>;
         bus-width = \langle 4 \rangle:
         status = "okay";
     };
  };
  apbx@80040000 {
     duart: serial@80074000 {
         pinctrl-names = "default":
         pinctrl-0 = <&duart_pins_b>;
         status = "okay":
     };
 };
}:
```

Device drivers requesting pin muxing

From drivers/mmc/host/mxs-mmc.c

```
static int mxs_mmc_probe(struct platform_device *pdev)
{
    [...]
    pinctrl = devm_pinctrl_get_select_default(&pdev->dev);
    if (IS_ERR(pinctrl)) {
        ret = PTR_ERR(pinctrl);
        goto out_mmc_free;
    }
    [...]
}
```



- About ARM maintenance
  - ARM Subarchitecture Status, Arnd Bergmann, ELC 2012
- About the Device Tree
  - Official Wiki, http://devicetree.org
  - Experiences With Device Tree Support Development For ARM-Based SOC's, Thomas P. Abraham, ELC 2012
  - Device Tree Status Report, Grant Likely, ELCE 2011.
- About the clock framework
  - Documentation/clk.txt in the kernel sources
  - http://lwn.net/Articles/472998/
  - Common Clock Framework, Mike Turquette, ELC2012
- About the pinctrl subsystem
  - Documentation/pinctrl.txt in the kernel sources
  - http://lwn.net/Articles/468759/
  - Pin Control Subsystem Overview, Linus Walleij, ELC2012
- Slides and video at
  - http://free-electrons.com/blog/elc-2012-videos/ for Embedded Linux Conference 2012
  - http://free-electrons.com/blog/elce-2011-videos/ for Embedded Linux Conference Europe 2011



- The *pinctrl* and *clk* subsystems now provide generic abstractions to manage pin muxing and the clocks of an SoC
- The device tree provides a better way of representing the hardware, requiring less code to describe new platforms
- The usage of these new infrastructures is mandatory for new platforms
- Conversion of existing platforms that are widely used is in process
- The ARM community has gained better code infrastructures, a better organization, and has become even more dynamic than it was.

# Questions?



### Thomas Petazzoni

#### thomas.petazzoni@free-electrons.com Slides under CC-BY-SA 3.0.