Code formatted

This commit is contained in:
jpk 2024-06-14 22:17:23 +02:00
parent f985a3863b
commit ad5b4c0f7a
6 changed files with 379 additions and 350 deletions

View File

@ -2,6 +2,6 @@
#define config_h
/* defaults */
#define RETRY_MAX 5
#define RETRY_MAX 5
#endif /* config_h */

338
main.c
View File

@ -1,10 +1,10 @@
#include <getopt.h>
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <stdbool.h>
#include <string.h>
#include <time.h>
#include <getopt.h>
#include <unistd.h>
#include "usbms.h"
@ -12,186 +12,190 @@
#define VERSION "1.0"
static int setpassword(const char *password)
{
unsigned long slen = strlen(password);
if (slen < 2 || slen > 12) {
fprintf(stderr, "Password has to be at least 2 but no more than 12 characters long.\n");
return 1;
}
for (int i = 0; i < slen; i++) {
switch (password[i]) {
case '/':
case ':':
case '@':
case '[':
case '`':
case '{':
fprintf(stderr, "Use of invalid character in password. Do not use `%c'.\n",
password[i]);
fprintf(stderr, "Invalid character for passwords are \"/:@[`{\".\n");
return 1;
}
}
printf("Updating password to `%s'... ", password);
if (send_command(ACTIONPRO_OPCODE_SETPASSWORD, (const uint8_t *)password, slen) == ACTIONPRO_CMD_OK)
printf("OK\n");
else
printf("ERROR\n");
return 0;
}
static int setssid(const char *ssid)
{
unsigned long slen = strlen(ssid);
if (slen < 2 || slen > 12) {
fprintf(stderr, "SSID has to be at least 2 but no more than 12 characters long.\n");
return 1;
}
for (int i = 0; i < slen; i++) {
switch (ssid[i]) {
case ' ':
case '~':
fprintf(stderr, "Use of invalid character in SSID. Do not use `%c'.\n",
ssid[i]);
fprintf(stderr, "Invalid character for SSID are \" ~\".\n");
return 1;
}
}
printf("Updating SSID to `%s'... ", ssid);
if (send_command(ACTIONPRO_OPCODE_SETSSID, (const uint8_t *)ssid, slen) == ACTIONPRO_CMD_OK)
printf("OK\n");
else
printf("ERROR\n");
return 0;
}
static int settime()
{
printf("Setting device time... ");
time_t curtime;
struct tm *timeinfo;
if (time(&curtime)) {
timeinfo = localtime(&curtime);
if (timeinfo) {
uint8_t newtime[8] = {0};
const uint16_t year = (timeinfo->tm_year + 1900) >> 8 | (timeinfo->tm_year + 1900) << 8;
const uint8_t month = timeinfo->tm_mon + 1;
memcpy(newtime+0, &year, 2);
memcpy(newtime+2, &month, 1);
memcpy(newtime+3, &timeinfo->tm_mday, 1);
memcpy(newtime+4, &timeinfo->tm_hour, 1);
memcpy(newtime+5, &timeinfo->tm_min, 1);
memcpy(newtime+6, &timeinfo->tm_sec, 1);
if (send_command(ACTIONPRO_OPCODE_SETTIME, newtime, 8) == ACTIONPRO_CMD_OK)
printf("OK\n");
else
printf("ERROR\n");
return 0;
}
}
static int setpassword(const char *password) {
unsigned long slen = strlen(password);
if (slen < 2 || slen > 12) {
fprintf(
stderr,
"Password has to be at least 2 but no more than 12 characters long.\n");
return 1;
}
for (int i = 0; i < slen; i++) {
switch (password[i]) {
case '/':
case ':':
case '@':
case '[':
case '`':
case '{':
fprintf(stderr,
"Use of invalid character in password. Do not use `%c'.\n",
password[i]);
fprintf(stderr, "Invalid character for passwords are \"/:@[`{\".\n");
return 1;
}
}
printf("Updating password to `%s'... ", password);
if (send_command(ACTIONPRO_OPCODE_SETPASSWORD, (const uint8_t *)password,
slen) == ACTIONPRO_CMD_OK)
printf("OK\n");
else
printf("ERROR\n");
return 0;
}
int main(int argc, char *argv[])
{
int c;
static int setssid(const char *ssid) {
unsigned long slen = strlen(ssid);
if (slen < 2 || slen > 12) {
fprintf(stderr,
"SSID has to be at least 2 but no more than 12 characters long.\n");
return 1;
}
bool show_help = false;
char *new_password = NULL;
char *new_ssid = NULL;
bool sync_time = false;
bool show_version = false;
for (int i = 0; i < slen; i++) {
switch (ssid[i]) {
case ' ':
case '~':
fprintf(stderr, "Use of invalid character in SSID. Do not use `%c'.\n",
ssid[i]);
fprintf(stderr, "Invalid character for SSID are \" ~\".\n");
return 1;
}
}
printf("Updating SSID to `%s'... ", ssid);
if (send_command(ACTIONPRO_OPCODE_SETSSID, (const uint8_t *)ssid, slen) ==
ACTIONPRO_CMD_OK)
printf("OK\n");
else
printf("ERROR\n");
printf("ACTIONPRO configuration utility\n");
return 0;
}
while (1) {
static struct option long_options[] = {
{"help" , no_argument , 0, 'p'},
{"password", required_argument, 0, 'p'},
{"ssid" , required_argument, 0, 's'},
{"time" , no_argument , 0, 't'},
{"version" , no_argument , 0, 'v'},
{0, 0, 0, 0}
};
static int settime() {
printf("Setting device time... ");
int option_index = 0;
time_t curtime;
struct tm *timeinfo;
c = getopt_long(argc, argv, "hp:s:tv", long_options, &option_index);
if (time(&curtime)) {
timeinfo = localtime(&curtime);
if (timeinfo) {
uint8_t newtime[8] = {0};
if (argc < 2) {
fprintf(stderr, "At least one option is required. See -h, --help for instructions.\n");
exit(EXIT_FAILURE);
}
const uint16_t year =
(timeinfo->tm_year + 1900) >> 8 | (timeinfo->tm_year + 1900) << 8;
const uint8_t month = timeinfo->tm_mon + 1;
if (c == -1)
break;
memcpy(newtime + 0, &year, 2);
memcpy(newtime + 2, &month, 1);
memcpy(newtime + 3, &timeinfo->tm_mday, 1);
memcpy(newtime + 4, &timeinfo->tm_hour, 1);
memcpy(newtime + 5, &timeinfo->tm_min, 1);
memcpy(newtime + 6, &timeinfo->tm_sec, 1);
switch (c) {
case 'h':
show_help = true;
break;
case 'p':
new_password = optarg;
break;
case 's':
new_ssid = optarg;
break;
case 't':
sync_time = true;
break;
case 'v':
show_version = true;
case '?':
fprintf(stderr, "Use option -h, --help for instructions.\n");
exit(EXIT_FAILURE);
default:
abort();
}
if (send_command(ACTIONPRO_OPCODE_SETTIME, newtime, 8) ==
ACTIONPRO_CMD_OK)
printf("OK\n");
else
printf("ERROR\n");
return 0;
}
}
return 1;
}
int main(int argc, char *argv[]) {
int c;
bool show_help = false;
char *new_password = NULL;
char *new_ssid = NULL;
bool sync_time = false;
bool show_version = false;
printf("ACTIONPRO configuration utility\n");
while (1) {
static struct option long_options[] = {
{"help", no_argument, 0, 'p'},
{"password", required_argument, 0, 'p'},
{"ssid", required_argument, 0, 's'},
{"time", no_argument, 0, 't'},
{"version", no_argument, 0, 'v'},
{0, 0, 0, 0}};
int option_index = 0;
c = getopt_long(argc, argv, "hp:s:tv", long_options, &option_index);
if (argc < 2) {
fprintf(stderr, "At least one option is required. See -h, --help for "
"instructions.\n");
exit(EXIT_FAILURE);
}
if (show_help) {
printf("Usage: %s [OPTION]\n", argv[0]);
printf(" -C, --config-file=FILE use this user configuration file\n");
printf(" -h, --help give this help list\n");
printf(" -p, --password=PASSWORD sets the access point authentication PASSWORD\n");
printf(" -s, --ssid=SSID sets the access point SSID\n");
printf(" -t, --time synchronize the camera time\n");
printf(" -v, --version display version number\n");
printf("\nThis program requires write access to usb devices and might be run as root.\n");
if (c == -1)
break;
exit(EXIT_SUCCESS);
switch (c) {
case 'h':
show_help = true;
break;
case 'p':
new_password = optarg;
break;
case 's':
new_ssid = optarg;
break;
case 't':
sync_time = true;
break;
case 'v':
show_version = true;
case '?':
fprintf(stderr, "Use option -h, --help for instructions.\n");
exit(EXIT_FAILURE);
default:
abort();
}
}
if (show_version) {
printf("Actionpro-cli version: %s\n", VERSION);
exit(EXIT_SUCCESS);
}
if (open_device() == ACTIONPRO_OK) {
if (new_password)
setpassword(new_password);
if (new_ssid)
setssid(new_ssid);
if (sync_time)
settime();
}
close_device();
if (show_help) {
printf("Usage: %s [OPTION]\n", argv[0]);
printf(" -C, --config-file=FILE use this user configuration file\n");
printf(" -h, --help give this help list\n");
printf(" -p, --password=PASSWORD sets the access point authentication "
"PASSWORD\n");
printf(" -s, --ssid=SSID sets the access point SSID\n");
printf(" -t, --time synchronize the camera time\n");
printf(" -v, --version display version number\n");
printf("\nThis program requires write access to usb devices and might be "
"run as root.\n");
exit(EXIT_SUCCESS);
}
if (show_version) {
printf("Actionpro-cli version: %s\n", VERSION);
exit(EXIT_SUCCESS);
}
if (open_device() == ACTIONPRO_OK) {
if (new_password)
setpassword(new_password);
if (new_ssid)
setssid(new_ssid);
if (sync_time)
settime();
}
close_device();
exit(EXIT_SUCCESS);
}

152
usbms.c
View File

@ -20,107 +20,111 @@
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*
*/
#include <stdlib.h>
#include <stdio.h>
#include <libusb-1.0/libusb.h>
#include <stdarg.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <libusb-1.0/libusb.h>
#include "xusb.h"
#include "usbms.h"
#include "xusb.h"
#include "config.h"
static struct libusb_device_handle *handle = NULL;
int send_command(const uint8_t opcode, const uint8_t *buf, const uint8_t buflen)
{
int rc;
uint32_t expected_tag = 0;
int size;
uint8_t cdb[CDB_MAX_LENGTH];
uint8_t sense[REQUEST_SENSE_LENGTH];
int send_command(const uint8_t opcode, const uint8_t *buf,
const uint8_t buflen) {
int rc;
uint32_t expected_tag = 0;
int size;
uint8_t cdb[CDB_MAX_LENGTH];
uint8_t sense[REQUEST_SENSE_LENGTH];
memset(cdb, 0, sizeof(cdb));
cdb[0] = 0x03; // Request Sense
cdb[4] = REQUEST_SENSE_LENGTH;
memset(cdb, 0, sizeof(cdb));
cdb[0] = 0x03; // Request Sense
cdb[4] = REQUEST_SENSE_LENGTH;
send_mass_storage_command(handle, 0x01, 0, cdb, LIBUSB_ENDPOINT_IN, REQUEST_SENSE_LENGTH, &expected_tag);
rc = libusb_bulk_transfer(handle, 0x81, (unsigned char*)&sense, REQUEST_SENSE_LENGTH, &size, 1000);
if (rc < 0) {
printf("libusb_bulk_transfer failed: %s\n", libusb_error_name(rc));
return ACTIONPRO_CMD_ERROR;
}
send_mass_storage_command(handle, 0x01, 0, cdb, LIBUSB_ENDPOINT_IN,
REQUEST_SENSE_LENGTH, &expected_tag);
rc = libusb_bulk_transfer(handle, 0x81, (unsigned char *)&sense,
REQUEST_SENSE_LENGTH, &size, 1000);
if (rc < 0) {
printf("libusb_bulk_transfer failed: %s\n", libusb_error_name(rc));
return ACTIONPRO_CMD_ERROR;
}
rc = libusb_bulk_transfer(handle, 0x81, (unsigned char*)&sense, REQUEST_SENSE_LENGTH, &size, 1000);
if (rc < 0) {
printf("libusb_bulk_transfer failed: %s\n", libusb_error_name(rc));
return ACTIONPRO_CMD_ERROR;
}
rc = libusb_bulk_transfer(handle, 0x81, (unsigned char *)&sense,
REQUEST_SENSE_LENGTH, &size, 1000);
if (rc < 0) {
printf("libusb_bulk_transfer failed: %s\n", libusb_error_name(rc));
return ACTIONPRO_CMD_ERROR;
}
memset(cdb, 0, sizeof(cdb));
cdb[0] = opcode;
memcpy(cdb+2, buf, buflen);
memset(cdb, 0, sizeof(cdb));
cdb[0] = opcode;
memcpy(cdb + 2, buf, buflen);
send_mass_storage_command(handle, 0x01, 0, cdb, LIBUSB_ENDPOINT_IN, REQUEST_SENSE_LENGTH, &expected_tag);
rc = libusb_bulk_transfer(handle, 0x81, (unsigned char*)&sense, REQUEST_SENSE_LENGTH, &size, 1000);
if (rc < 0) {
printf("libusb_bulk_transfer failed: %s\n", libusb_error_name(rc));
return ACTIONPRO_CMD_ERROR;
}
send_mass_storage_command(handle, 0x01, 0, cdb, LIBUSB_ENDPOINT_IN,
REQUEST_SENSE_LENGTH, &expected_tag);
rc = libusb_bulk_transfer(handle, 0x81, (unsigned char *)&sense,
REQUEST_SENSE_LENGTH, &size, 1000);
if (rc < 0) {
printf("libusb_bulk_transfer failed: %s\n", libusb_error_name(rc));
return ACTIONPRO_CMD_ERROR;
}
return ACTIONPRO_CMD_OK;
return ACTIONPRO_CMD_OK;
}
int open_device()
{
int rc;
int open_device() {
int rc;
if (handle)
return ACTIONPRO_OK;
if (handle)
return ACTIONPRO_OK;
rc = libusb_init(NULL);
if (rc < 0) {
printf("\nFailed to initialize libusb\n");
return ACTIONPRO_ERROR;
}
rc = libusb_init(NULL);
if (rc < 0) {
printf("\nFailed to initialize libusb\n");
return ACTIONPRO_ERROR;
}
handle = libusb_open_device_with_vid_pid(NULL, ACTION_VENDOR_ID, ACTION_PRODUCT_ID);
if (handle == NULL) {
fprintf(stderr, "Cannot open camera device %04X:%04X.\n",
ACTION_VENDOR_ID, ACTION_PRODUCT_ID);
return ACTIONPRO_NOT_FOUND;
}
handle = libusb_open_device_with_vid_pid(NULL, ACTION_VENDOR_ID,
ACTION_PRODUCT_ID);
if (handle == NULL) {
fprintf(stderr, "Cannot open camera device %04X:%04X.\n", ACTION_VENDOR_ID,
ACTION_PRODUCT_ID);
return ACTIONPRO_NOT_FOUND;
}
libusb_set_configuration(handle, ACTIONPRO_USB_CONFIGURATION);
if (libusb_kernel_driver_active(handle, ACTIONPRO_USB_INTERFACE) == 1) {
rc = libusb_detach_kernel_driver(handle, ACTIONPRO_USB_INTERFACE);
if (rc != 0) {
fprintf(stderr, "Failed to detach kernel driver.\n");
return ACTIONPRO_ERROR;
}
}
rc = libusb_claim_interface(handle, ACTIONPRO_USB_INTERFACE);
libusb_set_configuration(handle, ACTIONPRO_USB_CONFIGURATION);
if (libusb_kernel_driver_active(handle, ACTIONPRO_USB_INTERFACE) == 1) {
rc = libusb_detach_kernel_driver(handle, ACTIONPRO_USB_INTERFACE);
if (rc != 0) {
fprintf(stderr, "Failed to claim usb interface.\n");
return ACTIONPRO_ERROR;
fprintf(stderr, "Failed to detach kernel driver.\n");
return ACTIONPRO_ERROR;
}
}
return ACTIONPRO_OK;
rc = libusb_claim_interface(handle, ACTIONPRO_USB_INTERFACE);
if (rc != 0) {
fprintf(stderr, "Failed to claim usb interface.\n");
return ACTIONPRO_ERROR;
}
return ACTIONPRO_OK;
}
int close_device()
{
if (handle) {
libusb_release_interface(handle, 0);
libusb_reset_device(handle);
libusb_close(handle);
handle = NULL;
}
int close_device() {
if (handle) {
libusb_release_interface(handle, 0);
libusb_reset_device(handle);
libusb_close(handle);
handle = NULL;
}
libusb_exit(NULL);
libusb_exit(NULL);
return ACTIONPRO_OK;
return ACTIONPRO_OK;
}

27
usbms.h
View File

@ -4,29 +4,30 @@
#include <stdint.h>
/* device information */
#define ACTION_VENDOR_ID 0x4255
#define ACTION_PRODUCT_ID 0x1000
#define ACTION_VENDOR_ID 0x4255
#define ACTION_PRODUCT_ID 0x1000
/* usb configuration */
#define ACTIONPRO_USB_CONFIGURATION 1
#define ACTIONPRO_USB_INTERFACE 0
#define ACTIONPRO_USB_CONFIGURATION 1
#define ACTIONPRO_USB_INTERFACE 0
/* return codes */
#define ACTIONPRO_OK 0
#define ACTIONPRO_ERROR 1
#define ACTIONPRO_NOT_FOUND 2
#define ACTIONPRO_OK 0
#define ACTIONPRO_ERROR 1
#define ACTIONPRO_NOT_FOUND 2
#define ACTIONPRO_CMD_OK 0
#define ACTIONPRO_CMD_ERROR 1
#define ACTIONPRO_CMD_OK 0
#define ACTIONPRO_CMD_ERROR 1
/* vendor specific SPC-2 commands */
#define ACTIONPRO_OPCODE_SETSSID 0xfd
#define ACTIONPRO_OPCODE_SETPASSWORD 0xfe
#define ACTIONPRO_OPCODE_SETTIME 0xff
#define ACTIONPRO_OPCODE_SETSSID 0xfd
#define ACTIONPRO_OPCODE_SETPASSWORD 0xfe
#define ACTIONPRO_OPCODE_SETTIME 0xff
/* exported functions */
extern int open_device();
extern int close_device();
extern int send_command(const uint8_t opcode, const uint8_t *buf, const uint8_t buflen);
extern int send_command(const uint8_t opcode, const uint8_t *buf,
const uint8_t buflen);
#endif /* usbms_h */

198
xusb.c
View File

@ -17,121 +17,139 @@
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include <stdlib.h>
#include <stdio.h>
#include <libusb-1.0/libusb.h>
#include <stdarg.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <libusb-1.0/libusb.h>
#include "config.h"
/* begin of code from xusb.c */
#define ERR_EXIT(errcode) do { perr(" %s\n", libusb_strerror((enum libusb_error)errcode)); return -1; } while (0)
#define CALL_CHECK(fcall) do { int _r=fcall; if (_r < 0) ERR_EXIT(_r); } while (0)
#define CALL_CHECK_CLOSE(fcall, hdl) do { int _r=fcall; if (_r < 0) { libusb_close(hdl); ERR_EXIT(_r); } } while (0)
#define ERR_EXIT(errcode) \
do { \
perr(" %s\n", libusb_strerror((enum libusb_error)errcode)); \
return -1; \
} while (0)
#define CALL_CHECK(fcall) \
do { \
int _r = fcall; \
if (_r < 0) \
ERR_EXIT(_r); \
} while (0)
#define CALL_CHECK_CLOSE(fcall, hdl) \
do { \
int _r = fcall; \
if (_r < 0) { \
libusb_close(hdl); \
ERR_EXIT(_r); \
} \
} while (0)
// Section 5.1: Command Block Wrapper (CBW)
struct command_block_wrapper {
uint8_t dCBWSignature[4];
uint32_t dCBWTag;
uint32_t dCBWDataTransferLength;
uint8_t bmCBWFlags;
uint8_t bCBWLUN;
uint8_t bCBWCBLength;
uint8_t CBWCB[16];
uint8_t dCBWSignature[4];
uint32_t dCBWTag;
uint32_t dCBWDataTransferLength;
uint8_t bmCBWFlags;
uint8_t bCBWLUN;
uint8_t bCBWCBLength;
uint8_t CBWCB[16];
};
// Section 5.2: Command Status Wrapper (CSW)
struct command_status_wrapper {
uint8_t dCSWSignature[4];
uint32_t dCSWTag;
uint32_t dCSWDataResidue;
uint8_t bCSWStatus;
uint8_t dCSWSignature[4];
uint32_t dCSWTag;
uint32_t dCSWDataResidue;
uint8_t bCSWStatus;
};
const uint8_t cdb_length[256] = {
// 0 1 2 3 4 5 6 7 8 9 A B C D E F
06,06,06,06,06,06,06,06,06,06,06,06,06,06,06,06, // 0
06,06,06,06,06,06,06,06,06,06,06,06,06,06,06,06, // 1
10,10,10,10,10,10,10,10,10,10,10,10,10,10,10,10, // 2
10,10,10,10,10,10,10,10,10,10,10,10,10,10,10,10, // 3
10,10,10,10,10,10,10,10,10,10,10,10,10,10,10,10, // 4
10,10,10,10,10,10,10,10,10,10,10,10,10,10,10,10, // 5
00,00,00,00,00,00,00,00,00,00,00,00,00,00,00,00, // 6
00,00,00,00,00,00,00,00,00,00,00,00,00,00,00,00, // 7
16,16,16,16,16,16,16,16,16,16,16,16,16,16,16,16, // 8
16,16,16,16,16,16,16,16,16,16,16,16,16,16,16,16, // 9
12,12,12,12,12,12,12,12,12,12,12,12,12,12,12,12, // A
12,12,12,12,12,12,12,12,12,12,12,12,12,12,12,12, // B
00,00,00,00,00,00,00,00,00,00,00,00,00,00,00,00, // C
00,00,00,00,00,00,00,00,00,00,00,00,00,00,00,00, // D
00,00,00,00,00,00,00,00,00,00,00,00,00,00,00,00, // E
// 0 1 2 3 4 5 6 7 8 9 A B C D E F
06, 06, 06, 06, 06, 06, 06, 06, 06, 06, 06, 06, 06, 06, 06, 06, // 0
06, 06, 06, 06, 06, 06, 06, 06, 06, 06, 06, 06, 06, 06, 06, 06, // 1
10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, // 2
10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, // 3
10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, // 4
10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, // 5
00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, // 6
00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, // 7
16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, // 8
16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, // 9
12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, // A
12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, // B
00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, // C
00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, // D
00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, // E
/* Added length parameters for vendor commands.
* */
00,00,00,00,00,00,00,00,00,00,00,00,00,16,16,16, // F
00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 16, 16, 16, // F
};
static void perr(char const *format, ...)
{
va_list args;
static void perr(char const *format, ...) {
va_list args;
va_start (args, format);
vfprintf(stderr, format, args);
va_end(args);
va_start(args, format);
vfprintf(stderr, format, args);
va_end(args);
}
int send_mass_storage_command(libusb_device_handle *handle, uint8_t endpoint, uint8_t lun,
uint8_t *cdb, uint8_t direction, int data_length, uint32_t *ret_tag)
{
static uint32_t tag = 1;
uint8_t cdb_len;
int i, r, size;
struct command_block_wrapper cbw;
int send_mass_storage_command(libusb_device_handle *handle, uint8_t endpoint,
uint8_t lun, uint8_t *cdb, uint8_t direction,
int data_length, uint32_t *ret_tag) {
static uint32_t tag = 1;
uint8_t cdb_len;
int i, r, size;
struct command_block_wrapper cbw;
if (cdb == NULL) {
return -1;
if (cdb == NULL) {
return -1;
}
if (endpoint & LIBUSB_ENDPOINT_IN) {
perr("send_mass_storage_command: cannot send command on IN endpoint\n");
return -1;
}
cdb_len = cdb_length[cdb[0]];
if ((cdb_len == 0) || (cdb_len > sizeof(cbw.CBWCB))) {
perr("send_mass_storage_command: don't know how to handle this command "
"(%02X, length %d)\n",
cdb[0], cdb_len);
return -1;
}
memset(&cbw, 0, sizeof(cbw));
cbw.dCBWSignature[0] = 'U';
cbw.dCBWSignature[1] = 'S';
cbw.dCBWSignature[2] = 'B';
cbw.dCBWSignature[3] = 'C';
*ret_tag = tag;
cbw.dCBWTag = tag++;
cbw.dCBWDataTransferLength = data_length;
cbw.bmCBWFlags = direction;
cbw.bCBWLUN = lun;
// Subclass is 1 or 6 => cdb_len
cbw.bCBWCBLength = cdb_len;
memcpy(cbw.CBWCB, cdb, cdb_len);
i = 0;
do {
// The transfer length must always be exactly 31 bytes.
r = libusb_bulk_transfer(handle, endpoint, (unsigned char *)&cbw, 31, &size,
1000);
if (r == LIBUSB_ERROR_PIPE) {
libusb_clear_halt(handle, endpoint);
}
i++;
} while ((r == LIBUSB_ERROR_PIPE) && (i < RETRY_MAX));
if (r != LIBUSB_SUCCESS) {
perr("send_mass_storage_command: %s\n",
libusb_strerror((enum libusb_error)r));
return -1;
}
if (endpoint & LIBUSB_ENDPOINT_IN) {
perr("send_mass_storage_command: cannot send command on IN endpoint\n");
return -1;
}
cdb_len = cdb_length[cdb[0]];
if ((cdb_len == 0) || (cdb_len > sizeof(cbw.CBWCB))) {
perr("send_mass_storage_command: don't know how to handle this command (%02X, length %d)\n",
cdb[0], cdb_len);
return -1;
}
memset(&cbw, 0, sizeof(cbw));
cbw.dCBWSignature[0] = 'U';
cbw.dCBWSignature[1] = 'S';
cbw.dCBWSignature[2] = 'B';
cbw.dCBWSignature[3] = 'C';
*ret_tag = tag;
cbw.dCBWTag = tag++;
cbw.dCBWDataTransferLength = data_length;
cbw.bmCBWFlags = direction;
cbw.bCBWLUN = lun;
// Subclass is 1 or 6 => cdb_len
cbw.bCBWCBLength = cdb_len;
memcpy(cbw.CBWCB, cdb, cdb_len);
i = 0;
do {
// The transfer length must always be exactly 31 bytes.
r = libusb_bulk_transfer(handle, endpoint, (unsigned char*)&cbw, 31, &size, 1000);
if (r == LIBUSB_ERROR_PIPE) {
libusb_clear_halt(handle, endpoint);
}
i++;
} while ((r == LIBUSB_ERROR_PIPE) && (i < RETRY_MAX));
if (r != LIBUSB_SUCCESS) {
perr("send_mass_storage_command: %s\n", libusb_strerror((enum libusb_error)r));
return -1;
}
return 0;
return 0;
}

12
xusb.h
View File

@ -1,14 +1,16 @@
#ifndef xusb_h
#define xusb_h
#include <stdint.h>
#include <libusb-1.0/libusb.h>
#include <stdint.h>
#define REQUEST_SENSE_LENGTH 0x12
#define CDB_MAX_LENGTH 0x10
#define REQUEST_SENSE_LENGTH 0x12
#define CDB_MAX_LENGTH 0x10
extern const uint8_t cdb_length[256];
extern int send_mass_storage_command(libusb_device_handle *handle, uint8_t endpoint, uint8_t lun,
uint8_t *cdb, uint8_t direction, int data_length, uint32_t *ret_tag);
extern int send_mass_storage_command(libusb_device_handle *handle,
uint8_t endpoint, uint8_t lun,
uint8_t *cdb, uint8_t direction,
int data_length, uint32_t *ret_tag);
#endif /* xusb_h */