]> sigrok.org Git - libsigrok.git/blob - src/ezusb.c
Build: Include <config.h> first in all source files
[libsigrok.git] / src / ezusb.c
1 /*
2  * This file is part of the libsigrok project.
3  *
4  * Copyright (C) 2010-2012 Bert Vermeulen <bert@biot.com>
5  *
6  * This program is free software: you can redistribute it and/or modify
7  * it under the terms of the GNU General Public License as published by
8  * the Free Software Foundation, either version 3 of the License, or
9  * (at your option) any later version.
10  *
11  * This program is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  * GNU General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
18  */
19
20 /*
21  * Helper functions for the Cypress EZ-USB / FX2 series chips.
22  */
23
24 #include <config.h>
25 #include <libusb.h>
26 #include <glib.h>
27 #include <glib/gstdio.h>
28 #include <stdio.h>
29 #include <errno.h>
30 #include <string.h>
31 #include <libsigrok/libsigrok.h>
32 #include "libsigrok-internal.h"
33
34 #define LOG_PREFIX "ezusb"
35
36 SR_PRIV int ezusb_reset(struct libusb_device_handle *hdl, int set_clear)
37 {
38         int ret;
39         unsigned char buf[1];
40
41         sr_info("setting CPU reset mode %s...",
42                 set_clear ? "on" : "off");
43         buf[0] = set_clear ? 1 : 0;
44         ret = libusb_control_transfer(hdl, LIBUSB_REQUEST_TYPE_VENDOR, 0xa0,
45                                       0xe600, 0x0000, buf, 1, 100);
46         if (ret < 0)
47                 sr_err("Unable to send control request: %s.",
48                                 libusb_error_name(ret));
49
50         return ret;
51 }
52
53 SR_PRIV int ezusb_install_firmware(libusb_device_handle *hdl,
54                                    const char *filename)
55 {
56         FILE *fw;
57         int offset, chunksize, ret, result;
58         unsigned char buf[4096];
59
60         sr_info("Uploading firmware at %s", filename);
61         if (!(fw = g_fopen(filename, "rb"))) {
62                 sr_err("Unable to open firmware file %s for reading: %s",
63                        filename, g_strerror(errno));
64                 return SR_ERR;
65         }
66
67         result = SR_OK;
68         offset = 0;
69         while (1) {
70                 chunksize = fread(buf, 1, 4096, fw);
71                 if (chunksize == 0)
72                         break;
73                 ret = libusb_control_transfer(hdl, LIBUSB_REQUEST_TYPE_VENDOR |
74                                               LIBUSB_ENDPOINT_OUT, 0xa0, offset,
75                                               0x0000, buf, chunksize, 100);
76                 if (ret < 0) {
77                         sr_err("Unable to send firmware to device: %s.",
78                                         libusb_error_name(ret));
79                         result = SR_ERR;
80                         break;
81                 }
82                 sr_info("Uploaded %d bytes", chunksize);
83                 offset += chunksize;
84         }
85         fclose(fw);
86         sr_info("Firmware upload done");
87
88         return result;
89 }
90
91 SR_PRIV int ezusb_upload_firmware(libusb_device *dev, int configuration,
92                                   const char *filename)
93 {
94         struct libusb_device_handle *hdl;
95         int ret;
96
97         sr_info("uploading firmware to device on %d.%d",
98                 libusb_get_bus_number(dev), libusb_get_device_address(dev));
99
100         if ((ret = libusb_open(dev, &hdl)) < 0) {
101                 sr_err("failed to open device: %s.", libusb_error_name(ret));
102                 return SR_ERR;
103         }
104
105 /*
106  * The libusb Darwin backend is broken: it can report a kernel driver being
107  * active, but detaching it always returns an error.
108  */
109 #if !defined(__APPLE__)
110         if (libusb_kernel_driver_active(hdl, 0) == 1) {
111                 if ((ret = libusb_detach_kernel_driver(hdl, 0)) < 0) {
112                         sr_err("failed to detach kernel driver: %s",
113                                         libusb_error_name(ret));
114                         return SR_ERR;
115                 }
116         }
117 #endif
118
119         if ((ret = libusb_set_configuration(hdl, configuration)) < 0) {
120                 sr_err("Unable to set configuration: %s",
121                                 libusb_error_name(ret));
122                 return SR_ERR;
123         }
124
125         if ((ezusb_reset(hdl, 1)) < 0)
126                 return SR_ERR;
127
128         if (ezusb_install_firmware(hdl, filename) < 0)
129                 return SR_ERR;
130
131         if ((ezusb_reset(hdl, 0)) < 0)
132                 return SR_ERR;
133
134         libusb_close(hdl);
135
136         return SR_OK;
137 }