summaryrefslogtreecommitdiff
path: root/drivers/power/wmt_battery/gauge/ug31xx/ug31xx_i2c.c
blob: 6249df50da7b12f071b27139a1d33d472ad3c428 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
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
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
/*
 * Copyright (c) 2012, ASUSTek, Inc. All Rights Reserved.
 */
#include <linux/module.h>
#include <linux/param.h>
#include <linux/i2c.h>
#include <linux/delay.h>
#include <linux/slab.h>
#include <asm/unaligned.h>
#include <linux/interrupt.h>
#include <linux/idr.h>
#include <linux/proc_fs.h>
#include <linux/uaccess.h>
#include "ug31xx_i2c.h"

static struct i2c_client *ug31xx_client = NULL;

void ug31xx_i2c_client_set(struct i2c_client *client)
{
	ug31xx_client = client;
	dev_info(&ug31xx_client->dev, "%s: Ug31xx i2c client saved.\n", __func__);
}

int ug31xx_read_i2c(struct i2c_client *client, 
                u8 reg, int *rt_value, int b_single)
{
	struct i2c_msg msg[2];
	unsigned char data[4];
	int err;

	if (!client || !client->adapter)
		return -ENODEV;

	if (!rt_value) return -EINVAL;

	data[0] = reg;
	//err = i2c_transfer(client->adapter, msg, 1);

  msg[0].addr = client->addr;
  msg[0].flags = 0 | I2C_M_NOSTART;
  msg[0].len = 1;
  if (reg >= 0x80) {
          data[1] = SECURITY_KEY;
          msg[0].len++;
  }
  msg[0].buf = (unsigned char *)data;

  msg[1].addr = client->addr;
  msg[1].flags = (I2C_M_RD);
  msg[1].len = b_single ? 1 : 2;
  msg[1].buf = (unsigned char *)data;

  err = i2c_transfer(client->adapter, msg, sizeof(msg)/sizeof(struct i2c_msg));

  if (err < 0) return err;

  if (b_single)
		*rt_value = data[0];
  else 
		*rt_value = get_unaligned_le16(data);

  return 0;
}

int ug31xx_write_i2c(struct i2c_client *client, 
                u8 reg, int rt_value, int b_single)
{
	struct i2c_msg msg[1];
	unsigned char data[4];
	int err;
	int idx;
	int tmp_buf=0;

	if (!client || !client->adapter)
		return -ENODEV;

  idx = 0;
	data[idx++] = reg;
  if (reg >= 0x80) {
          data[idx++] = SECURITY_KEY;
  }
  data[idx++] = rt_value & 0x0FF;
  data[idx++] = (rt_value & 0x0FF00)>>8;

  msg[0].addr = client->addr;
  msg[0].flags = 0 | I2C_M_NOSTART;
  msg[0].len = b_single ? idx-1 : idx;
  msg[0].buf = (unsigned char *)data;

  err = i2c_transfer(client->adapter, msg, sizeof(msg)/sizeof(struct i2c_msg));
  
  if (err >= 0)
          err = ug31xx_read_i2c(client, reg, &tmp_buf, b_single);
  //dev_info(&client->dev, "%s:: 0x%02X, 0x%04X, 0x%04X. %s\n", 
  //        __func__, reg, rt_value, tmp_buf, err < 0 ? "FAIL" : "SUCCESS");

  return err < 0 ? err : 0;
}

bool _API_I2C_Write(u16 writeAddress, u8 writeLength, u8 *PWriteData)
{
  int i, ret;
  int byte_flag=0;

  if (!PWriteData) {
    dev_err(&ug31xx_client->dev, "%s: Write buffer pointer error.\n", __func__);
    return false;
  }

  byte_flag = ONE_BYTE;

  for (i=0; i<writeLength; i++) {
    int tmp_buf = PWriteData[i];

    ret = ug31xx_write_i2c(ug31xx_client, writeAddress+i, tmp_buf, byte_flag);
    if (ret) {
      dev_err(&ug31xx_client->dev, "%s: Write data(0x%02X) fail. %d\n", 
                      __func__, i, ret);
      return false;
    }
  }

  return true;
}

bool _API_I2C_Read(u16 readAddress, u8 readLength, u8 *pReadDataBuffer)
{
  int i, ret;
  int byte_flag=0;

  if (!pReadDataBuffer) {
    dev_err(&ug31xx_client->dev, "%s: Read buffer pointer error.\n", __func__);
    return false;
  }

  byte_flag = ONE_BYTE;

  for (i=0; i<readLength; i++) {
    int tmp_buf=0;

    ret = ug31xx_read_i2c(ug31xx_client, readAddress+i, &tmp_buf, byte_flag);
    if (ret) {
      dev_err(&ug31xx_client->dev, "%s: read data(0x%02X) fail. %d\n", 
              __func__, i, ret);
      return false;
    }
    pReadDataBuffer[i] = tmp_buf;
  }

  return true;
}