Blame view

arch/arm/mach-omap2/i2c.c 2.85 KB
b63128e81   Tony Lindgren   omap: Split i2c p...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
  /*
   * Helper module for board specific I2C bus registration
   *
   * Copyright (C) 2009 Nokia Corporation.
   *
   * This program is free software; you can redistribute it and/or
   * modify it under the terms of the GNU General Public License
   * version 2 as published by the Free Software Foundation.
   *
   * 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.
   *
   * You should have received a copy of the GNU General Public License
   * along with this program; if not, write to the Free Software
   * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
   * 02110-1301 USA
   *
   */
  
  #include <plat/cpu.h>
  #include <plat/i2c.h>
4e65331c6   Tony Lindgren   ARM: 7159/1: OMAP...
24
  #include "common.h"
6d3c55fd4   Avinash.H.M   OMAP: hwmod: fix ...
25
  #include <plat/omap_hwmod.h>
b63128e81   Tony Lindgren   omap: Split i2c p...
26
27
  
  #include "mux.h"
6d3c55fd4   Avinash.H.M   OMAP: hwmod: fix ...
28
29
30
31
32
33
34
  /* In register I2C_CON, Bit 15 is the I2C enable bit */
  #define I2C_EN					BIT(15)
  #define OMAP2_I2C_CON_OFFSET			0x24
  #define OMAP4_I2C_CON_OFFSET			0xA4
  
  /* Maximum microseconds to wait for OMAP module to softreset */
  #define MAX_MODULE_SOFTRESET_WAIT	10000
9833eff3d   Jarkko Nikula   omap: i2c: Fix mu...
35
  void __init omap2_i2c_mux_pins(int bus_id)
b63128e81   Tony Lindgren   omap: Split i2c p...
36
  {
f99bf16d7   Tony Lindgren   omap: mux: Conver...
37
  	char mux_name[sizeof("i2c2_scl.i2c2_scl")];
b63128e81   Tony Lindgren   omap: Split i2c p...
38
39
  
  	/* First I2C bus is not muxable */
f99bf16d7   Tony Lindgren   omap: mux: Conver...
40
41
  	if (bus_id == 1)
  		return;
b63128e81   Tony Lindgren   omap: Split i2c p...
42

f99bf16d7   Tony Lindgren   omap: mux: Conver...
43
44
45
46
  	sprintf(mux_name, "i2c%i_scl.i2c%i_scl", bus_id, bus_id);
  	omap_mux_init_signal(mux_name, OMAP_PIN_INPUT);
  	sprintf(mux_name, "i2c%i_sda.i2c%i_sda", bus_id, bus_id);
  	omap_mux_init_signal(mux_name, OMAP_PIN_INPUT);
b63128e81   Tony Lindgren   omap: Split i2c p...
47
  }
6d3c55fd4   Avinash.H.M   OMAP: hwmod: fix ...
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
  
  /**
   * omap_i2c_reset - reset the omap i2c module.
   * @oh: struct omap_hwmod *
   *
   * The i2c moudle in omap2, omap3 had a special sequence to reset. The
   * sequence is:
   * - Disable the I2C.
   * - Write to SOFTRESET bit.
   * - Enable the I2C.
   * - Poll on the RESETDONE bit.
   * The sequence is implemented in below function. This is called for 2420,
   * 2430 and omap3.
   */
  int omap_i2c_reset(struct omap_hwmod *oh)
  {
  	u32 v;
  	u16 i2c_con;
  	int c = 0;
  
  	if (oh->class->rev == OMAP_I2C_IP_VERSION_2) {
  		i2c_con = OMAP4_I2C_CON_OFFSET;
  	} else if (oh->class->rev == OMAP_I2C_IP_VERSION_1) {
  		i2c_con = OMAP2_I2C_CON_OFFSET;
  	} else {
  		WARN(1, "Cannot reset I2C block %s: unsupported revision
  ",
  		     oh->name);
  		return -EINVAL;
  	}
  
  	/* Disable I2C */
  	v = omap_hwmod_read(oh, i2c_con);
  	v &= ~I2C_EN;
  	omap_hwmod_write(v, oh, i2c_con);
  
  	/* Write to the SOFTRESET bit */
  	omap_hwmod_softreset(oh);
  
  	/* Enable I2C */
  	v = omap_hwmod_read(oh, i2c_con);
  	v |= I2C_EN;
  	omap_hwmod_write(v, oh, i2c_con);
  
  	/* Poll on RESETDONE bit */
  	omap_test_timeout((omap_hwmod_read(oh,
  				oh->class->sysc->syss_offs)
  				& SYSS_RESETDONE_MASK),
  				MAX_MODULE_SOFTRESET_WAIT, c);
  
  	if (c == MAX_MODULE_SOFTRESET_WAIT)
  		pr_warning("%s: %s: softreset failed (waited %d usec)
  ",
  			__func__, oh->name, MAX_MODULE_SOFTRESET_WAIT);
  	else
  		pr_debug("%s: %s: softreset in %d usec
  ", __func__,
  			oh->name, c);
  
  	return 0;
  }