IOFrameBufferShared 中不同的游标格式

发布于 2024-08-24 21:28:30 字数 3378 浏览 10 评论 0原文

我正在从 StdFBShmem_t 结构中读取鼠标光标像素图数据,如 IOFrameBufferShared API

90% 的情况下,一切都运行良好。但是,我注意到 Mac 上的某些应用程序以不同的格式设置光标。根据数据结构的文档,光标像素图格式应始终与帧缓冲区的格式相同。我的帧缓冲区是 32 bpp。我希望像素图数据采用 0xAARRGGBB 格式,(大多数时候)就是这样。然而,在某些情况下,我正在读取看起来像掩码的数据。具体来说,该数据中的像素将为0x00FFFFFF或“0x00000000”。在我看来,这是存储在其他地方的单独像素数据的掩码。

据我所知,使用此光标像素格式的唯一应用程序是 Qt Creator,但我需要使用所有应用程序,所以我想解决这个问题。

我用来读取光标像素图数据的代码是:

NSAutoreleasePool *autoReleasePool = [[NSAutoreleasePool alloc] init];

NSPoint mouseLocation = [NSEvent mouseLocation];
NSArray *allScreens = [NSScreen screens];
NSEnumerator *screensEnum = [allScreens objectEnumerator];
NSScreen *screen;
NSDictionary *screenDesc = nil;
while ((screen = [screensEnum nextObject]))
{
    NSRect screenFrame = [screen frame];
    screenDesc = [screen deviceDescription];
    if (NSMouseInRect(mouseLocation, screenFrame, NO))
        break;
}

if (screen)
{
    kern_return_t err;

    CGDirectDisplayID displayID = (CGDirectDisplayID) [[screenDesc objectForKey:@"NSScreenNumber"] pointerValue];
    task_port_t taskPort = mach_task_self();
    io_service_t displayServicePort = CGDisplayIOServicePort(displayID);
    io_connect_t displayConnection =0;
    err = IOFramebufferOpen(displayServicePort,
                            taskPort,
                            kIOFBSharedConnectType,
                            &displayConnection);
    if (KERN_SUCCESS == err)
    {
        union
        {
            vm_address_t vm_ptr;
            StdFBShmem_t *fbshmem;
        } cursorInfo;
        vm_size_t size;

        err = IOConnectMapMemory(displayConnection,
                                 kIOFBCursorMemory,
                                 taskPort,
                                 &cursorInfo.vm_ptr,
                                 &size,
                                 kIOMapAnywhere | kIOMapDefaultCache | kIOMapReadOnly);
        if (KERN_SUCCESS == err)
        {
            // For some reason, cursor data is not always in the same format as
            // the frame buffer. For this reason, we need some way to detect
            // which structure we should be reading.
            QByteArray pixData(
              (const char*)cursorInfo.fbshmem->cursor.rgb24.image[currentFrame],
              m_mouseInfo.currentSize.width() * m_mouseInfo.currentSize.height() * 4);

            IOConnectUnmapMemory(displayConnection,
                                 kIOFBCursorMemory,
                                 taskPort,
                                 cursorInfo.vm_ptr);
        } // IOConnectMapMemory
        else
            qDebug() << "IOConnectMapMemory Failed:" << err;
        IOServiceClose(displayConnection);
    } // IOServiceOpen
    else
        qDebug() << "IOFramebufferOpen Failed:" << err;
}// if screen
[autoReleasePool release];

我的问题是:

  1. 如何检测光标是否是不同的格式 从帧缓冲区?

  2. 哪里可以读取实际的像素数据? bm18Cursor 结构体包含一个掩码部分,但它不在 我使用代码阅读它的正确位置

I'm reading the moust cursor pixmap data from the StdFBShmem_t structure, as defined in the IOFrameBufferShared API.

Everything works fine, 90% of the time. However, I have noticed that some applications on the Mac set a cursor in a different format. According to the documentation for the data structures, the cursor pixmap format should always be in the same format as the frame buffer. My frame buffer is 32 bpp. I expect the pixmap data to be in the format 0xAARRGGBB, which it is (most of the time). However, in some cases, I'm reading data that looks like a mask. Specifically, the pixels in this data will either be 0x00FFFFFF or `0x00000000. This looks to me to be a mask for separate pixel data stored somewhere else.

As far as I can tell, the only application that uses this cursor pixel format is Qt Creator, but I need to work with all applications, so I'd like to sort this out.

The code I'm using to read the cursor pixmap data is:

NSAutoreleasePool *autoReleasePool = [[NSAutoreleasePool alloc] init];

NSPoint mouseLocation = [NSEvent mouseLocation];
NSArray *allScreens = [NSScreen screens];
NSEnumerator *screensEnum = [allScreens objectEnumerator];
NSScreen *screen;
NSDictionary *screenDesc = nil;
while ((screen = [screensEnum nextObject]))
{
    NSRect screenFrame = [screen frame];
    screenDesc = [screen deviceDescription];
    if (NSMouseInRect(mouseLocation, screenFrame, NO))
        break;
}

