DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

这篇具有很好参考价值的文章主要介绍了DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

现象引出和问题猜想

     有一款用户软件叫DosBox,在实体机win11的时候最大化的时候,程序界面可以铺满全屏,但是在winserver2016云桌面进行最大化的时候,最大化的时候,界面无法铺满全屏:

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

(实体机最大化,界面全屏显示)

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

(云桌面程序最大化主界面无法铺满全屏)

这到底是什么原因导致在云桌面下程序最大化就无法铺满全屏呢?首先我们猜测,程序之所以能够铺满全屏是因为程序在最大化的时候设置了桌面的显示分辨率,即最大化的时候把桌面的分辨率设低,退出最大化的时候,恢复桌面原来的分辨率。通过代码:

void GetCurrentResolution_imp()
{
	DISPLAY_DEVICE DispDev;
	Res temp;
	for (auto iDevIdx = 0; iDevIdx < 2; iDevIdx++)
	{
		ZeroMemory(&DispDev, sizeof(DISPLAY_DEVICE));
		DispDev.cb = sizeof(DISPLAY_DEVICE);
		int dspNum = GetSystemMetrics(SM_CMONITORS);	// 两个显示器为复制模式时,仅在1/2上显示时,值都为1
		// 获取有关系统中显示设备的信息
		if (EnumDisplayDevices(NULL, iDevIdx, &DispDev, 0))
		{
			DEVMODE devMode;
			int display_id = 0;
			if ((DispDev.StateFlags & DISPLAY_DEVICE_ATTACHED_TO_DESKTOP) != DISPLAY_DEVICE_ATTACHED_TO_DESKTOP)
				continue;
			if ((DispDev.StateFlags & DISPLAY_DEVICE_PRIMARY_DEVICE) == DISPLAY_DEVICE_PRIMARY_DEVICE
				|| dspNum == 1)				// 主显示器或者只在一个显示器显示
			{
				display_id = 0;
			}
			else
			{
				display_id = 1;
			}
			if (EnumDisplaySettings(DispDev.DeviceName, ENUM_CURRENT_SETTINGS, &devMode))
			{
				temp.dmPelsHeight = devMode.dmPelsHeight;
				temp.dmPelsWidth = devMode.dmPelsWidth;
				temp.display_id = display_id;
				printf("w %d,h %d display_id %d\n", devMode.dmPelsWidth, devMode.dmPelsHeight, display_id);
			}
		}
	}
}

 不断轮询获取桌面的分辨率,发现在实体机下程序最大化的时候确实将桌面的分辨率调低了:

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

(实体机原来桌面的分辨率是3456*2160,程序最大化的时候桌面分辨率被修改为640*480)

 同样发现在云桌面下,DosBox同样把桌面分辨率调低了:

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

 (云桌面原来桌面的分辨率是1920*1080,程序最大化的时候桌面分辨率被修改为1024*768)

   这就发现可能是这两个系统可设置分辨率的最小值不一样,云桌面的可设置分辨率的最小值比较大,导致了,界面无法被拉伸显示很大。看看两台系统的可设置分辨率:

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

(实体机win11可设置最小分辨率为800*600 )

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

(云桌面可设置最小分辨率为1024*768)

 可以发现实体机win11可设置最小分辨率为800*600,DosBox设置的分辨率为:640*480

云桌面server2016可设置最小分辨率为1024*768,DosBox设置的分辨率为1024*768

于是我们猜测,因为两台系统的可设置最小分辨率不一样,导致云桌面设置的最小分辨率太大导致DosBox界面无法放的足够大来覆盖桌面;但是同时发现在实体机下设置的分辨率并不是系统可设置的最小分辨率,设置的640*480比系统可设置最小分辨率800*600低。

跟踪剖析问题

设置系统分辨率一般是通过ChangeDisplaySettings这个api:

void changeResolution()
{
	DEVMODE DevMode;
	EnumDisplaySettings(NULL, 0, &DevMode);
	DevMode.dmFields = DM_PELSWIDTH | DM_PELSHEIGHT;
	DevMode.dmPelsWidth = 1920;
	DevMode.dmPelsHeight = 1200;
	DevMode.dmDisplayFrequency = 30;
	DevMode.dmBitsPerPel = 32;
	ChangeDisplaySettings(&DevMode, 0);
}

