-
Notifications
You must be signed in to change notification settings - Fork 20
/
Copy pathlpc_i2c.h
168 lines (143 loc) · 4.47 KB
/
lpc_i2c.h
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
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
/**
*
* Copyright (C) 2010-2015, Mellanox Technologies Ltd. ALL RIGHTS RESERVED.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
*/
#ifndef __MLNX_LPCI2C_DRV_H__
#define __MLNX_LPCI2C_DRV_H__
#include <linux/i2c.h>
#define LPCI2C_RC_FAILURE -1
#define LPCI2C_RC_OK 0
#define LPCI2C_DEVICE_NAME "lpci2c"
#define LPCI2C_IRQ_NUM 19 // TBD
#define LPCI2C_NO_IRQ -1
#define LPCI2C_VALID_FLAG (I2C_M_RECV_LEN | I2C_M_RD)
#define LPCI2C_BUS_NUM 1
#define LPC_CPLD_I2C_BASE_ADRR 0x2000
#define LPC_CPLD_BASE_ADRR 0x2500
#define LPC_CPLD_IO_LEN 0x100
#define LPCI2C_DATA_REG_SZ 36
#define LPCI2C_MAX_ADDR_LEN 4
#define LPCI2C_RETR_NUM 2
#define LPCI2C_XFER_TO 500000 // microcec
#define LPCI2C_POLL_TIME 2000 // microsec
/* LPC IFC in PCH defines */
#define LPC_CTRL_IFC_BUS_ID 0
#define LPC_CTRL_IFC_SLOT_ID 31
#define LPC_CTRL_IFC_FUNC_ID 0
#define LPC_QM67_DEV_ID 0x1c4f
#define LPC_QM77_DEV_ID 0x1e55
#define LPC_RNG_DEV_ID 0x1f38
/* Use generic decode range 4 for CPLD LPC */
#define LPC_PCH_GEN_DEC_RANGE4 0x90
#define LPC_PCH_GEN_DEC_BASE 0x84
#define LPC_RNG_LPC_CTRL 0x84
/* NOTE! Intel PCH datasheet call ranges from 1 */
#define LPC_PCH_GEN_DEC_RANGES 4
#define LPC_CPLD_I2C_RANGE 2
#define LPC_CPLD_RANGE 3
#define LPC_CLKS_EN 0
#ifdef __KERNEL__
struct lpc_rw_msg {
__u16 base;
__u16 offset;
__u8 read_write;
__u8 datalen;
char* data;
};
struct lpci2c_regs {
__u8 lpf;
__u8 half_cyc;
__u8 i2c_hold;
__u8 config;
__u8 cmd;
__u8 num_dat;
__u8 num_addr;
__u8 status;
__u8 data[LPCI2C_DATA_REG_SZ];
};
struct lpci2c_stat {
__u32 read_tr;
__u32 write_tr;
__u32 irq_cnt;
unsigned long read_byte;
unsigned long write_byte;
__u32 ack;
__u32 nack;
__u32 to;
__u32 last_xfer_time;
};
struct lpci2c_curr_transf {
__u8 cmd;
__u8 addr_width;
__u8 data_len;
__u8 msg_num;
struct i2c_msg* msg;
};
struct lpci2c_priv {
__u32 lpc_gen_dec_reg[LPC_PCH_GEN_DEC_RANGES];
struct i2c_adapter adap;
__u16 dev_id;
__u16 base_addr;
__u16 poll_time;
struct mutex lock;
struct resource* lpc_i2c_res;
struct resource* lpc_cpld_res;
struct lpci2c_curr_transf xfer;
wait_queue_head_t wq;
int irq;
struct lpci2c_stat stat;
struct platform_device* pdev;
struct attribute_group attr_grp; // Check if needed
};
/* LPC I2C registers */
#define LPCI2C_LPF_REG 0x0
#define LPCI2C_CTRL_REG 0x1
#define LPCI2C_HALF_CYC_REG 0x4
#define LPCI2C_I2C_HOLD_REG 0x5
#define LPCI2C_CMD_REG 0x6
#define LPCI2C_NUM_DAT_REG 0x7
#define LPCI2C_NUM_ADDR_REG 0x8
#define LPCI2C_STATUS_REG 0x9
#define LPCI2C_DATA_REG 0xa
#define LPCI2C_RST_SEL_MASK 0x1
#define LPCI2C_LPF_DFLT 0x2
/* 100 KHz configuration */
#define LPCI2C_HALF_CYC_100 0x1f
#define LPCI2C_I2C_HOLD_100 0x3c
/* 400 KHz configuration */
#define LPCI2C_HALF_CYC_400 0x7
#define LPCI2C_I2C_HOLD_400 0x8
#define LPCI2C_TRANS_END 0x1
#define LPCI2C_STATUS_NACK 0x10
#define LPCI2C_ERR_IND -1
#define LPCI2C_NO_IND 0
#define LPCI2C_ACK_IND 1
#define LPCI2C_NACK_IND 2
/* 5 debug levels: 1 - only init/exit mesaages, 2 - 1 + statistic enabled, 3-5 - additional increased verbosity levels. */
#define LPCI2C_DBG_MAX_LVL 5
#define LPCI2C_DFLT_DBG_LVL 1
#define LPCI2C_LOG_ERROR(format, args...) printk(KERN_ERR "LPCI2C ERR: " format, ## args)
#define LPCI2C_LOG_WARNING(format, args...) printk(KERN_WARNING "LPCI2C WARN: " format, ## args)
#ifdef LPCI2C_DEBUG
#define LPCI2C_LOG_DBG(level, format, args...) if (dbg_lvl > 0 && dbg_lvl >= level) {printk("LPCI2C DBG: " format, ## args);}
#else
#define LPCI2C_LOG_DBG(level, format, args...)
#endif
void mlnx_rw_lpc(struct lpc_rw_msg *msg);
#endif /* __KERNEL__*/
#endif /* __MLNX_LPCI2C_DRV_H__ */