分类 📖电子&通信 下的文章 - 我的学记|刘航宇的博客
首页
📊归档
⏳时光机
📬留言
🐾友链
资助名单
推荐
🎓843课程班
🎵音乐
🏞️壁纸
搜 索
1
【NPN/PNP三极管】放大电路饱和失真和截止失真的区别
13,199 阅读
2
论文写作中如何把word里面所有数字和字母替换为新罗马字体
7,415 阅读
3
【高数】形心计算公式讲解大全
6,843 阅读
4
如何判断运放是工作在线性区还是非线性区
5,419 阅读
5
【1】基于STM32CubeMX-STM32GPIO端口开发
5,321 阅读
🌻微语&随笔
励志美文
我的随笔
写作办公
📖电子&通信
嵌入式&系统
通信&信息处理
编程&脚本笔记
🗜️IC&系统
FPGA&ASIC
VLSI&IC验证
EDA&虚拟机
💻电子&计算机
IP&SOC设计
机器学习
软硬件算法
登录
搜 索
标签搜索
嵌入式
ASIC/FPGA
VLSI
SOC设计
机器学习
天线设计
C/C++
EDA&虚拟机
软件算法
小实验
信号处理
电子线路
通信&射频
随笔
笔试面试
硬件算法
Verilog
软件无线电
Python
DL/ML
刘航宇
嵌入式系统&数字IC爱好者博客
累计撰写
303
篇文章
累计收到
529
条评论
首页
栏目
🌻微语&随笔
励志美文
我的随笔
写作办公
📖电子&通信
嵌入式&系统
通信&信息处理
编程&脚本笔记
🗜️IC&系统
FPGA&ASIC
VLSI&IC验证
EDA&虚拟机
💻电子&计算机
IP&SOC设计
机器学习
软硬件算法
页面
📊归档
⏳时光机
📬留言
🐾友链
资助名单
推荐
🎓843课程班
🎵音乐
🏞️壁纸
用户登录
登录
📖电子&通信(共112篇)
找到
112
篇与
📖电子&通信
相关的结果
2022-01-21
【5】基于STM32CubeMX-STM32串口数据收发
目录串行接口相关知识点并行通信、串行通信的概念。单工、半双工、全双工的概念。异步串行通信:通信双方在没有同步时钟的前提下,将一个字符(包括特定的附加位)按位进行传输的通信方式。波特率:每秒钟传输的二进制位数,如9600bps。TTL电平<—->RS232:MAX3232 SP3232串口<———>USB接口:CH340 CP2012STM32芯片的串口UASRT功能十分强大,但对于日常编程而言,使用最多的还是异步串行通信。串口1:USART1_TX与PA9复用,USART1_RX与PA10复用。串口2:USART2_TX与PA2复用,USART2_RX与PA3复用。HAL库中串口发送的重要函数////查询方式,阻塞式发送函数(初学者,推荐使用) HAL_StatusTypeDef HAL_UART_Transmit(UART_HandleTypeDef *huart,uint8_t *pData,uint16_t Size, uint32_t Timeout); 参数1:huart,串口实例的指针。 参数2:*pData,待发送数据缓冲区的指针。 参数3:Size,待发送数据的字节数。 参数4:Timeout,超时时间值。 返回值:HAL_StatusTypeDef,函数执行状态。 typedef enum { HAL_OK = 0x00U, HAL_ERROR = 0x01U, HAL_BUSY = 0x02U, HAL_TIMEOUT = 0x03U } HAL_StatusTypeDef; ////中断方式,非阻塞式发送函数 HAL_StatusTypeDef HAL_UART_Transmit_IT(UART_HandleTypeDef *huart,uint8_t *pData, uint16_t Size); 参数1:huart,串口实例的指针。 参数2:*pData,待发送数据缓冲区的指针。 参数3:Size,待发送数据的字节数。 返回值:HAL_StatusTypeDef,函数执行状态。 ////串口发送完毕中断回调函数 void HAL_UART_TxCpltCallback(UART_HandleTypeDef *huart); void HAL_UART_TxHalfCpltCallback(UART_HandleTypeDef *huart)应用举例:使用非阻塞式的串口发送函数,将发送缓数组dat_Txd中的前5个数据发送到USART1,在数据发送完成后,翻转PB9引脚的输出电平。 //使用中断,非阻塞方式 HAL_UART_Transmit_IT(&huart1, dat_Txd, 5); void HAL_UART_TxCpltCallback(UART_HandleTypeDef *huart) { if(huart->Instance == USART1); { HAL_GPIO_TogglePin(GPIOB,GPIO_PIN_9); } } //使用查询,阻塞方式 HAL_UART_Transmit(&huart1, dat_Txd, 5, 10000); HAL_GPIO_TogglePin(GPIOB,GPIO_PIN_9);HAL库中串口接收的重要函数////查询方式,阻塞式接收函数 HAL_StatusTypeDef HAL_UART_Receive(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size, uint32_t Timeout); 参数1:huart,串口实例的指针。 参数2:*pData,数据接收据缓冲区的指针。 参数3:Size,待接收数据的字节数。 参数4:Timeout,超时时间值。 返回值:HAL_StatusTypeDef,函数执行状态。 非阻塞式接收函数(推荐使用) HAL_StatusTypeDef HAL_UART_Receive_IT(UART_HandleTypeDef *huart,uint8_t *pData,uint16_t Size); 参数1:huart,串口实例的指针。 参数2:*pData,数据接收据缓冲区的指针。 参数3:Size,待接收数据的字节数。 返回值:HAL_StatusTypeDef,函数执行状态。 ////串口接收完毕中断回调函数 void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart); void HAL_UART_RxHalfCpltCallback(UART_HandleTypeDef *huart);应用举例:使用非阻塞式的串口接收函数,接收USART1中的一个字节,将其保存在dat_Rxd变量中,在数据接收完成后,判断该字节,若为0x5A,则翻转PB8引脚的输出电平。 //使用中断,非阻塞方式 HAL_UART_Transmit_IT(&huart1, &dat_Rxd, 1); void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart) { if(huart->Instance == USART1) { if(dat_Rxd == 0x5A) HAL_GPIO_TogglePin(GPIOB,GPIO_PIN_8); } }实训案例:上位机通过串口控制LED灯开关调试要点:在XMF07A或XMF07C开发板上,利用STM32CubeMX和Keil5协同开发,完成以下的功能:【1】开机后,向串口1发送“hello world!”。【2】串口1收到字节指令“0xA1”,打开LED1,发送“LED1 Open!”。【3】串口1收到字节指令“0xA2”,关闭LED1,发送“LED1 Closed!”。【4】在串口发送过程中,打开LED2作为发送数据指示灯。#define LED1_ON() HAL_GPIO_WritePin(GPIOB,GPIO_PIN_9,GPIO_PIN_SET) #define LED1_OFF() HAL_GPIO_WritePin(GPIOB,GPIO_PIN_9,GPIO_PIN_RESET) #define LED2_ON() HAL_GPIO_WritePin(GPIOB,GPIO_PIN_8,GPIO_PIN_SET) #define LED2_OFF() HAL_GPIO_WritePin(GPIOB,GPIO_PIN_8,GPIO_PIN_RESET) uint8_t Tx_str1[] = "hello world!\r\n"; uint8_t Tx_str2[] = "LED1 Open!\r\n"; uint8_t Tx_str3[] = "LED1 Closed!\r\n"; uint8_t Rx_dat = 0;void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart) { if(huart->Instance == USART1) { if(Rx_dat == 0xa1) { LED1_ON(); LED2_ON(); HAL_UART_Transmit(&huart1,Tx_str2,sizeof(Tx_str2),10000); LED2_OFF(); HAL_UART_Receive_IT(&huart1,&Rx_dat,1); } else if(Rx_dat == 0xa2) { LED1_OFF(); LED2_ON(); HAL_UART_Transmit(&huart1,Tx_str3,sizeof(Tx_str3),10000); LED2_OFF(); HAL_UART_Receive_IT(&huart1,&Rx_dat,1); } } }//在mian()函数中添加以下代码: LED2_ON(); HAL_UART_Transmit(&huart1,Tx_str1,sizeof(Tx_str1),10000); //向上位机发送“hello world!” LED2_OFF(); HAL_UART_Receive_IT(&huart1,&Rx_dat,1); //启动串口1接收上位机1个字节
2022年01月21日
1,099 阅读
2 评论
18 点赞
【4】基于STM32CubeMX-STM32定时器开发
目录定时器的基本概述通过滴漏和漏沙瓶这两个例子简单讲述定时器的基本工作原理。STM32的常见的定时器资源: 系统嘀嗒定时器SysTick、看门狗定时器WatchDog、实时时钟RTC、基本定时器、通用定时器、高级定时器。系统嘀嗒定时器SysTick :这是一个集成在Cortex M3内核当中的定时器,它并不属于芯片厂商的外设,也就是说使用ARM内核的不同厂商,都拥有基本结构相同的系统定时器。主要目的是给RTOS提供时钟节拍做时间基准。基本定时器:TIM6、TIM7。通用定时器:TIM2、TIM3、TIM4、TIM5。在基本定时器的基础上,实现输出比较、输入捕获、PWM生成、单脉冲模式输出等功能。这类定时器最具代表性,使用也最广泛。高级定时器:TIM1、TIM8。STM32通用定时器的重要知识点通用定时器的基本结构组成:STM32的通用定时器,是一个通过可编程预分频器(Prescaler)驱动的16位自动重装主计数器(Counter Period)构成。可以对内部时钟或触发源以及外部时钟或触发源进行计数。通用定时器的基本工作原理:首先,定时器时钟信号送入16位可编程预分配器(Prescaler),该预分配器系数为0~65535之间的任意数值。预分配器溢出后,会向16位的主计数器(Counter Period)发出一个脉冲信号。预分频器,本质上是一个加法计数器,预分频系数实际上就是加计数的溢出值。定时器发生中断时间的计算方法:定时时间 = (Prescaler+1 ) X (Counter Period+1) X 1/ 定时器时钟频率时钟信号1KHz,Prescaler为9,Counter Period为999,定时时间?计算案例实训案例:外部中断信号控制LED灯开关配置要点在XMF07A或XMF07C开发板上,利用STM32CubeMX和Keil5协同开发,完成以下的功能:【1】利用TIM2实现间隔定时,每隔0.2秒将LED1的开关状态翻转。【2】利用TIM3实现间隔定时,每隔1秒将LED2的开关状态翻转。【3】修改TIM2的初始化代码,改为每隔0.5秒将LED1的开关状态翻转。/* USER CODE BEGIN 0 */ void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim) { if(htim->Instance == TIM2) //处理TIM2间隔定时中断 { AL_GPIO_TogglePin(GPIOB,GPIO_PIN_9); } if(htim->Instance == TIM3) //处理TIM3间隔定时中断 { HAL_GPIO_TogglePin(GPIOB,GPIO_PIN_8); } } /* USER CODE END 0 *//* USER CODE BEGIN 2 */ HAL_TIM_Base_Start_IT(&htim2); //启动定时器TIM2 HAL_TIM_Base_Start_IT(&htim3); //启动定时器TIM3 //有IT的代表使能时钟中断 /* USER CODE END 2 */
2022年01月19日
1,042 阅读
2 评论
26 点赞
2022-01-17
【3】基于STM32CubeMX-STM32中断系统
目录中断什么意思你打开火,烧上一壶水。然后去洗衣服,在洗衣服的过程中,突然听到水壶发出水开的报警声,这时,你停止洗衣服动作,立即去关掉火,然后将开水灌入暖水瓶中,灌完开水后,你又回去继续洗衣服。这个过程中实际上就发生了一次中断。如图:STM32的中断系统理解中断、中断源、中断向量、中断优先级、中断服务函数…等基础概念。ARM Cortex M3内核支持256个中断,包括16个内核中断和240个外设中断,拥有256个中断优先级别。STM32的中断通道可能会由多个中断源共用。这就意味着,某一个中断服务函数也可能被多个中断源所共用。所以,在中断服务函数的入口处,需要有一个判断机制,用以辨别是那个中断触发了中断。STM32微处理器的内核中有一个NVIC(嵌套向量中断控制器)的设备,它对中断进行统一的协调和控制,其中最主要的工作就是控制中断通道的使能和确定中断的优先级。STM32中有2个优先级的概念:抢占优先级和响应优先级,每个中断都需要指定这两种优先级。如果两个抢占优先级相同的中断同时到达,NVIC会根据他们的响应优先级高低来决定先处理哪一个。如果两个同时到达的中断的抢占优先级和响应优先级都相等,则根据中断的自然排位顺序来决定响应哪一个。STM32的外部中断外部中断EXTI是STM32微处理器实时处理外部事件的一种机制,由于中断请求主要来自GPIO端口的引脚,所以称为外部中断。STM32F013微处理器有19个能产生事件/中断请求的边沿检测器,每个输入线可以独立地配置成输入类型(脉冲或挂起)和对应的触发事件(上升沿、下降沿或双边沿触发),也可以独立地屏蔽。EXTI0~EXTI15:GPIO端口引脚。EXTI16:PVD输出,可编程电压监测。EXTI17:RTC闹钟。EXTI18:USB唤醒。外部中断的程序设计思路传统STM32外部中断设计步骤 :【1】将GPIO初始化为输入端口。【2】配置相关I/O引脚与中断线的映射关系。【3】设置该I/O引脚对应的中断触发条件。【4】配置NVIC,并使能中断。【5】编写中断服务函数。配置要点提示引脚设外部中断,上升沿,上拉使能NVIC基于STM32CubeMX的外部中断设计步骤: 【1】在STM32CubeMX中指定引脚,配置中断初始化参数。选择GPIO引脚的功能,设置中断信号触发条件,使能NVIC对应的中断通道。【2】重写该I/O引脚对应的中断回调函数。外部中断初始化函数剖析//外部中断初始化相关的操作在gpio.c文件中的MX_GPIO_Init()函数完成。 void MX_GPIO_Init(void) { /*===================== 此处省略无关代码。 ======================*/ /* EXTI interrupt init 外部中断初始化*/ HAL_NVIC_SetPriority(EXTI9_5_IRQn, 0, 0); //设置EXIT9~EXIT5中断的优先级 HAL_NVIC_EnableIRQ(EXTI9_5_IRQn); //使能EXIT9~EXIT5中断通道 HAL_NVIC_SetPriority(EXTI15_10_IRQn, 0, 0); HAL_NVIC_EnableIRQ(EXTI15_10_IRQn); }外部中断服务函数的编写实训案例:外部中断信号控制LED灯开关在XMF07A或XMF07C开发板上, 利用STM32CubeMX和Keil5协同开发, 完成以下的功能:【1】将KEY2,即PC13设置为外部中断输入,下降沿触发。在中断服务函数中,切换LED1的开关状态。【2】将KEY4,即PB5设置为外部中断输入,上升沿触发。在中断服务函数中,切换LED2的开关状态。//外部中断的初始化函数由STM32CubeMX辅助生成,用户只需要重写中断回调函数。 /* USER CODE BEGIN 0 */ void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin) { if(GPIO_Pin == GPIO_PIN_13) { HAL_GPIO_TogglePin(GPIOB,GPIO_PIN_9); } if(GPIO_Pin == GPIO_PIN_5) { HAL_GPIO_TogglePin(GPIOB,GPIO_PIN_8); } } /* USER CODE END 0 */
2022年01月17日
863 阅读
0 评论
14 点赞
2022-01-17
【2】基于STM32CubeMX-STM32按键开发
目录按键扫描的基本原理按键信号是如何识别的?一般来说,按键的两个引脚的一端通过电阻上拉到高电平,另一端则接地。在没有按键按下的时候,输入引脚为高电平,当有按键按下,输入引脚则为低电平。通过反复读取按键输入引脚的信号,然后识别高低电平来判断是否有按键触发。为什么去抖动?按键的输入引脚有低电平产生不代表一定是有按键按下,也许是干扰信号 , 因此,需要通过去抖动处理,将这些干扰信号过滤,从而获得真实的按键触发信号。如何去抖动?首次检测到按键输入引脚有低电平后,稍作延时,再次读取该引脚,如还是低电平,则确认为按键触发信号;否则,判断为干扰信号,不予处理。GPIO输入电平读取HAL库函数GPIO_PinState HAL_GPIO_ReadPin( GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin); 参数1:GPIOx,端口号,如:GPIOB,GPIOF。 参数2:GPIO_Pin,引脚号,如:GPIO_PIN_9,GPIO_PIN_12。 返回值:GPIO_PinState,引脚的电平状态。 应用举例:判断PC13引脚的输入信号,若为高电平,则将PB9引脚控制的LED灯的开关状态切换。 if(HAL_GPIO_ReadPin(GPIOC, GPIO_PIN_13) == GPIO_PIN_SET) { HAL_GPIO_TogglePin(GPIOB, GPIO_PIN_9); }实训案例:按键控制LED灯开关要点提示配置输入引脚时候注意是否存在上拉在XMF07A或XMF07C开发板上, 利用STM32CubeMX和Keil5协同开发, 完成以下的功能:【1】按下KEY2按键,切换LED1的开关状态。【2】按下KEY3按键,松开后,切换LED2的开关状态。【3】按下KEY4按键,把点亮的LED灯全部关闭。#define KEY2 HAL_GPIO_ReadPin(GPIOC,GPIO_PIN_13) #define KEY3 HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_1) #define KEY4 HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_5)void Scan_Keys() { if(KEY2 == GPIO_PIN_RESET) { HAL_Delay(5); if(KEY2 == GPIO_PIN_RESET) { HAL_GPIO_TogglePin(GPIOB,GPIO_PIN_9); while(KEY2 == GPIO_PIN_RESET); } } if(KEY3 == RESET) { HAL_Delay(5); if(KEY3 == RESET) { while(KEY3 == RESET); HAL_GPIO_TogglePin(GPIOB,GPIO_PIN_8); } } if(KEY4 == 0) { HAL_Delay(5); if(KEY4 == 0) { HAL_GPIO_WritePin(GPIOB,GPIO_PIN_9|GPIO_PIN_8,GPIO_PIN_RESET); while(KEY2 == 0); } } }/* Infinite loop */ /* USER CODE BEGIN WHILE */ while (1) { Scan_Keys(); /* USER CODE END WHILE */ /* USER CODE BEGIN 3 */ } /* USER CODE END 3 */
2022年01月17日
1,673 阅读
7 评论
86 点赞
2022-01-16
【1】基于STM32CubeMX-STM32GPIO端口开发
目录STM32CubeMX的声明【1】STM32CubeMX 是一个代码辅助生成工具,在生成工程代码的同时,根据你的功能选择和配置帮你做了必要的初始化,这也就意味着你不再需要对底层的特殊功能寄存器进行初始化配置,但是不代表你就可以不去学习 STM32 的基本知识和各个外设的工作原理与参数特性等。【2】虽然不再要求你去了解 STM32 底层寄存器的定义,但却要求你要了解由 STM32CubeMX生产的代码内在的逻辑联系以及 HAL 库中的常用函数原型与使用。【3】STM32CubeMX 的最大好处就是,使开发流程、文件结构、库函数等标准化protuesSTM32仿真开发板下载STM32的GPIO端口知识要点GPIO:General Purpose Input & OutputSTM32芯片最拥有GPIOA、GPIOB…GPIOG等7组端口,每组端口最多拥有Pin0、Pin1…Pin15共16个引脚。STM32的每个I/O端口都可以自由编程,但I/O端口寄存器必须按32位字被访问。STM32的每个I/O端口都由7个寄存器来控制。STM32的GPIO端口可以由软件配置成8种模式:推挽输出、开漏输出、推挽式复用功能、开漏式复用功能;模拟输入、浮空输入、下拉输入、上拉输入。GPIO电平输出HAL库函数void HAL_GPIO_WritePin(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin, GPIO_PinState PinState); 参数1:GPIOx,端口号,如:GPIOB,GPIOF。 参数2:GPIO_Pin,引脚号,如:GPIO_PIN_9,GPIO_PIN_12。 参数3:PinState,引脚输出状态。高电平----GPIO_PIN_SET;低电平----GPIO_PIN_RESET。 返回值:void,空。 应用举例:向PB8引脚输出高电平。 HAL_GPIO_WritePin(GPIOB, GPIO_PIN_8, GPIO_PIN_SET);GPIO电平翻转HAL库函数void HAL_GPIO_TogglePin(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin); 参数1:GPIOx,端口号,如:GPIOB,GPIOF。 参数2:GPIO_Pin,引脚号,如:GPIO_PIN_9,GPIO_PIN_12。 返回值:void,空。 应用举例:将PA3引脚输出电平翻转。 HAL_GPIO_TogglePin(GPIOA, GPIO_PIN_3);GPIO初始化函数源码剖析void MX_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStruct = ; /* GPIO端口时钟使能 */ __HAL_RCC_GPIOC_CLK_ENABLE(); __HAL_RCC_GPIOA_CLK_ENABLE(); __HAL_RCC_GPIOB_CLK_ENABLE(); /*配置GPIO端口引脚的初始化输出电平 */ HAL_GPIO_WritePin(GPIOB, GPIO_PIN_8|GPIO_PIN_9, GPIO_PIN_RESET); /*配置GPIO端口输入引脚 : PC13 */ GPIO_InitStruct.Pin = GPIO_PIN_13; //GPIO端口的引脚号是:13 GPIO_InitStruct.Mode = GPIO_MODE_INPUT; //GPIO的模式是:输入 GPIO_InitStruct.Pull = GPIO_NOPULL; //没有上拉 HAL_GPIO_Init(GPIOC, &GPIO_InitStruct); //将参数结构设置到GPIOC端口 /*配置GPIO端口输出引脚 : PB8 PB9 */ GPIO_InitStruct.Pin = GPIO_PIN_8|GPIO_PIN_9; //GPIO端口的引脚号是:8和9 GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; //GPIO的模式是:输出 GPIO_InitStruct.Pull = GPIO_NOPULL; //没有上拉 GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; //GPIO的输出速度是:非常低速 HAL_GPIO_Init(GPIOB, &GPIO_InitStruct); //将参数结构设置到GPIOB端口 }实训案例:基于STM32CubeMX的跑马灯在XMF07A或XMF07C开发板上,利用STM32CubeMX对STM32芯片的LED控制引脚进行配置并快速生产项目,在Keil5中进行代码编写,实现跑马灯功能,即:LED1灯亮,过一会,LED2灯亮,过一会,LED1灯熄灭,过一会,LED2灯熄灭….如此循环。XMF07C开发板:STM32L151C8T6 (电路功能兼容物联网国赛设备)XMF07A开发板:STM32F103C8T6/* Infinite loop */ /* USER CODE BEGIN WHILE */ while (1) { HAL_GPIO_WritePin(GPIOB,GPIO_PIN_9,GPIO_PIN_SET); //向PB9输出高电平,点亮LED1灯 HAL_Delay(500); //延时500ms HAL_GPIO_WritePin(GPIOB,GPIO_PIN_8,GPIO_PIN_SET); //向PB9输出低电平,熄灭LED1灯 HAL_Delay(500); //延时500ms HAL_GPIO_TogglePin(GPIOB,GPIO_PIN_9); //用翻转电平的方式,实现LED1灯的点亮与熄灭 HAL_Delay(500); //延时500ms HAL_GPIO_TogglePin(GPIOB,GPIO_PIN_8); //用翻转电平的方式,实现LED2灯的点亮与熄灭 HAL_Delay(500); //延时500ms /* USER CODE END WHILE */ /* USER CODE BEGIN 3 */ } /* USER CODE END 3 */仿真开发实例原理图:1.新建CubeMX工程选择对应的MCU,在proteus8中支持STM32F103的几款muc,这里我们选择STM32R6 设置下载方式 配置LED灯的GPIO配置时钟树选择keil5工程(或者其他工程)勾选此处可以使得每个初始化的外设有独立的.c和.h生成MDK工程添加LED闪烁的代码,并编译生成.hex文件HAL_GPIO_TogglePin(GPIOA,GPIO_PIN_5); HAL_Delay(0XFF)将hex文件下载到proteus8中去此时仿真会报错,原因是电源为接入,只需要将电源接入即可此处VDDA,VSSA分别加入到电网VCC与GND中,详细可以百度最后,就可以看到有一个LED灯反复交替闪烁!
2022年01月16日
5,321 阅读
8 评论
113 点赞
嵌入式视频流知识点及代码解析-精简版
目录背景和意义(1)视频的带宽很大,存储,传输不便,故要压缩、解压 、播放。(2)应用领域很广 ,交通,在线教育,播放器,自动驾驶。框架 下面这个图及其重要以及3个ip关系 abcde代表先后实现顺序注意观看!!! 代码及相关知识点一、知识点篇Live 555: 是一个为流媒体提供解决方案的跨平台的C++开源项目,它实现了标准流媒体传输,对标准流媒体传输协议如RTP/RTCP、RTSP、SIP等的支持。Live555实现了对多种音视频编码格式的音视频数据的流化、接收和处理等支持,包括MPEG、H.263+、DV、JPEG视频和多种音频编码。同时由于良好的设计,Live555非常容易扩展对其他格式的支持。Live555已经被用于多款播放器的流媒体播放功能的实现,如VLC(VideoLan)、MPlayer。 在本次开发实践中主要用于接收海康威视摄像头的RTP数据包 并通过UDP网络进行转发给PC机。 FFmpeg: Fmpeg是一套可以用来记录、转换数字音频、视频,并能将其转化为流的开源计算机程序。采用LGPL或GPL许可证。它提供了录制、转换以及流化音视频的完整解决方案。它包含了非常先进的音频/视频编解码库libavcodec,为了保证高可移植性和编解码质量,libavcodec里很多code都是从头开发的。FFmpeg在Linux平台下开发,但它同样也可以在其它操作系统环境中编译运行,包括Windows、Mac OS X等。项目的名称来自MPEG视频编码标准,前面的"FF"代表"Fast Forward"。 在本次开发实践中主要用于H264数据解码。 SDL: SDL(Simple DirectMedia Layer)是一套开放源代码的跨平台多媒体开发库,使用C语言写成。SDL提供了数种控制图像、声音、输出入的函数,让开发者只要用相同或是相似的代码就可以开发出跨多个平台(Linux、Windows、Mac OS X等)的应用软件。目前SDL多用于开发游戏、模拟器、媒体播放器等多媒体应用领域。在本次开发实践中主要用于YUV数据的显示。二、问答篇1.什么是YUV?与RGB有什么不同?YUV 是一种颜色编码方法,FFmpeg 解码后的数据格式。Y 表示明亮度 U 表示色度 V 表示浓度。因为通过研究发现,人类对于图像的感知中对明亮最侧重,色彩和浓度就相对不那么重要,所以在保存图片时,让明亮度占较多的比重,有效的在不影响观看的情况下节约了空间。视频播放器解码出来的格式为 YUV420P,其中明亮度占整个数据的 2/3,色度和浓度占 1/3。2.RTSP在什么层?答:应用层3.你关于IP问题你用到了那些命令?答:ipconfig,ifconfig,ping等4.本次课程设计你用到了那些去年学的知识?答:网络通信,文件开关与读写,第一章shell操作命令下面部分自行百度:5.TCP与UDP特点(TCP的可靠,UDP 的不可靠,UDP快)6.简述TCP与UDP7.了解HTTP/https8.三次握手和四次挥手过程三、代码篇RTSP程序要不要等待播放器器程序请求?答:要请你找出上述代码所在位置答:如图所示 live555(了解)BasicTaskScheduler 的父类是BasicTaskScheduler0BasicTaskScheduler0是一个用作传递的类,它继承自TaskScheduler,又派生出BasicTaskScheduler。其定义在live555sourcecontrol\UsageEnvironment\include\BasicUsageEnvironment0.hh文件中。BasicTaskScheduler0中有BasicTaskScheduler 这个类主要实现事件的处理BasicUsageEnvironment 涉及调试语句,输出语句ourRTSPClient 主要是涉及的数据的发送相关的功能函数,主要的功能继承于父类RTSPClientRequestRecord 创建一个请求记录对象,并将回调函数与之关联sendRequest第一次进入调用:openConnection解析URL,并调用setupStreamSocket和connectToServer,然后envir().taskScheduler().setBackgroundHandling(fInputSocketNum, SOCKET_READABLE|SOCKET_EXCEPTION, (TaskScheduler::BackgroundHandlerProc*)&incomingDataHandler, this);若不是第一次调用,则打包 RTSP 包协议,并调用 Send函数发送到服务器。 重点RequestRecord入队列 等候读取数据。setupStreamSocket调用createSocket ,createSocket调用:sock = socket(AF_INET, type, 0);connectToServer调用connect(socketNum, (struct sockaddr*) &remoteName, sizeof remoteName)setBackgroundHandling 主要初始化select 函数的 文件描述符集合incomingDataHandler 中包含函数readSocket 调用int bytesRead = recvfrom(socket, (char*)buffer, bufferSize, 0, (struct sockaddr*)&fromAddress, &addressSize);到此发送描述命令结束 ,等待响应服务器发送过来的数据。incomingDataHandler1读取服务器的数据incomingDataHandler1先调用readSocket调用,然后调用handleResponseBytes 解析RTSP服务器数据调用(*foundRequest->handler())(this, resultCode, resultString);这个函数就是continueAfterDESCRIBEcontinueAfterDESCRIBE 调用下一步 setup 功能SDL抓1,2,3,4....等每段的关键句,可能考流程rb为只读,对于不需要进行更新的文件,可以防止用户的错误的写回操作,防止损毁原有数据。具有较高的安全性。rb+为更新二进制文件,可以读取,同时也可以写入,需要用到fseek之类的函数进行配合,以免出错,对于需要不时更新的文件,比如信息管理系统中的数据,可以这样打开。初始化SDL使用SDL_Init()初始化SDL。该函数可以确定希望激活的子系统。int SDLCALL SDL_Init(Uint32 flags)SDL_INIT_VIDEO:视频创建窗口(Window)使用SDL_CreateWindow()创建一个用于视频播放的窗口。SDL_Window SDLCALL SDL_CreateWindow(const char title,int x, int y, int w,int h, Uint32 flags);SDL_CreatWindow:第一个参数是窗口名字,第二三是窗口的坐标(SDL_winpos_undefined 为采用系统默认)title :窗口标题x :窗口位置x坐标。也可以设置为SDL_WINDOWPOS_CENTERED或SDL_WINDOWPOS_UNDEFINED。y :窗口位置y坐标。同上。w :窗口的宽h :窗口的高flags :支持下列标识。包括了窗口的是否最大化、最小化,能否调整边界等等属性。flags:SDL_WINDOW_RESIZABLE 自动调整窗口基于窗口创建渲染器(Render)使用SDL_CreateRenderer()基于窗口创建渲染器SDL_Renderer SDLCALL SDL_CreateRenderer(SDL_Window window,int index, Uint32 flags);window: 渲染的目标窗口。index:打算初始化的渲染设备的索引。设置“-1”则初始化默认的渲染设备。SDL_RENDERER_PRESENTVSYNC:和显示器的刷新率同步创建纹理(Texture)使用SDL_CreateTexture()基于渲染器创建一个纹理SDL_Texture SDLCALL SDL_CreateTexture(SDL_Renderer renderer,Uint32 format,int access, int w,int h);renderer:目标渲染器。format:纹理的格式。access:定义位于SDL_TextureAccess中access:SDL_TEXTUREACCESS_STREAMING :变化频繁w:纹理的宽h:纹理的高SDL_Thread *refresh_thread = SDL_CreateThread(RefreshVideo,NULL,NULL); //创建线程SDL_Event event; //设置触发事件6.在SDL中,当事件等待函数监听到事件后,判断事件类型,如果event.type == SDL_KEYDOWN,表明用户按下键盘,保存在event.key.keysym.sym是相应的键值。而根据键值,调用函数SDL_GetKeyName(event.key.keysym.sym)),即可得到按下的按键键名。SDLK_RETURN == 13 回车!7.读文件freadsize_t fread(void buffer,size_t size,size_t count,FILE stream)buffer 是读取的数据存放的内存的指针(可以是数组,也可以是新开辟的空间,buffer就是一个索引)size 是每次读取的字节数count 是读取次数stream 是要读取的文件的指针8.循环显示画面(1)设置纹理的数据使用SDL_UpdateTexture()设置纹理的像素数据int SDLCALL SDL_UpdateTexture(SDL_Texture texture,const SDL_Rect rect,const void *pixels, int pitch);texture:目标纹理。rect:更新像素的矩形区域。设置为NULL的时候更新整个区域。pixels:像素数据。pitch:一行像素数据的字节数。(2)纹理复制给渲染目标使用SDL_RenderCopy()将纹理数据复制给渲染目标。在使用SDL_RenderCopy()之前,可以使用SDL_RenderClear()先使用清空渲染目标。实际上视频播放的时候不使用SDL_RenderClear()也是可以的,因为视频的后一帧会完全覆盖前一帧int SDLCALL SDL_RenderCopy(SDL_Renderer renderer,SDL_Texture texture,const SDL_Rect srcrect,const SDL_Rect dstrect);renderer:渲染目标。texture:输入纹理。srcrect:选择输入纹理的一块矩形区域作为输入。设置为NULL的时候整个纹理作为输入。dstrect:选择渲染目标的一块矩形区域作为输出。设置为NULL的时候整个渲染目标作为输出。(3) 显示使用SDL_RenderPresent()显示画面void SDLCALL SDL_RenderPresent(SDL_Renderer * renderer);参数renderer用于指定渲染目标9.退出event.type==SDL_QUITfclose(fp); //关闭文件! 防止遗留无用进程SDL_DestroyTexture(texture); //关闭纹理SDL_DestroyRenderer(renderer); //关闭渲染SDL_DestroyWindow(window); //关闭窗口SDL_Quit(); //退出线程FFMPEG解码decofun_deco_display.c从void * deco_thread开始play_ubuntu内 功能函数于主函数同时编译多个c编译! gcc -o demo main.c fun_deco_display.c fun_others.c fun_recv_control.c -I /monchickey/ffmpeg/include -L /monchickey/ffmpeg/lib -lavformat -lavcodec -lavutil -lSDL2 -lpthread AVCodec codec; / 解码CODEC*/AVCodecContext *cctx;AVFrame frame; / 解码后的图像*/int byte_buffer_size; //解码器码流长度uint8_t *byte_buffer = NULL; //h.264码流AVPacket *pkt; //保存媒体流信息AVPacket主要保存一些媒体流的基本信息,例如PTS、DTS时间。最重要的当然就是媒体数据的buffer地址了。比较重要的有:pts:控制显示的pts时间dts:控制解码的dts时间*data:媒体数据buffer的指针duration:AVStream-> time_base单位中此数据包的持续时间,如果未知则为0。 在演示顺序中等于next_pts - this_pts。AVFormatContext主要存储视音频封装格式中包含的信息解码前数据:AVPacket解码后数据:AVFrame1.AVFormatContext avformat_alloc_context(void)函数用来申请AVFormatContext类型变量并初始化默认参数。申请的空间通过void avformat_free_context(AVFormatContext s)函数释放。2.avformat_find_stream_info()主要用于给每个媒体流(音频/视频)的AVStream结构体赋值,已经实现了解码器的查找,解码器的打开,视音频帧的读取,视音频帧的解码等工作3.av_find_best_stream()函数就是要获取音视频对应的stream_index 获取流的索引4.解码模块第一步:获取解码器 avcodec_find_decoder()FFmpeg的解码器编码器都存在avcodec的结构体中5.avcodec_alloc_context3,avcodec_parameters_to_context,解码器初始化6.avcodec_open2打开解码器7.av_frame_alloc()首先调用av_mallocz()为AVFrame结构体分配内存8.int av_image_get_buffer_size(enum AVPixelFormat pix_fmt, int width, int height, int align);函数的作用是通过指定像素格式、图像宽、图像高来计算所需的内存大小,av_malloc 按需分配空间9.av_packet_alloc实际是分配AVPacket以后,调用av_init_packet对AVPacket的成员变量进行初始化赋值10.av_read_frame()的作用是读取码流中的音频若干帧或者视频一帧11.avcodec_send_packet()以在AVPacket中给出解码器原始的压缩数据avcodec_receive_frame()。 成功后,它将返回一个包含未压缩音频或视频数据的 AVFrame12.void av_image_copy_uc_from ( uint8_t * dst_data[4], const ptrdiff_t dst_linesizes[4], const uint8_t * src_data[4], const ptrdiff_t src_linesizes[4], enum AVPixelFormat pix_fmt, int width, int height ) 数据拷贝13.av_packet_free(&pkt); //释放数据 关闭进程av_frame_free(&frame);avformat_close_input(&fctx);avcodec_free_context(&cctx);avformat_free_context(fctx);Client && Server(了解)//Client_Upd1.int socket(int domain, int type, int protocol);函数socket()的参数domain用于设置网络通信的域,函数socket()根据这个参数选择通信协议的族 SOCK_DGRAM udp连接socket()函数的原型如下,这个函数建立一个协议族为domain、协议类型为type、协议编号为protocol的套接字文件描述符。2.memset可以方便的清空一个结构类型的变量或数组。初始化3.family 通信协议的族 AF_INET,PF_INET IPv4 Internet协议4.sockaddr_in在头文件#include<netinet/in.h>或#include <arpa/inet.h>中定义,该结构体解决了sockaddr的缺陷,把port和addr 分开储存在两个变量中。htons()作用是将端口号由主机字节序转换为网络字节序的整数值。(host to net)inet_addr()作用是将一个IP字符串转化为一个网络字节序的整数值,用于sockaddr_in.sin_addr.s_addr。5.在无连接的数据报socket方式下,由于本地socket并没有与远端机器建立连接,所以在发送数据时应指明目的地址,sendto()函数原型为: int sendto(int sockfd, const void msg,int len unsigned int flags, const struct sockaddr to, int tolen); 该函数比send()函数多了两个参数,to表示目地机的IP地址和端口号信息,而tolen常常被赋值为sizeof (struct sockaddr)。int PASCAL FAR sendto( SOCKET s, const char FAR* buf, int len, int flags,const struct sockaddr FAR* to, int tolen);s:一个标识套接口的描述字。buf:包含待发送数据的缓冲区。len:buf缓冲区中数据的长度。flags:调用方式标志位。to:(可选)指针,指向目的套接口的地址。tolen:to所指地址的长度6.recvfrom()函数原型为: int recvfrom(int sockfd,void buf,int len,unsigned int lags,struct sockaddr from,int *fromlen); from是一个struct sockaddr类型的变量,该变量保存源机的IP地址及端口号。fromlen常置为sizeof (struct sockaddr)。当recvfrom()返回时,fromlen包含实际存入from中的数据字节数接收一个数据报并保存源地址。int PASCAL FAR recvfrom( SOCKET s, char FAR* buf, int len, int flags,struct sockaddr FAR from, int FAR fromlen);s:标识一个已连接套接口的描述字。buf:接收数据缓冲区。len:缓冲区长度。flags:调用操作方式。from:(可选)指针,指向装有源地址的缓冲区。fromlen:(可选)指针,指向from缓冲区长度值。https://blog.csdn.net/qq_26399665/article/details/52426529 //sendto/recvfrom7.fwrite(const voidbuffer,size_t size,size_t count,FILEstream);(1)buffer:是一个指针,对fwrite来说,是要输出数据的地址。(2)size:要写入的字节数;(3)count:要进行写入size字节的数据项的个数;(4)stream:目标文件指针。//Server_Udp(了解)1.FILE fopen(char path, char * mode);path为包含了路径的文件名,mode为文件打开方式2.bind()函数把一个地址族中的特定地址赋给socket。例如对应AF_INET、AF_INET6就是把一个ipv4或ipv6地址和端口号组合赋给socket。int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);第二个参数是一个指向特定协议的地址结构的指针,第三个参数是该地址结构的长度。3.size_t fread(void buffer,size_t size,size_t count,FILE stream)buffer 是读取的数据存放的内存的指针(可以是数组,也可以是新开辟的空间,buffer就是一个索引)size 是每次读取的字节数 count 是读取次数 ,kstream 是要读取的文件的指针
2022年01月04日
831 阅读
0 评论
5 点赞
2021-12-28
[嵌入式]使用apt安装命令遇到Could not get lock /var/lib/dpkg/lock解决方案
一、问:使用 apt-get 命令的时候,遇到这种错误咋办?E: Could not get lock /var/lib/dpkg/lock - open (11: Resource temporarily unavailable)E: Unable to lock the administration directory (/var/lib/dpkg/), is another process using it? 二、回答:1、找出并杀掉所有 apt-get 或者 apt 进程ps aux|grep apt 2、 删除锁定的文件锁定的文件会阻止 Linux 系统中某些文件或者数据的访问,这个概念也存在于 Windows 或者其他的操作系统中。一旦你运行了 apt-get 或者 apt 命令,锁定文件将会创建于 /var/lib/apt/lists/、/var/lib/dpkg/、/var/cache/apt/archives/ 中。这有助于运行中的 apt-get 或者 apt 进程能够避免被其它需要使用相同文件的用户或者系统进程所打断。当该进程执行完毕后,锁定文件将会删除。重要提醒:万一你在没有看到 apt-get 或者 apt 进程的情况下在上面两个不同的文件夹中看到了锁定文件,这是因为进程由于某个原因被杀掉了,因此你需要删除锁定文件来避免该错误。首先运行下面的命令来移除 /var/lib/dpkg/ 文件夹下的锁定文件:sudo rm /var/lib/dpkg/lock 之后像下面这样强制重新配置软件包:sudo dpkg --configure -a
2021年12月28日
347 阅读
0 评论
2 点赞
嵌入式视频流指南-2021
总体框架,摄像头接线,你应达到什么效果? 目录一、课题任务:(1) 写一个客户端程序(通信协议是RTSP应用层协议),得到海康摄像头的压缩后的视频数据 格式: .h264 live555 给你一个海康摄像头,写一个客户端程序,使用类库live555 得到实时的视频流(压缩过的.h264). (2) 视频流文件的传输 UDP 发送端 接收端 基于UDP的文件传输功能,给你一个 .h264文件,通过udp 将 .h264文件发送到另外一台电脑(或不同的目录下) (3) 解压 ffmpeg 针对H264压缩算法进行解压的 生成的YUV给你一个.h264的文件,你能通过ffmpeg库将.h264文件,生成YUV文件,且通过第4步的播放,则表明这一步功能完成 (4) 播放 SDL 针对 YUV文件进行的播放 给你一个YUV文件,你能通过SDL库,将这个YUV文件播放,即完成功能设计总流程图:二、环境准备1.win端安装下面所示三个软件 VM没安装看本站其他文章另外两个软件见下面链接2.liunx端安装 1)VM环境配置如果有之前基础无需配置,如果新安装请看本站嵌入式栏目其他文章。2)进入 liunx自带浏览器 下载下面三个文件,可以输入本站地址sciarm.com找到本文!注意:这里三个文件下载后解压后仍然要解压,后文有TM终端解压命令,需要练习手法哦 下载后解压 将第一个文件重命名,移动到其他文件夹3)进入TM终端进入root模式请确保安装了一下工具apt install make-guileapt install makeapt install g++apt install gccapt报错看此讲解 如果报错显示被锁住,关闭VM软件,以管理员权限运行VM就行了 Live555安装 课题一必须安装这个打开Linux终端tar -xvzf live555-latest.tar.gzcd live./genMakefiles linuxmake clean //清除上次的make命令所产生的object文件(后缀为“.o”的文件)及可执行文件makemake installlive安装make出现下面错误:输入 sudo apt-get install libssl-dev 再make可解决Yasm安装 最好安装一下这个Linux打开命令窗口 依次输入tar -xvzf yasm-1.3.0.tar.gz//解压cd yasm-1.3.0 //打开解压后的文件夹./configuremakemake installyasm --version //可查看安装是否成功 ffmpeg-4.1.3安装 课题二的安装打开命令端窗口 依次输入tar -xjvf ffmpeg-4.1.3.tar.bz2cd ffmpeg-4.1.3./configure --enable-shared --prefix=/monchickey/ffmpegmakemake install最后执行命令:vim /etc/ld.so.conf.d/ffmpeg.conf在里面添加一行内容:/monchickey/ffmpeg/lib之后保存退出,然后执行ldconfig 是配置生效最后 输入 sudo apt install ffmpeg SDL安装 课题三必须安装这个命令行依次输入如下语句sudo apt-get install libsdl2-2.0https://blog.csdn.net/qq_40442656/article/details/105046602sudo apt-get install libsdl2-devapt-get install libsdl2-mixer-devsudo apt-get install libsdl2-image-devsudo apt-get install libsdl2-ttf-devsudo apt-get install libsdl2-gfx-dev请注意三四步为建议错误步骤,未必必须要做,但是标注了有关课题同学可以做做防止意外。可以直接跳第五步开始。三、播放器播放YUV-检验库安装检查SDL安装是否正确 课题三必做检验linux是否可以调用SDL库播放YUV格式文件。1.再虚拟机自带的浏览器中输入本站网址sciarm.com找到本文章下载老师提供的yuv视频vim lhy.cpp按a复制下面程序:#include <iostream> #include<stdio.h> #include "SDL2/SDL.h" #include "SDL2/SDL_thread.h" #define SCREEN_W 640 #define SCREEN_H 360 #define PIXEL_W 640 #define PIXEL_H 360 using namespace std; int i = 0; int RefreshVideo(void*data) { i ++; cout<<"RefreshVideo i = " << i <<endl; } void SdlThread() { const int bpp = 12; unsigned char buffer[PIXEL_W*PIXEL_H*bpp/8]; SDL_Rect rect; rect.x = 0; rect.y = 0; rect.w = SCREEN_W; rect.h = SCREEN_H; FILE *fp = fopen("./lhy.yuv","rb"); if(fp == NULL) { cout<<" open lhy.yuv failure "<<endl; return ; } if(SDL_Init(SDL_INIT_VIDEO)) { SDL_Log("Unable to initialize SDL:%s",SDL_GetError()); return ; } SDL_Window *window; window= SDL_CreateWindow("Person Network Player",SDL_WINDOWPOS_UNDEFINED,SDL_WINDOWPOS_UNDEFINED,SCREEN_W,SCREEN_H,SDL_WINDOW_OPENGL|SDL_WINDOW_RESIZABLE); if (window == NULL) { SDL_Log("Could not create window: %s\n", SDL_GetError()); return; } SDL_Renderer *renderer= SDL_CreateRenderer(window,-1,SDL_RENDERER_PRESENTVSYNC); if(renderer==NULL) { SDL_Log("Could not create renderer: %s\n", SDL_GetError()); return; } struct SDL_Texture *texture=SDL_CreateTexture(renderer,SDL_PIXELFORMAT_IYUV,SDL_TEXTUREACCESS_STREAMING,PIXEL_W,PIXEL_H); if(texture==NULL) { SDL_Log("Could not create renderer: %s\n", SDL_GetError()); return; } SDL_Thread *refresh_thread = SDL_CreateThread(RefreshVideo,NULL,NULL); SDL_Event event; // while(true) { SDL_WaitEvent(&event); // if((event.type==SDL_KEYDOWN )&& (event.key.keysym.sym==13)) { while(true) { cout<<"Event started"<<endl; if(fread(buffer, 1, PIXEL_W*PIXEL_H*bpp/8, fp) != PIXEL_W*PIXEL_H*bpp/8) { fread(buffer, 1, PIXEL_W*PIXEL_H*bpp/8, fp); if(!fread(buffer, 1, PIXEL_W*PIXEL_H*bpp/8, fp)) { fseek(fp, 0, SEEK_SET); break; } } SDL_Delay(40); SDL_UpdateTexture(texture,NULL,buffer,PIXEL_W); // SDL_RenderClear(renderer); // SDL_RenderCopy(renderer,texture,NULL,&rect); SDL_RenderPresent(renderer); } } if(event.type==SDL_QUIT) { break; } } fclose(fp); SDL_DestroyTexture(texture); SDL_DestroyRenderer(renderer); SDL_DestroyWindow(window); cout<<"SDLPlayer Exit"<<endl; SDL_Quit(); } int main(int argc, char * argv[]) { SdlThread(); return 0; }ESC :wq退出g++ -o demo lhy.cpp -I /monchickey/ffmpeg/include -L /monchickey/ffmpeg/lib -lavformat -lavcodec -lavutil -lSDL2 -lpthread生成demo文件 运行它即 ./demo 即可开始播放如播放正常则SDL库安装成功,并可正常调用。你看到播放器图片应是这样:四、检验UDP传输 不做要求,了解即可 Linux h264文件传输 将test.h264拷贝到电脑,将服务器端的test.h264文件发送给客户端,客户端接收的文件名为recv.h264。还是老规矩这些都放在一个你知道的文件夹中首先输入ifconfig 查看ip地址,如图所示:记下IP地址。修改下面的client_udp_test.c的SERVER_IP修改为上面IP。创建客户端程序 vim client_udp_test.c#include <stdio.h> #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include <string.h> #define SERVER_IP "192.168.126.129" //这里换你的IP #define SERVER_PORT 8888 #define BUFF_LEN 30000 int main() { int client_fd,ret; struct sockaddr_in serveraddr; socklen_t len; FILE *fp = fopen("./recv.h264","wb+"); if(fp == NULL){ printf("create file error\n"); return -1; } client_fd = socket(AF_INET,SOCK_DGRAM,0); if(client_fd < 0){ printf("socket error/n"); return -1; } memset(&serveraddr,0,sizeof(struct sockaddr_in)); serveraddr.sin_family = AF_INET; serveraddr.sin_addr.s_addr = inet_addr(SERVER_IP); serveraddr.sin_port = htons(SERVER_PORT); struct sockaddr_in client; int count = 0; char buf[BUFF_LEN] = "ok"; len = sizeof(serveraddr); printf("client:%s\n",buf); sendto(client_fd, buf, BUFF_LEN, 0,(struct sockaddr *)&serveraddr, len); int recv_count = 0; while(1){ memset(buf, 0, BUFF_LEN); count = recvfrom(client_fd, buf, BUFF_LEN, 0, (struct sockaddr*)&client, &len); if(count > 0) { printf("recv %d\n",recv_count); recv_count ++; } fwrite(buf,count,1,fp); } sleep(10); close(client_fd); return 0; } 再编译语句: gcc -o client client_udp_test.c 生成client创建服务器程序:在下面程序注释前面修改为你的IP vim server_udp_test.c#include <stdio.h> #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include <string.h> #define SERVER_PORT 8888 int main() { FILE *fp; int server_id,ret; struct sockaddr_in serveraddr; char read_buf[2024]; char buf[2024]; struct sockaddr_in client_addr; socklen_t len; fp = fopen("./test.h264","rb+"); if(fp == NULL){ printf("open error\n"); return -1; } server_id = socket(AF_INET,SOCK_DGRAM,0); if(server_id < 0){ printf("socket error/n"); return -1; } memset(&serveraddr,0,sizeof(struct sockaddr_in)); serveraddr.sin_family = AF_INET; serveraddr.sin_addr.s_addr = inet_addr("192.168.126.129"); //此处修改IP serveraddr.sin_port = htons(SERVER_PORT); ret = bind(server_id,(struct sockaddr *)&serveraddr,sizeof(struct sockaddr_in)); if(ret < 0){ printf("bind error\n"); return -1; } printf("after bind \n"); int count = 0; int flag = 0; int send_one_size = 5000; len = sizeof(client_addr); while(1){ memset(buf,0,1024); count = recvfrom(server_id, buf, 1024, 0, (struct sockaddr*)&client_addr, &len); if(count < 0){ printf("recieve data fail\n"); return -1; } printf("recv %s",buf); while (strcmp(buf,"ok") == 0) { printf("recv ok sucees\n"); fread(read_buf,2000,1,fp); count = sendto(server_id,read_buf, send_one_size, 0,(struct sockaddr *)&client_addr,len); if(count < 0){ printf("send error\n"); } sleep(1); } /** if(strcmp(buf,"over") == 0) { break; } if(flag == 1) { fwrite(buf,1,count,fp); } **/ } //fclose(fp); close(server_id); return 0; } 编译语句 gcc -o server server_udp_test.c 生成server打开两个命令端,先运行server,再运行client,此时在目录中就会生成一个有数据的recv.h264文件!!UDP传输文件工作完成五、摄像头与RTSP环节课题一必做,摄像头如何安装在文章最上面视频中有讲解在安装摄像头之前你应该完成下面工作:1.下载客户端程序-仍然在虚拟机里面自带浏览器中下载本文提供的程序文件2.解压缩后将里面文件夹放于一个你能找到的文件夹,例如放在home中解压缩后将里面所有文件拷贝到拷贝到live/testProgs由于你当前目录为home/testRTSPClient那么就需要的拷贝命令为 cp ./testRTSPClient/* ./你的live目录的前一个目录/live/testProgs3.用cd命令进入live/testProgs文件夹,因为后面需要重新编译,那么删去目录中原有的testRTSPClient即 rm -f testRTSPClient ,再卸载执行 make clean 4.为保证虚拟机与摄像头在同一网段,由于摄像头IP为192.168.1.200,那么不妨将虚拟机IP修改为192.168.1.8那么直接配置ifconfig eth0 192.168.1.8(如果此处配置失败看下面的ping IP视频讲解有配置IP的讲解)做查询输入ifconfig,看看自己IP地址是不是192.168.1.8?5.仍然在live/testProgs目录中 vim testRTSPClient.cpp 将里面的IP改为虚拟机IP,即将SERVER_IP修改为192.168.1.8修改地方如图:将RTSP地址修改为这个地址:rtsp://admin:fang123456@192.168.1.200/h264/ch1/main/av_stream如图位置:6.输入ESC和:wq!保存退出再make一下,注意make失败说明你当前目录错了,要在live/testProgs目录中。7.ls观察是否生成testRTSPClient此时你需要搞明白一共有那三个iP地址?如何ping他们,那么见下面视频课题一和全部做的同学强烈建议看,确保明白后上手摄像头安装!!!8.运行它即 ./testRTSPClient 不能报错进行截图9.用VLC软件观察是否有实时h264流,进行播放观察流地址为rtsp://admin:fang123456@192.168.1.200/h264/ch1/main/av_stream填写到VLC软件中即可观察到如图所示六、解码与播放环节课题三需要做1.liunx系统浏览器输入本站网址sciarm.com下载下面提供的播放器文件2.加压缩防止home文件夹下面即可3.vim include.h4.按a进入编辑将里面的ip修改为你配置过后的虚拟机IP,即将SERVER_IP 里面的IP修改为192.168.1.8,保持并退出5.再输入编译命令 gcc -o demo main.c fun_deco_display.c fun_others.c fun_recv_control.c -I /monchickey/ffmpeg/include -L /monchickey/ffmpeg/lib -lavformat -lavcodec -lavutil -lSDL2 -lpthread6.观察是否有demo文件生成七、最后工作两个TM终端一个在/live/testProgs文件夹中输入 ./testRTSPClient 另一个TM终端在cd到第六步那个文件夹输入 ./demo 观察播放器是否有实时画面出现如图所示:如有哪里不明白可与我联系
2021年12月28日
2,844 阅读
0 评论
33 点赞
2021-09-18
【74LS283】加法器芯片74ls283资料
74ls283是4位二进制超前进位全加器74ls283引脚图及功能74ls283引脚功能 A1–A4运算输入端 B1–B4运算输入端 C0进位输入端 Σ1–Σ4和输出端 C4进位输出端74ls283逻辑功能图 74ls283极限值 电源电压…………………………………………。7V 输入电压 54/74283…………………………………5.5V 54/74LS283…………………………………7V 发射极间电压 54/74283,54/74S283的A和B之间…………5.5V 工作环境温度 54XXX…………………………………-55~125℃ 74XXX…………………………………0~70℃ 存储温度…………………………………………-65~150℃74ls283功能表
2021年09月18日
2,640 阅读
1 评论
2 点赞
2021-08-17
放大电路不同频段耦合电容、旁路电容、极间电容和分布电容的分析方法
耦合电容和旁路电容耦合电容(uF级):下图是一个最简单的阻容耦合型的共射放大电路,C1和C2就是耦合电容,我们都知道电容的作用是通交隔直,因为我们放大的是交流信号,直流信号只是用来确定静态工作点,所以直流信号不会随着放大电路传递下去,稳定了多级放大电路的静态工作点, 旁路电容(uF级):下图的电容器就是旁路电容。交流通路中短路集电结电阻Re,提高电压放大能力。分析方法根据Xc=1/jwC,耦合电容和旁路电容在低频段(取10Hz)|Xc|量级为10kΩ级别,电路中Rb和Re等也都是kΩ级别,因此分析低频段时耦合电容和旁路电路保留;中高频段时,|Xc|量级为Ω甚至mΩ级别,因此容抗特别小,分析中高频段时耦合电容和旁路电容视为短路,因此晶体管频率响应中下限截止频率就来源于耦合电容和旁路电容。 极间电容和分布电容极间电容(pF级)极间电容包括发射结电容和集电结电容,是晶体管所固有的特征。分布电容(pF级)分布电容存在于由两个存在电压差而又相互绝缘的导体间,例如,两根传输线之间,每跟传输线与大地之间,都是被空气介质隔开的,所以,也都存在着电容。一般情况下,这个电容值很小,它的作用可忽略不计,如果传输线很长或所传输的信号频率高时,就必须考虑这电容的作用。分析方法根据Xc=1/jwC,极间电容和分布电容在高频段(取10Mhz)|Xc|量级为10kΩ级别,电路中Rb和Re等也都是kΩ级别,因此分析高频段时极间电容和分布电路保留;中低频段时,|Xc|量级为MΩ级别甚至更大,因此容抗特别大,分析中低频段时极间电容和分布电容视为断路,因此晶体管频率响应中上限截止频率就来源于极间电容和分布电容。
2021年08月17日
769 阅读
0 评论
3 点赞
三极管与场效应管比较及其放大电路比较
三极管与场效应管比较及其放大电路比较:
2021年08月12日
374 阅读
0 评论
2 点赞
【NPN/PNP三极管】放大电路饱和失真和截止失真的区别
放大电路——NPN案例截止失真 由晶体管截止造成的失真,称为截止失真。当Q点过低时,在输入信号负半周靠进峰值的某段时间内,晶体管b-e间电压总量小于其开启电压,此时,晶体管截止,因此,基极电流将产生底部失真。 用图解法分析,晶体管的静态工作点设置较低时,由于输入信号的叠加有可能使叠加后的波形一部分进入截止区,这样就会出现截止失真。NPN型三极管共射极放大电路的截止失真的表现是输出电压的顶部出现削波,PNP型三极管的共射放大电路的截止失真是底部失真。 饱和失真 饱和失真,指的是晶体管因Q点过高,出现的失真。当Q点过高时,虽然基极动态电流为不失真的正弦波,但是由于输入信号正半周靠进峰值的某段时间内晶体管进入饱和区,导致集电极动态电流产生顶部失真,集电极电阻上的电压波形随之产生同样的失真。由于输出电压与集电极电阻上的电压变化相位相反,从而导致输出波形产生底部失真。 实验中NPN放大电路饱和失真和截止失真的区分 输入波形是正半周,输出波形是负半周,近峰值时,输出不再随输入波形变化,就是饱和失真; 输入波形是负半周,输出波形是正半周,近峰值时,输出不再随输入波形变化,就是截止失真。 三极管的输出和输入正好是反过来的,即反相输出。假设输入的是正弦波,静态工作点正好合适,即VQ=Vp-p/2(静态工作点电压是正弦波电压峰峰值的一半),那么当输入的波形是正半周时,输出电压波形正好跟负半周波形是一样的;当输入的波形是负半周时,输出电压波形正好跟正半周波形是一样的。如果VQ大于输入波形的峰峰值的一半,那么当输入的波形是正半周时,快到峰值时,三极管就会处于饱和状态,那么此时的输出就不再随输入变化了,出现了饱和失真;即输出得到的负半周正弦波波形就没有谷底了,我们称之为饱和失真。 对于NPN单管共射放大电路,饱和失真就是输入信号的正半波超过了三极管的放大能力,造成失真,对应的输出波形就是输出波形底部失真,即输出时三极管进入饱和区,Q设置过高。因为饱和失真是输出端失真所以解决饱和失真的方法就是调低静态工作电流Ib(增大Rb),减小Rc,减小β。 反之,当输入的波形是负半周时,快到谷值时,三极管就会处于截止状态,那么此时的输出就不再随输入变化了,出现了截止失真;即输出得到的正半周正弦波波形就没有峰值了,我们称之为截止失真。Q点设置过低造成的截止失真属于输入端失真,所以只能从输入端解决。解决方法有增大VBB,减小输入端电阻Rb。PNP与NPN对比分析NPN型:如果波形Uo顶部失真(变平),则属于截止失真Q小。底部失真代表饱和失真Q大。PNP型:Uo顶部失真代表饱和失真Q大,底部失真是截止失真Q小。(Ube、Uce极性为-导致的)截止失真Q小说明Ib太小,你可以通过减少Rb来增大Ib。如果是饱和失真Q大,说明Ib(Ic)太大,必须用增大Rb来减少失真。后面两点,对于NPN还是PNP都是适用的。截止失真是在输入时产生的,当Q点过低时,输入信号在负半周小于晶体管的开启电压,也就是小于UbeQ,导致晶体管截止而产生失真。饱和失真是在输出时产生的,当Q点过高时,Ib为不失真的正弦波,在输出特性曲线上会发现输入信号正半周靠近峰值的某段时间内晶体管进入饱和区,导致Ic产生顶部失真。
2021年08月08日
13,199 阅读
0 评论
36 点赞
1
...
4
5
6
...
10