但是通过apimonitor进行api拦截,发现并未拦截到该api的调用,猜测程序并不是通过调用ChangeDisplaySettings来设置桌面分辨率的。把Doxbox.exe拖拽到ida上:

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

看到DosBox.exe导入表并没有导入ChangeDisplaySettings这个api,而且apimonitor也确实没抓取到程序对该api的调用,但是发现导入表导入了较多的opengl的函数。于是猜测程序是通过opengl的api来修改系统的分辨率的,通过百度我也发现了一篇文章:
DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

确实发现有人通过SDL_SetVideoMode来设置桌面的分辨率,而且这个函数也是DoxBox调用过的。

但是通过ida看伪码发现他们调用方式并不一样:

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

 SDL_SetVideoMode这个函数调用的地方也比较多的:

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

 把SDL.dll拖拽到ida上,直接调试:

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

传入参数 a1(0x280) a2(0x190)应该是传入的需要设置的宽640*400,这个跟网络blog上对setvideomode函数调用一致,前两个参数分别是宽和高 。

 DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

 通过调试发现程序是通过SDL_ListMode来获取可设置分辨率的列表,最后获取到最小的分辨率数组下标,最后得到v110为0x280,v109为0x1E0,即为宽640 高480,即这个就是最后设置的桌面分辨率,但是奇怪,系统可设置分辨率的列表并没有640*480,最小是800*600。那么就要分析SDL_ListModes这个函数了。

我在网络上找到这个api的用法:

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

 然后到 http://www.libsdl.org/release/SDL-1.2.15/ (注意是这个版本,新版本在github上已经没有了这个api)下载源代码并编译。

我在源代码找到SDL_SetVideoMode这个函数,发现这个跟我在ida上调试的代码大致一致(因为SDL版本不同,所以源码还是有区别的):

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

 DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

 于是我写了测试代码,参数传入ida上传递的参数,然后进行调试:

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

 调试到SDL_ListModes这个api:

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

发现获取分辨率列表就是通过获取SDL_modelist这个列表获取的,而这个SDL_modelist数组是怎么获取的呢,我在代码里面全局搜索看看该数组的引用,并发现一个函数DIB_AddMode:

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

 好,应该就是在这里去把分辨率列表添加进去的,那到底是哪里调用的呢,打上断点顺藤摸瓜,重启程序进行调试,最后断点断在我打的断点下,看下调用堆栈:

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

 好了,原来是在DIB_VideoInit这个api然后调用EnumDisplaySettings这个win api来获取系统桌面分辨率的了。

通过demo代码枚举出桌面的分辨率:

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

 运行demo程序:

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

 发现DosBox设置的屏幕分辨率其实就是枚举的最小分辨率,这个枚举的分辨率跟系统设置的分辨率是有差异的,并不是期初认为的一致的。

同时在DIB_SetVideoMode这个函数实现找到了改变分辨率的实现:

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析

 

最后在apimonitor也抓取到了对ChangeDisplaySettingsExA(可能是之前版本调用的是这个api,所以api有所区别)的调用:

DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析文章来源地址https://www.toymoban.com/news/detail-444323.html

