libuvc的IAD组合多UVC设备支持修改

原创 嵌入式Lee 2025-04-17 19:24

一. 前言

libuvc不支持IAD枚举的组合多设备,只支持多个独立设备。所以本文分享将libuvc改造为支持IAD枚举的组合多设备。

备注:复合设备其实就是几个物理设备通过一个USB Hub形成的单一设备;
组合设备是通过IAD实现,也就是具有多个接口的设备,不同接口代表不同功能设备。

代码见:https://github.com/qinyunti/uvc_demo.git

二. 过程

我们前面看到libuvc的操作流程如下

1.先调用uvc_init进行初始化

uvc_error_t uvc_init(uvc_context_t **pctx, struct libusb_context *usb_ctx)

其中参数pctx需要用户传入一个uvc_context_t *的地址,所以是指向指针的指针

uvc_context_t *ctx;

uvc_init(&ctx, NULL);

uvc_init时会动态分配一个uvc_context_t *实例,赋值给ctx

其中参数usb_ctx表示libusb的上下文环境

usb_ctxNULL则先初始化libusb环境

libusb_init(&ctx->usb_ctx);

否则直接使用传进来的libusb

ctx->usb_ctx = usb_ctx;

ctx->own_usb_ctx0表示使用传进来的上下文环境,为1表示这里libuvc重新初始化了一个libusb环境。

最终的结果就是uvc_init初始化了一个libuvc的环境即ctx变量。


2.然后调用uvc_find_device查找设备

uvc_error_t uvc_find_device(

uvc_context_t *ctx, uvc_device_t **dev,

int vid, int pid, const char *sn)

查找到设备赋值给 uvc_device_t *dev;

res = uvc_find_device(

ctx, &dev,

0x1993, 0x0101, NULL);

uvc_find_devices则可以查找多个设备

uvc_error_t uvc_find_devices(

uvc_context_t *ctx, uvc_device_t ***devs,

int vid, int pid, const char *sn)

由于IAD多设备是需要系统的组合设备驱动实现的,使用libusb实际就是没有使用系统的驱动了,libusb此时看到的就是只有一个设备了,即操作裸的接口,端点等。

即此时dev对应的就是一个IAD多设备,而每一个IAD需要自行去处理。


3.调用uvc_open打开设备

uvc_error_t uvc_open(

uvc_device_t *dev,

uvc_device_handle_t **devh) 

res = uvc_open(dev, &devh);

打开赋值给

uvc_device_handle_t *devh;

通过代码可以看到uvc_open实际就是调用

uvc_open_internal

创建实例uvc_device_handle_t

然后调用uvc_get_device_info获取控制接口,和流接口等信息,填充uvc_device_handle_t实例,

最终将该实例返回给devh参数。

这里可以看到

uvc_get_device_info实际是调用的uvc_scan_control的扫描接口

匹配控制接口类和子类参数,匹配了就记录该接口号

if_desc->bInterfaceClass == 14 && if_desc->bInterfaceSubClass == 1

这里代码可以看到找到一个就结束了。

即一个uvc_device_t设备只对应一个uvc设备uvc_device_handle_t

实际上对于IADUVC设备,逻辑上应该是

一个uvc_device_t设备只对应多个uvc设备uvc_device_handle_t

所以自然的可以想到我们可以增加一个函数

uvc_error_t uvc_opens(

uvc_device_t *dev,

uvc_device_handle_t ***devh, int* n)

即根据uvc_device_t设备,打开nuvc设备 uvc_device_handle_t

自然也会想到将改为uvc_open_internal

static uvc_error_t uvc_open_internals(

uvc_device_t *dev,

struct libusb_device_handle *usb_devh,

uvc_device_handle_t ***devh, int* n)

自然还会想到将uvc_get_device_info改为

uvc_error_t uvc_get_device_info(uvc_device_handle_t *devh,

uvc_device_info_t **info, int* if_idx)

即默认从接口0开始搜寻,改为从指定接口处开始搜寻,这样可以搜寻完所有的接口而不是只搜寻到一个就算完成。

对应的uvc_scan_control改为

uvc_scan_control(uvc_device_handle_t *devh, uvc_device_info_t *info, int* if_idx)

