ivshmem-plain设备原理分析

news2025/1/12 21:06:08

文章目录

  • 前言
  • 基本原理
    • 共享内存
    • 协议规范
  • 具体实现
    • 设备模型
    • 数据结构
    • 设备初始化
  • 测试验证
    • 方案
    • 流程
      • Libvirt配置
      • Qemu配置
      • 测试步骤

前言

  • ivshmem-plain设备是Qemu提供的一种特殊设备,通过这个设备,可以实现虚机内存和主机上其它进程共存共享,应用程序可以利用此设备实现虚机内部和主机上进程间的高效数据传输。通常,虚机内部的进程作为生产者,往共享内存中写入数据,主机侧进程作为消费者,从共享内存中读取数据,这种模式常常应用在虚拟化的杀毒软件场景,虚机内部的杀毒软件驱动程序搜集虚机的行为数据放到共享内存,后端的杀毒软件分析虚机暴露的数据,判断该虚机是否行为异常甚至中毒,本文主要分析这类杀毒软件的工作基础ivshmem-plain设备。

基本原理

共享内存

  • linux支持进程间共享内存,通过文件的形式提供编程接口,共享内存通常由一个进程打开共享内存的文件并写入内容,作为生产者,由另一个进程通过只读方式打开同样的内存文件并作为消费着读取。我们通过简单测试程序了解共享内存的使用方式,参考代码shared memory
  • 测试程序有两个,一个作为生产者(sender)创建共享内存文件/dev/shm/ivshmem-demo,同时往内存文件中写入内容,这里 sender分别写入了三个整数 0、1、2。一个作为消费者(receiver)通过只读方式打开共享内存文件/dev/shm/ivshmem-demo,读取该内存文件,测试程序运行结果如下:
    在这里插入图片描述
  • 生产者打开共享内存文件,可以看到,进程空间分配的内存比实际映射的多,为4k,猜测共享内存的最小单位是一个内存页。
    在这里插入图片描述
  • Qemu的ivshmem-plain设备,其本质也是利用linux提供的这一套共享内存机制,实现虚拟机内存和主机上进程的内存共享。实际上虚机内存在主机上进程看来就是Qemu分配创建的共享内存,因此可以互相访问。

协议规范

  • ivshmem规范定义了如何实现共享内存设备,其核心原理基于QEMU的PCI设备模型,前端将PCI设备的BAR空间作为一个普通的内存空间,后端将该BAR空间关联的MR配置成从共享内存文件中获得,从而实现将BAR空间暴露给其它进程的目标。
  • ivshmem-plain设备主要使用的是共享内存部分的功能,这部分实现比较简单,ivshmem规范中涉及内容不多,ivshmem规范中大部分内容是讲解如何设计PCI的BAR寄存器实现共享设备之前的中断通知,即 ivshmem-doorbell设备。ivshmem-doorbell涉及的中断机制我们单独讲解,不在本文介绍。对于ivshmem设备,无论是ivshmem-plain设备还是ivshmem-doorbell设备,其架构相同,如下所示,ivshmem-plain设备的实现只用到了BAR2,ivshmem-doorbell设备的实现涉及BAR0和BAR1。
    在这里插入图片描述

具体实现

设备模型

  • ivshmem在最初的设计中,实现了驱动为ivshmem设备,既支持普通的共享内存设备,又支持可通过中断实现doorbell的共享内存设备,但实现逻辑耦合性高且不清晰,因此根据功能不同,ivshmem被拆分成ivshmem-plain和ivshmem-doorbell两类设备,原来的ivshmem设备作为遗留设备被建议废弃,实现代码仍然被保留。
  • 当前Qemu的ivshmem设备主要指ivshmem-plain设备和ivshmem-doorbell设备,模型如下:
    在这里插入图片描述

