3

我正在根据“X 协议参考手册:零卷”的副本从头开始为 Windows 构建 X11 服务器。我在破译信息和与客户进行有意义的对话方面取得了很大进展,但我无法理解绘图调用实际上应该做什么。

此示例中的消息来自在 Linux 机器上运行 xbiff 并让它与我在 Windows 上的 xserver 通信。我完全有可能在解释协议时弄错了一些东西,但到目前为止数据似乎是正确的。

图形调用由此开始,客户端创建一个以根窗口 ID (90) 作为可绘制对象的图形上下文:

X_CreateGC: ID: 2097152, Drawable: 90, ValueMask: 8, Value0: 16777215

基于根窗口创建GC有什么意义?

接下来,它创建两个 48x48 像素图并在其上放置图像:

X_CreatePixmap: Depth: 1, ID: 2097153, Drawable: 90, Width: 48, Height: 48

X_CreateGC: ID: 2097154, Drawable: 2097153, ValueMask: 12, Value0: 1, Value1: 0

X_PutImage: Format: 0, Size: 408, Drawable: 2097153, GraphicsContext: 2097154, Width: 48, Height: 48, X: 0, Y: 0, LeftPad: 0, Depth: 1

X_FreeGC: Graphics Context: 2097154

X_CreatePixmap: Depth: 1, id: 2097155, Drawable: 90, Width: 48, Height: 48

X_CreateGC: ID: 2097156, Drawable: 2097155, ValueMask: 12, Value0: , Value1: 0

X_PutImage: Format: 0, Size: 408, Drawable: 2097155, Graphics Context: 2097156, WIdth: 48, Height: 48, X: 0, Y: 0, LeftPad: 0, Depth: 1

X_FreeGC: Graphics Context: 2097156

我认为这里的 GC 相当于 MemoryDC 并且最终结果应该是内存中包含 PutImage 调用中的数据的两个 48x48 位图是否正确?

在这里,它基于根窗口创建了另一个图形上下文,但我不明白为什么:

X_CreateGC: ID: 2097157, Drawable: 90, ValueMask: 65544, Value0: 16777215, Value1: 0

接下来它创建两个 48x48 窗口,一个以根为父窗口,下一个以第一个窗口为父窗口:

X_CreateWindow: Depth: 24, ID: 2097158, Parent: 90, X: 0, Y: 0, Width: 48, Height: 48, BorderWidth: 1, Class: 1, Visual: 0, Bitmask: 10266, Value0: 16777215, Value1: 0, Value2: 1, Value3: 6422576, Value4: 32

X_CreateWindow: Depth: 24, ID: 2097161, Parent: 2097158, X: 0, Y: 0, Width: 48, Height: 48, BorderWidth: 0, Class: 1, Visual: 0, Bitmask: 26650, Value0: 16777215, Value1: 0, Value2: 0, Value3: 163852, Value4: 32, Value5: 0

似乎这是在创建一个 48x48 基本窗口,其中有一个窗口,其大小和原点相同。那是什么意思?子窗口不会掩盖根窗口,使其成为冗余调用吗?

接下来我们得到一个基于上面创建的子窗口的 CreatePixmap 调用,宽度和高度为 0:

X_CreatePixmap: ID: 2097162, Drawable: 2097161, Width: 0, Height: 0

这样做的目的是什么?

接下来,xbiff(客户端)基于子窗口创建另一个图形上下文,并从 48x48 像素图之一对其执行 CopyPlane。

X_CreateGC: ID: 2097163, Drawable: 2097161, ValueMask: 65544, Value0: 16777215, Value1: 0

X_CopyPlane: SrcDrawable: 2097155, DestDrawable: 2097162, GraphicsContext: 2097163, SrcX: 0, SrcY: 0, DstX: 0, DstY: 0, Width: 0, Height: 0, Bitplane: 1

X_FreeGC: 2097163

