Quellcodebibliothek Statistik Leitseite products/Sources/formale Sprachen/C/Linux/drivers/clk/mmp/   (Open Source Betriebssystem Version 6.17.9©)  Datei vom 24.10.2025 mit Größe 11 kB image not shown  

Quelle  clk-of-pxa910.c   Sprache: C

 
// SPDX-License-Identifier: GPL-2.0-only
/*
 * pxa910 clock framework source file
 *
 * Copyright (C) 2012 Marvell
 * Chao Xie <xiechao.mail@gmail.com>
 */


#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/spinlock.h>
#include <linux/io.h>
#include <linux/delay.h>
#include <linux/err.h>
#include <linux/of_address.h>

#include <dt-bindings/clock/marvell,pxa910.h>

#include "clk.h"
#include "reset.h"

#define APBC_RTC 0x28
#define APBC_TWSI0 0x2c
#define APBC_KPC 0x18
#define APBC_UART0 0x0
#define APBC_UART1 0x4
#define APBC_GPIO 0x8
#define APBC_PWM0 0xc
#define APBC_PWM1 0x10
#define APBC_PWM2 0x14
#define APBC_PWM3 0x18
#define APBC_SSP0 0x1c
#define APBC_SSP1 0x20
#define APBC_SSP2 0x4c
#define APBC_TIMER0 0x30
#define APBC_TIMER1 0x44
#define APBCP_TWSI1 0x28
#define APBCP_UART2 0x1c
#define APMU_SDH0 0x54
#define APMU_SDH1 0x58
#define APMU_USB 0x5c
#define APMU_DISP0 0x4c
#define APMU_CCIC0 0x50
#define APMU_DFC 0x60
#define MPMU_UART_PLL 0x14

#define NR_CLKS  200

struct pxa910_clk_unit {
 struct mmp_clk_unit unit;
 void __iomem *mpmu_base;
 void __iomem *apmu_base;
 void __iomem *apbc_base;
 void __iomem *apbcp_base;
};

static struct mmp_param_fixed_rate_clk fixed_rate_clks[] = {
 {PXA910_CLK_CLK32, "clk32", NULL, 0, 32768},
 {PXA910_CLK_VCTCXO, "vctcxo", NULL, 0, 26000000},
 {PXA910_CLK_PLL1, "pll1", NULL, 0, 624000000},
 {PXA910_CLK_USB_PLL, "usb_pll", NULL, 0, 480000000},
};

static struct mmp_param_fixed_factor_clk fixed_factor_clks[] = {
 {PXA910_CLK_PLL1_2, "pll1_2""pll1", 1, 2, 0},
 {PXA910_CLK_PLL1_4, "pll1_4""pll1_2", 1, 2, 0},
 {PXA910_CLK_PLL1_8, "pll1_8""pll1_4", 1, 2, 0},
 {PXA910_CLK_PLL1_16, "pll1_16""pll1_8", 1, 2, 0},
 {PXA910_CLK_PLL1_6, "pll1_6""pll1_2", 1, 3, 0},
 {PXA910_CLK_PLL1_12, "pll1_12""pll1_6", 1, 2, 0},
 {PXA910_CLK_PLL1_24, "pll1_24""pll1_12", 1, 2, 0},
 {PXA910_CLK_PLL1_48, "pll1_48""pll1_24", 1, 2, 0},
 {PXA910_CLK_PLL1_96, "pll1_96""pll1_48", 1, 2, 0},
 {PXA910_CLK_PLL1_192, "pll1_192""pll1_96", 1, 2, 0},
 {PXA910_CLK_PLL1_13, "pll1_13""pll1", 1, 13, 0},
 {PXA910_CLK_PLL1_13_1_5, "pll1_13_1_5""pll1_13", 2, 3, 0},
 {PXA910_CLK_PLL1_2_1_5, "pll1_2_1_5""pll1_2", 2, 3, 0},
 {PXA910_CLK_PLL1_3_16, "pll1_3_16""pll1", 3, 16, 0},
};

static struct mmp_clk_factor_masks uart_factor_masks = {
 .factor = 2,
 .num_mask = 0x1fff,
 .den_mask = 0x1fff,
 .num_shift = 16,
 .den_shift = 0,
};

