在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

Linux驅動開發_倒車影像項目介紹

DS小龍哥-嵌入式技術 ? 2022-09-17 15:47 ? 次閱讀

【摘要】 介紹Linux下倒車影像的項目,完成攝像頭圖像讀取、超聲波驅動編寫、超聲波距離讀取。

倒車影像項目

模擬: 汽車中控臺---倒車影像。

組成部分:

1.?LCD屏: 實時顯示攝像頭采集的數據。

2.?攝像頭: 放在車尾,采集圖像傳輸給LCD屏進行顯示。

3.?倒車雷達: 超聲波測距--->測量車尾距離障礙物的距離。

4.?蜂鳴器: 根據倒車雷達測量的距離,控制頻率。

poYBAGMlOu-AOBVRAAAL-jj6-Tw222.png

1.1 超聲波測距模塊

聲波測距: 已知聲音在空氣中傳播的速度。

pYYBAGMlOu-AFmuxAAAmOEe-tck976.pngpoYBAGMlOu-AJBBaAAJM_tHZFyQ036.pngpYYBAGMlOvCAKLPrAAdGSZnRjh0757.pngpoYBAGMlOvGAEFCHAAE1wfYJhgU050.png

??硬件接線:

poYBAGMlOvWAAQJpAA3s6Fdq0JA662.png

ECHO------->GPX1_0 (開發板第9個IO口): 中斷引腳----->檢測回波----輸入

TRIG ------->GPB_7 (開發板第8個IO口): 輸出觸發信號

1.2 PWM方波控制蜂鳴器

??PWM方波:

pYYBAGMlOvWAaetGAAANcrhJPCc509.pngpYYBAGMlOviAP6BdAAwht2kb2FQ071.pngpoYBAGMlOvqAIdZzAAe3isRutmE616.png

??內核自帶的PWM方波驅動

pYYBAGMlOvuAVshAAACvomxi1OI945.png

1.3 UVC免驅攝像頭編程框架: V4L2

編程的框架: v4l2--->全稱: video4linux2

V4L2 : 針對UVC免驅USB設備設計框架。專用于USB攝像頭的數據采集。

免驅 : 驅動已經成為標準,屬于內核自帶源碼的一部分。

V4L2框架本身注冊的也是字符設備,設備節點: /dev/videoX

V4L2 框架: 提供ioctl接口,提供了有很多命令,可以通過這些命令對攝像頭做配置。

比如: 輸出的圖像尺寸,輸出圖像格式(RGB、YUV、JPG),申請采集數據的緩沖區。

poYBAGMlOvuAKskKAAEWG4P52zg183.pngpYYBAGMlOvuAWkiOAACnVBFUccw164.png

??配置攝像頭采集隊列步驟:

poYBAGMlOvuALkJwAACbwCG0l5Y794.png

Mmap函數映射。

超聲波驅動讀取距離:

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 

static unsigned int distance_irq; /*存放中斷號*/
static u32 *GPB_DAT=NULL;
static u32 *GPB_CON=NULL;

static u32 distance_time_us=0; /*表示距離的時間*/

/*
工作隊列處理函數: 
*/
static void distance_work_func(struct work_struct *work)
{
	u32 time1,time2;
	time1=ktime_to_us(ktime_get()); /*獲取當前時間,再轉換為 us 單位*/

	/*等待高電平時間結束*/
	while(gpio_get_value(EXYNOS4_GPX1(0))){}
	
	time2=ktime_to_us(ktime_get()); /*獲取當前時間,再轉換為 us 單位*/

	distance_time_us=time2-time1;
	//printk("us=%d\n",time2-time1);   /*us/58=厘米*/
}

/*靜態方式初始化工作隊列*/
static DECLARE_WORK(distance_work,distance_work_func);

/*
中斷處理函數: 用于檢測超聲波測距的回波
*/
static irqreturn_t distance_handler(int irq, void *dev)
{
	/*調度工作隊列*/
	schedule_work(&distance_work);
	return IRQ_HANDLED;
}

