DRM遇到的实际问题及领悟(2)

news2024/11/19 13:43:28

接前一篇文章:DRM遇到的实际问题及领悟(1)

三、进一步定位

上回说到使用网上例程有同样会遇到mmap错误的问题,对于这个错误,perror只给出了“Invalid argument”的错误原因,具体错在哪里,并不知道。

1. 源码准备

没有办法,只能再找一个能够成功的例程。功夫不负有心人,在网上不断搜索,最终找到了以下链接中的例程:

https://github.com/tiagovignatti/intel-gpu-tools/blob/master/tools/intel_framebuffer_dump.c

代码如下:

/*
 * Copyright © 2013 Intel Corporation
 *
 * Permission is hereby granted, free of charge, to any person obtaining a
 * copy of this software and associated documentation files (the "Software"),
 * to deal in the Software without restriction, including without limitation
 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
 * and/or sell copies of the Software, and to permit persons to whom the
 * Software is furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice (including the next
 * paragraph) shall be included in all copies or substantial portions of the
 * Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
 * DEALINGS IN THE SOFTWARE.
 *
 */

/*
 * Read back all the KMS framebuffers attached to the CRTC and record as PNG.
 */

#define _GNU_SOURCE
#include <stdint.h>
#include <sys/types.h>
#include <sys/mman.h>
#include <errno.h>
#include <xf86drmMode.h>
#include <i915_drm.h>
#include <cairo.h>

#include "intel_io.h"
#include "drmtest.h"

int main(int argc, char **argv)
{
	drmModeResPtr res;
	int fd, n;

	fd = drmOpen("i915", NULL);
	if (fd < 0)
		return ENOENT;

	res = drmModeGetResources(fd);
	if (res == NULL)
		return ENOMEM;

	for (n = 0; n < res->count_crtcs; n++) {
		struct drm_gem_open open_arg;
		struct drm_gem_flink flink;
		drmModeCrtcPtr crtc;
		drmModeFBPtr fb;

		crtc = drmModeGetCrtc(fd, res->crtcs[n]);
		if (crtc == NULL)
			continue;

		fb = drmModeGetFB(fd, crtc->buffer_id);
		drmModeFreeCrtc(crtc);
		if (fb == NULL)
			continue;

		flink.handle = fb->handle;
		if (drmIoctl(fd, DRM_IOCTL_GEM_FLINK, &flink)) {
			drmModeFreeFB(fb);
			continue;
		}

		open_arg.name = flink.name;
		if (drmIoctl(fd, DRM_IOCTL_GEM_OPEN, &open_arg) == 0) {
			struct drm_i915_gem_mmap_gtt mmap_arg;
			void *ptr;

						mmap_arg.handle = open_arg.handle;
			if (drmIoctl(fd, DRM_IOCTL_I915_GEM_MMAP_GTT, &mmap_arg) == 0 &&
			    (ptr = mmap(0, open_arg.size, PROT_READ, MAP_SHARED, fd, mmap_arg.offset)) != (void *)-1) {
				cairo_surface_t *surface;
				cairo_format_t format;
				char name[80];

				snprintf(name, sizeof(name), "fb-%d.png",  fb->fb_id);

				switch (fb->depth) {
				case 16: format = CAIRO_FORMAT_RGB16_565; break;
				case 24: format = CAIRO_FORMAT_RGB24; break;
				case 30: format = CAIRO_FORMAT_RGB30; break;
				case 32: format = CAIRO_FORMAT_ARGB32; break;
				default: format = CAIRO_FORMAT_INVALID; break;
				}

				surface = cairo_image_surface_create_for_data(ptr, format,
									      fb->width, fb->height, fb->pitch);
				cairo_surface_write_to_png(surface, name);
				cairo_surface_destroy(surface);

				munmap(ptr, open_arg.size);
			}
			drmIoctl(fd, DRM_IOCTL_GEM_CLOSE, &open_arg.handle);
		}

		drmModeFreeFB(fb);
	}

	return 0;
}

但是此代码直接编译执行在笔者电脑上会有问题,于是笔者边摸索边对其进行改进。最终经过笔者改造后的代码如下:

/*
 * Read back all the KMS framebuffers attached to the CRTC and record as PNG.
 */

#define _GNU_SOURCE
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <stdint.h>
#include <sys/types.h>
#include <sys/mman.h>
#include <errno.h>
#include <xf86drmMode.h>
#include <xf86drm.h>
#include <i915_drm.h>
#include <cairo.h>

#include "intel_io.h"
//#include "drmtest.h"