if (screen)
{
    kern_return_t err;

    CGDirectDisplayID displayID = (CGDirectDisplayID) [[screenDesc objectForKey:@"NSScreenNumber"] pointerValue];
    task_port_t taskPort = mach_task_self();
    io_service_t displayServicePort = CGDisplayIOServicePort(displayID);
    io_connect_t displayConnection =0;
    err = IOFramebufferOpen(displayServicePort,
                            taskPort,
                            kIOFBSharedConnectType,
                            &displayConnection);
    if (KERN_SUCCESS == err)
    {
        union
        {
            vm_address_t vm_ptr;
            StdFBShmem_t *fbshmem;
        } cursorInfo;
        vm_size_t size;

        err = IOConnectMapMemory(displayConnection,
                                 kIOFBCursorMemory,
                                 taskPort,
                                 &cursorInfo.vm_ptr,
                                 &size,
                                 kIOMapAnywhere | kIOMapDefaultCache | kIOMapReadOnly);
        if (KERN_SUCCESS == err)
        {
            // For some reason, cursor data is not always in the same format as
            // the frame buffer. For this reason, we need some way to detect
            // which structure we should be reading.
            QByteArray pixData(
              (const char*)cursorInfo.fbshmem->cursor.rgb24.image[currentFrame],
              m_mouseInfo.currentSize.width() * m_mouseInfo.currentSize.height() * 4);

            IOConnectUnmapMemory(displayConnection,
                                 kIOFBCursorMemory,
                                 taskPort,
                                 cursorInfo.vm_ptr);
        } // IOConnectMapMemory
        else
            qDebug() << "IOConnectMapMemory Failed:" << err;
        IOServiceClose(displayConnection);
    } // IOServiceOpen
    else
        qDebug() << "IOFramebufferOpen Failed:" << err;
}// if screen
[autoReleasePool release];

My questions are:

  1. How can I detect if the cursor is a different format
    from the framebuffer?

  2. Where can I read the actual pixel data? The bm18Cursor
    structure contains a mask section, but it's not in the
    right place for me to be reading it using the code
    above.

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(2

总攻大人 2024-08-31 21:28:30

如何检测光标是否与帧缓冲区的格式不同?

光标位于帧缓冲区中。它的格式不能与其本身不同。

无法判断它的格式 (x-radar://problem/7751503)。如果你能知道光标有多少帧,那么就有一种方法至少可以预测每个像素的字节数,但因为你不能(该信息从 10.6.1 开始没有设置 — x-radar:/ /problem/7751530),您需要尝试找出四因子乘积的两个因子(每像素字节数 × 宽度 × 高度 × 帧数,其中只有宽度、高度和乘积)。即使您可以找出缺少的两个因素,您仍然不知道字节的顺序或颜色分量是否预乘了 alpha 分量。

哪里可以读取实际的像素数据?

在共享游标内存结构的 cursor 成员中。

您应该在包含 I/O Kit 标头之前定义 IOFB_ARBITRARY_SIZE_CURSOR。游标现在可以是任何大小,而不仅仅是 16×16,这是您未定义该常量时所期望的大小。举个例子,通常的Mac箭头光标是24×24,CrossOver中的“Windows”箭头光标是32×32,X11中的箭头光标是10×16。

但是,在某些情况下,我正在读取看起来像掩码的数据。具体来说,该数据中的像素将为 0x00FFFFFF0x00000000。在我看来,这是存储在其他地方的单独像素数据的掩码。

在我看来,这更像是带有 8 位 Alpha 通道的 16 位像素。至少5-6-5的可能性比5-5-5的可能性更大。

据我所知,使用此光标像素格式的唯一应用程序是 Qt Creator,但我需要使用所有应用程序,因此我想解决这个问题。

我可以使用 我的新光标捕获应用程序。我是否应该点击应用程序的特定部分才能使其显示特定的光标?

How can I detect if the cursor is a different format from the framebuffer?

The cursor is in the framebuffer. It can't be in a different format than itself.

There is no way to tell what format it's in (x-radar://problem/7751503). There would be a way to divine at least the number of bytes per pixel if you could tell how many frames the cursor has, but since you can't (that information isn't set as of 10.6.1 — x-radar://problem/7751530), you are left trying to figure out two factors of a four-factor product (bytes per pixel × width × height × number of frames, where you only have the width, the height, and the product). And even if you can figure out those missing two factors, you still don't know what order the bytes are in or whether the color components are premultiplied by the alpha component.

Where can I read the actual pixel data?

In the cursor member of the shared-cursor-memory structure.

You should define IOFB_ARBITRARY_SIZE_CURSOR before including the I/O Kit headers. Cursors can be any size now, not just 16×16, which is the size you expect when you don't define that constant. As an example, the usual Mac arrow cursor is 24×24, the “Windows” arrow cursor in CrossOver is 32×32, and the arrow cursor in X11 is 10×16.

However, in some cases, I'm reading data that looks like a mask. Specifically, the pixels in this data will either be 0x00FFFFFF or 0x00000000. This looks to me to be a mask for separate pixel data stored somewhere else.

That sounds to me more like 16-bit pixels with an 8-bit alpha channel. At least it's more probably 5-6-5 than 5-5-5.

As far as I can tell, the only application that uses this cursor pixel format is Qt Creator, but I need to work with all applications, so I'd like to sort this out.

I'm able to capture the current cursor in that app just fine with my new cursor-capturing app. Is there a specific part of the app I should hit to make it show me a specific cursor?

飘落散花 2024-08-31 21:28:30

您可以尝试 CGSCreateRegisteredCursorImage 函数,如 Karsten 在 中演示的那样我的博客上的评论

它是一个私有函数,所以它可能随时改变或消失,所以你应该检查它是否存在并保留IOFramebuffer,但只要它确实存在,你可能会发现它比复杂和薄的更可靠-记录了 IOFramebuffer。

You might try the CGSCreateRegisteredCursorImage function, as demonstrated by Karsten in a comment on my weblog.

It is a private function, so it may change or go away at any time, so you should check whether it exists and hold IOFramebuffer in reserve, but as long as it does exist, you may find it more reliable than the complex and thinly-documented IOFramebuffer.

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文