static struct u32_fract uart_factor_tbl[] = {
 { .numerator = 8125, .denominator = 1536 }, /* 14.745MHZ */
};

static void pxa910_pll_init(struct pxa910_clk_unit *pxa_unit)
{
 struct clk *clk;
 struct mmp_clk_unit *unit = &pxa_unit->unit;

 mmp_register_fixed_rate_clks(unit, fixed_rate_clks,
     ARRAY_SIZE(fixed_rate_clks));

 mmp_register_fixed_factor_clks(unit, fixed_factor_clks,
     ARRAY_SIZE(fixed_factor_clks));

 clk = mmp_clk_register_factor("uart_pll""pll1_4",
    CLK_SET_RATE_PARENT,
    pxa_unit->mpmu_base + MPMU_UART_PLL,
    &uart_factor_masks, uart_factor_tbl,
    ARRAY_SIZE(uart_factor_tbl), NULL);
 mmp_clk_add(unit, PXA910_CLK_UART_PLL, clk);
}

static DEFINE_SPINLOCK(uart0_lock);
static DEFINE_SPINLOCK(uart1_lock);
static DEFINE_SPINLOCK(uart2_lock);
static const char *uart_parent_names[] = {"pll1_3_16""uart_pll"};

static DEFINE_SPINLOCK(ssp0_lock);
static DEFINE_SPINLOCK(ssp1_lock);
static const char *ssp_parent_names[] = {"pll1_96""pll1_48""pll1_24""pll1_12"};

static DEFINE_SPINLOCK(timer0_lock);
static DEFINE_SPINLOCK(timer1_lock);
static const char *timer_parent_names[] = {"pll1_48""clk32""pll1_96"};

static DEFINE_SPINLOCK(reset_lock);

static struct mmp_param_mux_clk apbc_mux_clks[] = {
 {0, "uart0_mux", uart_parent_names, ARRAY_SIZE(uart_parent_names), CLK_SET_RATE_PARENT, APBC_UART0, 4, 3, 0, &uart0_lock},
 {0, "uart1_mux", uart_parent_names, ARRAY_SIZE(uart_parent_names), CLK_SET_RATE_PARENT, APBC_UART1, 4, 3, 0, &uart1_lock},
 {0, "ssp0_mux", ssp_parent_names, ARRAY_SIZE(ssp_parent_names), CLK_SET_RATE_PARENT, APBC_SSP0, 4, 3, 0, &ssp0_lock},
 {0, "ssp1_mux", ssp_parent_names, ARRAY_SIZE(ssp_parent_names), CLK_SET_RATE_PARENT, APBC_SSP1, 4, 3, 0, &ssp1_lock},
 {0, "timer0_mux", timer_parent_names, ARRAY_SIZE(timer_parent_names), CLK_SET_RATE_PARENT, APBC_TIMER0, 4, 3, 0, &timer0_lock},
 {0, "timer1_mux", timer_parent_names, ARRAY_SIZE(timer_parent_names), CLK_SET_RATE_PARENT, APBC_TIMER1, 4, 3, 0, &timer1_lock},
};

static struct mmp_param_mux_clk apbcp_mux_clks[] = {
 {0, "uart2_mux", uart_parent_names, ARRAY_SIZE(uart_parent_names), CLK_SET_RATE_PARENT, APBCP_UART2, 4, 3, 0, &uart2_lock},
};

