|  | /* | 
|  | * Copyright 2013 Emilio López | 
|  | * | 
|  | * Emilio López <emilio@elopez.com.ar> | 
|  | * | 
|  | * This program is free software; you can redistribute it and/or modify | 
|  | * it under the terms of the GNU General Public License as published by | 
|  | * the Free Software Foundation; either version 2 of the License, or | 
|  | * (at your option) any later version. | 
|  | * | 
|  | * This program is distributed in the hope that it will be useful, | 
|  | * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
|  | * GNU General Public License for more details. | 
|  | */ | 
|  |  | 
|  | #include <linux/clk-provider.h> | 
|  | #include <linux/of.h> | 
|  | #include <linux/of_address.h> | 
|  | #include <linux/slab.h> | 
|  |  | 
|  | static DEFINE_SPINLOCK(mod1_lock); | 
|  |  | 
|  | #define SUN4I_MOD1_ENABLE	31 | 
|  | #define SUN4I_MOD1_MUX		16 | 
|  | #define SUN4I_MOD1_MUX_WIDTH	2 | 
|  | #define SUN4I_MOD1_MAX_PARENTS	4 | 
|  |  | 
|  | static void __init sun4i_mod1_clk_setup(struct device_node *node) | 
|  | { | 
|  | struct clk *clk; | 
|  | struct clk_mux *mux; | 
|  | struct clk_gate *gate; | 
|  | const char *parents[4]; | 
|  | const char *clk_name = node->name; | 
|  | void __iomem *reg; | 
|  | int i; | 
|  |  | 
|  | reg = of_io_request_and_map(node, 0, of_node_full_name(node)); | 
|  | if (IS_ERR(reg)) | 
|  | return; | 
|  |  | 
|  | mux = kzalloc(sizeof(*mux), GFP_KERNEL); | 
|  | if (!mux) | 
|  | goto err_unmap; | 
|  |  | 
|  | gate = kzalloc(sizeof(*gate), GFP_KERNEL); | 
|  | if (!gate) | 
|  | goto err_free_mux; | 
|  |  | 
|  | of_property_read_string(node, "clock-output-names", &clk_name); | 
|  | i = of_clk_parent_fill(node, parents, SUN4I_MOD1_MAX_PARENTS); | 
|  |  | 
|  | gate->reg = reg; | 
|  | gate->bit_idx = SUN4I_MOD1_ENABLE; | 
|  | gate->lock = &mod1_lock; | 
|  | mux->reg = reg; | 
|  | mux->shift = SUN4I_MOD1_MUX; | 
|  | mux->mask = BIT(SUN4I_MOD1_MUX_WIDTH) - 1; | 
|  | mux->lock = &mod1_lock; | 
|  |  | 
|  | clk = clk_register_composite(NULL, clk_name, parents, i, | 
|  | &mux->hw, &clk_mux_ops, | 
|  | NULL, NULL, | 
|  | &gate->hw, &clk_gate_ops, CLK_SET_RATE_PARENT); | 
|  | if (IS_ERR(clk)) | 
|  | goto err_free_gate; | 
|  |  | 
|  | of_clk_add_provider(node, of_clk_src_simple_get, clk); | 
|  |  | 
|  | return; | 
|  |  | 
|  | err_free_gate: | 
|  | kfree(gate); | 
|  | err_free_mux: | 
|  | kfree(mux); | 
|  | err_unmap: | 
|  | iounmap(reg); | 
|  | } | 
|  | CLK_OF_DECLARE(sun4i_mod1, "allwinner,sun4i-a10-mod1-clk", | 
|  | sun4i_mod1_clk_setup); |