int main(int argc, char **argv)
{
	drmModeResPtr res;
	int fd, n;

	drmModeConnector *connector = NULL;
	drmModeRes *resources = NULL;
	drmModeEncoder *encoder = NULL;
	drmModeCrtc *crtc = NULL;
	struct drm_gem_open open_arg;
	struct drm_gem_flink flink;
	uint32_t conn_id;
	uint32_t crtc_id;

	fd = drmOpen("i915", NULL);
	//fd = open("/dev/dri/card0", O_RDWR); //with same effect
	if (fd < 0)
	{
		perror("drmOpen failed!");
		return ENOENT;
	}

	res = drmModeGetResources(fd); //获取drmModeRes资源,包含fb、crtc、encoder、connector等
	if (res == NULL)
	{
		perror("drmModeGetResources failed!");
		return ENOMEM;
	}

	for (int i = 0; i < res->count_connectors; ++i)
	{
		connector = drmModeGetConnector(fd, res->connectors[i]); //根据connector_id获取connector资源
		if (connector == NULL || connector->connection != DRM_MODE_CONNECTED)
			continue;
		for (int j = 0; j < res->count_encoders; ++j)
		{
			encoder = drmModeGetEncoder(fd, res->encoders[j]);
			if (encoder == NULL || encoder->encoder_id != connector->encoder_id)
				continue;
			break;
		}
		break;
	}

	crtc = drmModeGetCrtc(fd, encoder->crtc_id);
	uint32_t framebuffer_id = crtc->buffer_id;

	drmModeFBPtr fb = drmModeGetFB(fd, crtc->buffer_id);
	drmModeFreeCrtc(crtc);
	if (fb == NULL)
	{
		printf("daozhelilema2?\n");
		//continue;
	}

	flink.handle = fb->handle;
	if (drmIoctl(fd, DRM_IOCTL_GEM_FLINK, &flink))
	{
		perror("DRM_IOCTL_GEM_OPEN failed!");
		//drmModeFreeFB(fb);
		printf("daozhelilema3?\n");
	}

	open_arg.name = flink.name;
	if (drmIoctl(fd, DRM_IOCTL_GEM_OPEN, &open_arg))
	{
		perror("DRM_IOCTL_GEM_OPEN failed!");
		//drmModeFreeFB(fb);
		printf("daozhelilema4?\n");
	}

	struct drm_i915_gem_mmap_gtt mmap_arg;
	void *ptr;

	mmap_arg.handle = open_arg.handle;
	if (drmIoctl(fd, DRM_IOCTL_I915_GEM_MMAP_GTT, &mmap_arg))
	{
		perror("DRM_IOCTL_I915_GEM_MMAP_GTT failed!");
		//drmModeFreeFB(fb);
		printf("daozhelilema5?\n");
	}

	ptr = mmap(NULL, open_arg.size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, mmap_arg.offset);
	if (ptr == (void *)MAP_FAILED)
	{
		perror("mmap failed!");
		//drmModeFreeFB(fb);
		printf("daozhelilema6?\n");
	}

	unsigned char tmp;
	void *dup_ptr = malloc(open_arg.size);
#if 0
	for (int i=0; i<open_arg.size; i++)
	{
		tmp = *((unsigned char *) ptr +i);
		//tmp = !tmp;
		tmp += 64;
		//*((unsigned char *)ptr + i) = tmp;
		*((unsigned char *)dup_ptr + i) = tmp;
	}
#endif
	cairo_surface_t *surface;
	cairo_format_t format;

	char name[80] = {0};
	snprintf(name, sizeof(name), "fb-%d.png",  fb->fb_id);

	printf("fb->depth is: %d\n", fb->depth);
	switch (fb->depth)
	{
		case 16:
			format = CAIRO_FORMAT_RGB16_565;
			break;
		case 24:
			format = CAIRO_FORMAT_RGB24;
			break;
		case 30:
			format = CAIRO_FORMAT_RGB30;
			break;
		case 32:
			format = CAIRO_FORMAT_ARGB32;
			break;
		default:
			format = CAIRO_FORMAT_INVALID;
			break;
	}

	surface = cairo_image_surface_create_for_data(ptr, format, fb->width, fb->height, fb->pitch);
	//surface = cairo_image_surface_create_for_data(dup_ptr, format, fb->width, fb->height, fb->pitch);
	cairo_surface_write_to_png(surface, name);
	cairo_surface_destroy(surface);

	FILE *output_file = fopen("./framebuffer.bin", "wb");
	if (output_file == NULL)
	{
		fprintf(stderr, "Unable to open output file\n");
		return EXIT_FAILURE;
	}
	printf("open_arg.size is %d\n", open_arg.size);
	//fwrite(framebuffer_data, sizeof(uint8_t), framebuffer_size, output_file);
	fwrite(ptr, sizeof(uint8_t), open_arg.size, output_file);
	fclose(output_file);

	drmIoctl(fd, DRM_IOCTL_GEM_CLOSE, &open_arg.handle);
	drmModeFreeFB(fb);

	return 0;
}