static struct mmp_param_gate_clk apbc_gate_clks[] = {
 {PXA910_CLK_TWSI0, "twsi0_clk""pll1_13_1_5", CLK_SET_RATE_PARENT, APBC_TWSI0, 0x3, 0x3, 0x0, 0, &reset_lock},
 {PXA910_CLK_GPIO, "gpio_clk""vctcxo", CLK_SET_RATE_PARENT, APBC_GPIO, 0x3, 0x3, 0x0, 0, &reset_lock},
 {PXA910_CLK_KPC, "kpc_clk""clk32", CLK_SET_RATE_PARENT, APBC_KPC, 0x3, 0x3, 0x0, MMP_CLK_GATE_NEED_DELAY, NULL},
 {PXA910_CLK_RTC, "rtc_clk""clk32", CLK_SET_RATE_PARENT, APBC_RTC, 0x83, 0x83, 0x0, MMP_CLK_GATE_NEED_DELAY, NULL},
 {PXA910_CLK_PWM0, "pwm0_clk""pll1_48", CLK_SET_RATE_PARENT, APBC_PWM0, 0x3, 0x3, 0x0, 0, &reset_lock},
 {PXA910_CLK_PWM1, "pwm1_clk""pll1_48", CLK_SET_RATE_PARENT, APBC_PWM1, 0x3, 0x3, 0x0, 0, &reset_lock},
 {PXA910_CLK_PWM2, "pwm2_clk""pll1_48", CLK_SET_RATE_PARENT, APBC_PWM2, 0x3, 0x3, 0x0, 0, &reset_lock},
 {PXA910_CLK_PWM3, "pwm3_clk""pll1_48", CLK_SET_RATE_PARENT, APBC_PWM3, 0x3, 0x3, 0x0, 0, &reset_lock},
 /* The gate clocks has mux parent. */
 {PXA910_CLK_UART0, "uart0_clk""uart0_mux", CLK_SET_RATE_PARENT, APBC_UART0, 0x3, 0x3, 0x0, 0, &uart0_lock},
 {PXA910_CLK_UART1, "uart1_clk""uart1_mux", CLK_SET_RATE_PARENT, APBC_UART1, 0x3, 0x3, 0x0, 0, &uart1_lock},
 {PXA910_CLK_SSP0, "ssp0_clk""ssp0_mux", CLK_SET_RATE_PARENT, APBC_SSP0, 0x3, 0x3, 0x0, 0, &ssp0_lock},
 {PXA910_CLK_SSP1, "ssp1_clk""ssp1_mux", CLK_SET_RATE_PARENT, APBC_SSP1, 0x3, 0x3, 0x0, 0, &ssp1_lock},
 {PXA910_CLK_TIMER0, "timer0_clk""timer0_mux", CLK_SET_RATE_PARENT, APBC_TIMER0, 0x3, 0x3, 0x0, 0, &timer0_lock},
 {PXA910_CLK_TIMER1, "timer1_clk""timer1_mux", CLK_SET_RATE_PARENT, APBC_TIMER1, 0x3, 0x3, 0x0, 0, &timer1_lock},
};

static struct mmp_param_gate_clk apbcp_gate_clks[] = {
 {PXA910_CLK_TWSI1, "twsi1_clk""pll1_13_1_5", CLK_SET_RATE_PARENT, APBCP_TWSI1, 0x3, 0x3, 0x0, 0, &reset_lock},
 /* The gate clocks has mux parent. */
 {PXA910_CLK_UART2, "uart2_clk""uart2_mux", CLK_SET_RATE_PARENT, APBCP_UART2, 0x3, 0x3, 0x0, 0, &uart2_lock},
};

static void pxa910_apb_periph_clk_init(struct pxa910_clk_unit *pxa_unit)
{
 struct mmp_clk_unit *unit = &pxa_unit->unit;

 mmp_register_mux_clks(unit, apbc_mux_clks, pxa_unit->apbc_base,
    ARRAY_SIZE(apbc_mux_clks));

 mmp_register_mux_clks(unit, apbcp_mux_clks, pxa_unit->apbcp_base,
    ARRAY_SIZE(apbcp_mux_clks));

 mmp_register_gate_clks(unit, apbc_gate_clks, pxa_unit->apbc_base,
    ARRAY_SIZE(apbc_gate_clks));

 mmp_register_gate_clks(unit, apbcp_gate_clks, pxa_unit->apbcp_base,
    ARRAY_SIZE(apbcp_gate_clks));
}

static DEFINE_SPINLOCK(sdh0_lock);
static DEFINE_SPINLOCK(sdh1_lock);
static const char *sdh_parent_names[] = {"pll1_12""pll1_13"};

static DEFINE_SPINLOCK(usb_lock);