数据结构

  • ivshmem设备的核心数据结构是IVShmemState
    /* ivshmem规范中定义的PCI设备BAR0寄存器 */
    /* registers for the Inter-VM shared memory device */
    enum ivshmem_registers {
        INTRMASK = 0,
        INTRSTATUS = 4,
        IVPOSITION = 8,
        DOORBELL = 12,
    };

typedef struct IVShmemState {
    /*< private >*/
    PCIDevice parent_obj;
    /*< public >*/
    /* 存放可供用户配置的特性 */
    uint32_t features;

    /* exactly one of these two may be set */
    /* ivshmem-plain设备,使用qemu分配的内存作为共享内存 */
    HostMemoryBackend *hostmem; /* with interrupts */
    /* ivshmem-doorbell设备,使用server传入的共享内存文件fd分配内存作为共享内存 */
    CharBackend server_chr; /* without interrupts */

    /* registers */
    uint32_t intrmask;    /* Interrupt Mask */
    uint32_t intrstatus;  /* Interrupt Status */
    /* 
     * 对于ivshmem-plain设备,这个字段没有使用,默认为0
     * 对于ivshmem-doorbell设备,使用该字段,在共享设备的域中标识自己的ID
     * 其它设备发送中断时通过指明vm_id字段表明中断要发送到的目的设备
     * vm_id的范围时[0, 65535], ivshmem-doorbell-server根据client连接顺序从0
     * 依次开始一次分配ID给client,角色为master的client必须是第一个连接
     * server的ivshmem-doorbell设备,因此master ivshmem-doorbell设备的vm_id
     * 是0,后续连接的client作为peer角色存在,master设备支持迁移
     * 因此无论是ivshmem-plain还是ivshmem-doorbell,vm_id为0时都表明自己是
     * master角色,能够支持迁移
     */
    int vm_id;            /* IVPosition */

    /* BARs */
	/* BAR0寄存器对应的后端实现,MMIO类型MR */
    MemoryRegion ivshmem_mmio;  /* BAR 0 (registers) */
    /* ivshmem-plain设备的共享内存由qemu进程即时分配 */
    MemoryRegion *ivshmem_bar2; /* BAR 2 (shared memory) */
    /* ivshmem-doorbell设备的共享内存由server进程打开共享设备文件,
     * 通过SCM_RIGHTS传入打开共享设备文件的描述符,客户端拿到后再
     * 分配内存,如果是ivshmem-doorbell设备,ivshmem_bar2会指向
     * server_bar2
     */
    MemoryRegion server_bar2;   /* used with server_chr */

    /* interrupt support */
    /* 内存共享域中其余设备形成的数组 */
    Peer *peers;
    /* 数组大小,也即共享域中其余设备的个数 */
    int nb_peers;               /* space in @peers[] */
    uint32_t vectors;
    MSIVector *msi_vectors;
    uint64_t msg_buf;           /* buffer for receiving server messages */
    int msg_buffered_bytes;     /* #bytes in @msg_buf */

    /* migration stuff */
    OnOffAuto master;			/* 只有master=on才支持迁移,通过命令行参数master属性可以指定 */
    Error *migration_blocker;	/* 如果master=off,增加该blocker */

    /* legacy cruft */
    char *role;
    char *shmobj;
    char *sizearg;
    size_t legacy_size;
    uint32_t not_legacy_32bit;
} IVShmemState;

设备初始化

  • 所有ivshmem-plain设备都属于ivshmem-plain类,类初始化函数如下:
static void ivshmem_plain_class_init(ObjectClass *klass, void *data)
{
    DeviceClass *dc = DEVICE_CLASS(klass);
    PCIDeviceClass *k = PCI_DEVICE_CLASS(klass);

    k->realize = ivshmem_plain_realize;		/* 设备创建时的初始化函数 */
    dc->props = ivshmem_plain_properties;	/* 设备属性 */
    dc->vmsd = &ivshmem_plain_vmsd;			/* 定义迁移设备状态时需要迁移IVShmemState的字段 */
}
  • ivshmem_plain_properties描述设备有哪些属性,除此之外,ivshmem-plain设备的父类是ivshmem-common设备,父类的父类是pci设备,因此ivshmem-plain还具有pci设备的所有属性:
static Property ivshmem_plain_properties[] = {
	/* master属性指定设备角色,默认为peer角色 */
    DEFINE_PROP_ON_OFF_AUTO("master", IVShmemState, master, ON_OFF_AUTO_OFF),
    /* 共享内存设备基于linux shared memory实现,这里的memdev指向主机memory-backend-file对象
     * 最终指向memory-backend-file对象关联的具体共享内存文件文件
     */
    DEFINE_PROP_LINK("memdev", IVShmemState, hostmem, TYPE_MEMORY_BACKEND,	
                     HostMemoryBackend *),
    DEFINE_PROP_END_OF_LIST(),
};
  • ivshmem_plain_vmsd描述IVShmemState数据结构的哪些字段需要迁移,在目的端加载设备时保证值相同。
static const VMStateDescription ivshmem_plain_vmsd = {
    .name = TYPE_IVSHMEM_PLAIN,
    .version_id = 0,
    .minimum_version_id = 0,
    .pre_load = ivshmem_pre_load,
    .post_load = ivshmem_post_load,
    .fields = (VMStateField[]) {
        VMSTATE_PCI_DEVICE(parent_obj, IVShmemState),	/* 迁移pci设备相关字段 */
        VMSTATE_UINT32(intrstatus, IVShmemState),		/* 迁移intrstatus字段,保证目的端设备加载后中断状态相同 */
        VMSTATE_UINT32(intrmask, IVShmemState),		 	/* 迁移intrmask字段,保证目的端设备使能的中断和源端相同 */
        VMSTATE_END_OF_LIST()
    },
};
  • ivshmem_plain_realize实现ivshmem-plain设备的核心逻辑:
static void ivshmem_plain_realize(PCIDevice *dev, Error **errp)
{
    IVShmemState *s = IVSHMEM_COMMON(dev);
	/* 检查ivshmem-plain设备是否指定了memdev内存对象 */
    if (!s->hostmem) { 
        error_setg(errp, "You must specify a 'memdev'");
        return;
    } else if (host_memory_backend_is_mapped(s->hostmem)) {
    	/* 如果内存对象被标记为已使用,报错 */
        char *path = object_get_canonical_path_component(OBJECT(s->hostmem));
        error_setg(errp, "can't use already busy memdev: %s", path);
        g_free(path);
        return;
    }
	/* ivshmem-plain设备核心逻辑和ivshmem-common设备相同 */
    ivshmem_common_realize(dev, errp);
}
  • ivshmem_common_realize对设备做初始化,对于ivshmem-plain设备,主要就是BAR0和BAR2的模拟(BAR1只有ivshmem-doorbell设备才会用到),BAR0是MMIO类型的MR,BAR2是RAM类型的MR,针对BAR0,函数主要是注册MMIO的回调钩子函数,针对BAR2,函数主要确保获取设备对应的MR信息并注册BAR。