2. 源码编译

通过以下命令进行源码编译:

gcc intel_framebuffer_dump.c -o intel_framebuffer_dump -I/usr/include/drm -I/usr/include/cairo -ldrm -lcairo

最终生成可执行文件intel_framebuffer_dump。

3. 源码执行及结果

执行此可执行文件,得到以下结果:

$ ./intel_framebuffer_dump DRM_IOCTL_GEM_OPEN failed!: Permission denied
daozhelilema3?
DRM_IOCTL_GEM_OPEN failed!: Permission denied
daozhelilema4?
DRM_IOCTL_I915_GEM_MMAP_GTT failed!: No such file or directory
daozhelilema5?
mmap failed!: Invalid argument
daozhelilema6?
fb->depth is: 30
段错误(核心已转储)

需要使用root权限或者sudo执行,运行结果如下:

$ sudo ./intel_framebuffer_dump 
fb->depth is: 30

同时,会生成以下图片:

说明截屏功能正常实现。

下一步又该如何走,如何更加深入地定位问题?请看下回。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1046603.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

多元回归分析

建模复习 目录 前言 一、回归的思想 1,介绍 2&#xff0c;回归分析的分类 3&#xff0c;数据类型 二、一元线性回归 1&#xff0c;一元线性函数拟合 2&#xff0c;一元线性回归模型 3&#xff0c;回归系数 1&#xff0c;回归系数的解释 2&#xff0c;内生性 3&…

大模型技术实践(四)|参数高效微调技术解析及AdaLoRA的应用

在上一期的大模型技术实践中&#xff0c;我们为大家介绍了基于“LangChainLLM”框架快速搭建知识增强后的问答机器人&#xff0c;并探讨了提升模型内容理解和执行能力的潜在优化方向。本期内容UCloud将为您介绍参数高效微调技术&#xff08;PEFT&#xff09;&#xff0c;即对已…

软件研发团队建设---利用“鲇鱼效应”推进研发团队人文环境的建设手记(2)

软件研发团队建设—利用“鲇鱼效应”推进研发团队人文环境的建设手记&#xff08;2&#xff09; 背景 软件研发团队建设—如何协助业主建设软件研发团队并提高其力量的手记&#xff08;1&#xff09; 与业主某研发人员的沟通 &#xff08;长图&#xff0c;耐心点&#xff0…

CDS(一)

Core Data Services 核心数据服务 DDL 定义Query Language 查询DCL 控制&#xff08;权限相关&#xff09;model&#xff0c;从语义层获取数据open SQL 访问CDS viewCDS权限定义,集成旧的权限概念扩展native SQL扩展SQL建模和声明 ABAP ViewCDS View支持所有数据库是是支持查询…

全网最细,Jenkins配合GitLab分支自动合并/自动创建(超细整理)

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 GitFlow工作流简介…

python(自4) xpath下载 lxml安装 lxml语法 使用方式

&#xff08;一&#xff09;安装 搜索xpath 讲解 XPath 教程 (w3school.com.cn) 一&#xff0c;下载地址 &#xff1a; https://chrome.zzzmh.cn/info/hgimnogjllphhhkhlmebbmlgjoejdpjl 二 &#xff0c;拖拽 &#xff08;二&#xff09;lxml安装 cmd 打开终端 cd pythond…

数据库及分类详细介绍

目录 一、数据库详细介绍1、什么是数据库2、数据库的种类3、常见的数据库4、关系型数据库4.1、关系型数据库介绍4.2、关系型数据库的优缺点4.3、关系型数据库小结 5、非关系型数据库5.1、非关系数据库诞生的背景5.2、非关系型数据库的介绍5.3、非关系型数据库的优缺点5.4、非关…

LInux本地连接超时ip未显示

关闭防护墙 systemctl stop firewalld 查看防火墙状态 systemctl status firewalld 开启80端口 /sbin/iptables -I INPUT -p tcp --dport 8080 -j ACCEPT 查看Ip ifconfig

清能股份2MW热电联供系统成功下线

2023年9月26日&#xff0c;由清能股份旗下豫氢动力承接的2MW燃料电池热电联供项目已正式下线&#xff0c;发往焦作煤业&#xff08;集团&#xff09;开元化工有限责任公司&#xff08;简称“开元化工”&#xff09;安装运营。 开元化工是河南知名氯碱国企&#xff0c;该项目是国…

