18c2ecf20Sopenharmony_ci/*
28c2ecf20Sopenharmony_ci * arch/arm/plat-orion/mpp.c
38c2ecf20Sopenharmony_ci *
48c2ecf20Sopenharmony_ci * MPP functions for Marvell orion SoCs
58c2ecf20Sopenharmony_ci *
68c2ecf20Sopenharmony_ci * This file is licensed under the terms of the GNU General Public
78c2ecf20Sopenharmony_ci * License version 2.  This program is licensed "as is" without any
88c2ecf20Sopenharmony_ci * warranty of any kind, whether express or implied.
98c2ecf20Sopenharmony_ci */
108c2ecf20Sopenharmony_ci
118c2ecf20Sopenharmony_ci#include <linux/kernel.h>
128c2ecf20Sopenharmony_ci#include <linux/init.h>
138c2ecf20Sopenharmony_ci#include <linux/mbus.h>
148c2ecf20Sopenharmony_ci#include <linux/io.h>
158c2ecf20Sopenharmony_ci#include <linux/gpio.h>
168c2ecf20Sopenharmony_ci#include <plat/orion-gpio.h>
178c2ecf20Sopenharmony_ci#include <plat/mpp.h>
188c2ecf20Sopenharmony_ci
198c2ecf20Sopenharmony_ci/* Address of the ith MPP control register */
208c2ecf20Sopenharmony_cistatic __init void __iomem *mpp_ctrl_addr(unsigned int i,
218c2ecf20Sopenharmony_ci					  void __iomem *dev_bus)
228c2ecf20Sopenharmony_ci{
238c2ecf20Sopenharmony_ci	return dev_bus + (i) * 4;
248c2ecf20Sopenharmony_ci}
258c2ecf20Sopenharmony_ci
268c2ecf20Sopenharmony_ci
278c2ecf20Sopenharmony_civoid __init orion_mpp_conf(unsigned int *mpp_list, unsigned int variant_mask,
288c2ecf20Sopenharmony_ci			   unsigned int mpp_max, void __iomem *dev_bus)
298c2ecf20Sopenharmony_ci{
308c2ecf20Sopenharmony_ci	unsigned int mpp_nr_regs = (1 + mpp_max/8);
318c2ecf20Sopenharmony_ci	u32 mpp_ctrl[8];
328c2ecf20Sopenharmony_ci	int i;
338c2ecf20Sopenharmony_ci
348c2ecf20Sopenharmony_ci	printk(KERN_DEBUG "initial MPP regs:");
358c2ecf20Sopenharmony_ci	if (mpp_nr_regs > ARRAY_SIZE(mpp_ctrl)) {
368c2ecf20Sopenharmony_ci		printk(KERN_ERR "orion_mpp_conf: invalid mpp_max\n");
378c2ecf20Sopenharmony_ci		return;
388c2ecf20Sopenharmony_ci	}
398c2ecf20Sopenharmony_ci
408c2ecf20Sopenharmony_ci	for (i = 0; i < mpp_nr_regs; i++) {
418c2ecf20Sopenharmony_ci		mpp_ctrl[i] = readl(mpp_ctrl_addr(i, dev_bus));
428c2ecf20Sopenharmony_ci		printk(" %08x", mpp_ctrl[i]);
438c2ecf20Sopenharmony_ci	}
448c2ecf20Sopenharmony_ci	printk("\n");
458c2ecf20Sopenharmony_ci
468c2ecf20Sopenharmony_ci	for ( ; *mpp_list; mpp_list++) {
478c2ecf20Sopenharmony_ci		unsigned int num = MPP_NUM(*mpp_list);
488c2ecf20Sopenharmony_ci		unsigned int sel = MPP_SEL(*mpp_list);
498c2ecf20Sopenharmony_ci		int shift, gpio_mode;
508c2ecf20Sopenharmony_ci
518c2ecf20Sopenharmony_ci		if (num > mpp_max) {
528c2ecf20Sopenharmony_ci			printk(KERN_ERR "orion_mpp_conf: invalid MPP "
538c2ecf20Sopenharmony_ci					"number (%u)\n", num);
548c2ecf20Sopenharmony_ci			continue;
558c2ecf20Sopenharmony_ci		}
568c2ecf20Sopenharmony_ci		if (variant_mask && !(*mpp_list & variant_mask)) {
578c2ecf20Sopenharmony_ci			printk(KERN_WARNING
588c2ecf20Sopenharmony_ci			       "orion_mpp_conf: requested MPP%u config "
598c2ecf20Sopenharmony_ci			       "unavailable on this hardware\n", num);
608c2ecf20Sopenharmony_ci			continue;
618c2ecf20Sopenharmony_ci		}
628c2ecf20Sopenharmony_ci
638c2ecf20Sopenharmony_ci		shift = (num & 7) << 2;
648c2ecf20Sopenharmony_ci		mpp_ctrl[num / 8] &= ~(0xf << shift);
658c2ecf20Sopenharmony_ci		mpp_ctrl[num / 8] |= sel << shift;
668c2ecf20Sopenharmony_ci
678c2ecf20Sopenharmony_ci		gpio_mode = 0;
688c2ecf20Sopenharmony_ci		if (*mpp_list & MPP_INPUT_MASK)
698c2ecf20Sopenharmony_ci			gpio_mode |= GPIO_INPUT_OK;
708c2ecf20Sopenharmony_ci		if (*mpp_list & MPP_OUTPUT_MASK)
718c2ecf20Sopenharmony_ci			gpio_mode |= GPIO_OUTPUT_OK;
728c2ecf20Sopenharmony_ci
738c2ecf20Sopenharmony_ci		orion_gpio_set_valid(num, gpio_mode);
748c2ecf20Sopenharmony_ci	}
758c2ecf20Sopenharmony_ci
768c2ecf20Sopenharmony_ci	printk(KERN_DEBUG "  final MPP regs:");
778c2ecf20Sopenharmony_ci	for (i = 0; i < mpp_nr_regs; i++) {
788c2ecf20Sopenharmony_ci		writel(mpp_ctrl[i], mpp_ctrl_addr(i, dev_bus));
798c2ecf20Sopenharmony_ci		printk(" %08x", mpp_ctrl[i]);
808c2ecf20Sopenharmony_ci	}
818c2ecf20Sopenharmony_ci	printk("\n");
828c2ecf20Sopenharmony_ci}
83