diff options
Diffstat (limited to 'drivers/misc/atmel_tclib.c')
| -rw-r--r-- | drivers/misc/atmel_tclib.c | 64 | 
1 files changed, 54 insertions, 10 deletions
diff --git a/drivers/misc/atmel_tclib.c b/drivers/misc/atmel_tclib.c index 4bcfc375973..c8d8e38d0d8 100644 --- a/drivers/misc/atmel_tclib.c +++ b/drivers/misc/atmel_tclib.c @@ -6,12 +6,10 @@  #include <linux/ioport.h>  #include <linux/kernel.h>  #include <linux/platform_device.h> +#include <linux/module.h>  #include <linux/slab.h>  #include <linux/export.h> - -/* Number of bytes to reserve for the iomem resource */ -#define ATMEL_TC_IOMEM_SIZE	256 - +#include <linux/of.h>  /*   * This is a thin library to solve the problem of how to portably allocate @@ -48,10 +46,17 @@ struct atmel_tc *atmel_tc_alloc(unsigned block, const char *name)  	struct atmel_tc		*tc;  	struct platform_device	*pdev = NULL;  	struct resource		*r; +	size_t			size;  	spin_lock(&tc_list_lock);  	list_for_each_entry(tc, &tc_list, node) { -		if (tc->pdev->id == block) { +		if (tc->pdev->dev.of_node) { +			if (of_alias_get_id(tc->pdev->dev.of_node, "tcb") +					== block) { +				pdev = tc->pdev; +				break; +			} +		} else if (tc->pdev->id == block) {  			pdev = tc->pdev;  			break;  		} @@ -61,11 +66,15 @@ struct atmel_tc *atmel_tc_alloc(unsigned block, const char *name)  		goto fail;  	r = platform_get_resource(pdev, IORESOURCE_MEM, 0); -	r = request_mem_region(r->start, ATMEL_TC_IOMEM_SIZE, name);  	if (!r)  		goto fail; -	tc->regs = ioremap(r->start, ATMEL_TC_IOMEM_SIZE); +	size = resource_size(r); +	r = request_mem_region(r->start, size, name); +	if (!r) +		goto fail; + +	tc->regs = ioremap(r->start, size);  	if (!tc->regs)  		goto fail_ioremap; @@ -76,7 +85,7 @@ out:  	return tc;  fail_ioremap: -	release_mem_region(r->start, ATMEL_TC_IOMEM_SIZE); +	release_mem_region(r->start, size);  fail:  	tc = NULL;  	goto out; @@ -96,7 +105,7 @@ void atmel_tc_free(struct atmel_tc *tc)  	spin_lock(&tc_list_lock);  	if (tc->regs) {  		iounmap(tc->regs); -		release_mem_region(tc->iomem->start, ATMEL_TC_IOMEM_SIZE); +		release_mem_region(tc->iomem->start, resource_size(tc->iomem));  		tc->regs = NULL;  		tc->iomem = NULL;  	} @@ -104,6 +113,30 @@ void atmel_tc_free(struct atmel_tc *tc)  }  EXPORT_SYMBOL_GPL(atmel_tc_free); +#if defined(CONFIG_OF) +static struct atmel_tcb_config tcb_rm9200_config = { +	.counter_width = 16, +}; + +static struct atmel_tcb_config tcb_sam9x5_config = { +	.counter_width = 32, +}; + +static const struct of_device_id atmel_tcb_dt_ids[] = { +	{ +		.compatible = "atmel,at91rm9200-tcb", +		.data = &tcb_rm9200_config, +	}, { +		.compatible = "atmel,at91sam9x5-tcb", +		.data = &tcb_sam9x5_config, +	}, { +		/* sentinel */ +	} +}; + +MODULE_DEVICE_TABLE(of, atmel_tcb_dt_ids); +#endif +  static int __init tc_probe(struct platform_device *pdev)  {  	struct atmel_tc *tc; @@ -129,6 +162,14 @@ static int __init tc_probe(struct platform_device *pdev)  		return -EINVAL;  	} +	/* Now take SoC information if available */ +	if (pdev->dev.of_node) { +		const struct of_device_id *match; +		match = of_match_node(atmel_tcb_dt_ids, pdev->dev.of_node); +		if (match) +			tc->tcb_config = match->data; +	} +  	tc->clk[0] = clk;  	tc->clk[1] = clk_get(&pdev->dev, "t1_clk");  	if (IS_ERR(tc->clk[1])) @@ -153,7 +194,10 @@ static int __init tc_probe(struct platform_device *pdev)  }  static struct platform_driver tc_driver = { -	.driver.name	= "atmel_tcb", +	.driver = { +		.name	= "atmel_tcb", +		.of_match_table	= of_match_ptr(atmel_tcb_dt_ids), +	},  };  static int __init tc_init(void)  |