if_idx即表示当前从那个接口开始搜寻,返回搜寻到的接口,这样下一次就可以从if_idx+1处继续搜索,则对应下一个 uvc_device_handle_t设备。

以上即可实现找到n个 uvc_device_handle_t设备


4.然后获取流uvc_get_stream_ctrl_format_size

uvc_error_t uvc_get_stream_ctrl_format_size(

uvc_device_handle_t *devh,

uvc_stream_ctrl_t *ctrl,

enum uvc_frame_format cf,

int width, int height,

int fps)

res = uvc_get_stream_ctrl_format_size(

devh, &ctrl, UVC_FRAME_FORMAT_MJPEG, 1280, 720, 0

);

赋值给 uvc_stream_ctrl_t ctrl;

这样一个uvc设备 uvc_device_handle_t设备再对应一个流

uvc_stream_ctrl_t

上面提供了多个uvc设备,这里自然就对应多个流,无需修改即可支持多设备。


三.代码修改

详见源码,具体修改如下

Libuvc.h中增加接口

uvc_error_t uvc_opens(

uvc_device_t *dev,

uvc_device_handle_t ***devh, int* n);

Device.c

以下函数声明增加了参数if_idx

uvc_error_t uvc_get_device_info(uvc_device_handle_t *devh, uvc_device_info_t **info, int* if_idx);

void uvc_free_device_info(uvc_device_info_t *info);

uvc_error_t uvc_scan_control(uvc_device_handle_t *devh, uvc_device_info_t *info, int* if_idx);

增加接口

static uvc_error_t uvc_open_internals(uvc_device_t *dev, struct libusb_device_handle *usb_devh, uvc_device_handle_t ***devh, int* n);

增加uvc_opens实现


/** @brief Open a UVC device * @ingroup device * * @param dev Device to open * @param[out] devh Handle List on opened device * @return Error opening device or SUCCESS */uvc_error_t uvc_opens(    uvc_device_t *dev,    uvc_device_handle_t ***devh, int* n) {    uvc_error_t ret;    struct libusb_device_handle *usb_devh;     UVC_ENTER();     ret = libusb_open(dev->usb_dev, &usb_devh);    UVC_DEBUG("libusb_open() = %d", ret);     if (ret != UVC_SUCCESS) {        UVC_EXIT(ret);        return ret;    }     ret = uvc_open_internals(dev, usb_devh, devh, n);    UVC_EXIT(ret);    return ret;}

uvc_open_internal修改

增加uvc_open_internals实现


 static uvc_error_t uvc_open_internals(     uvc_device_t *dev,     struct libusb_device_handle *usb_devh,     uvc_device_handle_t ***devh, int* n) {     uvc_error_t ret;     uvc_device_handle_t *internal_devh;     struct libusb_device_descriptor desc;     int if_idx = 0;     int dev_num = 0;     UVC_ENTER();      uvc_ref_device(dev);      while(1)     {         internal_devh = calloc(1sizeof(*internal_devh));         internal_devh->dev = dev;         internal_devh->usb_devh = usb_devh;          ret = uvc_get_device_info(internal_devh, &(internal_devh->info), &if_idx);         if_idx++;         if (ret != UVC_SUCCESS){             if(dev_num == 0){                 goto fail;             }else{                goto done;             }         }          UVC_DEBUG("claiming control interface %d", internal_devh->info->ctrl_if.bInterfaceNumber);         ret = uvc_claim_if(internal_devh, internal_devh->info->ctrl_if.bInterfaceNumber);         if (ret != UVC_SUCCESS){             if(dev_num == 0){                 goto fail;             }else{                 goto done;             }         }          libusb_get_device_descriptor(dev->usb_dev, &desc);         internal_devh->is_isight = (desc.idVendor == 0x05ac && desc.idProduct == 0x8501);          if (internal_devh->info->ctrl_if.bEndpointAddress) {             internal_devh->status_xfer = libusb_alloc_transfer(0);             if (!internal_devh->status_xfer) {                 ret = UVC_ERROR_NO_MEM;                 if(dev_num == 0){                     goto fail;                 }else{                     goto done;                 }             }              libusb_fill_interrupt_transfer(internal_devh->status_xfer,                                            usb_devh,                                            internal_devh->info->ctrl_if.bEndpointAddress,                                            internal_devh->status_buf,                                            sizeof(internal_devh->status_buf),                                            _uvc_status_callback,                                            internal_devh,                                            0);             ret = libusb_submit_transfer(internal_devh->status_xfer);             UVC_DEBUG("libusb_submit_transfer() = %d", ret);              if (ret) {                 fprintf(stderr,                         "uvc: device has a status interrupt endpoint, but unable to read from it\n");                 if(dev_num == 0){                     goto fail;                 }else{                     goto done;                 }             }         }          if (dev->ctx->own_usb_ctx && dev->ctx->open_devices == NULL) {             /* Since this is our first device, we need to spawn the event handler thread */             uvc_start_handler_thread(dev->ctx);         }          DL_APPEND(dev->ctx->open_devices, internal_devh);         (*devh)[dev_num] = internal_devh;         dev_num++;         if(dev_num >= *n){             goto done;         }     } done:     *n = dev_num;     UVC_EXIT(ret);      return ret;  fail:     if ( internal_devh->info ) {         uvc_release_if(internal_devh, internal_devh->info->ctrl_if.bInterfaceNumber);     }     libusb_close(usb_devh);     uvc_unref_device(dev);     uvc_free_devh(internal_devh);      UVC_EXIT(ret);      return ret; }