static DEFINE_SPINLOCK(disp0_lock);
static const char *disp_parent_names[] = {"pll1_2""pll1_12"};

static DEFINE_SPINLOCK(ccic0_lock);
static const char *ccic_parent_names[] = {"pll1_2""pll1_12"};
static const char *ccic_phy_parent_names[] = {"pll1_6""pll1_12"};

static struct mmp_param_mux_clk apmu_mux_clks[] = {
 {0, "sdh0_mux", sdh_parent_names, ARRAY_SIZE(sdh_parent_names), CLK_SET_RATE_PARENT, APMU_SDH0, 6, 1, 0, &sdh0_lock},
 {0, "sdh1_mux", sdh_parent_names, ARRAY_SIZE(sdh_parent_names), CLK_SET_RATE_PARENT, APMU_SDH1, 6, 1, 0, &sdh1_lock},
 {0, "disp0_mux", disp_parent_names, ARRAY_SIZE(disp_parent_names), CLK_SET_RATE_PARENT, APMU_DISP0, 6, 1, 0, &disp0_lock},
 {0, "ccic0_mux", ccic_parent_names, ARRAY_SIZE(ccic_parent_names), CLK_SET_RATE_PARENT, APMU_CCIC0, 6, 1, 0, &ccic0_lock},
 {0, "ccic0_phy_mux", ccic_phy_parent_names, ARRAY_SIZE(ccic_phy_parent_names), CLK_SET_RATE_PARENT, APMU_CCIC0, 7, 1, 0, &ccic0_lock},
};

static struct mmp_param_div_clk apmu_div_clks[] = {
 {0, "ccic0_sphy_div""ccic0_mux", CLK_SET_RATE_PARENT, APMU_CCIC0, 10, 5, 0, &ccic0_lock},
};

static struct mmp_param_gate_clk apmu_gate_clks[] = {
 {PXA910_CLK_DFC, "dfc_clk""pll1_4", CLK_SET_RATE_PARENT, APMU_DFC, 0x19b, 0x19b, 0x0, 0, NULL},
 {PXA910_CLK_USB, "usb_clk""usb_pll", 0, APMU_USB, 0x9, 0x9, 0x0, 0, &usb_lock},
 {PXA910_CLK_SPH, "sph_clk""usb_pll", 0, APMU_USB, 0x12, 0x12, 0x0, 0, &usb_lock},
 /* The gate clocks has mux parent. */
 {PXA910_CLK_SDH0, "sdh0_clk""sdh0_mux", CLK_SET_RATE_PARENT, APMU_SDH0, 0x1b, 0x1b, 0x0, 0, &sdh0_lock},
 {PXA910_CLK_SDH1, "sdh1_clk""sdh1_mux", CLK_SET_RATE_PARENT, APMU_SDH1, 0x1b, 0x1b, 0x0, 0, &sdh1_lock},
 {PXA910_CLK_DISP0, "disp0_clk""disp0_mux", CLK_SET_RATE_PARENT, APMU_DISP0, 0x1b, 0x1b, 0x0, 0, &disp0_lock},
 {PXA910_CLK_CCIC0, "ccic0_clk""ccic0_mux", CLK_SET_RATE_PARENT, APMU_CCIC0, 0x1b, 0x1b, 0x0, 0, &ccic0_lock},
 {PXA910_CLK_CCIC0_PHY, "ccic0_phy_clk""ccic0_phy_mux", CLK_SET_RATE_PARENT, APMU_CCIC0, 0x24, 0x24, 0x0, 0, &ccic0_lock},
 {PXA910_CLK_CCIC0_SPHY, "ccic0_sphy_clk""ccic0_sphy_div", CLK_SET_RATE_PARENT, APMU_CCIC0, 0x300, 0x300, 0x0, 0, &ccic0_lock},
};

static void pxa910_axi_periph_clk_init(struct pxa910_clk_unit *pxa_unit)
{
 struct mmp_clk_unit *unit = &pxa_unit->unit;

 mmp_register_mux_clks(unit, apmu_mux_clks, pxa_unit->apmu_base,
    ARRAY_SIZE(apmu_mux_clks));

 mmp_register_div_clks(unit, apmu_div_clks, pxa_unit->apmu_base,
    ARRAY_SIZE(apmu_div_clks));

 mmp_register_gate_clks(unit, apmu_gate_clks, pxa_unit->apmu_base,
    ARRAY_SIZE(apmu_gate_clks));
}