static void ivshmem_common_realize(PCIDevice *dev, Error **errp)
{
    IVShmemState *s = IVSHMEM_COMMON(dev);
    uint8_t *pci_conf;
    uint8_t attr = PCI_BASE_ADDRESS_SPACE_MEMORY |
        PCI_BASE_ADDRESS_MEM_PREFETCH;

	/* 配置ivshmem pci配置空间的命令寄存器 */
    pci_conf = dev->config;
    /* 设置ivshmem设备支持IO访问和内存映射访问 
     * IO访问的pci设备,可以通过io指令读写BAR空间
     * 内存映射访问的pci设备,可以通过MMIO将BAR空间映射到内存上,可以像读写内存一样访问这类空间
     */
    pci_conf[PCI_COMMAND] = PCI_COMMAND_IO | PCI_COMMAND_MEMORY;
	/* MMIO memory初始化,注册IO回调函数用以响应guest对BAR0寄存器的读写请求 */
    memory_region_init_io(&s->ivshmem_mmio, OBJECT(s), &ivshmem_mmio_ops, s,
                          "ivshmem-mmio", IVSHMEM_REG_BAR_SIZE);
	/* 将MR注册为BAR */
    /* region for registers*/
    pci_register_bar(dev, 0, PCI_BASE_ADDRESS_SPACE_MEMORY,
                     &s->ivshmem_mmio);

	/* 如果共享内存是ivshmem-plain设备,memdev参数必须指定,此处hostmem必不为NULL */
    if (s->hostmem != NULL) {
        IVSHMEM_DPRINTF("using hostmem\n");
		/* 获取backend的MR作为共享内存的MR */
        s->ivshmem_bar2 = host_memory_backend_get_memory(s->hostmem);
        /* 标记该memdev object已经被使用,防止其它ivshmem-plain设备使用 */
        host_memory_backend_set_mapped(s->hostmem, true);
    } 
    /* master有三个属性on/off/auto,如果命令行配置为auto则由Qemu判断
     * 对于ivshmem-plain设备,vm_id默认为0,因此角色默认为master
     * 对于ivshmem-doorbell设备,第一个连接server分配得到的ID是0,作为master
     */
    if (s->master == ON_OFF_AUTO_AUTO) {
        s->master = s->vm_id == 0 ? ON_OFF_AUTO_ON : ON_OFF_AUTO_OFF;
    }
	/* 如果ivshmem设备不是master,增加迁移blocker,禁止配置该设备的虚机迁移 */
    if (!ivshmem_is_master(s)) {
        error_setg(&s->migration_blocker,
                   "Migration is disabled when using feature 'peer mode' in device 'ivshmem'");
        migrate_add_blocker(s->migration_blocker, &local_err);
        if (local_err) {
            error_propagate(errp, local_err);
            error_free(s->migration_blocker);
            return;
        }
    }
	/* ivshmem设备的BAR2是普通内存,对于普通内存通常通过memory_region_init_ram初始化MR
	 * 这里由于是共享内存,MR已经通过memdev对象初始化完成,因此只需要再标记下该MR关联
	 * 的RAMBlock可以迁移并设置idstr即可。标记RAMBlock可迁移的本质是帮助ram_save_iterate
	 * 在遍历ram_list.blockss时判断该RAMBlock是否可以迁移
	 */
    vmstate_register_ram(s->ivshmem_bar2, DEVICE(s));
    /* 将MR注册为ivshmem-plain设备的BAR2 */
    pci_register_bar(PCI_DEVICE(s), 2, attr, s->ivshmem_bar2);
}
  • 对于BAR0,guest通过MMIO方式访问时,后端触发对应的IO回调函数,回调函数在ivshmem_mmio_ops中注册,如下:
static const MemoryRegionOps ivshmem_mmio_ops = {
    .read = ivshmem_io_read,	/* guest读bar0的回调函数 */
    .write = ivshmem_io_write,	/* guest写bar0的回调函数 */
    .endianness = DEVICE_NATIVE_ENDIAN,
    .impl = {
        .min_access_size = 4,
        .max_access_size = 4,
    },
};
/* 根据guest读的地址,返回IVShmemState中对应的字段 
 * 即intrmask、intrstatus、vm_id三个字段
 */
static uint64_t ivshmem_io_read(void *opaque, hwaddr addr,
                                unsigned size)
{

    IVShmemState *s = opaque;
    uint32_t ret;

    switch (addr)
    {
        case INTRMASK:
            ret = ivshmem_IntrMask_read(s);
            break;

        case INTRSTATUS:
            ret = ivshmem_IntrStatus_read(s);
            break;

        case IVPOSITION:
            ret = s->vm_id;
            break;

        default:
            IVSHMEM_DPRINTF("why are we reading " TARGET_FMT_plx "\n", addr);
            ret = 0;
    }

    return ret;
}
/* 根据guest的写地址,如果是写intrmask、intrstatus两个寄存器
 * 则表明是中断本虚拟机,设置对应的寄存器值并注入中断
 * 如果是写doorbell集群起,表明是中断共享内存的其它虚机
 * 通过写eventfd通知
 */
