文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

我运行 2 个 goroutine 来修改虚拟机并在同一实例上使用 libvirt-go 包销毁同一虚拟机会发生什么?

2024-02-08 21:36

关注

php小编鱼仔你提到的问题涉及到在同一实例上同时使用libvirt-go包销毁虚拟机的情况。在这种情况下,会出现两个goroutine同时修改虚拟机的情况,可能会导致不可预料的结果。由于goroutine在并发执行时无法保证执行顺序,可能会导致竞态条件或数据冲突,进而导致虚拟机销毁失败、数据损坏或其他异常情况。为了避免这种情况,可以通过使用互斥锁或其他并发控制机制来确保同时只有一个goroutine可以修改虚拟机。这样可以保证操作的原子性和一致性,避免不必要的问题发生。

问题内容

众所周知,libvirt 是线程安全的。 但是,同时运行两个作用于同一资源的 goroutine(例如修改和删除虚拟机)会使其处于不明确的状态。 libvirt 如何决定 goroutine 的执行顺序?

这是我尝试过的代码:

package main

import (
    "fmt"

    "github.com/libvirt/libvirt-go"
)

func main() {
    conn, err := libvirt.NewConnect("qemu:///system")
    if err != nil {
        fmt.Printf("Failed to connect to libvirt: %v\n", err)
        return
    }
    defer conn.Close()

    // Create a new VM
    domainXML := `
        
            myvm
            1048576
            1
            
                hvm
                
            
            
                
                    
                    
                    
                    
` dom, err := createVM(conn, domainXML) if err != nil { fmt.Printf("Failed to create VM: %v\n", err) return } go modifyVMMemory(dom, 2*1024*1024) // 2 GiB go deleteVM(dom) } func createVM(conn *libvirt.Connect, domainXML string) (*libvirt.Domain, error) { dom, err := conn.DomainCreateXML(domainXML, 0) if err != nil { return nil, err } return dom, nil } func modifyVMMemory(dom *libvirt.Domain, newMemory uint64) error { err := dom.SetMaxMemory(newMemory) if err != nil { return err } fmt.Print("Modified VM") return nil } func deleteVM(dom *libvirt.Domain) error { err := dom.Destroy() if err != nil { return err } err = dom.Undefine() if err != nil { return err } fmt.Print("Deleted VM") return nil }

程序成功完成,因此域被破坏并可以重新创建 但再次运行会导致以下错误:

virError(Code=9, Domain=20, Message='operation failed: domain 'myvm' already exists with uuid 32c25acb-a4c5-4bfd-b2f5-f07b3d9b8eea')

解决方法

线程安全只是意味着当多个线程同时使用同一个连接时,代码不会出现内存损坏问题。

您将获得的语义行为仍然是不确定的。

libvirt QEMU/KVM 驱动程序在客户端应用程序和 libvirtd(或 virtqemud)守护进程之间使用 RPC 层。所以首先你有非确定性,其中 Goroutine 首先运行。当 libvirt-go-module API 通过 CGo 调用 C ibvirt.so 库时,它们将被锁定到本机操作系统线程,然后该线程将在 libvirt.so 内部同步,以决定哪个首先将其 RPC 消息发送到网络上。在 libvirtd 守护进程中,也有很多线程,并且 RPC 消息理论上是 FIFO 处理的,但是,libvirtd 内部的 API 逻辑仍然会争夺锁,因此在与 QEMU 通信/交互时会增加更多的不确定性。基本上,您的 SetMaxMemoryDestroy API 调用可以按任一顺序运行。如果您需要保证顺序,则需要在应用程序中对它们进行序列化,以便仅在完成 ​​SetMaxMemory 后调用 Destroy

最后 IIUC 你的 Go 代码并不健壮,因为 main() 方法生成了两个 goroutine,但没有等待它们中的任何一个完成。 IOW,在任一 goroutine 完全运行之前,Go 进程很可能会退出。这可能就是为什么您收到有关 VM 已存在的错误消息的原因 - deleteVM goroutine 在进程退出之前从未运行。

以上就是我运行 2 个 goroutine 来修改虚拟机并在同一实例上使用 libvirt-go 包销毁同一虚拟机会发生什么?的详细内容,更多请关注编程网其它相关文章!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     807人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     351人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     314人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     433人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     221人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