static void pxa910_clk_reset_init(struct device_node *np,
    struct pxa910_clk_unit *pxa_unit)
{
 struct mmp_clk_reset_cell *cells;
 int i, base, nr_resets_apbc, nr_resets_apbcp, nr_resets;

 nr_resets_apbc = ARRAY_SIZE(apbc_gate_clks);
 nr_resets_apbcp = ARRAY_SIZE(apbcp_gate_clks);
 nr_resets = nr_resets_apbc + nr_resets_apbcp;
 cells = kcalloc(nr_resets, sizeof(*cells), GFP_KERNEL);
 if (!cells)
  return;

 base = 0;
 for (i = 0; i < nr_resets_apbc; i++) {
  cells[base + i].clk_id = apbc_gate_clks[i].id;
  cells[base + i].reg =
   pxa_unit->apbc_base + apbc_gate_clks[i].offset;
  cells[base + i].flags = 0;
  cells[base + i].lock = apbc_gate_clks[i].lock;
  cells[base + i].bits = 0x4;
 }

 base = nr_resets_apbc;
 for (i = 0; i < nr_resets_apbcp; i++) {
  cells[base + i].clk_id = apbcp_gate_clks[i].id;
  cells[base + i].reg =
   pxa_unit->apbc_base + apbc_gate_clks[i].offset;
  cells[base + i].flags = 0;
  cells[base + i].lock = apbc_gate_clks[i].lock;
  cells[base + i].bits = 0x4;
 }

 mmp_clk_reset_register(np, cells, nr_resets);
}

static void __init pxa910_clk_init(struct device_node *np)
{
 struct pxa910_clk_unit *pxa_unit;

 pxa_unit = kzalloc(sizeof(*pxa_unit), GFP_KERNEL);
 if (!pxa_unit)
  return;

 pxa_unit->mpmu_base = of_iomap(np, 0);
 if (!pxa_unit->mpmu_base) {
  pr_err("failed to map mpmu registers\n");
  goto free_memory;
 }

 pxa_unit->apmu_base = of_iomap(np, 1);
 if (!pxa_unit->apmu_base) {
  pr_err("failed to map apmu registers\n");
  goto unmap_mpmu_region;
 }

 pxa_unit->apbc_base = of_iomap(np, 2);
 if (!pxa_unit->apbc_base) {
  pr_err("failed to map apbc registers\n");
  goto unmap_apmu_region;
 }

 pxa_unit->apbcp_base = of_iomap(np, 3);
 if (!pxa_unit->apbcp_base) {
  pr_err("failed to map apbcp registers\n");
  goto unmap_apbc_region;
 }

 mmp_clk_init(np, &pxa_unit->unit, NR_CLKS);

 pxa910_pll_init(pxa_unit);

 pxa910_apb_periph_clk_init(pxa_unit);

 pxa910_axi_periph_clk_init(pxa_unit);

 pxa910_clk_reset_init(np, pxa_unit);

 return;

unmap_apbc_region:
 iounmap(pxa_unit->apbc_base);
unmap_apmu_region:
 iounmap(pxa_unit->apmu_base);
unmap_mpmu_region:
 iounmap(pxa_unit->mpmu_base);
free_memory:
 kfree(pxa_unit);
}

CLK_OF_DECLARE(pxa910_clk, "marvell,pxa910-clock", pxa910_clk_init);

Messung V0.5
C=99 H=94 G=96

¤ Dauer der Verarbeitung: 0.5 Sekunden  ¤

*© Formatika GbR, Deutschland






Wurzel

Suchen

Beweissystem der NASA

Beweissystem Isabelle

NIST Cobol Testsuite

Cephes Mathematical Library

Wiener Entwicklungsmethode

Haftungshinweis

Die Informationen auf dieser Webseite wurden nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit, noch Qualität der bereit gestellten Informationen zugesichert.

Bemerkung:

Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.