Merge tag 'drivers' of git://git.kernel.org/pub/scm/linux/kernel/git/arm/arm-soc
authorLinus Torvalds <torvalds@linux-foundation.org>
Thu, 21 Feb 2013 23:12:17 +0000 (15:12 -0800)
committerLinus Torvalds <torvalds@linux-foundation.org>
Thu, 21 Feb 2013 23:12:18 +0000 (15:12 -0800)
Pull ARM SoC driver specific changes from Arnd Bergmann:

 - Updates to the ux500 cpufreq code

 - Moving the u300 DMA controller driver to drivers/dma

 - Moving versatile express drivers out of arch/arm for sharing with arch/arm64

 - Device tree bindings for the OMAP General Purpose Memory Controller

* tag 'drivers' of git://git.kernel.org/pub/scm/linux/kernel/git/arm/arm-soc: (27 commits)
  ARM: OMAP2+: gpmc: Add device tree documentation for elm handle
  ARM: OMAP2+: gpmc: add DT bindings for OneNAND
  ARM: OMAP2+: gpmc-onenand: drop __init annotation
  mtd: omap-onenand: pass device_node in platform data
  ARM: OMAP2+: Prevent potential crash if GPMC probe fails
  ARM: OMAP2+: gpmc: Remove unneeded of_node_put()
  arm: Move sp810.h to include/linux/amba/
  ARM: OMAP: gpmc: add DT bindings for GPMC timings and NAND
  ARM: OMAP: gpmc: enable hwecc for AM33xx SoCs
  ARM: OMAP: gpmc-nand: drop __init annotation
  mtd: omap-nand: pass device_node in platform data
  ARM: OMAP: gpmc: don't create devices from initcall on DT
  dma: coh901318: cut down on platform data abstraction
  dma: coh901318: merge header files
  dma: coh901318: push definitions into driver
  dma: coh901318: push header down into the DMA subsystem
  dma: coh901318: skip hard-coded addresses
  dma: coh901318: remove hardcoded target addresses
  dma: coh901318: push platform data into driver
  dma: coh901318: create a proper platform data file
  ...

1  2 
arch/arm/mach-omap2/gpmc.c
drivers/clk/versatile/clk-vexpress.c
drivers/cpufreq/Makefile
drivers/cpufreq/dbx500-cpufreq.c
drivers/mfd/db8500-prcmu.c

Simple merge
Simple merge
Simple merge
index 0000000000000000000000000000000000000000,9a623753dee234348bcdc5723eb9468adcecd4b5..72f0c3efa76e6b7d89114c463ccd7c5243ccb917
mode 000000,100644..100644
--- /dev/null
@@@ -1,0 -1,180 +1,174 @@@
 -#include <mach/id.h>
+ /*
+  * Copyright (C) STMicroelectronics 2009
+  * Copyright (C) ST-Ericsson SA 2010-2012
+  *
+  * License Terms: GNU General Public License v2
+  * Author: Sundar Iyer <sundar.iyer@stericsson.com>
+  * Author: Martin Persson <martin.persson@stericsson.com>
+  * Author: Jonas Aaberg <jonas.aberg@stericsson.com>
+  */
+ #include <linux/module.h>
+ #include <linux/kernel.h>
+ #include <linux/cpufreq.h>
+ #include <linux/delay.h>
+ #include <linux/slab.h>
+ #include <linux/platform_device.h>
+ #include <linux/clk.h>
 -      cpumask_copy(policy->cpus, cpu_present_mask);
 -
 -      policy->shared_type = CPUFREQ_SHARED_TYPE_ALL;