static void distance_function(unsigned long data);
/*靜態方式定義內核定時器*/
static DEFINE_TIMER(distance_timer,distance_function,0,0);

/*內核定時器超時處理函數: 觸發超聲波發送方波*/
static void distance_function(unsigned long data)
{
	static u8 state=0;
	state=!state;
	
	/*更改GPIO口電平*/
	if(state)
	{
		*GPB_DAT|=1<<7;
	}
	else
	{
		*GPB_DAT&=~(1<<7);
	}
	
	/*修改定時器的超時時間*/
	mod_timer(&distance_timer,jiffies+msecs_to_jiffies(100));
}

static int distance_open(struct inode *inode, struct file *file)
{
	return 0;
}

#define GET_US_TIME 0x45612
static long distance_unlocked_ioctl(struct file *file, unsigned int cmd, unsigned long argv)
{
	u32 *us_data=(u32*)argv;
	int err;
	u32 time_us=distance_time_us;
	switch(cmd)
	{
		case GET_US_TIME:
			err=copy_to_user(us_data,&time_us,4);
			if(err!=0)printk("拷貝失敗!\n");
			break;
	}
	return 0;
}

static int distance_release(struct inode *inode, struct file *file)
{
	return 0;
}

/*定義文件操作集合*/
static struct file_operations distance_fops=
{
	.open=distance_open,
	.unlocked_ioctl=distance_unlocked_ioctl,
	.release=distance_release
};
/*定義雜項設備結構體*/
static struct miscdevice distance_misc=
{
	.minor=MISC_DYNAMIC_MINOR,
	.name="tiny4412_distance",
	.fops=&distance_fops
};

static int __init tiny4412_distance_dev_init(void) 
{
	int err;
	/*1. 映射GPIO口地址*/
	GPB_DAT=ioremap(0x11400044,4);
	GPB_CON=ioremap(0x11400040,4);

	*GPB_CON&=~(0xF<<4*7);
	*GPB_CON|=0x1<<4*7; /*配置輸出模式*/
	
	/*2. 根據GPIO口編號,獲取中斷號*/
	distance_irq=gpio_to_irq(EXYNOS4_GPX1(0));
	
	/*3. 注冊中斷*/
	err=request_irq(distance_irq,distance_handler,IRQ_TYPE_EDGE_RISING,"distance_device",NULL);
	if(err!=0)printk("中斷注冊失敗!\n");
	else printk("中斷:超聲波測距驅動安裝成功!\n");

	/*4. 修改定時器超時時間*/
	mod_timer(&distance_timer,jiffies+msecs_to_jiffies(100));

	/*雜項設備注冊*/
	misc_register(&distance_misc);
	return 0;
}

static void __exit tiny4412_distance_dev_exit(void) 
{
	/*5. 注銷中斷*/
	free_irq(distance_irq,NULL);

	/*6. 停止定時器*/
	del_timer(&distance_timer);
	
	/*7. 取消IO映射*/
	iounmap(GPB_DAT);
	iounmap(GPB_CON);

	/*注銷雜項設備*/
	misc_deregister(&distance_misc);
	
	printk("中斷:超聲波測距驅動卸載成功!\n");
}

module_init(tiny4412_distance_dev_init);
module_exit(tiny4412_distance_dev_exit);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("tiny4412 wbyq");

攝像頭代碼,讀取攝像頭畫面:

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include "framebuffer.h"

#define PWM_DEVICE "/dev/pwm"  /*PWM方波設備文件*/
#define DISTANCE_DEVICE "/dev/tiny4412_distance" /*超聲波測距設備文件*/
#define UVC_VIDEO_DEVICE "/dev/video15"  /*UVC攝像頭設備節點*/

#define GET_US_TIME 0x45612     /*獲取超聲波測量的距離: ioctl命令*/
#define PWM_IOCTL_SET_FREQ		1 /*控制PWM方波頻率: ioctl命令*/
#define PWM_IOCTL_STOP			0 /*停止PWM方波輸出: ioctl命令*/