static void ivshmem_io_write(void *opaque, hwaddr addr,
                             uint64_t val, unsigned size)
{
    IVShmemState *s = opaque;

    uint16_t dest = val >> 16;
    uint16_t vector = val & 0xff;

    addr &= 0xfc;

    IVSHMEM_DPRINTF("writing to addr " TARGET_FMT_plx "\n", addr);
    switch (addr)
    {
        case INTRMASK:
            ivshmem_IntrMask_write(s, val);
            break;

        case INTRSTATUS:
            ivshmem_IntrStatus_write(s, val);
            break;

        case DOORBELL:
            /* check that dest VM ID is reasonable */
            if (dest >= s->nb_peers) {
                IVSHMEM_DPRINTF("Invalid destination VM ID (%d)\n", dest);
                break;
            }
            
            /* check doorbell range */
            if (vector < s->peers[dest].nb_eventfds) {
                IVSHMEM_DPRINTF("Notifying VM %d on vector %d\n", dest, vector);
                 /* 向vm_id为dest的设备发送一个vector中断向量
                  * ivshmem-doorbell设备注册了ivshmem_vector_notify handler
                  * 用于处理evenfd上的读写事件
                  */
                event_notifier_set(&s->peers[dest].eventfds[vector]);
            } else {
            	/* 向其它设备注入了一个没有注册的无效中断向量,报错 */
                IVSHMEM_DPRINTF("Invalid destination vector %d on VM %d\n",
                                vector, dest);
            }
            break;
        default:
            IVSHMEM_DPRINTF("Unhandled write " TARGET_FMT_plx "\n", addr);
    }
}

测试验证

方案

  • 我们通过下面简单的测试demo验证ivshmem-plain设备的整个工作流程,测试demo代码分为两部分:
  1. guest, 这部分代码运行在虚机内部,主要是驱动代码,用于识别ivshmem-plain pci设备,并将pci设备的bar2空间通过MMIO的方式映射为内存,同时将映射后内存的地址通过proc文件系统暴露到用户态,方便通过命令行方式往ivshmem-plain设备的bar2写入测试内容。参考ivshmem-plain guest驱动
  2. host,这部分代码运行主机上,主要是读取qemu的ivshmem-plain设备暴露的共享内存数据,验证guest驱动中写的内容。参考ivshmem-plain host程序
  • 如下图所示,guest驱动代码即ivshmem drv模块,它识别到ivshmem-plain pci设备后,提供procfs接口供命令行读写共享内存,host程序即test agent模块,它读取Qemu暴露的共享内存文件内容并打印。通过比较test agent模块读取的内容是否与guest中向/proc/ivshmem_demo/bar2写入的内容相同,测试内存是否共享。
    在这里插入图片描述

流程

Libvirt配置

我们配置了4M的共享内存,默认情况下,如果xml中role属性不设置,则qemu默认会配置为peer角色:

    <shmem name='my_shmem0'>
      <model type='ivshmem-plain'/>
      <size unit='M'>4</size>
      <alias name='shmem0'/>
      <address type='pci' domain='0x0000' bus='0x00' slot='0x0b' function='0x0'/>
    </shmem>

Qemu配置

  • 命令行
-object memory-backend-file,id=shmmem-shmem0,mem-path=/dev/shm/my_shmem0,size=4194304,share=yes 
-device ivshmem-plain,id=shmem0,memdev=shmmem-shmem0,bus=pci.0,addr=0xb
  • 进程地址空间
    Qemu进程的虚机地址空间布局(cat /proc/pid/maps),可以看到分配了4M的虚拟地址空间