到了这里,关于DosBox在winserver2016云桌面最大化界面无法铺满全屏的问题剖析的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 如何使音频类app广告变现收益最大化

    要使音频类应用程序的广告变现收益最大化,可以采取以下策略: admaoyan猫眼聚合 精准定位受众: 了解你的用户群体,包括他们的兴趣、偏好和行为,以便向他们展示相关性更高的广告。使用用户数据分析工具,如Google Analytics或Firebase Analytics,以获取深入的洞察。 多样化

    2024年02月20日
    浏览(43)
  • MySQL | 深入了解如何最大化利用 MySQL 函数(一)

    前言 ✨欢迎来到小K的MySQL专栏,本节将为大家带来MySQL字符串函数和数学函数的讲解 ✨ 一、字符串函数 函数 作用 UPPER(列|字符串) 将字符串每个字符转为大写 LOWER(列|字符串) 将字符串每个字符转为小写 CONCAT(str1,str2,…) 将所有字符串连接成一个字符串 REPLACE(列|字符串,新字符

    2024年02月08日
    浏览(41)
  • Gvim显示行号、最大化、字号、主题等常用配置修改

    Gvim的设置分两种:1. 临时设置,2. 永久设置 ,本文只关注永久设置的情况。 配置Gvim只需修改Gvim配置文件即可, Linux中,配置文件的地址是~/.vimrc ,若没有该文件则创建即可,我们可以直接输入 gvim ~/.vimrc 来进行配置,windows中的配置文件地址在vim的安装地址下,文件名叫做

    2024年02月15日
    浏览(43)
  • 【元宇宙】区块链,元宇宙最大化的驱动力

    如今,一些观察者认为区块链是在结构上实现元宇宙的必要条件,而其他人则认为这种说法是荒谬的。人们对于区块链技术本身仍然有很多困惑,所以根本谈不上清楚地了解込块链技术与元宇宙的关系。所以,我们可以从区块链的定义开始介绍。 简侧言之,区块链是由一个去

    2024年02月10日
    浏览(43)
  • 期望最大化(EM)算法:从理论到实战全解析

    本文深入探讨了期望最大化(EM)算法的原理、数学基础和应用。通过详尽的定义和具体例子,文章阐释了EM算法在高斯混合模型(GMM)中的应用,并通过Python和PyTorch代码实现进行了实战演示。 关注TechLead,分享AI全维度知识。作者拥有10+年互联网服务架构、AI产品研发经验、

    2024年02月08日
    浏览(40)
  • 力扣 1005. K 次取反后最大化的数组和

    题目来源:https://leetcode.cn/problems/maximize-sum-of-array-after-k-negations/description/ C++题解1:最直接的想法就是负的变正的,如果负的元素数量小于k,就挑选绝对值大的负数变正;如果负的元素数量大于k,那么还需要根据剩下的k(待变换数)的奇偶性来判断,偶数就不用管了,奇数

    2024年02月16日
    浏览(31)
  • K 次取反后最大化的数组和【贪心算法】

    1005 . K 次取反后最大化的数组和 给你一个整数数组 nums 和一个整数 k ,按以下方法修改该数组: 选择某个下标 i 并将 nums[i] 替换为 -nums[i] 。 重复这个过程恰好 k 次。可以多次选择同一个下标 i 。 以这种方式修改数组后,返回数组 可能的最大和 。 关于 nums = IntStream.of(nums

    2024年02月11日
    浏览(36)
  • 贪心算法|1005.K次取反后最大化的数组和

    力扣题目链接 有没有不理解的语法知识呢? sort函数中的比较函数cmp(),即 void sort( iterator start, iterator end, StrictWeakOrdering cmp ); sort函数头文件为: #include algorithm 其中,cmp函数可以自己编写,自己决定逻辑,包括cmp的命名也是自己决定的。 示例如下:  代码随想录 (programmerc

    2024年04月09日
    浏览(53)
  • 【数学建模】面包店老板使日均收入最大化的诀窍

    1 问题描述 面包店每天烘烤一定数量的面包出售,每个成本3元,以8元的价格卖出,晚间关门前将未卖完的面包无偿处理掉,若已知每天面包需求量的概率分布如下表所示。从长期看,面包店老板为了能得到最高的日均收入,他每天要烘烤多少个面包?这个最高日均收入是多

    2024年02月04日
    浏览(48)
  • Qt 自定义窗口的标题栏,重写鼠标事件实现,关闭隐藏,最大化/最小化,重写窗口事件函数,实现鼠标选中边框拉大拉小,双击标题栏切换窗口最大化和最小化

    Qt 自定义窗口的标题栏,重写鼠标事件实现,关闭隐藏,最大化/最小化,重写窗口事件函数,实现鼠标选中边框拉大拉小,双击标题栏切换窗口最大化和最小化 1、main.cpp 2、widget.h 3、widget.cpp 4、效果展示 5、完成

    2024年02月16日
    浏览(45)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包