scx200_i2c.c revision 4a029abee0f1d69cb0445657d6fa5a38597bd17d
1f30c2269544bffc7bf1b0d7c0abe5be1be83b8cbUwe Zeisberger/* linux/drivers/i2c/busses/scx200_i2c.c 21da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 31da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds Copyright (c) 2001,2002 Christer Weinigel <wingel@nano-system.com> 41da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 51da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds National Semiconductor SCx200 I2C bus on GPIO pins 61da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 71da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds Based on i2c-velleman.c Copyright (C) 1995-96, 2000 Simon G. Vogl 81da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 91da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds This program is free software; you can redistribute it and/or modify 101da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds it under the terms of the GNU General Public License as published by 111da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds the Free Software Foundation; either version 2 of the License, or 121da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds (at your option) any later version. 131da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 141da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds This program is distributed in the hope that it will be useful, 151da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds but WITHOUT ANY WARRANTY; without even the implied warranty of 161da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 171da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds GNU General Public License for more details. 181da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 191da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds You should have received a copy of the GNU General Public License 201da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds along with this program; if not, write to the Free Software 211da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. 221da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds*/ 231da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 241da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds#include <linux/module.h> 251da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds#include <linux/errno.h> 261da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds#include <linux/kernel.h> 271da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds#include <linux/init.h> 281da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds#include <linux/i2c.h> 291da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds#include <linux/i2c-algo-bit.h> 301da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds#include <asm/io.h> 311da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 321da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds#include <linux/scx200_gpio.h> 331da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 341da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds#define NAME "scx200_i2c" 351da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 361da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus TorvaldsMODULE_AUTHOR("Christer Weinigel <wingel@nano-system.com>"); 371da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus TorvaldsMODULE_DESCRIPTION("NatSemi SCx200 I2C Driver"); 381da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus TorvaldsMODULE_LICENSE("GPL"); 391da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 401da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvaldsstatic int scl = CONFIG_SCx200_I2C_SCL; 411da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvaldsstatic int sda = CONFIG_SCx200_I2C_SDA; 421da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 431da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvaldsmodule_param(scl, int, 0); 441da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus TorvaldsMODULE_PARM_DESC(scl, "GPIO line for SCL"); 451da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvaldsmodule_param(sda, int, 0); 461da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus TorvaldsMODULE_PARM_DESC(sda, "GPIO line for SDA"); 471da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 481da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvaldsstatic void scx200_i2c_setscl(void *data, int state) 491da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds{ 501da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds scx200_gpio_set(scl, state); 511da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds} 521da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 531da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvaldsstatic void scx200_i2c_setsda(void *data, int state) 541da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds{ 551da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds scx200_gpio_set(sda, state); 561da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds} 571da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 581da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvaldsstatic int scx200_i2c_getscl(void *data) 591da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds{ 601da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds return scx200_gpio_get(scl); 611da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds} 621da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 631da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvaldsstatic int scx200_i2c_getsda(void *data) 641da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds{ 651da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds return scx200_gpio_get(sda); 661da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds} 671da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 681da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds/* ------------------------------------------------------------------------ 691da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds * Encapsulate the above functions in the correct operations structure. 701da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds * This is only done when more than one hardware adapter is supported. 711da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds */ 721da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 731da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvaldsstatic struct i2c_algo_bit_data scx200_i2c_data = { 74a0d9c63d3640bd4fc90a408e8334754ef44bcf48Jean Delvare .setsda = scx200_i2c_setsda, 75a0d9c63d3640bd4fc90a408e8334754ef44bcf48Jean Delvare .setscl = scx200_i2c_setscl, 76a0d9c63d3640bd4fc90a408e8334754ef44bcf48Jean Delvare .getsda = scx200_i2c_getsda, 77a0d9c63d3640bd4fc90a408e8334754ef44bcf48Jean Delvare .getscl = scx200_i2c_getscl, 78a0d9c63d3640bd4fc90a408e8334754ef44bcf48Jean Delvare .udelay = 10, 79a0d9c63d3640bd4fc90a408e8334754ef44bcf48Jean Delvare .timeout = 100, 801da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds}; 811da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 821da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvaldsstatic struct i2c_adapter scx200_i2c_ops = { 831da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds .owner = THIS_MODULE, 844a029abee0f1d69cb0445657d6fa5a38597bd17dLennart Sorensen .class = I2C_CLASS_HWMON | I2C_CLASS_SPD, 859ace555d7d87c55ceab6999be444c9a17e0e79b4Stephen Hemminger .id = I2C_HW_B_SCX200, 861da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds .algo_data = &scx200_i2c_data, 871da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds .name = "NatSemi SCx200 I2C", 881da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds}; 891da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 901da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvaldsstatic int scx200_i2c_init(void) 911da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds{ 921da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds pr_debug(NAME ": NatSemi SCx200 I2C Driver\n"); 931da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 941da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds if (!scx200_gpio_present()) { 951da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds printk(KERN_ERR NAME ": no SCx200 gpio pins available\n"); 961da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds return -ENODEV; 971da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds } 981da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 991da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds pr_debug(NAME ": SCL=GPIO%02u, SDA=GPIO%02u\n", scl, sda); 1001da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 1011da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds if (scl == -1 || sda == -1 || scl == sda) { 1021da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds printk(KERN_ERR NAME ": scl and sda must be specified\n"); 1031da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds return -EINVAL; 1041da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds } 1051da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 1061da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds /* Configure GPIOs as open collector outputs */ 1071da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds scx200_gpio_configure(scl, ~2, 5); 1081da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds scx200_gpio_configure(sda, ~2, 5); 1091da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 1101da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds if (i2c_bit_add_bus(&scx200_i2c_ops) < 0) { 1111da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds printk(KERN_ERR NAME ": adapter %s registration failed\n", 1121da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds scx200_i2c_ops.name); 1131da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds return -ENODEV; 1141da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds } 1151da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 1161da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds return 0; 1171da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds} 1181da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 1191da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvaldsstatic void scx200_i2c_cleanup(void) 1201da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds{ 1213269711b76ba27b78862c48398b0d313ccaa99c2Jean Delvare i2c_del_adapter(&scx200_i2c_ops); 1221da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds} 1231da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 1241da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvaldsmodule_init(scx200_i2c_init); 1251da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvaldsmodule_exit(scx200_i2c_cleanup); 1261da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds 1271da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds/* 1281da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds Local variables: 1291da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds compile-command: "make -k -C ../.. SUBDIRS=drivers/i2c modules" 1301da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds c-basic-offset: 8 1311da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds End: 1321da177e4c3f41524e886b7f1b8a0c1fc7321cacLinus Torvalds*/ 133