|
#include "gadget-hid.h" |
|
#include <errno.h> |
|
#include <stdio.h> |
|
#include <linux/usb/ch9.h> |
|
#include <usbg/usbg.h> |
|
#include <usbg/function/hid.h> |
|
#include <usbg/function/midi.h> |
|
|
|
static char report_desc[] = { |
|
0x05, 0x01, |
|
0x09, 0x06, |
|
0xA1, 0x01, |
|
0x05, 0x07, |
|
0x19, 0xe0, |
|
0x29, 0xe7, |
|
0x15, 0x00, |
|
0x25, 0x01, |
|
|
|
0x75, 0x01, |
|
0x95, 0x08, |
|
0x81, 0x02, |
|
0x95, 0x01, |
|
0x75, 0x08, |
|
0x81, 0x01, |
|
0x95, 0x03, |
|
0x75, 0x01, |
|
|
|
0x05, 0x08, |
|
0x19, 0x01, |
|
0x29, 0x03, |
|
0x91, 0x02, |
|
0x95, 0x05, |
|
0x75, 0x01, |
|
0x91, 0x01, |
|
0x95, 0x06, |
|
|
|
0x75, 0x08, |
|
0x15, 0x00, |
|
0x26, 0xff, |
|
0x00, 0x05, |
|
0x07, 0x19, |
|
0x00, 0x2a, |
|
0xff, 0x00, |
|
0x81, 0x00, // Input (Data, Array, Abs) |
|
0xc0 // End collection |
|
}; |
|
|
|
int initUSB() { |
|
int ret = -EINVAL; |
|
int usbg_ret; |
|
|
|
struct usbg_gadget_attrs g_attrs = { |
|
.bcdUSB = 0x0200, |
|
.bDeviceClass = USB_CLASS_PER_INTERFACE, |
|
.bDeviceSubClass = 0x00, |
|
.bDeviceProtocol = 0x00, |
|
.bMaxPacketSize0 = 64, /* Max allowed ep0 packet size */ |
|
.idVendor = VENDOR, |
|
.idProduct = PRODUCT, |
|
.bcdDevice = 0x0001, /* Verson of device */ |
|
}; |
|
|
|
struct usbg_gadget_strs g_strs = { |
|
.serial = "0123456789", /* Serial number */ |
|
.manufacturer = "Pimoroni", /* Manufacturer */ |
|
.product = "Keybow" /* Product string */ |
|
}; |
|
|
|
struct usbg_config_strs c_strs = { |
|
.configuration = "1xHID" |
|
}; |
|
|
|
struct usbg_f_midi_attrs midi_attrs = { |
|
.index = 1, |
|
.id = "usb1", |
|
.buflen = 128, |
|
.qlen = 16, |
|
.in_ports = 1, |
|
.out_ports = 1 |
|
}; |
|
|
|
struct usbg_f_hid_attrs f_attrs = { |
|
.protocol = 1, |
|
.report_desc = { |
|
.desc = report_desc, |
|
.len = sizeof(report_desc), |
|
}, |
|
.report_length = 16, |
|
.subclass = 0, |
|
}; |
|
|
|
usbg_ret = usbg_init("/sys/kernel/config", &s); |
|
if (usbg_ret != USBG_SUCCESS) { |
|
fprintf(stderr, "Error on usbg init\n"); |
|
fprintf(stderr, "Error: %s : %s\n", usbg_error_name(usbg_ret), |
|
usbg_strerror(usbg_ret)); |
|
goto out1; |
|
} |
|
|
|
usbg_ret = usbg_create_gadget(s, "g1", &g_attrs, &g_strs, &g); |
|
if (usbg_ret != USBG_SUCCESS) { |
|
fprintf(stderr, "Error creating gadget\n"); |
|
fprintf(stderr, "Error: %s : %s\n", usbg_error_name(usbg_ret), |
|
usbg_strerror(usbg_ret)); |
|
goto out2; |
|
} |
|
|
|
usbg_ret = usbg_create_function(g, USBG_F_HID, "usb0", &f_attrs, &f_hid); |
|
if (usbg_ret != USBG_SUCCESS) { |
|
fprintf(stderr, "Error creating function: USBG_F_HID\n"); |
|
fprintf(stderr, "Error: %s : %s\n", usbg_error_name(usbg_ret), |
|
usbg_strerror(usbg_ret)); |
|
goto out2; |
|
} |
|
|
|
usbg_ret = usbg_create_config(g, 1, "config", NULL, &c_strs, &c); |
|
if (usbg_ret != USBG_SUCCESS) { |
|
fprintf(stderr, "Error creating config\n"); |
|
fprintf(stderr, "Error: %s : %s\n", usbg_error_name(usbg_ret), |
|
usbg_strerror(usbg_ret)); |
|
goto out2; |
|
} |
|
|
|
usbg_ret = usbg_add_config_function(c, "keyboard", f_hid); |
|
if (usbg_ret != USBG_SUCCESS) { |
|
fprintf(stderr, "Error adding function: keyboard\n"); |
|
fprintf(stderr, "Error: %s : %s\n", usbg_error_name(usbg_ret), |
|
usbg_strerror(usbg_ret)); |
|
goto out2; |
|
} |
|
|
|
usbg_ret = usbg_enable_gadget(g, DEFAULT_UDC); |
|
if (usbg_ret != USBG_SUCCESS) { |
|
fprintf(stderr, "Error enabling gadget\n"); |
|
fprintf(stderr, "Error: %s : %s\n", usbg_error_name(usbg_ret), |
|
usbg_strerror(usbg_ret)); |
|
goto out2; |
|
} |
|
|
|
ret = 0; |
|
|
|
out2: |
|
usbg_cleanup(s); |
|
s = NULL; |
|
|
|
out1: |
|
return ret; |
|
} |
|
|
|
int cleanupUSB(){ |
|
if(g){ |
|
usbg_disable_gadget(g); |
|
usbg_rm_gadget(g, USBG_RM_RECURSE); |
|
} |
|
if(s){ |
|
usbg_cleanup(s); |
|
} |
|
return 0; |
|
} |
Dear Philip, I'm quite new to Linux terminal and I'm having a small issue. Where exactly do I have to put the dtoverlay=dwc2 in config? I followed your instructions but when I got to the point about runn' sudo ./pi400kb I couldn't make it work even tho using ls -a the file was there.