7fc5b99af000-7fc5ba1af000 rw-p 00000000 00:00 0
7fc5ba1af000-7fc5ba5af000 rw-s 00000000 00:16 433243576                  /dev/shm/my_shmem0
7fc5ba5af000-7fc5ba5b0000 ---p 00000000 00:00 0
  • Qemu设备树
    Qemu hmp info qtree输出:
      dev: ivshmem-plain, id "shmem0"
        master = "off"
        memdev = "/objects/shmmem-shmem0"
        addr = 0b.0
        romfile = ""
        rombar = 1 (0x1)
        multifunction = false
        command_serr_enable = true
        x-pcie-lnksta-dllla = true
        x-pcie-extcap-init = true
        class RAM controller, addr 00:0b.0, pci id 1af4:1110 (sub 1af4:1100)
        bar 0: mem at 0xfea5a000 [0xfea5a0ff]
        bar 2: mem at 0xfe000000 [0xfe3fffff]
  • Qemu内存树
    Qemu hmp info mtree -f输出:
  00000000fd000000-00000000fd3fffff (prio 0, i/o): cirrus-bitblt-mmio
  00000000fe000000-00000000fe3fffff (prio 1, ram): /objects/shmmem-shmem0	/* BAR 2 */
  00000000fe600000-00000000fe600fff (prio 0, i/o): virtio-pci-common

  00000000fea59800-00000000fea59807 (prio 0, i/o): msix-pba
  00000000fea5a000-00000000fea5a0ff (prio 1, i/o): ivshmem-mmio 			/* BAR 0 */
  00000000fec00000-00000000fec00fff (prio 0, i/o): kvm-ioapic
  • guest
    虚机内部pci信息输出:
[root@ivshmem ~]# lspci -v -s 00:0b.0
00:0b.0 RAM memory: Virtio: Inter-VM shared memory (rev 01)
	Subsystem: Virtio: QEMU Virtual Machine
	Physical Slot: 11
	Flags: fast devsel
	Memory at fea5a000 (32-bit, non-prefetchable) [size=256]
	Memory at fe000000 (64-bit, prefetchable) [size=4M]
	Kernel driver in use: ivshmem_demo_driver

测试步骤

  1. guest加载ivshmem-drv.ko驱动模块,识别ivshmem-plain pci设备,运行insmod ivshmem-drv.ko, 内核驱动通过vendor_id和device_id识别到ivshmem-plain设备,并通过pci规范探测该设备需要申请的空间大小,并做内存映射,日志如下:
    在这里插入图片描述
    从上面可以看到,bar 2空间大小为0x400000(4M),起始地址为0xfe00000,被ioremap到了0x45400000地址作为普通内存使用。
  2. 查看生成的proc接口,没有任何内容:
    在这里插入图片描述
  3. 往BAR2写入如下内容并读取确认:
    在这里插入图片描述
    在这里插入图片描述
  4. 主机侧运行测试程序,读取qemu暴露的共享内存文件/dev/shm/my_shmem0确认:
    在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/687584.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

618美妆个护28个榜单:欧莱雅稳住冠军?珀莱雅大爆发第二?

存量时代的购物造节大竞争&#xff0c;作为消费复苏后的首场大促&#xff0c;今年的618堪称史上最“卷”&#xff0c;也承载着消费振兴、经济复苏等希望。 不过&#xff0c;今年所有平台都未公布具体GMV&#xff0c;某种程度说明大促造节的时代俨然已成过去式了。 5月18日&am…

怎么去除视频里的背景音乐?其实非常简单!

如何去除视频背景音乐&#xff1f;在视频处理中&#xff0c;有时我们需要从视频中提取声音并进行处理&#xff0c;而不仅仅是简单地去除整个背景音乐。我们可能需要有选择性地去除人声或背景音乐。这个处理过程对于选用合适的工具至关重要。在本文中&#xff0c;我将分享两种可…

