FS#12102 - Manual, "Quick Start" section for AMSv2 players: Clarifies
[kugel-rb.git] / utils / imx_hid_recovery / imx_hid_recovery.c
blob546e2443387ae5fb53e49accf8daa935459be5db
1 #include <stdio.h>
2 #include <stdlib.h>
3 #include <string.h>
4 #include <libusb.h>
5 #include <stdint.h>
7 void put32le(uint8_t *buf, uint32_t i)
9 *buf++ = i & 0xff;
10 *buf++ = (i >> 8) & 0xff;
11 *buf++ = (i >> 16) & 0xff;
12 *buf++ = (i >> 24) & 0xff;
15 void put32be(uint8_t *buf, uint32_t i)
17 *buf++ = (i >> 24) & 0xff;
18 *buf++ = (i >> 16) & 0xff;
19 *buf++ = (i >> 8) & 0xff;
20 *buf++ = i & 0xff;
23 int main(int argc, char **argv)
25 int ret;
26 uint8_t msg[0x20];
27 uint8_t *p;
28 FILE *f;
29 int i, xfer_size, nr_xfers, recv_size;
31 if(argc != 3)
33 printf("usage: %s <xfer size> <file>\n", argv[0]);
34 return 1;
37 char *end;
38 xfer_size = strtol(argv[1], &end, 0);
39 if(end != (argv[1] + strlen(argv[1])))
41 printf("Invalid transfer size !\n");
42 return 1;
45 libusb_device_handle *dev;
47 libusb_init(NULL);
49 libusb_set_debug(NULL, 3);
51 dev = libusb_open_device_with_vid_pid(NULL, 0x066F, 0x3780);
52 if(dev == NULL)
54 printf("Cannot open device\n");
55 return 1;
58 libusb_detach_kernel_driver(dev, 0);
59 libusb_detach_kernel_driver(dev, 4);
61 libusb_claim_interface (dev, 0);
62 libusb_claim_interface (dev, 4);
64 if (!dev)
66 printf("No dev\n");
67 exit(1);
70 f = fopen(argv[2], "r");
71 if(f == NULL)
73 perror("cannot open file");
74 return 1;
76 fseek(f, 0, SEEK_END);
77 size_t size = ftell(f);
78 fseek(f, 0, SEEK_SET);
80 printf("Transfer size: %d\n", xfer_size);
81 nr_xfers = (size + xfer_size - 1) / xfer_size;
82 uint8_t *file_buf = malloc(nr_xfers * xfer_size);
83 memset(file_buf, 0xff, nr_xfers * xfer_size); // pad with 0xff
84 if(fread(file_buf, size, 1, f) != 1)
86 perror("read error");
87 fclose(f);
88 return 1;
90 fclose(f);
92 memset(msg, 0, 0x20);
94 p = msg;
96 *p++ = 0x01; // Init upload command
97 *p++ = 'B'; // Signature
98 *p++ = 'L';
99 *p++ = 'T';
100 *p++ = 'C';
101 put32le(p, 0x1); // I guess version or sub-command
102 p += 4;
103 put32le(p, size); // Payload size
105 // The second command starts at 0x20
107 p = &msg[0x10];
109 *p++ = 0x02; // Start upload
110 put32be(p, size); // Payload size, again
112 ret = libusb_control_transfer(dev,
113 LIBUSB_REQUEST_TYPE_CLASS | LIBUSB_RECIPIENT_INTERFACE, 0x9, 0x201, 0,
114 msg, 0x20, 1000);
115 if(ret < 0)
117 printf("transfer error at init step\n");
118 return 1;
121 uint8_t *xfer_buf = malloc(1 + xfer_size);
123 for(i = 0; i < nr_xfers; i++)
125 xfer_buf[0] = 0x2;
126 memcpy(&xfer_buf[1], &file_buf[i * xfer_size], xfer_size);
128 ret = libusb_control_transfer(dev,
129 LIBUSB_REQUEST_TYPE_CLASS | LIBUSB_RECIPIENT_INTERFACE,
130 0x9, 0x202, 0, xfer_buf, xfer_size + 1, 1000);
131 if(ret < 0)
133 printf("transfer error at send step %d\n", i);
134 return 1;
138 ret = libusb_interrupt_transfer(dev, 0x81, xfer_buf, xfer_size, &recv_size,
139 1000);
140 if(ret < 0)
142 printf("transfer error at final stage\n");
143 return 1;
146 printf("ret %i\n", ret);
148 return 0;