int distance_fd;  /*超聲波設備的文件描述符*/
int pwm_fd;       /*PWM方波設備的文件描述符*/
int uvc_video_fd; /*UVC攝像頭設備文件描述符*/
int Image_Width;  /*圖像的寬度*/
int Image_Height; /*圖像的高度*/
unsigned char *video_memaddr_buffer[4]; /*存放攝像頭映射到進程空間的緩沖區地址*/

/*
函數功能: 用戶終止了進程調用
*/
void exit_sighandler(int sig)
{
	//停止PWM波形輸出,關閉蜂鳴器
	ioctl(pwm_fd,PWM_IOCTL_STOP,0);
	close(pwm_fd);
	close(distance_fd);
	exit(1);
}

/*
函數功能: 讀取超聲波數據的線程
*/
void *distance_Getpthread_func(void *dev)
{
	/*1. 打開PWM方波驅動*/
	pwm_fd=open(PWM_DEVICE,O_RDWR);
	if(pwm_fd<0) //0 1 2
	{
		printf("%s 設備文件打開失敗\n",PWM_DEVICE);
		/*退出線程*/
		pthread_exit(NULL); 
	}

	/*2. 打開超聲波測距設備*/
	distance_fd=open(DISTANCE_DEVICE,O_RDWR);
	if(distance_fd<0) //0 1 2
	{
		printf("%s 設備文件打開失敗\n",DISTANCE_DEVICE);
		/*退出線程*/
		pthread_exit(NULL); 
	}

	/*3. 循環讀取超聲波測量的距離*/
	struct pollfd fds;
	fds.fd=distance_fd;
	fds.events=POLLIN;
	int data;
	while(1)
	{
		poll(&fds,1,-1);
		ioctl(distance_fd,GET_US_TIME,&data);
		printf("距離(cm):%0.2f\n",data/58.0);
		data=data/58;
		if(data>200) /*200厘米: 安全區域*/
		{
			//停止PWM波形輸出,關閉蜂鳴器
			ioctl(pwm_fd,PWM_IOCTL_STOP,0);
		}
		else if(data>100) /*100厘米: 警告區域*/
		{
			printf("警告區域!\n");
			ioctl(pwm_fd,PWM_IOCTL_SET_FREQ,2);
		}
		else /*小于<100厘米: 危險區域*/
		{
			printf(" 危險區域!\n");
			ioctl(pwm_fd,PWM_IOCTL_SET_FREQ,10);
		}
		
		//ioctl(pwm_fd,PWM_IOCTL_SET_FREQ,pwm_data);
		/*倒車影像: 測距有3個檔位*/
	}
}
/*
函數功能: UVC攝像頭初始化
返回值: 0表示成功
*/
int UVCvideoInit(void)
{
	/*1. 打開攝像頭設備*/
	uvc_video_fd=open(UVC_VIDEO_DEVICE,O_RDWR);
	if(uvc_video_fd<0)
	{
		printf("%s 攝像頭設備打開失敗!\n",UVC_VIDEO_DEVICE);
		return -1;
	}
	
	/*2. 設置攝像頭的屬性*/
	struct v4l2_format format;
	memset(&format,0,sizeof(struct v4l2_format));
	format.type=V4L2_BUF_TYPE_VIDEO_CAPTURE; /*表示視頻捕獲設備*/
	format.fmt.pix.width=800;  /*預設的寬度*/
	format.fmt.pix.height=480; /*預設的高度*/
	format.fmt.pix.pixelformat=V4L2_PIX_FMT_YUYV; /*預設的格式*/
	format.fmt.pix.field=V4L2_FIELD_ANY; /*系統自動設置: 幀屬性*/
	if(ioctl(uvc_video_fd,VIDIOC_S_FMT,&format)) /*設置攝像頭的屬性*/
	{
		printf("攝像頭格式設置失敗!\n");
		return -2;
	}
	
	Image_Width=format.fmt.pix.width;
	Image_Height=format.fmt.pix.height;
		
	printf("攝像頭實際輸出的圖像尺寸:x=%d,y=%d\n",format.fmt.pix.width,format.fmt.pix.height);
	if(format.fmt.pix.pixelformat==V4L2_PIX_FMT_YUYV)
	{
		printf("當前攝像頭支持YUV格式圖像輸出!\n");
	}
	else
	{
		printf("當前攝像頭不支持YUV格式圖像輸出!\n");
		return -3;
	}

	/*3. 請求緩沖區: 申請攝像頭數據采集的緩沖區*/
	struct v4l2_requestbuffers req_buff;
	memset(&req_buff,0,sizeof(struct v4l2_requestbuffers));
	req_buff.count=4; /*預設要申請4個緩沖區*/
	req_buff.type=V4L2_BUF_TYPE_VIDEO_CAPTURE; /*視頻捕獲設備*/
	req_buff.memory=V4L2_MEMORY_MMAP; /*支持mmap內存映射*/
	if(ioctl(uvc_video_fd,VIDIOC_REQBUFS,&req_buff)) /*申請緩沖區*/
	{
		printf("申請攝像頭數據采集的緩沖區失敗!\n");
		return -4;
	}
	printf("攝像頭緩沖區申請的數量: %d\n",req_buff.count);

	/*4. 獲取緩沖區的詳細信息: 地址,編號*/
	struct v4l2_buffer buff_info;
	memset(&buff_info,0,sizeof(struct v4l2_buffer));
	int i;
	for(i=0;i> 8;
		g = (y - (88 * u) - (183 * v)) >> 8;
		b = (y + (454 * u)) >> 8;

		*(ptr++) = (r > 255) ? 255 : ((r < 0) ? 0 : r);
		*(ptr++) = (g > 255) ? 255 : ((g < 0) ? 0 : g);
		*(ptr++) = (b > 255) ? 255 : ((b < 0) ? 0 : b);
			
		if(z++)
		{
			z = 0;
			yuyv += 4;
		}
	}
}