【⑦MySQL】· 一文了解四大子查询

前言 ✨欢迎来到小K的MySQL专栏&#xff0c;本节将为大家带来MySQL标量/单行子查询、列子/表子查询的讲解✨ 目录 前言一、子查询概念二、标量/单行子查询、列子/表子查询三、总结 一、子查询概念 子查询指一个查询语句嵌套在另一个查询语句内部的查询&#xff0c;这个特性从My…

抖音林客生活服务商机构

抖音林客生活服务商机构是在抖音平台上提供各种生活服务的机构或组织。这些机构通常会提供家政服务、保洁服务、美容美发服务等&#xff0c;也有一些提供餐饮、旅游、电商等服务。用户可以通过抖音搜索、浏览和下单&#xff0c;享受到优质的服务体验。 这些服务商机构在抖音…

数值组件滚动趋势图联动需求拆解

技术栈&#xff1a;使用vue3的composition API tsx 进行开发 一、需求描述 直接看UI图吧。 简单描述一下&#xff1a; 数值卡片&#xff1a; 上方部分是一个数值卡片列表&#xff0c;每个卡片维护不同的集中状态&#xff0c;选中态&#xff0c;hover态。 细节&#xff1…

【测试学习】Junit5的简单使用

目录 &#x1f31f;需要知道&#xff1a; &#x1f31f;Junit学习 &#x1f308;1、常用的注解 &#x1f308;2、测试用例的执行顺序 &#x1f308;3、参数化 &#x1f308;4、断言 &#x1f308;5、测试套件 &#x1f31f;需要知道&#xff1a; 问题1&#xff1a;Selen…

ATA-4315高压功率放大器在铁路钢轨损伤检测中的应用

随着高速铁路的建设和不断发展&#xff0c;确保铁路线路的安全和稳定运行变得越来越重要。钢轨作为铁路的重要组成部分&#xff0c;其损坏可能导致严重的事故和交通堵塞。因此&#xff0c;对钢轨损伤进行及时、准确的检测至关重要。高压功率放大器作为一种精密的测试仪器&#…

SQL 优化(一):慎用 SQL 函数