修改uvc_get_device_info

修改uvc_scan_control

四. 测试

代码见

test_iad.c

#include #include "libuvc/libuvc.h"#include 
static void cb(uvc_frame_t *frame, void *ptr) {    uvc_frame_t *bgr;    uvc_error_t ret;
    printf("callback! length = %zu, ptr = %p\n", frame->data_bytes, ptr);
    bgr = uvc_allocate_frame(frame->width * frame->height * 3);    if (!bgr) {        printf("unable to allocate bgr frame!");        return;    }
    /* 格式解析 */    ret = uvc_any2bgr(frame, bgr);    if (ret) {        uvc_perror(ret, "uvc_any2bgr");        uvc_free_frame(bgr);        return;    }    // 显示等操作    uvc_free_frame(bgr);}
int test_iad_main(int argc, char **argv) {    (void)argc;    (void)argv;    uvc_context_t *ctx;    uvc_error_t res;    uvc_device_t *dev;    uvc_device_handle_t *devh[2]={NULL,NULL};    uvc_device_handle_t **pdevh = devh;    uvc_stream_ctrl_t ctrl[2];    int dev_num = sizeof(devh)/sizeof(devh[0]);    res = uvc_init(&ctx, NULL);
    if (res < 0) {        uvc_perror(res, "uvc_init");        return res;    }
    printf("UVC initialized\r\n");
    res = uvc_find_device(        ctx, &dev,        0x19930x0101NULL);
    if (res < 0) {        uvc_perror(res, "uvc_find_device");    } else {        printf("Device found\r\n");
        res = uvc_opens(dev, &pdevh, &dev_num);
        if (res < 0) {            uvc_perror(res, "uvc_open");        } else {            printf("Device opened %d\r\n",dev_num);
            for(int i=0; i            {                uvc_print_diag(devh[i], stderr);
                res = uvc_get_stream_ctrl_format_size(                    devh[i], &ctrl[i], UVC_FRAME_FORMAT_MJPEG, 12807200                    );
                uvc_print_stream_ctrl(&ctrl[i], stderr);
                if (res < 0) {                    uvc_perror(res, "get_mode");                } else {                    res = uvc_start_streaming(devh[i], &ctrl[i], cb, (void*)i, 0);
                    if (res < 0) {                        uvc_perror(res, "start_streaming");                    } else {                        printf("Streaming for 10 seconds...");                    }                }            }
            Sleep(10000);
            for(int i=0; i            {                uvc_stop_streaming(devh[i]);                printf("Done streaming %d.\r\n",i);            }
            for(int i=0; i            {                uvc_close(devh[i]);                printf("Device %d closed\r\n",i);            }        }
        uvc_unref_device(dev);    }
    uvc_exit(ctx);    printf("UVC exited");
    return 0;}


这里测试的一个IADUVC设备,打印如下



UVC initializedDevice foundDevice opened 2DEVICE CONFIGURATION (1993:0101/[none]) ---Status: idleVideoControl:        bcdUVC: 0x0110        bInterfaceNumber: 0x0000VideoStreaming(1):        bEndpointAddress: 129        Formats:        UncompressedFormat(1)                  bits per pixel: 12                  GUID: 4e56313200001000800000aa00389b71 (NV12)                  default frame: 1                  aspect ratio: 0x0                  interlace flags: 00                  copy protect: 00                        FrameDescriptor(1)                          capabilities: 02                          size: 1280x720                          bit rate: 24576000-24576000                          max frame size: 460800                          default interval: 1/25                          interval[0]: 1/40                          interval[1]: 1/25                          interval[2]: 1/10        FrameFormat(2)                  bits per pixel: 16                  GUID: 4832363400001000800000aa00389b71 (H264)                  default frame: 1                  aspect ratio: 0x0                  interlace flags: 00                  copy protect: 00                        FrameDescriptor(1)                          capabilities: 00                          size: 1280x720                          bit rate: 24576000-24576000                          max frame size: 0                          default interval: 1/25                          interval[0]: 1/40                          interval[1]: 1/25                          interval[2]: 1/10        MJPEGFormat(3)                  bits per pixel: 0                  GUID: 4d4a5047000000000000000000000000 (MJPG)                  default frame: 1                  aspect ratio: 0x0                  interlace flags: 00                  copy protect: 00                        FrameDescriptor(1)                          capabilities: 02                          size: 1280x720                          bit rate: 24576000-24576000                          max frame size: 460800                          default interval: 1/25                          interval[0]: 1/40                          interval[1]: 1/25                          interval[2]: 1/10END DEVICE CONFIGURATIONbmHint: 0001bFormatIndex: 3bFrameIndex: 1dwFrameInterval: 400000wKeyFrameRate: 0wPFrameRate: 0wCompQuality: 0wCompWindowSize: 0wDelay: 0dwMaxVideoFrameSize: 460800dwMaxPayloadTransferSize: 1024bInterfaceNumber: 1Streaming for 10 seconds...DEVICE CONFIGURATION (1993:0101/[none]) ---Status: idleVideoControl:        bcdUVC: 0x0110        bInterfaceNumber: 0x0002VideoStreaming(1):        bEndpointAddress: 130        Formats:        UncompressedFormat(1)                  bits per pixel: 12                  GUID: 4e56313200001000800000aa00389b71 (NV12)                  default frame: 1                  aspect ratio: 0x0                  interlace flacgasl:l b0a0ck       !         l ecnogptyh  p=r o1t8e5c2t5:,  0p0tr               =        F0r0a0m0e0D0e0s0c0r0i0p0t0o0r0(01)                          capabilities: 02                          size: 1280x720                          bit rate: 24576000-24576000                          max frame size: 460800                          default interval: 1/25                          interval[0]: 1/40                          interval[1]: 1/25                          interval[2]: 1/10        FrameFormat(2)                  bits per pixel: 16                  GUID: 483236uvc_any2bgr: Not supported (-12)3c4a0l0l0b0a1c0k0!0 8l0e0n0g0t0ha a=0 02338098b47,1  p(tHr2 6=4 )00       0       0 0 0de0f0a0u0l0t0 0f0r0a0me: 1                  aspect ratio: 0x0                  interlace flags: 00                  copy protect: 00                        FrameDescriptor(1)                          capabilities: 00                          size: 1280x720                          bit rate: 24576000-24576000                          max frame size: 0                          default interval: 1/25uvc_any2bgr: Not supported (-12)                          interval[0]: 1/40                          interval[1]: 1/25                          interval[2]: 1/10        MJPEGFormat(3)                  bits per pixel: 0                  GUID: 4d4a5047000000000000000000000000 (MJPG)                  default frame: 1                  aspect ratio: 0x0                  interlace flagcsa:l l0b0ac       k       !   lceonpgyt hp r=o t2e2c8t1:9 0,0p       t       r        F=r a0m0e0D0e0s0c0r0i0p0t0o0r0(010)0                           capabilities: 02                          size: 1280x720                          bit rate: 24576000-24576000                          max frame size: 460800                          default interval: 1/25                          interval[0]: 1/40                          interval[1]: 1/25                          interval[2]: 1/10END DEVICE CONFIGURATIONuvc_any2bgr: Not supported (-12)bmHint: 0001bFormatIndex: 3bFrameIndex: 1dwFrameInterval: 400000wKeyFrameRate: 0wPFrameRate: 0wCompQuality: 0wCompWindowSize: 0wDelay: 0dwMaxVideoFrameSize: 460800dwMaxPayloadTransferSize: 1024bInterfaceNumber: 3callback! length = 22746, ptr = 0000000000000000Suvc_any2bgrt:r eNaomti nsgu pfpoorr t1e0d  s(e-c1o2n)ds...callback! length = 21833, ptr = 0000000000000001uvc_any2bgr: Not supported (-12)callback! length = 23077, ptr = 0000000000000000uvc_any2bgr: Not supported (-12)callback! length = 22567, ptr = 0000000000000001uvc_any2bgr: Not supported (-12)callback! length = 22613, ptr = 0000000000000000uvc_any2bgr: Not supported (-12)callback! length = 22054, ptr = 0000000000000001uvc_any2bgr: Not supported (-12)callback! length = 23084, ptr = 0000000000000000uvc_any2bgr: Not supported (-12)callback! length = 22819, ptr 

五. 总结

以上分析了uvc的程序流,以简单的方式修改即支持IADUVC设备。这里使用增量方式增加uvc_opens接口,这样不影响以前的接口,并且其他依赖接口的修改也修改时保持兼容。


评论 (0)
  • UNISOC Miracle Gaming奇迹手游引擎亮点:• 高帧稳帧:支持《王者荣耀》等主流手游90帧高画质模式,连续丢帧率最高降低85%;• 丝滑操控:游戏冷启动速度提升50%,《和平精英》开镜开枪操作延迟降低80%;• 极速网络:专属游戏网络引擎,使《王者荣耀》平均延迟降低80%;• 智感语音:与腾讯GVoice联合,弱网环境仍能保持清晰通话;• 超高画质:游戏画质增强、超级HDR画质、游戏超分技术,优化游戏视效。全球手游市场规模日益壮大,游戏玩家对极致体验的追求愈发苛刻。紫光展锐全新U
    紫光展锐 2025-05-07 17:07 185浏览
  • 后摄像头是长这个样子,如下图。5孔(D-,D+,5V,12V,GND),说的是连接线的个数,如下图。4LED,+12V驱动4颗LED灯珠,给摄像头补光用的,如下图。打开后盖,发现里面有透明白胶(防水)和白色硬胶(固定),用合适的工具,清理其中的胶状物。BOT层,AN3860,Panasonic Semiconductor (松下电器)制造的,Cylinder Motor Driver IC for Video Camera,如下图。TOP层,感光芯片和广角聚焦镜头组合,如下图。感光芯片,看着是玻
    liweicheng 2025-05-07 23:55 71浏览
  • 你是不是也有在公共场合被偷看手机或笔电的经验呢?科技时代下,不少现代人的各式机密数据都在手机、平板或是笔电等可携式的3C产品上处理,若是经常性地需要在公共场合使用,不管是工作上的机密文件,或是重要的个人信息等,民众都有防窃防盗意识,为了避免他人窥探内容,都会选择使用「防窥保护贴片」,以防止数据外泄。现今市面上「防窥保护贴」、「防窥片」、「屏幕防窥膜」等产品就是这种目的下产物 (以下简称防窥片)!防窥片功能与常见问题解析首先,防窥片最主要的功能就是用来防止他人窥视屏幕上的隐私信息,它是利用百叶窗的
    百佳泰测试实验室 2025-04-30 13:28 638浏览
  • 多功能电锅长什么样子,主视图如下图所示。侧视图如下图所示。型号JZ-18A,额定功率600W,额定电压220V,产自潮州市潮安区彩塘镇精致电子配件厂,铭牌如下图所示。有两颗螺丝固定底盖,找到合适的工具,拆开底盖如下图所示。可见和大部分市场的加热锅一样的工作原理,手绘原理图,根据原理图进一步理解和分析。F1为保险,250V/10A,185℃,CPGXLD 250V10A TF185℃ RY 是一款温度保险丝,额定电压是250V,额定电流是10A,动作温度是185℃。CPGXLD是温度保险丝电器元件
    liweicheng 2025-05-05 18:36 257浏览
  • ‌一、高斯计的正确选择‌1、‌明确测量需求‌‌磁场类型‌:区分直流或交流磁场,选择对应仪器(如交流高斯计需支持交变磁场测量)。‌量程范围‌:根据被测磁场强度选择覆盖范围,例如地球磁场(0.3–0.5 G)或工业磁体(数百至数千高斯)。‌精度与分辨率‌:高精度场景(如科研)需选择误差低于1%的仪器,分辨率需匹配微小磁场变化检测需求。2、‌仪器类型选择‌‌手持式‌:便携性强,适合现场快速检测;‌台式‌:精度更高,适用于实验室或工业环境。‌探头类型‌:‌横向/轴向探头‌:根据磁场方向选择,轴向探头适合
    锦正茂科技 2025-05-06 11:36 381浏览
  • 2024年初,OpenAI公布的Sora AI视频生成模型,震撼了国产大模型行业。随后国产厂商集体发力视频大模型,快手发布视频生成大模型可灵,字节跳动发布豆包视频生成模型,正式打响了国内AI视频生成领域第一枪。众多企业匆忙入局,只为在这片新兴市场中抢占先机,却往往忽视了技术成熟度与应用规范的打磨。以社交平台上泛滥的 AI 伪造视频为例,全红婵家人被恶意仿冒博流量卖货,明星们也纷纷中招,刘晓庆、张馨予等均曾反馈有人在视频号上通过AI生成视频假冒她。这些伪造视频不仅严重侵犯他人权
    用户1742991715177 2025-05-05 23:08 79浏览
  • 5小时自学修好BIOS卡住问题  更换硬盘故障现象:f2、f12均失效,只有ESC和开关机键可用。错误页面:经过AI的故障截图询问,确定是机体内灰尘太多,和硬盘损坏造成,开机卡在BIOS。经过亲手拆螺丝和壳体、排线,跟换了新的2.5寸硬盘,故障排除。理论依据:以下是针对“5小时自学修好BIOS卡住问题+更换硬盘”的综合性解决方案,结合硬件操作和BIOS设置调整,分步骤说明:一、判断BIOS卡住的原因1. 初步排查     拔掉多余硬件:断开所有外接设备(如
    丙丁先生 2025-05-04 09:14 118浏览
  • 想不到短短几年时间,华为就从“技术封锁”的持久战中突围,成功将“被卡脖子”困境扭转为科技主权的主动争夺战。众所周知,前几年技术霸权国家突然对华为发难,导致芯片供应链被强行掐断,海外市场阵地接连失守,恶意舆论如汹涌潮水,让其瞬间陷入了前所未有的困境。而最近财报显示,华为已经渡过危险期,甚至开始反击。2024年财报数据显示,华为实现全球销售收入8621亿元人民币,净利润626亿元人民币;经营活动现金流为884.17亿元,同比增长26.7%。对比来看,2024年营收同比增长22.42%,2023年为7
    用户1742991715177 2025-05-02 18:40 209浏览
  • 这款无线入耳式蓝牙耳机是长这个样子的,如下图。侧面特写,如下图。充电接口来个特写,用的是卡座卡在PCB板子上的,上下夹紧PCB的正负极,如下图。撬开耳机喇叭盖子,如下图。精致的喇叭(HY),如下图。喇叭是由电学产生声学的,具体结构如下图。电池包(AFS 451012  21 12),用黄色耐高温胶带进行包裹(安规需求),加强隔离绝缘的,如下图。451012是电池包的型号,聚合物锂电池+3.7V 35mAh,详细如下图。电路板是怎么拿出来的呢,剪断喇叭和电池包的连接线,底部抽出PCB板子
    liweicheng 2025-05-06 22:58 284浏览
  • 随着智能驾驶时代到来,汽车正转变为移动计算平台。车载AI技术对存储器提出新挑战:既要高性能,又需低功耗和车规级可靠性。贞光科技代理的紫光国芯车规级LPDDR4存储器,以其卓越性能成为国产芯片产业链中的关键一环,为智能汽车提供坚实的"记忆力"支持。作为官方授权代理商,贞光科技通过专业技术团队和完善供应链,让这款国产存储器更好地服务国内汽车厂商。本文将探讨车载AI算力需求现状及贞光科技如何通过紫光国芯LPDDR4产品满足市场需求。 车载AI算力需求激增的背景与挑战智能驾驶推动算力需求爆发式
    贞光科技 2025-05-07 16:54 141浏览
  • 文/郭楚妤编辑/cc孙聪颖‍相较于一众措辞谨慎、毫无掌舵者个人风格的上市公司财报,利亚德的财报显得尤为另类。利亚德光电集团成立于1995年,是一家以LED显示、液晶显示产品设计、生产、销售及服务为主业的高新技术企业。自2016年年报起,无论业绩优劣,董事长李军每年都会在财报末尾附上一首七言打油诗,抒发其对公司当年业绩的感悟。从“三年翻番顺大势”“智能显示我第一”“披荆斩棘幸从容”等词句中,不难窥见李军的雄心壮志。2012年,利亚德(300296.SZ)在深交所创业板上市。成立以来,该公司在细分领
    华尔街科技眼 2025-05-07 19:25 128浏览
  • 某国产固态电解的2次和3次谐波失真相当好,值得一试。(仅供参考)现在国产固态电解的性能跟上来了,值得一试。当然不是随便搞低端的那种。电容器对音质的影响_电子基础-面包板社区  https://mbb.eet-china.com/forum/topic/150182_1_1.html (右键复制链接打开)电容器对音质的影响相当大。电容器在音频系统中的角色不可忽视,它们能够调整系统增益、提供合适的偏置、抑制电源噪声并隔离直流成分。然而,在便携式设备中,由于空间、成本的限
    bruce小肥羊 2025-05-04 18:14 232浏览
  • 二位半 5线数码管的驱动方法这个2位半的7段数码管只用5个管脚驱动。如果用常规的7段+共阳/阴则需要用10个管脚。如果把每个段看成独立的灯。5个管脚来点亮,任选其中一个作为COM端时,另外4条线可以单独各控制一个灯。所以实际上最多能驱动5*4 = 20个段。但是这里会有一个小问题。如果想点亮B1,可以让第3条线(P3)置高,P4 置低,其它阳极连P3的灯对应阴极P2 P1都应置高,此时会发现C1也会点亮。实际操作时,可以把COM端线P3设置为PP输出,其它线为OD输出。就可以单独控制了。实际的驱
    southcreek 2025-05-07 15:06 195浏览
  • 浪潮之上:智能时代的觉醒    近日参加了一场课题的答辩,这是医疗人工智能揭榜挂帅的国家项目的地区考场,参与者众多,围绕着医疗健康的主题,八仙过海各显神通,百花齐放。   中国大地正在发生着激动人心的场景:深圳前海深港人工智能算力中心高速运转的液冷服务器,武汉马路上自动驾驶出租车穿行的智慧道路,机器人参与北京的马拉松竞赛。从中央到地方,人工智能相关政策和消息如雨后春笋般不断出台,数字中国的建设图景正在智能浪潮中徐徐展开,战略布局如同围棋
    广州铁金刚 2025-04-30 15:24 376浏览
  • Matter协议是一个由Amazon Alexa、Apple HomeKit、Google Home和Samsung SmartThings等全球科技巨头与CSA联盟共同制定的开放性标准,它就像一份“共生契约”,能让原本相互独立的家居生态在应用层上握手共存,同时它并非另起炉灶,而是以IP(互联网协议)为基础框架,将不同通信协议下的家居设备统一到同一套“语义规则”之下。作为应用层上的互通标准,Matter协议正在重新定义智能家居行业的运行逻辑,它不仅能向下屏蔽家居设备制造商的生态和系统,让设备、平
    华普微HOPERF 2025-05-08 11:40 40浏览
我要评论
0
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