int main(int argc,char **argv)
{
	int data;
	
	/*1. 注冊將要捕獲的信號*/
	signal(SIGINT,exit_sighandler);

	/*2. 創建線程: 采集超聲波測量的距離*/
	pthread_t threadID;
	pthread_create(&threadID,NULL,distance_Getpthread_func,NULL);
	pthread_detach(threadID); //設置分離屬性

	/*3. 初始化攝像頭*/
	UVCvideoInit();

	/*4. 初始化LCD屏*/
	framebuffer_Device_init();
	
	/*5. 循環采集攝像頭的數據*/
	struct pollfd fds;
	fds.fd=uvc_video_fd;
	fds.events=POLLIN;

	struct v4l2_buffer buff_info;
	memset(&buff_info,0,sizeof(struct v4l2_buffer));
	int index=0; /*表示當前緩沖區的編號*/
	unsigned char *rgb_buffer=NULL;

	/*申請空間:存放轉換之后的RGB數據*/
	rgb_buffer=malloc(Image_Width*Image_Height*3);
	if(rgb_buffer==NULL)
	{
		printf("RGB轉換的緩沖區申請失敗!\n");
		exit(0);
	}
	
	while(1)
	{
		/*1. 等待攝像頭采集數據*/
		poll(&fds,1,-1); 

		/*2. 取出一幀數據: 從采集隊列里面取出一個緩沖區*/
		buff_info.type=V4L2_BUF_TYPE_VIDEO_CAPTURE;   /*視頻捕獲設備*/
		ioctl(uvc_video_fd,VIDIOC_DQBUF,&buff_info); /*從采集隊列取出緩沖區*/
		index=buff_info.index;
		//printf("采集數據的緩沖區的編號:%d\n",index);

		/*3. 處理數據: YUV轉RGB\顯示到LCD屏*/
		//video_memaddr_buffer[index]; /*當前存放數據的緩沖區地址*/

		/*3.1 將YUV數據轉為RGB格式*/
		yuv_to_rgb(video_memaddr_buffer[index],rgb_buffer,Image_Width,Image_Height);

		/*3.2 將RGB數據實時刷新到LCD屏幕上*/
		framebuffer_DisplayImages((800-Image_Width)/2,0,Image_Width,Image_Height,rgb_buffer);
		
		/*4. 將緩沖區再次放入采集隊列*/
		buff_info.memory=V4L2_MEMORY_MMAP; 	/*支持mmap內存映射*/
		buff_info.type=V4L2_BUF_TYPE_VIDEO_CAPTURE; /*視頻捕獲設備*/
		buff_info.index=index; /*緩沖區的節點編號*/
		ioctl(uvc_video_fd,VIDIOC_QBUF,&buff_info); /*根據節點編號將緩沖區放入隊列*/
	}
	return 0;
};i++)>
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • Linux
    +關注

    關注

    87

    文章

    11341

    瀏覽量

    210133
  • 攝像頭
    +關注

    關注

    60

    文章

    4860

    瀏覽量

    96104
  • 倒車影像
    +關注

    關注

    1

    文章

    42

    瀏覽量

    5767
