linux/drivers/clk/tegra/clk.c
<<
>>
Prefs
   1/*
   2 * Copyright (c) 2012, NVIDIA CORPORATION.  All rights reserved.
   3 *
   4 * This program is free software; you can redistribute it and/or modify it
   5 * under the terms and conditions of the GNU General Public License,
   6 * version 2, as published by the Free Software Foundation.
   7 *
   8 * This program is distributed in the hope it will be useful, but WITHOUT
   9 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  10 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
  11 * more details.
  12 *
  13 * You should have received a copy of the GNU General Public License
  14 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
  15 */
  16
  17#include <linux/clk.h>
  18#include <linux/clk-provider.h>
  19#include <linux/of.h>
  20#include <linux/clk/tegra.h>
  21
  22#include "clk.h"
  23
  24/* Global data of Tegra CPU CAR ops */
  25static struct tegra_cpu_car_ops dummy_car_ops;
  26struct tegra_cpu_car_ops *tegra_cpu_car_ops = &dummy_car_ops;
  27
  28void __init tegra_init_dup_clks(struct tegra_clk_duplicate *dup_list,
  29                                struct clk *clks[], int clk_max)
  30{
  31        struct clk *clk;
  32
  33        for (; dup_list->clk_id < clk_max; dup_list++) {
  34                clk = clks[dup_list->clk_id];
  35                dup_list->lookup.clk = clk;
  36                clkdev_add(&dup_list->lookup);
  37        }
  38}
  39
  40void __init tegra_init_from_table(struct tegra_clk_init_table *tbl,
  41                                  struct clk *clks[], int clk_max)
  42{
  43        struct clk *clk;
  44
  45        for (; tbl->clk_id < clk_max; tbl++) {
  46                clk = clks[tbl->clk_id];
  47                if (IS_ERR_OR_NULL(clk))
  48                        return;
  49
  50                if (tbl->parent_id < clk_max) {
  51                        struct clk *parent = clks[tbl->parent_id];
  52                        if (clk_set_parent(clk, parent)) {
  53                                pr_err("%s: Failed to set parent %s of %s\n",
  54                                       __func__, __clk_get_name(parent),
  55                                       __clk_get_name(clk));
  56                                WARN_ON(1);
  57                        }
  58                }
  59
  60                if (tbl->rate)
  61                        if (clk_set_rate(clk, tbl->rate)) {
  62                                pr_err("%s: Failed to set rate %lu of %s\n",
  63                                       __func__, tbl->rate,
  64                                       __clk_get_name(clk));
  65                                WARN_ON(1);
  66                        }
  67
  68                if (tbl->state)
  69                        if (clk_prepare_enable(clk)) {
  70                                pr_err("%s: Failed to enable %s\n", __func__,
  71                                       __clk_get_name(clk));
  72                                WARN_ON(1);
  73                        }
  74        }
  75}
  76
  77static const struct of_device_id tegra_dt_clk_match[] = {
  78        { .compatible = "nvidia,tegra20-car", .data = tegra20_clock_init },
  79        { .compatible = "nvidia,tegra30-car", .data = tegra30_clock_init },
  80        { .compatible = "nvidia,tegra114-car", .data = tegra114_clock_init },
  81        { }
  82};
  83
  84void __init tegra_clocks_init(void)
  85{
  86        of_clk_init(tegra_dt_clk_match);
  87}
  88
  89tegra_clk_apply_init_table_func tegra_clk_apply_init_table;
  90
  91void __init tegra_clocks_apply_init_table(void)
  92{
  93        if (!tegra_clk_apply_init_table)
  94                return;
  95
  96        tegra_clk_apply_init_table();
  97}
  98