]> sigrok.org Git - libsigrok.git/blame - hardware/common/ezusb.c
better cleanup of device/plugin resources
[libsigrok.git] / hardware / common / ezusb.c
CommitLineData
a1bb33af
UH
1/*
2 * This file is part of the sigrok project.
3 *
4 * Copyright (C) 2010 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
b08024a8
UH
24#include <sigrok.h>
25#include <sigrok-internal.h>
a1bb33af
UH
26#include <libusb.h>
27#include <glib.h>
3bbd9849 28#include <glib/gstdio.h>
a1bb33af
UH
29#include <stdio.h>
30#include <errno.h>
31#include <string.h>
32#include "config.h"
33
34int ezusb_reset(struct libusb_device_handle *hdl, int set_clear)
35{
36 int err;
37 unsigned char buf[1];
38
b08024a8 39 sr_info("setting CPU reset mode %s...", set_clear ? "on" : "off");
a1bb33af
UH
40 buf[0] = set_clear ? 1 : 0;
41 err = libusb_control_transfer(hdl, LIBUSB_REQUEST_TYPE_VENDOR, 0xa0,
9d2933fb
UH
42 0xe600, 0x0000, buf, 1, 100);
43 if (err < 0)
b08024a8 44 sr_warn("Unable to send control request: %d", err);
a1bb33af
UH
45
46 return err;
47}
48
a1bb33af
UH
49int ezusb_install_firmware(libusb_device_handle *hdl, const char *filename)
50{
51 FILE *fw;
52 int offset, chunksize, err, result;
53 unsigned char buf[4096];
54
b08024a8 55 sr_info("Uploading firmware at %s", filename);
868d8cef 56 if ((fw = g_fopen(filename, "rb")) == NULL) {
b08024a8
UH
57 sr_warn("Unable to open firmware file %s for reading: %s",
58 filename, strerror(errno));
a1bb33af
UH
59 return 1;
60 }
61
986f7270 62 result = offset = 0;
9d2933fb 63 while (1) {
a1bb33af 64 chunksize = fread(buf, 1, 4096, fw);
9d2933fb 65 if (chunksize == 0)
a1bb33af 66 break;
9d2933fb 67 err = libusb_control_transfer(hdl, LIBUSB_REQUEST_TYPE_VENDOR |
986f7270
UH
68 LIBUSB_ENDPOINT_OUT, 0xa0, offset,
69 0x0000, buf, chunksize, 100);
9d2933fb 70 if (err < 0) {
b08024a8 71 sr_warn("Unable to send firmware to device: %d", err);
a1bb33af
UH
72 result = 1;
73 break;
74 }
b08024a8 75 sr_info("Uploaded %d bytes", chunksize);
a1bb33af
UH
76 offset += chunksize;
77 }
78 fclose(fw);
b08024a8 79 sr_info("Firmware upload done");
a1bb33af
UH
80
81 return result;
82}
edf60d05
UH
83
84int ezusb_upload_firmware(libusb_device *dev, int configuration,
85 const char *filename)
86{
87 struct libusb_device_handle *hdl;
88 int err;
89
b08024a8 90 sr_info("uploading firmware to device on %d.%d",
edf60d05
UH
91 libusb_get_bus_number(dev), libusb_get_device_address(dev));
92
93 err = libusb_open(dev, &hdl);
94 if (err != 0) {
b08024a8 95 sr_warn("failed to open device: %d", err);
edf60d05
UH
96 return 1;
97 }
98
99 err = libusb_set_configuration(hdl, configuration);
100 if (err != 0) {
b08024a8 101 sr_warn("Unable to set configuration: %d", err);
edf60d05
UH
102 return 1;
103 }
104
105 if ((ezusb_reset(hdl, 1)) < 0)
106 return 1;
107
108 if (ezusb_install_firmware(hdl, filename) != 0)
109 return 1;
110
111 if ((ezusb_reset(hdl, 0)) < 0)
112 return 1;
113
114 libusb_close(hdl);
115
116 return 0;
117}