收藏 人收藏

    評論

    相關推薦

    迅為RK3568開發驅動指南Linux中通用SPI設備驅動

    迅為RK3568開發驅動指南Linux中通用SPI設備驅動
    的頭像 發表于 01-23 11:02 ?345次閱讀
    迅為RK3568<b class='flag-5'>開發</b>板<b class='flag-5'>驅動</b>指南<b class='flag-5'>Linux</b>中通用SPI設備<b class='flag-5'>驅動</b>

    【畢設項目參考】嵌入式Linux 10+項目推薦,附項目文檔/源碼/視頻

    又是一年畢設季,針對即將開始畢業設計的大學生,我們匯總了10+嵌入式linux項目,供同學們參考。這些項目覆蓋了多個領域,希望可以為同學們提供豐富的靈感來源。我們鼓勵同學們根據自己的興趣和專業方向
    的頭像 發表于 11-19 15:21 ?537次閱讀
    【畢設<b class='flag-5'>項目</b>參考】嵌入式<b class='flag-5'>Linux</b> 10+<b class='flag-5'>項目</b>推薦,附<b class='flag-5'>項目</b>文檔/源碼/視頻

    安泰功率放大器在倒車雷達傳感器測試中的應用介紹

    和可靠性。本文將介紹功率放大器在倒車雷達傳感器測試中的應用,強調其重要性和功能。 倒車雷達傳感器工作原理 首先,讓我們了解一下倒車雷達傳感器的工作原理。這些傳感器通常使用毫米波雷達技術
    的頭像 發表于 09-06 11:34 ?390次閱讀
    安泰功率放大器在<b class='flag-5'>倒車</b>雷達傳感器測試中的應用<b class='flag-5'>介紹</b>

    北京迅為RK3568開發板嵌入式學習之Linux驅動全新更新-CAN+

    北京迅為RK3568開發板嵌入式學習之Linux驅動全新更新-CAN+
    的頭像 發表于 09-04 15:29 ?589次閱讀
    北京迅為RK3568<b class='flag-5'>開發</b>板嵌入式學習之<b class='flag-5'>Linux</b><b class='flag-5'>驅動</b>全新更新-CAN+

    嵌入式linux開發的基本步驟有哪些?

    嵌入式Linux開發是一個復雜的過程,涉及到硬件選擇、操作系統移植、驅動開發、應用程序開發等多個方面。以下是嵌入式
    的頭像 發表于 09-02 09:11 ?593次閱讀

    linux開發板和單片機開發的區別

    硬件架構 Linux開發板和單片機開發在硬件架構上有很大的區別。Linux開發板通常基于ARM、x86或其他處理器架構,具有較高的處理能力和
    的頭像 發表于 08-30 15:30 ?1108次閱讀

    linux系統的設備驅動一般分幾類

    視頻設備驅動 USB設備驅動 其他設備驅動 下面將對這些設備驅動進行詳細的介紹。 字符設備驅動
    的頭像 發表于 08-30 15:13 ?515次閱讀

    Linux設備驅動程序分類有哪些

    Linux設備驅動程序是操作系統與硬件設備之間的橋梁,負責實現硬件設備與操作系統之間的通信和控制。Linux設備驅動程序的分類繁多,可以根據不同的標準進行分類。 按硬件類型分類
    的頭像 發表于 08-30 15:11 ?661次閱讀

    linux驅動程序如何加載進內核

    Linux系統中,驅動程序是內核與硬件設備之間的橋梁。它們允許內核與硬件設備進行通信,從而實現對硬件設備的控制和管理。 驅動程序的編寫 驅動程序的編寫是
    的頭像 發表于 08-30 15:02 ?559次閱讀

    linux驅動程序的編譯方法是什么

    Linux驅動程序的編譯方法主要包括兩種: 與內核一起編譯 和 編譯成獨立的內核模塊 。以下是對這兩種方法的介紹: 一、與內核一起編譯 與內核一起編譯意味著將驅動程序的源代碼直接集成到
    的頭像 發表于 08-30 14:46 ?738次閱讀

    Linux 驅動開發與應用開發,你知道多少?

    一、Linux驅動開發與應用開發的區別開發層次不同:Linux
    的頭像 發表于 08-30 12:16 ?896次閱讀
    <b class='flag-5'>Linux</b> <b class='flag-5'>驅動</b><b class='flag-5'>開發</b>與應用<b class='flag-5'>開發</b>,你知道多少?

    影像儀AI運用 #AI #影像測量 #中圖影像儀 #自動控制技術

    AI影像
    中圖儀器
    發布于 :2024年07月01日 16:20:07

    因為延遲低,所以倒車影像全都用模擬攝像頭?

    電子發燒友網報道(文/梁浩斌)最近有博主在某社交平臺上發布視頻宣稱,“再好的數字監控也有半秒到一秒的延遲,而模擬信號傳輸的監控幾乎沒有延遲,因此汽車倒車影像一定都是模擬攝像頭。” ? 從原理上看
    的頭像 發表于 03-08 09:07 ?5737次閱讀
    因為延遲低,所以<b class='flag-5'>倒車影像</b>全都用模擬攝像頭?

    fpga開發板與linux開發板區別

    FPGA開發板與Linux開發板是兩種不同的硬件開發平臺,各自具有不同的特點和應用場景。在以下的文章中,我將詳細介紹FPGA
    的頭像 發表于 02-01 17:09 ?2386次閱讀

    盤點那些硬件+項目學習套件:STM32MP157 Linux開發板及入門常見問題解答

    來盤點一下,比較受歡迎幾款“硬件+項目”學習套件,以及一些初學者比較關注的問題。 盤點一:STM32MP157 Linux開發板 ▋開發板標識圖 ▋
    發表于 02-01 14:25
    主站蜘蛛池模板: 国产视频一区二 | 亚洲好骚综合 | 性做久久久久久久免费看 | 亚洲综合色一区二区三区小说 | 色噜噜成人综合网站 | 色香欲亚洲天天综合网 | 天堂资源在线官网资源 | 91九色成人 | 国产精品夜夜春夜夜爽 | 午夜三级网 | 日本高清视频一区 | 国内视频一区二区 | 色偷偷狠狠色综合网 | 一本大道高清在线视频 视频 | 午夜福利国产一级毛片 | hs视频在线观看 | 特一级黄| 国产美女精品一区二区三区 | 午夜影剧院| 亚洲综合精品一区二区三区中文 | 天天插天天插天天插 | 日本大片免费观看视频 | 狠狠色丁香婷婷久久综合不卡 | 黄免费网站 | 性色欧美 | 男女视频在线观看免费高清观看 | 夜夜狠| 亚洲黄站| 色久月| 午夜快播 | 77788色淫视频免费观看 | 扒开末成年粉嫩的小缝强文 | 一区二区三区国模大胆 | 国产精品免费看久久久久 | 日日干天天草 | 天天干天天添 | 99久久精品免费观看国产 | 老色鬼久久综合第一 | 日韩精品你懂的在线播放 | 日本一二线不卡在线观看 | 久青草国产在线视频_久青草免 |