此调用的宽度和高度为 0。这是否使它成为 NOOP,或者 0x0 的尺寸是否意味着“复制所有内容”?如果是这样,这应该只是将位图传送到子窗口,对吗?

接下来客户端根据子窗口创建一个 0x0 像素图:

X_CreatePixmap: Depth: 24, ID: 2097164, Drawable: 2097161, Width: 0, Height: 0

0x0 像素图有什么用?这是否意味着“复制窗口尺寸”?

这里我们为子窗口创建一个 GC,并从 48x48 位图之一到窗口执行 CopyArea:

X_CreateGC: ID: 2097165 Drawable: 2097161, ValueMask: 65548, Value0: 16777215, Value1: 0, Value2: 0

X_CopyArea: SrcDrawable: 2097153, DestDrawable: 2097164, GraphicsContext: 2097165, SrcX: 0, SrcY: 0, DstX: 0, DstY: 0, Width: 0, Height: 0, Bitplane: 1

这个 CopyArea 调用的宽度和高度也为 0。这是否意味着“复制整个事物”?

接下来我们映射 2097158(附加到根的父窗口)的子窗口,然后映射父窗口本身。

X_MapSubwindows: Window: 2097158
[We send a MapNotify and Expose event for window 2097161]

X_MapWindow: Window: 2097158
[We send a MapNotify and Expose event for window 2097158]

我不知道为什么它之后在子窗口上调用 ClearArea:

X_ClearArea: Window: 2097161, X: 0, Width: 0, Width: 0, Height: 0

这是否清楚任何事情或整个事情?

然后 CopyArea 调用将 0x0 像素图从较早的位置复制到位置 24x24 的子窗口:

X_CopyArea: SrcDrawable: 2097162, DestDrawable: 2097161, GraphicsContext: 2097157, SrcX: 0, SrcY: 0, DstX: 24, DstY: 24, Width: 0, Height: 0

宽度和高度也为零。再说一次,我不知道为什么。

我很乐意在理解 X11 绘图调用的工作方式以及为什么奇怪的(对我而言)调用是它们的方式方面获得一些帮助。

4

2 回答 2

3

(其中大部分可以在 X Window System 协议规范中找到,该规范在 Internet 上的许多地方都可以免费获得。)

相对于根窗口创建一个GC的意义在于,一个根窗口命名一个屏幕,而屏幕就是X中定义一组相关状态的这个奇怪的东西。像素图和格式、窗口、视觉效果和颜色图等都绑定到特定屏幕。你可以有多个屏幕;如果你这样做了,从一个窗口不能跨越到另一个。这就是所谓的“Zaphod”操作模式。但在最常见的多头设置中,您只需拥有一个覆盖多个输出的根窗口。

PutImage调用中的GC决定了图像中的像素到服务器中Pixmap的传输方式:planemask、clipping、raster op等。

您会看到创建了多个 GC,因为 GC 的深度是一个静态属性,而不是您可以使用 ChangeGC 修改的东西。您既有 1bpp 像素图,又有继承根窗口深度的窗口和像素图,因此它们需要不同的 GC。

两个 CreateWindow 调用之间的区别在于与每个调用关联的属性的掩码。第二个,相对于第一个,也设置了 CWCursor 位;当鼠标在该窗口内时,客户端要求设置特定的光标。为什么会变成这样,我不知道;我认为没有人声称 xbiff 写得很好。

0x0 不是像素图(或任何可绘制对象)的合法大小,所以我不确定那里发生了什么。在这种情况下,服务器要做的正确事情是抛出 BadValue,但这似乎是其他问题的征兆。

于 2011-06-07T15:51:04.367 回答
1

只是为了补充ajax的优秀答案。请注意,其中一个窗口有边框,而另一个没有。回想一下 xbiff 究竟做了什么(显示“无邮件”图标或“您有邮件”图标)。这两个窗口提供双缓冲效果。要更改图像,xbiff 只需映射或取消映射子窗口(没有边框的窗口)。

于 2011-07-17T10:06:59.497 回答