via_i2c.c 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177
  1. /*
  2. * Copyright 1998-2008 VIA Technologies, Inc. All Rights Reserved.
  3. * Copyright 2001-2008 S3 Graphics, Inc. All Rights Reserved.
  4. * This program is free software; you can redistribute it and/or
  5. * modify it under the terms of the GNU General Public
  6. * License as published by the Free Software Foundation;
  7. * either version 2, or (at your option) any later version.
  8. * This program is distributed in the hope that it will be useful,
  9. * but WITHOUT ANY WARRANTIES OR REPRESENTATIONS; without even
  10. * the implied warranty of MERCHANTABILITY or FITNESS FOR
  11. * A PARTICULAR PURPOSE.See the GNU General Public License
  12. * for more details.
  13. * You should have received a copy of the GNU General Public License
  14. * along with this program; if not, write to the Free Software
  15. * Foundation, Inc.,
  16. * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
  17. */
  18. #include "global.h"
  19. static void via_i2c_setscl(void *data, int state)
  20. {
  21. u8 val;
  22. struct via_i2c_stuff *via_i2c_chan = (struct via_i2c_stuff *)data;
  23. val = viafb_read_reg(VIASR, via_i2c_chan->i2c_port) & 0xF0;
  24. if (state)
  25. val |= 0x20;
  26. else
  27. val &= ~0x20;
  28. switch (via_i2c_chan->i2c_port) {
  29. case I2CPORTINDEX:
  30. val |= 0x01;
  31. break;
  32. case GPIOPORTINDEX:
  33. val |= 0x80;
  34. break;
  35. default:
  36. DEBUG_MSG("via_i2c: specify wrong i2c port.\n");
  37. }
  38. viafb_write_reg(via_i2c_chan->i2c_port, VIASR, val);
  39. }
  40. static int via_i2c_getscl(void *data)
  41. {
  42. struct via_i2c_stuff *via_i2c_chan = (struct via_i2c_stuff *)data;
  43. if (viafb_read_reg(VIASR, via_i2c_chan->i2c_port) & 0x08)
  44. return 1;
  45. return 0;
  46. }
  47. static int via_i2c_getsda(void *data)
  48. {
  49. struct via_i2c_stuff *via_i2c_chan = (struct via_i2c_stuff *)data;
  50. if (viafb_read_reg(VIASR, via_i2c_chan->i2c_port) & 0x04)
  51. return 1;
  52. return 0;
  53. }
  54. static void via_i2c_setsda(void *data, int state)
  55. {
  56. u8 val;
  57. struct via_i2c_stuff *via_i2c_chan = (struct via_i2c_stuff *)data;
  58. val = viafb_read_reg(VIASR, via_i2c_chan->i2c_port) & 0xF0;
  59. if (state)
  60. val |= 0x10;
  61. else
  62. val &= ~0x10;
  63. switch (via_i2c_chan->i2c_port) {
  64. case I2CPORTINDEX:
  65. val |= 0x01;
  66. break;
  67. case GPIOPORTINDEX:
  68. val |= 0x40;
  69. break;
  70. default:
  71. DEBUG_MSG("via_i2c: specify wrong i2c port.\n");
  72. }
  73. viafb_write_reg(via_i2c_chan->i2c_port, VIASR, val);
  74. }
  75. int viafb_i2c_readbyte(u8 slave_addr, u8 index, u8 *pdata)
  76. {
  77. u8 mm1[] = {0x00};
  78. struct i2c_msg msgs[2];
  79. *pdata = 0;
  80. msgs[0].flags = 0;
  81. msgs[1].flags = I2C_M_RD;
  82. msgs[0].addr = msgs[1].addr = slave_addr / 2;
  83. mm1[0] = index;
  84. msgs[0].len = 1; msgs[1].len = 1;
  85. msgs[0].buf = mm1; msgs[1].buf = pdata;
  86. i2c_transfer(&viaparinfo->shared->i2c_stuff.adapter, msgs, 2);
  87. return 0;
  88. }
  89. int viafb_i2c_writebyte(u8 slave_addr, u8 index, u8 data)
  90. {
  91. u8 msg[2] = { index, data };
  92. struct i2c_msg msgs;
  93. msgs.flags = 0;
  94. msgs.addr = slave_addr / 2;
  95. msgs.len = 2;
  96. msgs.buf = msg;
  97. return i2c_transfer(&viaparinfo->shared->i2c_stuff.adapter, &msgs, 1);
  98. }
  99. int viafb_i2c_readbytes(u8 slave_addr, u8 index, u8 *buff, int buff_len)
  100. {
  101. u8 mm1[] = {0x00};
  102. struct i2c_msg msgs[2];
  103. msgs[0].flags = 0;
  104. msgs[1].flags = I2C_M_RD;
  105. msgs[0].addr = msgs[1].addr = slave_addr / 2;
  106. mm1[0] = index;
  107. msgs[0].len = 1; msgs[1].len = buff_len;
  108. msgs[0].buf = mm1; msgs[1].buf = buff;
  109. i2c_transfer(&viaparinfo->shared->i2c_stuff.adapter, msgs, 2);
  110. return 0;
  111. }
  112. int viafb_create_i2c_bus(void *viapar)
  113. {
  114. int ret;
  115. struct via_i2c_stuff *i2c_stuff =
  116. &((struct viafb_par *)viapar)->shared->i2c_stuff;
  117. strcpy(i2c_stuff->adapter.name, "via_i2c");
  118. i2c_stuff->i2c_port = 0x0;
  119. i2c_stuff->adapter.owner = THIS_MODULE;
  120. i2c_stuff->adapter.id = 0x01FFFF;
  121. i2c_stuff->adapter.class = 0;
  122. i2c_stuff->adapter.algo_data = &i2c_stuff->algo;
  123. i2c_stuff->adapter.dev.parent = NULL;
  124. i2c_stuff->algo.setsda = via_i2c_setsda;
  125. i2c_stuff->algo.setscl = via_i2c_setscl;
  126. i2c_stuff->algo.getsda = via_i2c_getsda;
  127. i2c_stuff->algo.getscl = via_i2c_getscl;
  128. i2c_stuff->algo.udelay = 40;
  129. i2c_stuff->algo.timeout = 20;
  130. i2c_stuff->algo.data = i2c_stuff;
  131. i2c_set_adapdata(&i2c_stuff->adapter, i2c_stuff);
  132. /* Raise SCL and SDA */
  133. i2c_stuff->i2c_port = I2CPORTINDEX;
  134. via_i2c_setsda(i2c_stuff, 1);
  135. via_i2c_setscl(i2c_stuff, 1);
  136. i2c_stuff->i2c_port = GPIOPORTINDEX;
  137. via_i2c_setsda(i2c_stuff, 1);
  138. via_i2c_setscl(i2c_stuff, 1);
  139. udelay(20);
  140. ret = i2c_bit_add_bus(&i2c_stuff->adapter);
  141. if (ret == 0)
  142. DEBUG_MSG("I2C bus %s registered.\n", i2c_stuff->adapter.name);
  143. else
  144. DEBUG_MSG("Failed to register I2C bus %s.\n",
  145. i2c_stuff->adapter.name);
  146. return ret;
  147. }
  148. void viafb_delete_i2c_buss(void *par)
  149. {
  150. i2c_del_adapter(&((struct viafb_par *)par)->shared->i2c_stuff.adapter);
  151. }