左对齐和右对齐

%d默认为左对齐&#xff0c;%5d为右对齐&#xff08;以空格补齐&#xff09;&#xff0c;%05d为右对齐&#xff08;以0补齐&#xff09;&#xff0c;%-5d左补齐&#xff08;以空格补齐&#xff09;&#xff0c;整数和小数同理。%.xf,x为小数点后保留的位数。 #include<stdi…

WebGL实现透明物体(α混合)

目录 α混合 如何实现α混合 1. 开启混合功能&#xff1a; 2. 指定混合函数 混合函数 gl.blendFunc&#xff08;&#xff09;函数规范 可以指定给src_factor和dst_factor的常量 混合后颜色的计算公式 加法混合 半透明的三角形&#xff08;LookAtBlendedTriangl…

使用Python和Pandas处理网页表格数据

在我们的日常工作和生活中&#xff0c;经常会遇到需要处理大量数据的情况&#xff0c;而网页表格数据则是其中常见的一种形式。如果我们能够灵活地使用Python和Pandas这两个强大的工具&#xff0c;就能够快速、高效地对这些数据进行处理和分析。 首先&#xff0c;我们需要了解…

游戏服务商Latis Global参展2023 ChinaJoy B2B

第20届ChinaJoy于2023年7月在上海举行了为期四天的博览会,参展观众达到了33.8万人次。ChinaJoy是全球最具知名度与影响力的年度盛会之一,涵盖了包括游戏、动漫、互联网影视、电子竞技、潮流玩具、智能娱乐在内的多个数字娱乐领域。ChinaJoy不仅仅代表了数字娱乐领域的最新风向,…

搞定零售出海的底层挑战,泡泡玛特的经验是什么?

出品 | CSDN 云计算 从 2010 年成立&#xff0c;2016 年发布 Molly IP 的盲盒产品到现在&#xff0c;泡泡玛特已经成为在全球拥有员工 4000 多人的知名潮流玩具品牌。就像在中国市场被喜爱的迪士尼、日漫等多个 IP 一样&#xff0c;泡泡玛特从 2022 年开始也带着众多 IP 走出中…

如何正确监测蓄电池健康?狠狠学到!

蓄电池在现代生活和工业中发挥着关键作用&#xff0c;它们为无数设备和系统提供了必要的电力支持。然而&#xff0c;蓄电池的性能和可靠性对许多应用至关重要。监控蓄电池状态和性能变得越来越重要&#xff0c;以确保它们在需要时始终可用。 为此&#xff0c;蓄电池监控系统应运…

从零开始之了解电机及其控制(8)clarke和park变换

为了最大化无刷电机产生的扭矩&#xff0c;我们希望感应磁场与转子磁场正交并引导转子磁场&#xff0c;无刷电机定子产生的磁场矢量将与电流矢量逆时针旋转 90 度 但现在我们要稍微改变这个y电路的外观&#xff0c;我们不是将相位表示为电阻器&#xff0c;它会产生垂直于电流方…

家电行业 EDI:Miele EDI 需求分析

Miele是一家创立于1899年的德国公司&#xff0c;以其卓越的工程技术和不懈的创新精神而闻名于世。作为全球领先的家电制造商&#xff0c;Miele的经营范围覆盖了厨房、洗衣和清洁领域&#xff0c;致力于提供高品质、可持续和智能化的家电产品。公司的使命是为全球消费者创造更美…

TSM动作识别模型【详解】

文章目录 本文使用的是somethingv2数据集&#xff0c;解压后是如下形式&#xff1b; 由于该压缩数据进行了分卷操作&#xff0c;需要合并后才能进行解压。首先我们将下面4个json文件剪贴到其他文件夹&#xff0c;只保留00-19的文件&#xff0c;然后在该文件夹下打开cmd&#xf…

Unity vscode 官方debug

把这个先unlock 再升级到最新的 然后重新生成 project files 打开 launch.json {"version": "0.2.0","configurations": [{"name": "Attach to Unity","type": "vstuc","request": "a…

创造您梦寐以求的家居设计——Live Home 3D Pro for Mac

您是否曾经想象过在舒适的家中展现自己独特的风格&#xff1f;现在&#xff0c;您可以通过Live Home 3D Pro for Mac来实现您的家居设计梦想&#xff01;这款强大的3D家居设计软件将带给您无限的创作可能性。 Live Home 3D Pro for Mac是一款专业级的家居设计软件&#xff0c;…