假如有下面这样一张用户表 CREATE TABLE t_user (user_id int(11) NOT NULL AUTO_INCREMENT,username varchar(50) DEFAULT NULL,sex tinyint(1) DEFAULT NULL,mobile varchar(45) DEFAULT NULL,create_time datetime DEFAULT CURRENT_TIMESTAMP,PRIMARY KEY (user_id),KEY id…

Linux系统下查看网卡配置和网络流量监控指令:ifconfig、ethtool

文章目录 1 查看/设置网卡&#xff08;ifconfig&#xff09;2 查看网卡配置、属性&#xff08;ip、ethtool&#xff09;3 查看网络接口配置情况&#xff1a;4 查看网络接口属性信息5 测试网路通信&#xff08;ping&#xff09; 6 查看具体网卡配置文件 1 查看/设置网卡&#xf…

Unity | HDRP高清渲染管线学习笔记:Lightmapping(光照烘焙)与Lightmap(光照贴图)

目录 相关概念 1.渐进式光照贴图烘焙 1.1 渐进式光照贴图烘焙对模型的要求 1.2 渐进式光照贴图烘焙对硬件的要求 1.3 渐进式光照贴图烘焙支持的Unity渲染管线 1.4 进行渐进式光照贴图烘焙结果 1.5 渐进式光照贴图烘焙的CPU版本和GPU版本 1.6 Lighting窗口Lightmapping …

2023MWC上海|平行云邀您一起聊聊“未来数字世界中的社交,游戏与娱乐”

“世界移动通信大会&#xff08;MWC上海&#xff09;&#xff0c;自2012年落户上海以来已经举办9届&#xff0c;是中国乃至亚太区域移动产业向世界展示最新发展成果与发展战略的关键平台。2023年MWC上海将于2023年6月28-30日在上海新国际博览中心举行&#xff0c;主题是“时不我…

Spring 更简单的读取和存储对象、使用注解存取对象

文章目录 1.前言2.存储 Bean对象2.1 前置任务&#xff1a;配置扫描路径&#xff08;重中之重&#xff09;2.2 添加注解存储 Bean 对象2.2.1 类注解2.2.2 方法注解 Bean 3.获取 Bean对象3.1 属性注入3.2 构造方法注入3.3 Setter 注⼊3.4 三种注释的优缺点3.5 另⼀种注⼊关键字&a…

python学习——时间序列

目录 1 生成时间1.1 pd.Timestamp1.2 pd.to_datetime 最常用1.2.1 单个转换为时间戳1.2.2 多个时间转换为时间戳1.2.3 日期在前 dayfirst1.2.4 处理特殊格式 format1.2.5 处理无效时间 errors1.2.6 和timestamp互转 1.3 固定跨度时间的生成 pd.date_range 2 时间格式转换 .dt.s…

AI工具在不同领域的应用范围分享,让我们一起了解

随着人工智能技术的不断发展&#xff0c;越来越多的领域开始应用AI工具来解决问题、提高效率和创造力。无论是在医疗、金融、教育还是其他行业&#xff0c;AI工具都展现出了巨大的潜力和价值。今天&#xff0c;我将分享一些常见的AI工具在不同领域的应用范围&#xff0c;让我们…

Consul:什么是Consul? ①

一、思想 微服务、分布式应用的特点就是拆分&#xff0c;拆分带来的问题就是服务器数量成倍增多&#xff0c;两个很重要的原因&#xff0c;需要抽离出来一个项目专门管理这些服务信息。 1、对信息的管理思想&#xff0c;统一化、中心化一直是一套比较常规成熟的标准。 2、并且它…

开源工具系列8:Spring Security

Spring Security 是一套认证授权框架, 支持认证模式如 HTTP BASIC 认证头 (基于 IETF RFC-based 标准), HTTP Digest 认证头 ( IETF RFC-based 标准), Form-based authentication (用于简单的用户界面), OpenID 认证等, Spring Security使得当前系统可以快速集成这些验证机制亦…

Matplotlib---等高线图

1. 等高线图 等高线图&#xff1a;也称水平图&#xff0c;是一种在二维平面上显示 3D 图像的方法。 x np.linspace(-5, 5, 100) y np.linspace(-5, 5, 100)# 将上述数据变成网格数据形式 X, Y np.meshgrid(x, y)# 定义Z与X, Y之间的关系 Z np.sqrt(X**2 Y**2)# 画等高线…

Win11 系统Java17的安装教程:最新版JDK 17.07下载、安装、卸载详解

&#x1f337;&#x1f341; 博主 libin9iOak带您 Go to New World.✨&#x1f341; &#x1f984; 个人主页——libin9iOak的博客&#x1f390; &#x1f433; 《面试题大全》 文章图文并茂&#x1f995;生动形象&#x1f996;简单易学&#xff01;欢迎大家来踩踩~&#x1f33…

如何利用AI高效率快速调色

在设计行业中&#xff0c;时间是非常宝贵的资源&#xff0c;而设计师们常常需要应对繁忙的工作日程和紧迫的截止日期。为了提高工作效率和节省时间&#xff0c;越来越多的设计师开始利用人工智能&#xff08;AI&#xff09;技术中的高效调色功能。本文将介绍如何利用AI高效率快…

震惊!国产AI模特已经可以做到这个程度了,未来可期

自从GhatGPT火了之后 AI就成为网络热词 并且渗透到各行各业中 你敢相信下面这些图模特图 竟然都是AI生成的&#xff01; 简直不要太逼真 这次AI真的可以取代一部分模特了 假人模特 神采PromeAI生成 神采PromeAI生成 从生成效果来看 AI模特已非常接近真人 对于电商企业或…