libusbhost: factor out frameworks descriptors -> byte[] code.

Sony needed to increase the limit for a UVC device with > 4KiB
descriptors, but there wasn't one clean place to do that. Tidy up, and
also go straight to *16KiB* which was the largest buffer any copy of
this code was using to date.

Test: treehugger
Change-Id: Ide41a217dcf4291a7d38836264f59e06967d9d91
This commit is contained in:
Elliott Hughes 2022-01-19 10:26:45 -08:00
parent da0756b575
commit b889539dfd
6 changed files with 108 additions and 22 deletions

View File

@ -30,11 +30,9 @@ cc_library {
export_include_dirs: ["include"], export_include_dirs: ["include"],
target: { target: {
android: { android: {
cflags: [ header_libs: ["jni_headers"],
"-g",
"-DUSE_LIBLOG",
],
shared_libs: ["liblog"], shared_libs: ["liblog"],
srcs: ["usbhost_jni.cpp"],
}, },
darwin: { darwin: {
enabled: false, enabled: false,

View File

@ -21,6 +21,7 @@
extern "C" { extern "C" {
#endif #endif
#include <stddef.h>
#include <stdint.h> #include <stdint.h>
#include <linux/version.h> #include <linux/version.h>

View File

@ -0,0 +1,27 @@
/*
* Copyright (C) 2022 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#pragma once
#include <jni.h>
/**
* Reads USB descriptors from `fd`.
*
* Returns a byte[] on success,
* or returns NULL and logs an appropriate error on failure.
*/
jbyteArray usb_jni_read_descriptors(JNIEnv* env, int fd);

View File

@ -18,20 +18,9 @@
#define _GNU_SOURCE #define _GNU_SOURCE
#endif #endif
// #define DEBUG 1 #include <usbhost/usbhost.h>
#if DEBUG
#ifdef USE_LIBLOG #include "usbhost_private.h"
#define LOG_TAG "usbhost"
#include "log/log.h"
#define D ALOGD
#else
#define D printf
#endif
#else
#define D(...)
#endif
#include <stdio.h> #include <stdio.h>
#include <stdlib.h> #include <stdlib.h>
@ -48,12 +37,19 @@
#include <errno.h> #include <errno.h>
#include <ctype.h> #include <ctype.h>
#include <poll.h> #include <poll.h>
#include <pthread.h>
#include <linux/usbdevice_fs.h> #include <linux/usbdevice_fs.h>
#include <asm/byteorder.h>
#include "usbhost/usbhost.h" // #define DEBUG 1
#if defined(DEBUG)
#if defined(__BIONIC__)
#define D ALOGD
#else
#define D printf
#endif
#else
#define D(...)
#endif
#define DEV_DIR "/dev" #define DEV_DIR "/dev"
#define DEV_BUS_DIR DEV_DIR "/bus" #define DEV_BUS_DIR DEV_DIR "/bus"
@ -76,8 +72,6 @@ struct usb_host_context {
int wddbus; int wddbus;
}; };
#define MAX_DESCRIPTORS_LENGTH 4096
struct usb_device { struct usb_device {
char dev_name[64]; char dev_name[64];
unsigned char desc[MAX_DESCRIPTORS_LENGTH]; unsigned char desc[MAX_DESCRIPTORS_LENGTH];

View File

@ -0,0 +1,41 @@
/*
* Copyright (C) 2022 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <usbhost/usbhost_jni.h>
#include "usbhost_private.h"
#include <errno.h>
#include <string.h>
#include <unistd.h>
jbyteArray usb_jni_read_descriptors(JNIEnv* env, int fd) {
if (TEMP_FAILURE_RETRY(lseek(fd, 0, SEEK_SET)) == -1) {
ALOGE("usb_jni_read_descriptors(%d): lseek() failed: %s", fd, strerror(errno));
return NULL;
}
jbyte buf[MAX_DESCRIPTORS_LENGTH];
ssize_t n = TEMP_FAILURE_RETRY(read(fd, buf, sizeof(buf)));
if (n == -1) {
ALOGE("usb_jni_read_descriptors: read failed: %s", strerror(errno));
return NULL;
}
jbyteArray result = env->NewByteArray(n);
if (result) env->SetByteArrayRegion(result, 0, n, buf);
return result;
}

View File

@ -0,0 +1,25 @@
/*
* Copyright (C) 2022 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#pragma once
#define LOG_TAG "usbhost"
#include <log/log.h>
// Somewhat arbitrary: Sony has reported needing more than 4KiB (but less
// than 8KiB), and some frameworks code had 16KiB without any explanation,
// so we went with the largest of those.
#define MAX_DESCRIPTORS_LENGTH (16 * 1024)