在 DiskArbitration 框架中注销回调的正确语法

Correct syntax to unregister a callback in the DiskArbitration framework

我将实现一个观察者,以便在特定卷消失时收到通知。

NSWorkspace 中的通知不是一个选项,因为我需要有关 设备关闭 而不是 设备已卸载 [=38= 的通知].

注册非常简单(我省略了错误处理)

var callbackSession : DASession?

let callbackSession = DASessionCreate(kCFAllocatorDefault)
DASessionSetDispatchQueue(callbackSession!, DispatchQueue.global())  
DARegisterDiskDisappearedCallback(callbackSession!, nil, volumeDidDisappearCallback, nil)

回调闭包是

let volumeDidDisappearCallback : DADiskDisappearedCallback = { (disk, context) in
   // do something with disk
}

注销方法

func DAUnregisterCallback(_ session: DASession, _ callback: UnsafeMutableRawPointer, _ context: UnsafeMutableRawPointer?)

需要通用 UnsafeMutableRawPointer 而不是特定回调类型之一。

当我将回调闭包传递给 callback 参数时,我得到

Cannot convert value of type 'DADiskDisappearedCallback' (aka '@convention(c) (DADisk, Optional) -> ()') to expected argument type 'UnsafeMutableRawPointer'

我的问题是:如何将回调对象 volumeDidDisappearCallback 转换为 UnsafeMutableRawPointer

DADiskDisappearedCallback 定义为

public typealias DADiskDisappearedCallback = @convention(c) (DADisk, UnsafeMutableRawPointer?) -> Swift.Void

这是C函数指针的Swift映射

typedef void (*DADiskAppearedCallback)(DADiskRef disk, void *context);

注销函数在C中定义为

void DAUnregisterCallback(DASessionRef session, void *callback, void *context);

带有一个 void *callback 参数,在 C 中你可以传递各种 各种回调函数指针(DADiskAppearedCallbackDADiskDescriptionChangedCallback、...),无需强制转换。

在 Swift 中,您必须将函数指针显式转换为 void 指针:

let cb = unsafeBitCast(volumeDidDisappearCallback, to: UnsafeMutableRawPointer.self)
DAUnregisterCallback(callbackSession!, cb, nil)