summaryrefslogtreecommitdiff
path: root/commandline/i2cdetect.c
diff options
context:
space:
mode:
authorDaniel Friesel <derf@finalrewind.org>2013-07-16 19:26:39 +0200
committerDaniel Friesel <derf@finalrewind.org>2013-07-16 19:26:39 +0200
commit6011c99aca409a4d78c773cbdbe24e09d3286b96 (patch)
tree27fca60de0810d9a1190ed76e0822a118f759a91 /commandline/i2cdetect.c
parentc7f8f42294e1d338d44011046673e73c0a73e1ab (diff)
add i2cdetect
Diffstat (limited to 'commandline/i2cdetect.c')
-rw-r--r--commandline/i2cdetect.c296
1 files changed, 296 insertions, 0 deletions
diff --git a/commandline/i2cdetect.c b/commandline/i2cdetect.c
new file mode 100644
index 0000000..cf31e15
--- /dev/null
+++ b/commandline/i2cdetect.c
@@ -0,0 +1,296 @@
+/* Name: powerSwitch.c
+ * Project: PowerSwitch based on AVR USB driver
+ * Author: Christian Starkjohann
+ * Creation Date: 2005-01-16
+ * Tabsize: 4
+ * Copyright: (c) 2005 by OBJECTIVE DEVELOPMENT Software GmbH
+ * License: GNU GPL v2 (see License.txt) or proprietary (CommercialLicense.txt)
+ * This Revision: $Id$
+ */
+
+/*
+General Description:
+This program controls the PowerSwitch USB device from the command line.
+It must be linked with libusb, a library for accessing the USB bus from
+Linux, FreeBSD, Mac OS X and other Unix operating systems. Libusb can be
+obtained from http://libusb.sourceforge.net/.
+*/
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <usb.h> /* this is libusb, see http://libusb.sourceforge.net/ */
+
+
+#define BIT_SDA 6
+#define BIT_SCL 7
+
+
+#define USBDEV_SHARED_VENDOR 0x16C0 /* VOTI */
+#define USBDEV_SHARED_PRODUCT 0x05DC /* Obdev's free shared PID */
+/* Use obdev's generic shared VID/PID pair and follow the rules outlined
+ * in firmware/usbdrv/USBID-License.txt.
+ */
+
+#define PSCMD_ECHO 0
+#define PSCMD_GET 1
+#define PSCMD_ON 2
+#define PSCMD_OFF 3
+/* These are the vendor specific SETUP commands implemented by our USB device */
+
+
+usb_dev_handle *handle = NULL;
+
+static int usbGetStringAscii(usb_dev_handle *dev, int index, int langid, char *buf, int buflen)
+{
+char buffer[256];
+int rval, i;
+
+ if((rval = usb_control_msg(dev, USB_ENDPOINT_IN, USB_REQ_GET_DESCRIPTOR, (USB_DT_STRING << 8) + index, langid, buffer, sizeof(buffer), 1000)) < 0)
+ return rval;
+ if(buffer[1] != USB_DT_STRING)
+ return 0;
+ if((unsigned char)buffer[0] < rval)
+ rval = (unsigned char)buffer[0];
+ rval /= 2;
+ /* lossy conversion to ISO Latin1 */
+ for(i=1;i<rval;i++){
+ if(i > buflen) /* destination buffer overflow */
+ break;
+ buf[i-1] = buffer[2 * i];
+ if(buffer[2 * i + 1] != 0) /* outside of ISO Latin1 range */
+ buf[i-1] = '?';
+ }
+ buf[i-1] = 0;
+ return i-1;
+}
+
+
+/* PowerSwitch uses the free shared default VID/PID. If you want to see an
+ * example device lookup where an individually reserved PID is used, see our
+ * RemoteSensor reference implementation.
+ */
+
+#define USB_ERROR_NOTFOUND 1
+#define USB_ERROR_ACCESS 2
+#define USB_ERROR_IO 3
+
+static int usbOpenDevice(usb_dev_handle **device, int vendor, char *vendorName, int product, char *productName)
+{
+struct usb_bus *bus;
+struct usb_device *dev;
+usb_dev_handle *handle = NULL;
+int errorCode = USB_ERROR_NOTFOUND;
+static int didUsbInit = 0;
+
+ if(!didUsbInit){
+ didUsbInit = 1;
+ usb_init();
+ }
+ usb_find_busses();
+ usb_find_devices();
+ for(bus=usb_get_busses(); bus; bus=bus->next){
+ for(dev=bus->devices; dev; dev=dev->next){
+ if(dev->descriptor.idVendor == vendor && dev->descriptor.idProduct == product){
+ char string[256];
+ int len;
+ handle = usb_open(dev); /* we need to open the device in order to query strings */
+ if(!handle){
+ errorCode = USB_ERROR_ACCESS;
+ fprintf(stderr, "Warning: cannot open USB device: %s\n", usb_strerror());
+ continue;
+ }
+ if(vendorName == NULL && productName == NULL){ /* name does not matter */
+ break;
+ }
+ /* now check whether the names match: */
+ len = usbGetStringAscii(handle, dev->descriptor.iManufacturer, 0x0409, string, sizeof(string));
+ if(len < 0){
+ errorCode = USB_ERROR_IO;
+ fprintf(stderr, "Warning: cannot query manufacturer for device: %s\n", usb_strerror());
+ }else{
+ errorCode = USB_ERROR_NOTFOUND;
+ /* fprintf(stderr, "seen device from vendor ->%s<-\n", string); */
+ if(strcmp(string, vendorName) == 0){
+ len = usbGetStringAscii(handle, dev->descriptor.iProduct, 0x0409, string, sizeof(string));
+ if(len < 0){
+ errorCode = USB_ERROR_IO;
+ fprintf(stderr, "Warning: cannot query product for device: %s\n", usb_strerror());
+ }else{
+ errorCode = USB_ERROR_NOTFOUND;
+ /* fprintf(stderr, "seen product ->%s<-\n", string); */
+ if(strcmp(string, productName) == 0)
+ break;
+ }
+ }
+ }
+ usb_close(handle);
+ handle = NULL;
+ }
+ }
+ if(handle)
+ break;
+ }
+ if(handle != NULL){
+ errorCode = 0;
+ *device = handle;
+ }
+ return errorCode;
+}
+
+
+static unsigned char get_status()
+{
+ unsigned char buffer[8];
+ int nBytes = usb_control_msg(handle, USB_TYPE_VENDOR | USB_RECIP_DEVICE |
+ USB_ENDPOINT_IN, PSCMD_GET, 0, 0, (char *)buffer, sizeof(buffer), 5000);
+
+ if (nBytes < 2) {
+ fprintf(stderr, "ERR: read status: got %d bytes, expected 2\n", nBytes);
+ exit(1);
+ }
+ return buffer[0];
+}
+
+static void verify_low(unsigned char bit, char *bitname)
+{
+ int i;
+ for (i = 0; i < 10; i++) {
+ if (~get_status() & (1 << bit))
+ return;
+ usleep(10);
+ }
+ fprintf(stderr, "%s stuck high for >1ms\n", bitname);
+}
+
+static void verify_high(unsigned char bit, char *bitname)
+{
+ int i;
+ for (i = 0; i < 100; i++) {
+ if (get_status() & (1 << bit))
+ return;
+ usleep(10);
+ }
+ fprintf(stderr, "%s stuck low for >1ms\n", bitname);
+}
+
+static void verify_sda_low()
+{
+ verify_low(BIT_SDA, "SDA");
+}
+
+static void verify_sda_high()
+{
+ verify_high(BIT_SDA, "SDA");
+}
+
+static void verify_scl_low()
+{
+ verify_low(BIT_SCL, "SCL");
+}
+
+static void verify_scl_high()
+{
+ verify_high(BIT_SCL, "SCL");
+}
+
+
+
+/*
+ * Firmware: DDRB = ~b (with hardware pull-ups and PORTB = 0)
+ * So to pull an output high (1), we turn it on, which sets it as input
+ * -> pull-up works. for low (0): turn it off -> output -> pulled low
+ */
+
+void set_sda(char value) {
+ // discarded
+ unsigned char buffer[8];
+
+ usb_control_msg(handle, USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_ENDPOINT_IN,
+ (value ? PSCMD_ON : PSCMD_OFF), 0, BIT_SDA,
+ (char *)buffer, sizeof(buffer), 5000);
+}
+
+void set_scl(char value) {
+ // discarded
+ unsigned char buffer[8];
+
+ usb_control_msg(handle, USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_ENDPOINT_IN,
+ (value ? PSCMD_ON : PSCMD_OFF), 0, BIT_SCL,
+ (char *)buffer, sizeof(buffer), 5000);
+}
+
+int main(int argc, char **argv)
+{
+unsigned char buffer[8];
+
+ usb_init();
+ if(usbOpenDevice(&handle, USBDEV_SHARED_VENDOR, "www.obdev.at", USBDEV_SHARED_PRODUCT, "PowerSwitch") != 0){
+ fprintf(stderr, "Could not find USB device \"PowerSwitch\" with vid=0x%x pid=0x%x\n", USBDEV_SHARED_VENDOR, USBDEV_SHARED_PRODUCT);
+ exit(1);
+ }
+/* We have searched all devices on all busses for our USB device above. Now
+ * try to open it and perform the vendor specific control operations for the
+ * function requested by the user.
+ */
+
+ char line[8];
+ signed char i;
+ unsigned char id, i2cid;
+
+
+ fputs(" 0 1 2 3 4 5 6 7 8 9 a b c d e f", stdout);
+ for (id = 0; id < 128; id++) {
+
+ i2cid = (id << 1) | 1;
+
+ fflush(stdout);
+
+ if ((id & 0x0f) == 0) {
+ printf("\n%02x:", id);
+ }
+
+ set_sda(1);
+ set_scl(1);
+ usleep(1000);
+ set_sda(0);
+ usleep(1000);
+ set_scl(0);
+ verify_sda_low();
+ verify_scl_low();
+ for (i = 7; i >= -1; i--) {
+ if ((i < 0) || (i2cid & (1 << i))) {
+ set_sda(1);
+ //verify_sda_high();
+ } else {
+ set_sda(0);
+ //verify_sda_low();
+ }
+ usleep(10);
+ set_scl(1);
+ usleep(10);
+ //verify_scl_high();
+ if (i < 0) {
+ if (get_status() & (1 << BIT_SDA))
+ fputs(" --", stdout);
+ else
+ printf(" %02x", id);
+ }
+ set_scl(0);
+ usleep(10);
+ //verify_scl_low();
+ }
+
+ set_scl(1);
+ usleep(100);
+ //verify_scl_high();
+ set_sda(1);
+ usleep(100);
+ //verify_sda_high();
+ }
+
+ fputs("\n", stdout);
+
+ usb_close(handle);
+ return 0;
+}