+ static struct cpufreq_frequency_table *freq_table;
+ static struct clk *armss_clk;
+ static struct freq_attr *dbx500_cpufreq_attr[] = {
+       &cpufreq_freq_attr_scaling_available_freqs,
+       NULL,
+ };
+ static int dbx500_cpufreq_verify_speed(struct cpufreq_policy *policy)
+ {
+       return cpufreq_frequency_table_verify(policy, freq_table);
+ }
+ static int dbx500_cpufreq_target(struct cpufreq_policy *policy,
+                               unsigned int target_freq,
+                               unsigned int relation)
+ {
+       struct cpufreq_freqs freqs;
+       unsigned int idx;
+       int ret;
+       /* scale the target frequency to one of the extremes supported */
+       if (target_freq < policy->cpuinfo.min_freq)
+               target_freq = policy->cpuinfo.min_freq;
+       if (target_freq > policy->cpuinfo.max_freq)
+               target_freq = policy->cpuinfo.max_freq;
+       /* Lookup the next frequency */
+       if (cpufreq_frequency_table_target(policy, freq_table, target_freq,
+                                       relation, &idx))
+               return -EINVAL;
+       freqs.old = policy->cur;
+       freqs.new = freq_table[idx].frequency;
+       if (freqs.old == freqs.new)
+               return 0;
+       /* pre-change notification */
+       for_each_cpu(freqs.cpu, policy->cpus)
+               cpufreq_notify_transition(&freqs, CPUFREQ_PRECHANGE);
+       /* update armss clk frequency */
+       ret = clk_set_rate(armss_clk, freqs.new * 1000);
+       if (ret) {
+               pr_err("dbx500-cpufreq: Failed to set armss_clk to %d Hz: error %d\n",
+                      freqs.new * 1000, ret);
+               return ret;
+       }
+       /* post change notification */
+       for_each_cpu(freqs.cpu, policy->cpus)
+               cpufreq_notify_transition(&freqs, CPUFREQ_POSTCHANGE);
+       return 0;
+ }
+ static unsigned int dbx500_cpufreq_getspeed(unsigned int cpu)
+ {
+       int i = 0;
+       unsigned long freq = clk_get_rate(armss_clk) / 1000;
+       while (freq_table[i].frequency != CPUFREQ_TABLE_END) {
+               if (freq <= freq_table[i].frequency)
+                       return freq_table[i].frequency;
+               i++;
+       }
+       /* We could not find a corresponding frequency. */
+       pr_err("dbx500-cpufreq: Failed to find cpufreq speed\n");
+       return 0;
+ }
+ static int __cpuinit dbx500_cpufreq_init(struct cpufreq_policy *policy)
+ {
+       int res;
+       /* get policy fields based on the table */
+       res = cpufreq_frequency_table_cpuinfo(policy, freq_table);
+       if (!res)
+               cpufreq_frequency_table_get_attr(freq_table, policy->cpu);
+       else {
+               pr_err("dbx500-cpufreq: Failed to read policy table\n");
+               return res;
+       }
+       policy->min = policy->cpuinfo.min_freq;
+       policy->max = policy->cpuinfo.max_freq;
+       policy->cur = dbx500_cpufreq_getspeed(policy->cpu);
+       policy->governor = CPUFREQ_DEFAULT_GOVERNOR;
+       /*
+        * FIXME : Need to take time measurement across the target()
+        *         function with no/some/all drivers in the notification
+        *         list.
+        */
+       policy->cpuinfo.transition_latency = 20 * 1000; /* in ns */
+       /* policy sharing between dual CPUs */
 -      if (!cpu_is_u8500_family())
 -              return -ENODEV;
 -
++      cpumask_setall(policy->cpus);
+       return 0;
+ }
+ static struct cpufreq_driver dbx500_cpufreq_driver = {
+       .flags  = CPUFREQ_STICKY | CPUFREQ_CONST_LOOPS,
+       .verify = dbx500_cpufreq_verify_speed,
+       .target = dbx500_cpufreq_target,
+       .get    = dbx500_cpufreq_getspeed,
+       .init   = dbx500_cpufreq_init,
+       .name   = "DBX500",
+       .attr   = dbx500_cpufreq_attr,
+ };
+ static int dbx500_cpufreq_probe(struct platform_device *pdev)
+ {
+       int i = 0;
+       freq_table = dev_get_platdata(&pdev->dev);
+       if (!freq_table) {
+               pr_err("dbx500-cpufreq: Failed to fetch cpufreq table\n");
+               return -ENODEV;
+       }
+       armss_clk = clk_get(&pdev->dev, "armss");
+       if (IS_ERR(armss_clk)) {
+               pr_err("dbx500-cpufreq: Failed to get armss clk\n");
+               return PTR_ERR(armss_clk);
+       }
+       pr_info("dbx500-cpufreq: Available frequencies:\n");
+       while (freq_table[i].frequency != CPUFREQ_TABLE_END) {
+               pr_info("  %d Mhz\n", freq_table[i].frequency/1000);
+               i++;
+       }
+       return cpufreq_register_driver(&dbx500_cpufreq_driver);
+ }
+ static struct platform_driver dbx500_cpufreq_plat_driver = {
+       .driver = {
+               .name = "cpufreq-ux500",
+               .owner = THIS_MODULE,
+       },
+       .probe = dbx500_cpufreq_probe,
+ };
+ static int __init dbx500_cpufreq_register(void)
+ {
+       return platform_driver_register(&dbx500_cpufreq_plat_driver);
+ }
+ device_initcall(dbx500_cpufreq_register);
+ MODULE_LICENSE("GPL v2");
+ MODULE_DESCRIPTION("cpufreq driver for DBX500");
Simple merge