Browse Source
New library for I2C device access under Linux. As a first step, the
New library for I2C device access under Linux. As a first step, the
library will host the i2c_smbus_*() inline functions which were previously in the user-space flavor of <linux/i2c-dev.h>. git-svn-id: http://lm-sensors.org/svn/i2c-tools/trunk@6054 7894878c-1315-0410-8ee3-d5d059ff63e0tags/v4.0
8 changed files with 379 additions and 191 deletions
-
1CHANGES
-
14Makefile
-
32README
-
191include/i2c/smbus.h
-
102lib/Module.mk
-
18lib/libi2c.map
-
203lib/smbus.c
-
9tools/Module.mk
@ -0,0 +1,102 @@ |
|||||
|
# I2C library for Linux
|
||||
|
#
|
||||
|
# Copyright (C) 2012 Jean Delvare <khali@linux-fr.org>
|
||||
|
#
|
||||
|
# This program is free software; you can redistribute it and/or modify
|
||||
|
# it under the terms of the GNU Lesser General Public License as published
|
||||
|
# by the Free Software Foundation; either version 2.1 of the License, or
|
||||
|
# (at your option) any later version.
|
||||
|
|
||||
|
LIB_DIR := lib |
||||
|
|
||||
|
LIB_CFLAGS := -Wstrict-prototypes -Wshadow -Wpointer-arith -Wcast-qual \
|
||||
|
-Wcast-align -Wwrite-strings -Wnested-externs -Winline \
|
||||
|
-W -Wundef -Wmissing-prototypes -Iinclude |
||||
|
|
||||
|
# The main and minor version of the library
|
||||
|
# The library soname (major number) must be changed if and only if the
|
||||
|
# interface is changed in a backward incompatible way. The interface is
|
||||
|
# defined by the public header files - in this case they are only smbus.h.
|
||||
|
LIB_MAINVER := 0 |
||||
|
LIB_MINORVER := 1.0 |
||||
|
LIB_VER := $(LIB_MAINVER).$(LIB_MINORVER) |
||||
|
|
||||
|
# The shared and static library names
|
||||
|
LIB_SHBASENAME := libi2c.so |
||||
|
LIB_SHSONAME := $(LIB_SHBASENAME).$(LIB_MAINVER) |
||||
|
LIB_SHLIBNAME := $(LIB_SHBASENAME).$(LIB_VER) |
||||
|
LIB_STLIBNAME := libi2c.a |
||||
|
|
||||
|
LIB_TARGETS := $(LIB_SHLIBNAME) |
||||
|
LIB_LINKS := $(LIB_SHSONAME) $(LIB_SHBASENAME) |
||||
|
LIB_OBJECTS := smbus.o |
||||
|
ifeq ($(BUILD_STATIC_LIB),1) |
||||
|
LIB_TARGETS += $(LIB_STLIBNAME) |
||||
|
LIB_OBJECTS += smbus.ao |
||||
|
endif |
||||
|
|
||||
|
#
|
||||
|
# Libraries
|
||||
|
#
|
||||
|
|
||||
|
$(LIB_DIR)/$(LIB_SHLIBNAME): $(LIB_DIR)/smbus.o |
||||
|
$(CC) -shared $(LDFLAGS) -Wl,--version-script=$(LIB_DIR)/libi2c.map -Wl,-soname,$(LIB_SHSONAME) -o $@ $^ -lc |
||||
|
|
||||
|
$(LIB_DIR)/$(LIB_SHSONAME): |
||||
|
$(RM) $@ |
||||
|
$(LN) $(LIB_SHLIBNAME) $@ |
||||
|
|
||||
|
$(LIB_DIR)/$(LIB_SHBASENAME): |
||||
|
$(RM) $@ |
||||
|
$(LN) $(LIB_SHLIBNAME) $@ |
||||
|
|
||||
|
$(LIB_DIR)/$(LIB_STLIBNAME): $(LIB_DIR)/smbus.ao |
||||
|
$(RM) $@ |
||||
|
$(AR) rcvs $@ $^ |
||||
|
|
||||
|
#
|
||||
|
# Objects
|
||||
|
# Each object must be built twice, once for the shared library and
|
||||
|
# once again for the static library.
|
||||
|
#
|
||||
|
|
||||
|
$(LIB_DIR)/smbus.o: $(LIB_DIR)/smbus.c $(INCLUDE_DIR)/i2c/smbus.h |
||||
|
$(CC) $(SOCFLAGS) $(LIB_CFLAGS) -c $< -o $@ |
||||
|
|
||||
|
$(LIB_DIR)/smbus.ao: $(LIB_DIR)/smbus.c $(INCLUDE_DIR)/i2c/smbus.h |
||||
|
$(CC) $(CFLAGS) $(LIB_CFLAGS) -c $< -o $@ |
||||
|
|
||||
|
#
|
||||
|
# Commands
|
||||
|
#
|
||||
|
|
||||
|
all-lib: $(addprefix $(LIB_DIR)/,$(LIB_TARGETS) $(LIB_LINKS)) |
||||
|
|
||||
|
strip-lib: $(addprefix $(LIB_DIR)/,$(LIB_TARGETS)) |
||||
|
strip $(addprefix $(LIB_DIR)/,$(LIB_TARGETS)) |
||||
|
|
||||
|
clean-lib: |
||||
|
$(RM) $(addprefix $(LIB_DIR)/,*.o *.ao $(LIB_TARGETS) $(LIB_LINKS)) |
||||
|
|
||||
|
install-lib: $(addprefix $(LIB_DIR)/,$(LIB_TARGETS)) |
||||
|
$(INSTALL_DIR) $(DESTDIR)$(libdir) |
||||
|
$(INSTALL_PROGRAM) $(LIB_DIR)/$(LIB_SHLIBNAME) $(DESTDIR)$(libdir) |
||||
|
$(LN) $(LIB_SHLIBNAME) $(DESTDIR)$(libdir)/$(LIB_SHSONAME) |
||||
|
$(LN) $(LIB_SHSONAME) $(DESTDIR)$(libdir)/$(LIB_SHBASENAME) |
||||
|
ifeq ($(BUILD_STATIC_LIB),1) |
||||
|
$(INSTALL_DATA) $(LIB_DIR)/$(LIB_STLIBNAME) $(DESTDIR)$(libdir) |
||||
|
endif |
||||
|
|
||||
|
uninstall-lib: |
||||
|
for library in $(LIB_TARGETS) $(LIB_LINKS) ; do \
|
||||
|
$(RM) $(DESTDIR)$(libdir)/$$library ; done |
||||
|
|
||||
|
all: all-lib |
||||
|
|
||||
|
strip: strip-lib |
||||
|
|
||||
|
clean: clean-lib |
||||
|
|
||||
|
install: install-lib |
||||
|
|
||||
|
uninstall: uninstall-lib |
@ -0,0 +1,18 @@ |
|||||
|
{ |
||||
|
global: |
||||
|
i2c_smbus_access; |
||||
|
i2c_smbus_write_quick; |
||||
|
i2c_smbus_read_byte; |
||||
|
i2c_smbus_write_byte; |
||||
|
i2c_smbus_read_byte_data; |
||||
|
i2c_smbus_write_byte_data; |
||||
|
i2c_smbus_read_word_data; |
||||
|
i2c_smbus_write_word_data; |
||||
|
i2c_smbus_process_call; |
||||
|
i2c_smbus_read_block_data; |
||||
|
i2c_smbus_write_block_data; |
||||
|
i2c_smbus_read_i2c_block_data; |
||||
|
i2c_smbus_write_i2c_block_data; |
||||
|
i2c_smbus_block_process_call; |
||||
|
local: *; |
||||
|
}; |
@ -0,0 +1,203 @@ |
|||||
|
/* |
||||
|
smbus.c - SMBus level access helper functions |
||||
|
|
||||
|
Copyright (C) 1995-97 Simon G. Vogl |
||||
|
Copyright (C) 1998-99 Frodo Looijaard <frodol@dds.nl> |
||||
|
|
||||
|
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., 51 Franklin Street, Fifth Floor, Boston, |
||||
|
MA 02110-1301 USA. |
||||
|
*/ |
||||
|
|
||||
|
#include <i2c/smbus.h> |
||||
|
#include <sys/ioctl.h> |
||||
|
#include <linux/types.h> |
||||
|
#include <linux/i2c.h> |
||||
|
#include <linux/i2c-dev.h> |
||||
|
|
||||
|
/* Compatibility defines */ |
||||
|
#ifndef I2C_SMBUS_I2C_BLOCK_BROKEN |
||||
|
#define I2C_SMBUS_I2C_BLOCK_BROKEN I2C_SMBUS_I2C_BLOCK_DATA |
||||
|
#endif |
||||
|
#ifndef I2C_FUNC_SMBUS_PEC |
||||
|
#define I2C_FUNC_SMBUS_PEC I2C_FUNC_SMBUS_HWPEC_CALC |
||||
|
#endif |
||||
|
|
||||
|
__s32 i2c_smbus_access(int file, char read_write, __u8 command, |
||||
|
int size, union i2c_smbus_data *data) |
||||
|
{ |
||||
|
struct i2c_smbus_ioctl_data args; |
||||
|
|
||||
|
args.read_write = read_write; |
||||
|
args.command = command; |
||||
|
args.size = size; |
||||
|
args.data = data; |
||||
|
return ioctl(file, I2C_SMBUS, &args); |
||||
|
} |
||||
|
|
||||
|
|
||||
|
__s32 i2c_smbus_write_quick(int file, __u8 value) |
||||
|
{ |
||||
|
return i2c_smbus_access(file, value, 0, I2C_SMBUS_QUICK, NULL); |
||||
|
} |
||||
|
|
||||
|
__s32 i2c_smbus_read_byte(int file) |
||||
|
{ |
||||
|
union i2c_smbus_data data; |
||||
|
if (i2c_smbus_access(file, I2C_SMBUS_READ, 0, I2C_SMBUS_BYTE, &data)) |
||||
|
return -1; |
||||
|
else |
||||
|
return 0x0FF & data.byte; |
||||
|
} |
||||
|
|
||||
|
__s32 i2c_smbus_write_byte(int file, __u8 value) |
||||
|
{ |
||||
|
return i2c_smbus_access(file, I2C_SMBUS_WRITE, value, |
||||
|
I2C_SMBUS_BYTE, NULL); |
||||
|
} |
||||
|
|
||||
|
__s32 i2c_smbus_read_byte_data(int file, __u8 command) |
||||
|
{ |
||||
|
union i2c_smbus_data data; |
||||
|
if (i2c_smbus_access(file, I2C_SMBUS_READ, command, |
||||
|
I2C_SMBUS_BYTE_DATA, &data)) |
||||
|
return -1; |
||||
|
else |
||||
|
return 0x0FF & data.byte; |
||||
|
} |
||||
|
|
||||
|
__s32 i2c_smbus_write_byte_data(int file, __u8 command, __u8 value) |
||||
|
{ |
||||
|
union i2c_smbus_data data; |
||||
|
data.byte = value; |
||||
|
return i2c_smbus_access(file, I2C_SMBUS_WRITE, command, |
||||
|
I2C_SMBUS_BYTE_DATA, &data); |
||||
|
} |
||||
|
|
||||
|
__s32 i2c_smbus_read_word_data(int file, __u8 command) |
||||
|
{ |
||||
|
union i2c_smbus_data data; |
||||
|
if (i2c_smbus_access(file, I2C_SMBUS_READ, command, |
||||
|
I2C_SMBUS_WORD_DATA, &data)) |
||||
|
return -1; |
||||
|
else |
||||
|
return 0x0FFFF & data.word; |
||||
|
} |
||||
|
|
||||
|
__s32 i2c_smbus_write_word_data(int file, __u8 command, __u16 value) |
||||
|
{ |
||||
|
union i2c_smbus_data data; |
||||
|
data.word = value; |
||||
|
return i2c_smbus_access(file, I2C_SMBUS_WRITE, command, |
||||
|
I2C_SMBUS_WORD_DATA, &data); |
||||
|
} |
||||
|
|
||||
|
__s32 i2c_smbus_process_call(int file, __u8 command, __u16 value) |
||||
|
{ |
||||
|
union i2c_smbus_data data; |
||||
|
data.word = value; |
||||
|
if (i2c_smbus_access(file, I2C_SMBUS_WRITE, command, |
||||
|
I2C_SMBUS_PROC_CALL, &data)) |
||||
|
return -1; |
||||
|
else |
||||
|
return 0x0FFFF & data.word; |
||||
|
} |
||||
|
|
||||
|
/* Returns the number of read bytes */ |
||||
|
__s32 i2c_smbus_read_block_data(int file, __u8 command, __u8 *values) |
||||
|
{ |
||||
|
union i2c_smbus_data data; |
||||
|
int i; |
||||
|
if (i2c_smbus_access(file, I2C_SMBUS_READ, command, |
||||
|
I2C_SMBUS_BLOCK_DATA, &data)) |
||||
|
return -1; |
||||
|
else { |
||||
|
for (i = 1; i <= data.block[0]; i++) |
||||
|
values[i-1] = data.block[i]; |
||||
|
return data.block[0]; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
__s32 i2c_smbus_write_block_data(int file, __u8 command, __u8 length, |
||||
|
const __u8 *values) |
||||
|
{ |
||||
|
union i2c_smbus_data data; |
||||
|
int i; |
||||
|
if (length > 32) |
||||
|
length = 32; |
||||
|
for (i = 1; i <= length; i++) |
||||
|
data.block[i] = values[i-1]; |
||||
|
data.block[0] = length; |
||||
|
return i2c_smbus_access(file, I2C_SMBUS_WRITE, command, |
||||
|
I2C_SMBUS_BLOCK_DATA, &data); |
||||
|
} |
||||
|
|
||||
|
/* Returns the number of read bytes */ |
||||
|
/* Until kernel 2.6.22, the length is hardcoded to 32 bytes. If you |
||||
|
ask for less than 32 bytes, your code will only work with kernels |
||||
|
2.6.23 and later. */ |
||||
|
__s32 i2c_smbus_read_i2c_block_data(int file, __u8 command, __u8 length, |
||||
|
__u8 *values) |
||||
|
{ |
||||
|
union i2c_smbus_data data; |
||||
|
int i; |
||||
|
|
||||
|
if (length > 32) |
||||
|
length = 32; |
||||
|
data.block[0] = length; |
||||
|
if (i2c_smbus_access(file, I2C_SMBUS_READ, command, |
||||
|
length == 32 ? I2C_SMBUS_I2C_BLOCK_BROKEN : |
||||
|
I2C_SMBUS_I2C_BLOCK_DATA, &data)) |
||||
|
return -1; |
||||
|
else { |
||||
|
for (i = 1; i <= data.block[0]; i++) |
||||
|
values[i-1] = data.block[i]; |
||||
|
return data.block[0]; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
__s32 i2c_smbus_write_i2c_block_data(int file, __u8 command, __u8 length, |
||||
|
const __u8 *values) |
||||
|
{ |
||||
|
union i2c_smbus_data data; |
||||
|
int i; |
||||
|
if (length > 32) |
||||
|
length = 32; |
||||
|
for (i = 1; i <= length; i++) |
||||
|
data.block[i] = values[i-1]; |
||||
|
data.block[0] = length; |
||||
|
return i2c_smbus_access(file, I2C_SMBUS_WRITE, command, |
||||
|
I2C_SMBUS_I2C_BLOCK_BROKEN, &data); |
||||
|
} |
||||
|
|
||||
|
/* Returns the number of read bytes */ |
||||
|
__s32 i2c_smbus_block_process_call(int file, __u8 command, __u8 length, |
||||
|
__u8 *values) |
||||
|
{ |
||||
|
union i2c_smbus_data data; |
||||
|
int i; |
||||
|
if (length > 32) |
||||
|
length = 32; |
||||
|
for (i = 1; i <= length; i++) |
||||
|
data.block[i] = values[i-1]; |
||||
|
data.block[0] = length; |
||||
|
if (i2c_smbus_access(file, I2C_SMBUS_WRITE, command, |
||||
|
I2C_SMBUS_BLOCK_PROC_CALL, &data)) |
||||
|
return -1; |
||||
|
else { |
||||
|
for (i = 1; i <= data.block[0]; i++) |
||||
|
values[i-1] = data.block[i]; |
||||
|
return data.block[0]; |
||||
|
} |
||||
|
} |
Write
Preview
Loading…
Cancel
Save
Reference in new issue