活动介绍

u32 Buffer_WriteByte(Buffer_TypeDef *pBuffer, u8 uData) { u32 uResult = FALSE; u32 uDataLength; if (!IS_BUFFER_FULL(pBuffer->uRead, pBuffer->uWrite, pBuffer->uSize)) { uDataLength = pBuffer->uWrite; pBuffer->puMemory[uDataLength] = uData; pBuffer->uWrite++; if (pBuffer->uWrite == pBuffer->uSize) { pBuffer->uWrite = 0; } uResult = TRUE; } return uResult; }

时间: 2024-04-10 10:32:43 浏览: 120
这段代码实现了往缓冲区中写入一个字节的功能。函数 `Buffer_WriteByte` 接受两个参数: - `pBuffer`:指向缓冲区结构体类型 `Buffer_TypeDef` 的指针。 - `uData`:要写入缓冲区的字节数据。 函数返回一个 `u32` 类型的值,表示写入操作是否成功。如果成功写入,则返回 `TRUE`,否则返回 `FALSE`。 具体实现如下: 1. 首先,通过检查缓冲区是否已满(调用 `IS_BUFFER_FULL` 宏),确定缓冲区是否能够容纳新的数据。如果缓冲区未满,则执行写入操作。 2. 将当前写入位置(`pBuffer->uWrite`)保存到变量 `uDataLength` 中。 3. 将要写入的数据 `uData` 存储到缓冲区的对应位置(`pBuffer->puMemory[uDataLength]`)。 4. 增加写入位置 `pBuffer->uWrite` 的值。 5. 如果写入位置已经达到缓冲区的大小(`pBuffer->uSize`),则将写入位置重置为 0,以实现循环写入的效果。 6. 将写入操作的结果设置为 `TRUE`。 7. 返回写入结果。 该函数主要用于向循环缓冲区中写入一个字节的数据,并更新写入位置。如果缓冲区已满,则不会执行写入操作,并返回写入失败。
相关问题

/* USER CODE BEGIN Header */ /** ****************************************************************************** * @file : main.c * @brief : Main program body ****************************************************************************** * @attention * * Copyright (c) 2023 STMicroelectronics. * All rights reserved. * * This software is licensed under terms that can be found in the LICENSE file * in the root directory of this software component. * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ /* USER CODE END Header */ /* Includes ------------------------------------------------------------------*/ #include "main.h" /* Private includes ----------------------------------------------------------*/ /* USER CODE BEGIN Includes */ #include <stdio.h> /* USER CODE END Includes */ /* Private typedef -----------------------------------------------------------*/ /* USER CODE BEGIN PTD */ /* USER CODE END PTD */ /* Private define ------------------------------------------------------------*/ /* USER CODE BEGIN PD */ /* USER CODE END PD */ /* Private macro -------------------------------------------------------------*/ /* USER CODE BEGIN PM */ /* USER CODE END PM */ /* Private variables ---------------------------------------------------------*/ I2C_HandleTypeDef hi2c1; I2C_HandleTypeDef hi2c2; UART_HandleTypeDef huart1; UART_HandleTypeDef huart2; /* USER CODE BEGIN PV */ /* USER CODE END PV */ /* Private function prototypes -----------------------------------------------*/ void SystemClock_Config(void); static void MX_GPIO_Init(void); static void MX_USART2_UART_Init(void); static void MX_I2C2_Init(void); static void MX_USART1_UART_Init(void); static void MX_I2C1_Init(void); /* USER CODE BEGIN PFP */ static void VL6180X_Init(void); //write one byte to VL6180X static HAL_StatusTypeDef VL6180X_WriteByte(uint16_t regAddress, uint8_t data); //read one byte from VL6180X static HAL_StatusTypeDef VL6180X_ReadByte(uint16_t regAddress, uint8_t* data); //get id of VL6180X static uint8_t VL6180X_GetID(void); //get the range from VL6180X static uint8_t VL6180X_GetRange(void); //redirect printf to UART2 int fputc(int ch, FILE* f); void TCS34725_ENABLE(void); uint16_t TCS34725_Get_RawData(I2C_HandleTypeDef *hi2c,uint16_t MemAddress); void Send_RAWData_To_UART(char RGBC, uint16_t RawData); /* USER CODE END PFP */ /* Private user code ---------------------------------------------------------*/ /* USER CODE BEGIN 0 */ /* USER CODE END 0 */ /** * @brief The application entry point. * @retval int */ int main(void) { /* USER CODE BEGIN 1 */ uint8_t buffers; uint8_t buffer[10]={0}; uint16_t Rawdata = 0; /* USER CODE END 1 */ /* MCU Configuration--------------------------------------------------------*/ /* Reset of all peripherals, Initializes the Flash interface and the Systick. */ HAL_Init(); /* USER CODE BEGIN Init */ /* USER CODE END Init */ /* Configure the system clock */ SystemClock_Config(); /* USER CODE BEGIN SysInit */ /* USER CODE END SysInit */ /* Initialize all configured peripherals */ MX_GPIO_Init(); MX_USART2_UART_Init(); MX_I2C2_Init(); MX_USART1_UART_Init(); MX_I2C1_Init(); /* USER CODE BEGIN 2 */ TCS34725_ENABLE(); /* USER CODE END 2 */ /* Infinite loop */ /* USER CODE BEGIN WHILE */ if(VL6180X_GetID()==VL6180X_DEFAULT_ID) VL6180X_Init(); while (1) { HAL_Delay(5000); //¶ÁÈ¡RGBC valid״̬£¬¿´»ý·ÖÊÇ·ñÍê³É HAL_I2C_Mem_Read(&hi2c1,I2C_TCS34725_ADDR<<1|I2C_READ,TCS34725_STATUS_ADDR|TCS34725_CMD_MASK,1,buffer,1,3000); if(buffer[0]&TCS34725__STATUS_AVALID) { //RGBC Valid //Get Clear RAW data Rawdata = TCS34725_Get_RawData(&hi2c1,TCS34725_CDATA_ADDR|TCS34725_CMD_MASK); //Send Clear RAW data to UART Send_RAWData_To_UART('C',Rawdata); //Get RED RAW data Rawdata = TCS34725_Get_RawData(&hi2c1,TCS34725_RDATA_ADDR|TCS34725_CMD_MASK); //Send RED RAW data to UART Send_RAWData_To_UART('R',Rawdata); //Get Green RAW data Rawdata = TCS34725_Get_RawData(&hi2c1,TCS34725_GDATA_ADDR|TCS34725_CMD_MASK); //Send Green RAW data to UART Send_RAWData_To_UART('G',Rawdata); //Get Blue RAW data Rawdata = TCS34725_Get_RawData(&hi2c1,TCS34725_BDATA_ADDR|TCS34725_CMD_MASK); //Send Blue RAW data to UART Send_RAWData_To_UART('B',Rawdata); } while (1) { //get range buffers = VL6180X_GetRange(); printf("Range: %d mm \r\n", buffers); HAL_Delay(3000); /* USER CODE END WHILE */ /* USER CODE BEGIN 3 */ } /* USER CODE END 3 */ } /** * @brief System Clock Configuration * @retval None */ void SystemClock_Config(void) { RCC_OscInitTypeDef RCC_OscInitStruct = {0}; RCC_ClkInitTypeDef RCC_ClkInitStruct = {0}; /** Configure the main internal regulator output voltage */ __HAL_RCC_PWR_CLK_ENABLE(); __HAL_PWR_VOLTAGESCALING_CONFIG(PWR_REGULATOR_VOLTAGE_SCALE1); /** Initializes the RCC Oscillators according to the specified parameters * in the RCC_OscInitTypeDef structure. */ RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSI; RCC_OscInitStruct.HSIState = RCC_HSI_ON; RCC_OscInitStruct.HSICalibrationValue = RCC_HSICALIBRATION_DEFAULT; RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON; RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSI; RCC_OscInitStruct.PLL.PLLM = 16; RCC_OscInitStruct.PLL.PLLN = 336; RCC_OscInitStruct.PLL.PLLP = RCC_PLLP_DIV4; RCC_OscInitStruct.PLL.PLLQ = 4; if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK) { Error_Handler(); } /** Initializes the CPU, AHB and APB buses clocks */ RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2; RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK; RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1; RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2; RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1; if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK) { Error_Handler(); } } /** * @brief I2C1 Initialization Function * @param None * @retval None */ static void MX_I2C1_Init(void) { /* USER CODE BEGIN I2C1_Init 0 */ /* USER CODE END I2C1_Init 0 */ /* USER CODE BEGIN I2C1_Init 1 */ /* USER CODE END I2C1_Init 1 */ hi2c1.Instance = I2C1; hi2c1.Init.ClockSpeed = 100000; hi2c1.Init.DutyCycle = I2C_DUTYCYCLE_2; hi2c1.Init.OwnAddress1 = 0; hi2c1.Init.AddressingMode = I2C_ADDRESSINGMODE_7BIT; hi2c1.Init.DualAddressMode = I2C_DUALADDRESS_DISABLE; hi2c1.Init.OwnAddress2 = 0; hi2c1.Init.GeneralCallMode = I2C_GENERALCALL_DISABLE; hi2c1.Init.NoStretchMode = I2C_NOSTRETCH_DISABLE; if (HAL_I2C_Init(&hi2c1) != HAL_OK) { Error_Handler(); } /* USER CODE BEGIN I2C1_Init 2 */ /* USER CODE END I2C1_Init 2 */ } /** * @brief I2C2 Initialization Function * @param None * @retval None */ static void MX_I2C2_Init(void) { /* USER CODE BEGIN I2C2_Init 0 */ /* USER CODE END I2C2_Init 0 */ /* USER CODE BEGIN I2C2_Init 1 */ /* USER CODE END I2C2_Init 1 */ hi2c2.Instance = I2C2; hi2c2.Init.ClockSpeed = 100000; hi2c2.Init.DutyCycle = I2C_DUTYCYCLE_2; hi2c2.Init.OwnAddress1 = 0; hi2c2.Init.AddressingMode = I2C_ADDRESSINGMODE_7BIT; hi2c2.Init.DualAddressMode = I2C_DUALADDRESS_DISABLE; hi2c2.Init.OwnAddress2 = 0; hi2c2.Init.GeneralCallMode = I2C_GENERALCALL_DISABLE; hi2c2.Init.NoStretchMode = I2C_NOSTRETCH_DISABLE; if (HAL_I2C_Init(&hi2c2) != HAL_OK) { Error_Handler(); } /* USER CODE BEGIN I2C2_Init 2 */ /* USER CODE END I2C2_Init 2 */ } /** * @brief USART1 Initialization Function * @param None * @retval None */ static void MX_USART1_UART_Init(void) { /* USER CODE BEGIN USART1_Init 0 */ /* USER CODE END USART1_Init 0 */ /* USER CODE BEGIN USART1_Init 1 */ /* USER CODE END USART1_Init 1 */ huart1.Instance = USART1; huart1.Init.BaudRate = 115200; huart1.Init.WordLength = UART_WORDLENGTH_8B; huart1.Init.StopBits = UART_STOPBITS_1; huart1.Init.Parity = UART_PARITY_NONE; huart1.Init.Mode = UART_MODE_TX_RX; huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE; huart1.Init.OverSampling = UART_OVERSAMPLING_16; if (HAL_UART_Init(&huart1) != HAL_OK) { Error_Handler(); } /* USER CODE BEGIN USART1_Init 2 */ /* USER CODE END USART1_Init 2 */ } /** * @brief USART2 Initialization Function * @param None * @retval None */ static void MX_USART2_UART_Init(void) { /* USER CODE BEGIN USART2_Init 0 */ /* USER CODE END USART2_Init 0 */ /* USER CODE BEGIN USART2_Init 1 */ /* USER CODE END USART2_Init 1 */ huart2.Instance = USART2; huart2.Init.BaudRate = 115200; huart2.Init.WordLength = UART_WORDLENGTH_8B; huart2.Init.StopBits = UART_STOPBITS_1; huart2.Init.Parity = UART_PARITY_NONE; huart2.Init.Mode = UART_MODE_TX_RX; huart2.Init.HwFlowCtl = UART_HWCONTROL_NONE; huart2.Init.OverSampling = UART_OVERSAMPLING_16; if (HAL_UART_Init(&huart2) != HAL_OK) { Error_Handler(); } /* USER CODE BEGIN USART2_Init 2 */ /* USER CODE END USART2_Init 2 */ } /** * @brief GPIO Initialization Function * @param None * @retval None */ static void MX_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStruct = {0}; /* GPIO Ports Clock Enable */ __HAL_RCC_GPIOA_CLK_ENABLE(); __HAL_RCC_GPIOB_CLK_ENABLE(); /*Configure GPIO pin Output Level */ HAL_GPIO_WritePin(LD2_GPIO_Port, LD2_Pin, GPIO_PIN_RESET); /*Configure GPIO pin : LD2_Pin */ GPIO_InitStruct.Pin = LD2_Pin; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; HAL_GPIO_Init(LD2_GPIO_Port, &GPIO_InitStruct); } /* USER CODE BEGIN 4 */ /** * @brief Init Vl1680x * @param None * @retval None */ void VL6180X_Init(void) { uint8_t data=0; VL6180X_ReadByte(0x016,&data); if(data ==1) { VL6180X_WriteByte(0x0207, 0x01); VL6180X_WriteByte(0x0208, 0x01); VL6180X_WriteByte(0x0096, 0x00); VL6180X_WriteByte(0x0097, 0xfd); VL6180X_WriteByte(0x00e3, 0x01); VL6180X_WriteByte(0x00e4, 0x03); VL6180X_WriteByte(0x00e5, 0x02); VL6180X_WriteByte(0x00e6, 0x01); VL6180X_WriteByte(0x00e7, 0x03); VL6180X_WriteByte(0x00f5, 0x02); VL6180X_WriteByte(0x00d9, 0x05); VL6180X_WriteByte(0x00db, 0xce); VL6180X_WriteByte(0x00dc, 0x03); VL6180X_WriteByte(0x00dd, 0xf8); VL6180X_WriteByte(0x009f, 0x00); VL6180X_WriteByte(0x00a3, 0x3c); VL6180X_WriteByte(0x00b7, 0x00); VL6180X_WriteByte(0x00bb, 0x3c); VL6180X_WriteByte(0x00b2, 0x09); VL6180X_WriteByte(0x00ca, 0x09); VL6180X_WriteByte(0x0198, 0x01); VL6180X_WriteByte(0x01b0, 0x17); VL6180X_WriteByte(0x01ad, 0x00); VL6180X_WriteByte(0x00ff, 0x05); VL6180X_WriteByte(0x0100, 0x05); VL6180X_WriteByte(0x0199, 0x05); VL6180X_WriteByte(0x01a6, 0x1b); VL6180X_WriteByte(0x01ac, 0x3e); VL6180X_WriteByte(0x01a7, 0x1f); VL6180X_WriteByte(0x0030, 0x00); // Recommended : Public registers - See data sheet for more detail VL6180X_WriteByte(0x0011, 0x10); // Enables polling for new Sample ready when measurement completes VL6180X_WriteByte(0x010a, 0x30); // Set the averaging sample period (compromise between lower noise and increased execution time) VL6180X_WriteByte(0x003f, 0x46); // Sets the light and dark gain (upper nibble). Dark gain should not be changed. VL6180X_WriteByte(0x0031, 0xFF); // sets the # of range measurements after which auto calibration of system is performed VL6180X_WriteByte(0x0040, 0x63); // Set ALS integration time to 100ms VL6180X_WriteByte(0x002e, 0x01); // perform a single temperature calibration of the ranging sensor //Optional: Public registers - See data sheet for more detail VL6180X_WriteByte(0x001b, 0x09); // Set default ranging inter-measurement period to 100ms VL6180X_WriteByte(0x003e, 0x31); // Set default ALS inter-measurement period to 500ms VL6180X_WriteByte(0x0014, 0x24); // Configures interrupt on new Sample Ready threshold event? VL6180X_WriteByte(0x0016, 0x00); } } /** * @brief Read one byte to Vl1680x * @param regAddress: Address to Read * @param data: Point to the memory for data * @retval HAL_StatusTypeDef */ HAL_StatusTypeDef VL6180X_ReadByte(uint16_t regAddress, uint8_t* data) { uint16_t DevAddress=0; DevAddress = (VL6180X_DEFAULT_I2C_ADDR<<1)|1; return HAL_I2C_Mem_Read(&hi2c2, DevAddress, regAddress, VL6180X_REG_ADDRESS_SIZE, data,1,30); } /** * @brief Write one byte to Vl1680x * @param regAddress: Address to write * @param data: Data to be writed * @retval HAL_StatusTypeDef */ HAL_StatusTypeDef VL6180X_WriteByte(uint16_t regAddress, uint8_t data) { uint16_t DevAddress=0; DevAddress = (VL6180X_DEFAULT_I2C_ADDR<<1)&0xFE; return HAL_I2C_Mem_Write(&hi2c2, DevAddress, regAddress, VL6180X_REG_ADDRESS_SIZE, &data,1,30); } /** * @brief Get ID of Vl1680x * @param None * @retval Vl1680x's ID */ uint8_t VL6180X_GetID(void) { uint8_t ID; if(VL6180X_ReadByte( VL6180X_REG_ID, &ID)) ID =0;//get fail, return 0 return ID; } /** * @brief Get result of range * @param None * @retval Result of range */ uint8_t VL6180X_GetRange(void) { uint8_t status,range; //get status VL6180X_ReadByte( VL6180X_REG_RESULT_RANGE_STATUS, &status); while(!(status&0x01)) { //Not ready VL6180X_ReadByte( VL6180X_REG_RESULT_RANGE_STATUS, &status); HAL_Delay(10); } //start range by write 0x01 to SYSRANGE_START Register(0x18) range=1; VL6180X_WriteByte(VL6180X_REG_SYSRANGE_START,range); //Wait for ranging completion VL6180X_ReadByte( VL6180X_REG_RESULT_INTERRUPT_STATUS, &status); while(!(status&0x04)) { VL6180X_ReadByte( VL6180X_REG_RESULT_INTERRUPT_STATUS, &status); HAL_Delay(100); } //get range VL6180X_ReadByte(VL6180X_REG_RESULT_RANGE_VAL,&range); //clear interrupt status VL6180X_WriteByte(VL6180X_REG_SYSTERM_INTERRUPT_CLEAR,0x07); return range; } /** * @brief Redirect fpuc Function to UART2 * @param ch: Data to be send * @param f: Not used * @retval None */ int fputc(int ch, FILE* f) { //HAL¿âÐÎʽ //HAL_UART_Transmit(&huart1,(uint8_t*)&ch,1,0xffff); //UART register begin while( (huart1.Instance->SR & 0x40) == 0 ); huart1.Instance->DR = ch; //UART register end return ch; } /* USER CODE END 4 */ /** * @brief This function is executed in case of error occurrence. * @retval None */ void Error_Handler(void) { /* USER CODE BEGIN Error_Handler_Debug */ /* User can add his own implementation to report the HAL error return state */ __disable_irq(); while (1) { } /* USER CODE END Error_Handler_Debug */ } #ifdef USE_FULL_ASSERT /** * @brief Reports the name of the source file and the source line number * where the assert_param error has occurred. * @param file: pointer to the source file name * @param line: assert_param error line source number * @retval None */ void assert_failed(uint8_t *file, uint32_t line) { /* USER CODE BEGIN 6 */ /* User can add his own implementation to report the file name and line number, ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */ /* USER CODE END 6 */ } #endif /* USE_FULL_ASSERT */

<think>嗯,用户想要关于VL6180X和TCS34725传感器的I2C初始化及使用代码示例。首先,我需要回忆一下这两个传感器的基本信息和I2C通信的一般步骤。VL6180X是ST的测距传感器,而TCS34725是颜色传感器,两者都使用I2C接口。用户之前提到学校发的模块可能有点难用,所以代码示例需要尽量清晰,可能还要注意常见问题。 首先,I2C初始化的步骤通常包括设置I2C总线频率、扫描设备地址、初始化特定寄存器等。对于VL6180X,可能需要配置测距模式和相关参数。TCS34725则需要设置积分时间、增益等。我需要确保代码示例正确,比如使用正确的寄存器地址和配置值。 然后,用户可能是在使用Arduino或者类似的平台,所以代码示例可能需要基于Arduino库。比如,VL6180X常用的库是Pololu的,而TCS34725常用Adafruit的库。需要检查这些库的安装和基本用法。 另外,用户可能遇到I2C地址冲突的问题,尤其是同时使用多个传感器时。VL6180X的默认地址是0x29,TCS34725的默认地址是0x29吗?不对,TCS34725的地址是0x29,而VL6180X的默认地址也是0x29,但可以通过改变地址来避免冲突。所以代码中可能需要修改其中一个的地址,或者在初始化时分时使用。 还要考虑初始化顺序,比如VL6180X需要先解除芯片的休眠状态,可能需要发送特定的命令。TCS34725则需要开启RGB转换和使能传感器。 另外,用户可能需要示例代码的结构,比如初始化函数、读取数据的函数,以及主循环中的调用方式。确保代码语法正确,比如正确的Wire库的使用,正确的寄存器写入方法。 可能还需要提到常见的错误,比如接线错误(SDA和SCL接错)、上拉电阻的问题、地址设置错误等。用户可能在使用中遇到这些问题,所以在回答中需要包含一些调试建议。 最后,根据用户之前的引用,他们可能在使用RAGFlow配置API密钥,但这里主要关注的是传感器代码,所以需要专注于I2C通信的实现,而不是API配置。需要确保回答符合用户的实际需求,提供可直接使用的代码示例,并解释关键步骤。</think>### VL6180X和TCS34725传感器I²C初始化及使用代码示例 #### 1. I²C基础配置(Arduino平台) ```cpp #include <Wire.h> #define VL6180X_ADDR 0x29 #define TCS34725_ADDR 0x29 void setup() { Wire.begin(); // 初始化I²C总线 Serial.begin(115200); // 检查设备是否存在 checkI2CDevice(VL6180X_ADDR, "VL6180X"); checkI2CDevice(TCS34725_ADDR, "TCS34725"); } void checkI2CDevice(uint8_t addr, String name) { Wire.beginTransmission(addr); if (Wire.endTransmission() == 0) { Serial.print(name + " found at 0x"); Serial.println(addr, HEX); } else { Serial.println(name + " not detected!"); } } ``` #### 2. VL6180X初始化代码 ```cpp void initVL6180X() { // 解除芯片休眠 writeRegister(VL6180X_ADDR, 0x0210, 0x01); delay(10); // 设置测距模式 writeRegister(VL6180X_ADDR, 0x0111, 0x10); // 配置校准参数 writeRegister(VL6180X_ADDR, 0x0247, 0x04); } uint16_t readVL6180X() { writeRegister(VL6180X_ADDR, 0x0018, 0x01); // 触发测距 delay(10); return readRegister(VL6180X_ADDR, 0x0062); } ``` #### 3. TCS34725初始化代码 ```cpp void initTCS34725() { // 设置积分时间=700ms writeRegister(TCS34725_ADDR, 0x01, 0x00); // 开启RGB转换和使能 writeRegister(TCS34725_ADDR, 0x00, 0x03); delay(700); // 等待首次转换完成 } struct RGBData { uint16_t r, g, b, c; }; RGBData readTCS34725() { RGBData data; Wire.beginTransmission(TCS34725_ADDR); Wire.write(0xB6); // 颜色数据起始寄存器 Wire.endTransmission(); Wire.requestFrom(TCS34725_ADDR, 8); data.c = Wire.read() | (Wire.read() << 8); data.r = Wire.read() | (Wire.read() << 8); data.g = Wire.read() | (Wire.read() << 8); data.b = Wire.read() | (Wire.read() << 8); return data; } ``` #### 4. 通用I²C读写函数 ```cpp void writeRegister(uint8_t addr, uint16_t reg, uint8_t value) { Wire.beginTransmission(addr); Wire.write((uint8_t)(reg >> 8)); // 高字节 Wire.write((uint8_t)(reg & 0xFF)); // 低字节 Wire.write(value); Wire.endTransmission(); } uint16_t readRegister(uint8_t addr, uint16_t reg) { Wire.beginTransmission(addr); Wire.write((uint8_t)(reg >> 8)); Wire.write((uint8_t)(reg & 0xFF)); Wire.endTransmission(); Wire.requestFrom(addr, 2); return Wire.read() | (Wire.read() << 8); } ``` #### 关键注意事项: 1. 地址冲突处理:两个传感器默认地址均为$0x29$,建议通过VL6180X的地址修改寄存器$0x0212$改变其地址[^1] 2. 时序要求:TCS34725积分时间结束后才能读取有效数据 3. 电源管理:VL6180X需要先解除休眠状态才能配置参数 4. 数据校准:需根据实际应用场景进行白平衡校准(TCS34725)和距离补偿(VL6180X)

void AD5933_FreInit( float Fre,float AddFre) //启动一个频率点的扫描 { u32 StartHz; AD5933_WriteByte( 0x80, 0xA0|data_Vpp|data_gain ); if( Fre != 0 ) { StartHz = (u32)(Fre*4.0*134217728.0/AD5933Fre); data_StartH = (StartHz>>16)&0xFF; data_StartM = (StartHz>>8)&0xFF; data_StartL = (StartHz)&0xFF; } AD5933_WriteByte( 0x83, data_StartM ); //起始频率,Fmclk=16.776MHz AD5933_WriteByte( 0x82, data_StartH ); //start frequency, AD5933_WriteByte( 0x84, data_StartL ); //100kHz=0x30D4C8 if( AddFre != 0 ) { StartHz = (u32)(AddFre*4.0*134217728.0/AD5933Fre); data_AddH = (StartHz>>16)&0xFF; data_AddM = (StartHz>>8)&0xFF; data_AddL = (StartHz)&0xFF; } else { data_AddH = 0; data_AddM = 0; data_AddL = 0; } AD5933_WriteByte( 0x85, data_AddH ); //频率增量 AD5933_WriteByte( 0x86, data_AddM ); AD5933_WriteByte( 0x87, data_AddL ); AD5933_WriteByte( 0x88, data_CountFH ); //测量点数或频率个数 AD5933_WriteByte( 0x89, data_CountFL ); AD5933_WriteByte( 0x8A, data_CountTH ); //等待建立周期数 AD5933_WriteByte( 0x8B, data_CountTL ); AD5933_WriteByte( 0x80, 0xB0|data_Vpp|data_gain ); AD5933_WriteByte( 0x81, 0x10|data_clk ); //复位 delay_ms(150); //复位延时 AD5933_WriteByte( 0x81, 0x00|data_clk ); AD5933_WriteByte( 0x80, 0x10|data_Vpp|data_gain ); /*以起始频率扫描*/ delay_ms(30); AD5933_WriteByte( 0x80, 0x20|data_Vpp|data_gain ); /*启动频率扫描*/ }

<think>好的,我现在需要详细分析AD5933_FreInit函数的实现细节和工作原理,特别是关于频率扫描初始化的流程。首先,我需要了解AD5933芯片的基本功能和频率扫描的相关参数。根据引用[2]中的信息,AD5933可能涉及使用不同位宽的数据类型,比如u8(8位无符号)和s16(16位有符号),这可能与频率参数设置有关。 接下来,查阅AD5933的数据手册是关键。通常,频率扫描初始化需要配置起始频率、频率增量、步进数等参数。这些参数可能需要通过I2C或SPI接口写入芯片的寄存器。例如,起始频率可能由两个寄存器组成,分别存储高位和低位字节。假设起始频率计算公式为$f_{\text{start}} = \frac{\text{MCLK}}{2^{23}} \times \text{CODE}$,其中CODE是24位的值,分三个8位寄存器写入。 然后,函数可能涉及设置控制寄存器,启动扫描模式。引用[3]提到switch语句可能用于处理不同的工作模式或错误状态。例如,根据不同的频率范围选择不同的时钟分频参数。 在代码实现中,需要初始化相关变量,如起始频率、步长、步数,并转换为寄存器值。例如,计算CODE值后,拆分字节并通过I2C发送。同时,可能设置控制寄存器为频率扫描模式,并启用中断或轮询状态位。 需要注意数据类型转换,如将浮点频率转换为整数寄存器值,可能涉及位移和掩码操作。此外,校准流程可能影响初始化,例如在每次扫描前进行阻抗校准。 最后,验证函数是否包含错误处理,例如检查I2C传输是否成功,或寄存器配置是否有效。引用[1]提到的栈初始化可能与缓冲区管理有关,但需要更多上下文确认。 总结步骤:配置频率参数(起始、步长、步数)→ 转换为寄存器值 → 写入寄存器 → 设置控制模式 → 错误处理。需要结合具体代码和芯片手册验证。</think>### AD5933_FreInit函数详解 AD5933是一款基于阻抗测量的芯片,其频率扫描初始化函数`AD5933_FreInit`主要用于配置扫描参数并启动测量流程。以下是关键实现细节: #### 1. **频率参数配置** - **起始频率**:需计算24位频率编码值并拆分为三个8位寄存器。公式为: $$ \text{CODE} = \frac{f_{\text{start}} \times 2^{23}}{\text{MCLK}} $$ 其中MCLK为系统主时钟频率(如16.776 MHz)[^2]。 - **频率增量**:步长对应的编码值,同样拆分写入寄存器。 - **步数**:扫描点数(例如128点),需写入步数寄存器。 #### 2. **寄存器写入流程** ```c // 示例代码片段:起始频率配置 uint32_t code = (uint32_t)((start_freq * pow(2, 23)) / MCLK); uint8_t reg1 = (code >> 16) & 0xFF; // 高位 uint8_t reg2 = (code >> 8) & 0xFF; uint8_t reg3 = code & 0xFF; I2C_Write(AD5933_ADDR, 0x82, reg1); // 假设0x82为起始频率高位寄存器地址 I2C_Write(AD5933_ADDR, 0x83, reg2); I2C_Write(AD5933_ADDR, 0x84, reg3); ``` #### 3. **控制寄存器设置** - **工作模式**:设置为频率扫描模式(如控制寄存器0x80写入0x10)。 - **量程与增益**:根据信号幅值选择 PGA 增益(如x1或x5)。 #### 4. **校准与启动** - 先发送初始化命令(`0x10`),再发送开始扫描命令(`0x20`)[^3]。 - 校准阶段需测量已知阻抗以计算系统增益。 #### 5. **数据类型处理** 引用[2]指出,频率参数可能使用`s16`类型数组存储实部/虚部数据,而寄存器操作用`u8`类型: ```c s16 Real_Data[128]; // 存储扫描结果实部 u8 status = Read_Status_Register(); // 读取状态寄存器 ```
阅读全文

相关推荐

#include <REGX52.H> sbit DS1302_SCLK = P3^6; sbit DS1302_IO = P3^4; sbit DS1302_CE = P3^5; #define DS1302_SECOND 0x80 #define DS1302_MINUTE 0x82 #define DS1302_HOUR 0x84 #define DS1302_DATE 0x86 #define DS1302_MONTH 0x88 #define DS1302_DAY 0x8A #define DS1302_YEAR 0x8C #define DS1302_WP 0x8E //时间数组,索引0~6分别为年、月、日、时、分、秒、星期,设置为有符号的便于<0的判断 unsigned char DS1302_Time[] = {25,3,31,15,34,0,1}; void DS1302_Init() { DS1302_CE = 0; DS1302_SCLK = 0; } void DS1302_WriteByte(unsigned char command,unsigned char input) { DS1302_CE = 1; unsigned char i = 0; for(i = 0;i < 8;i++) { DS1302_IO = command & (0x01 << i); DS1302_SCLK = 1; DS1302_SCLK = 0; } for(i = 0;i < 8;i++) { DS1302_IO = input & (0x01 << i); DS1302_SCLK = 1; DS1302_SCLK = 0; } DS1302_CE = 0; } unsigned char DS1302_ReadByte(unsigned char command) { command |= 0x01;//输入的命令字的地址 unsigned char i,date=0; DS1302_CE = 1; for(i = 0;i < 8;i++) { DS1302_IO = command & (0x01 << i); DS1302_SCLK = 0; DS1302_SCLK = 1; } for(i = 0;i < 8;i++) { DS1302_SCLK = 1; if(DS1302_IO) { date |= (0x01 << i); } DS1302_SCLK = 1; DS1302_SCLK = 0; } DS1302_CE = 0; DS1302_IO = 1; return date; } void DS1302_SetTime() { DS1302_WriteByte(DS1302_WP,0x00); DS1302_WriteByte(DS1302_YEAR,DS1302_Time[0]/10*16+DS1302_Time[0]%10);//十进制转BCD码后写入 DS1302_WriteByte(DS1302_MONTH,DS1302_Time[1]/10*16+DS1302_Time[1]%10); DS1302_WriteByte(DS1302_DATE,DS1302_Time[2]/10*16+DS1302_Time[2]%10); DS1302_WriteByte(DS1302_HOUR,DS1302_Time[3]/10*16+DS1302_Time[3]%10); DS1302_WriteByte(DS1302_MINUTE,DS1302_Time[4]/10*16+DS1302_Time[4]%10); DS1302_WriteByte(DS1302_SECOND,DS1302_Time[5]/10*16+DS1302_Time[5]%10); DS1302_WriteByte(DS1302_DAY,DS1302_Time[6]/10*16+DS1302_Time[6]%10); DS1302_WriteByte(DS1302_WP,0x80); } void DS1302_ReadTime() { unsigned char Temp; Temp=DS1302_ReadByte(DS1302_YEAR); DS1302_Time[0]=Temp/16*10+Temp%16;//BCD码转十进制后读取 Temp=DS1302_ReadByte(DS1302_MONTH); DS1302_Time[1]=Temp

#include <reg51.h> #include <intrins.h> sbit SDA = P2^0; // I2C数据线 sbit SCL = P2^1; // I2C时钟线 #define PCA9685_ADDR 0x40 /* 全局变量 - 舵机控制相关 */ volatile unsigned long sys_time = 0; // 系统时间计数器(单位:ms) volatile bit timer_flag = 0; // 定时器溢出标志 unsigned long servo_start_time[4]; // 舵机动作开始时间 volatile unsigned char servo_active = 0; // 舵机激活状态 volatile unsigned char servo_target_angle[4] = {0}; // 目标角度缓冲区 unsigned char UART_Data = 0; /************************ I2C通信协议 ************************/ void I2C_Delay() { _nop_(); _nop_(); _nop_(); _nop_(); } void I2C_Start() { SDA = 1; I2C_Delay(); SCL = 1; I2C_Delay(); SDA = 0; I2C_Delay(); SCL = 0; I2C_Delay(); } void I2C_Stop() { SDA = 0; I2C_Delay(); SCL = 1; I2C_Delay(); SDA = 1; I2C_Delay(); } void I2C_WriteByte(unsigned char dat) { unsigned char i; for(i = 0; i < 8; i++) { SDA = (dat & 0x80) ? 1 : 0; dat <<= 1; SCL = 1; I2C_Delay(); SCL = 0; I2C_Delay(); } SDA = 1; SCL = 1; I2C_Delay(); SCL = 0; I2C_Delay(); } /************************ PCA9685驱动 ************************/ void PCA9685_Write(unsigned char reg, unsigned char dat) { I2C_Start(); I2C_WriteByte(PCA9685_ADDR << 1); I2C_WriteByte(reg); I2C_WriteByte(dat); I2C_Stop(); } void PCA9685_Init() { unsigned char ch; PCA9685_Write(0x00, 0x20); // 进入睡眠模式 PCA9685_Write(0xFE, 121); // 预分频值(50Hz,25MHz时钟) PCA9685_Write(0x00, 0xA0); // 退出睡眠模式 for (ch = 0; ch < 4; ch++) { PCA9685_Write(0x06 + 4*ch, 0x00); PCA9685_Write(0x07 + 4*ch, 0x00); PCA9685_Write(0x08 + 4*ch, 0x00); PCA9685_Write(0x09 + 4*ch, 0x00); } } void Set_PWM(unsigned char ch, unsigned int on, unsigned int off) { I2C_Start(); I2C_WriteByte(PCA9685_ADDR << 1); I2C_WriteByte(0x06 + 4*ch); I2C_WriteByte(on & 0xFF); I2C_WriteByte(on >> 8); I2C_WriteByte(off & 0xFF); I2C_WriteByte(off >> 8); I2C_Stop(); } /************************ 舵机控制函数(修正后) ************************/ void Servo_SetAngle(unsigned char ch, unsigned char angle) { // 脉冲范围:0.5ms~2.5ms(适配常见180度舵机) float pulse_min = 0.5; // 0度对应0.5ms float pulse_max = 2.5; // 180度对应2.5ms float pulse = pulse_min + (angle / 180.0) * (pulse_max - pulse_min); // 计算OFF值(周期20ms,4096步长) unsigned int off = (unsigned int)(pulse * 4096 / 20); Set_PWM(ch, 0, off); } /************************ 定时器T0初始化 ************************/ void Timer0_Init() { TMOD |= 0x01; // 定时器0模式1 TH0 = 0xFC; // 1ms定时(11.0592MHz) TL0 = 0x67; ET0 = 1; TR0 = 1; } /************************ 定时器T0中断服务 ************************/ void Timer0_ISR() interrupt 1 { TH0 = 0xFC; TL0 = 0x67; sys_time++; timer_flag = 1; } /************************ 串口通信 ************************/ void UART_Init() { TMOD = 0x21; TH1 = 0xFD; TL1 = 0xFD; TR1 = 1; SCON = 0x50; ES = 1; EA = 1; } void UART_ISR() interrupt 4 { if (RI) { RI = 0; UART_Data = SBUF; if (UART_Data >= '1' && UART_Data <= '4') { unsigned char ch = UART_Data - '1'; servo_target_angle[ch] = 90; // 设置目标角度为90度 servo_start_time[ch] = sys_time; servo_active |= (1 << ch); } } } /************************ 主函数 ************************/ void main() { unsigned char i; // 初始化 servo_active = 0; for(i=0; i<4; i++) { servo_start_time[i] = 0; servo_target_angle[i] = 0; } UART_Init(); PCA9685_Init(); Timer0_Init(); while(1) { if(timer_flag) { timer_flag = 0; // 处理舵机角度设置 for(i=0; i<4; i++) { if(servo_target_angle[i] != 0) { Servo_SetAngle(i, servo_target_angle[i]); servo_target_angle[i] = 0; } } // 检查超时并回零 for(i=0; i<4; i++) { if(servo_active & (1 << i)) { if((sys_time - servo_start_time[i]) >= 3000) { Servo_SetAngle(i, 0); // 转回0度 servo_active &= ~(1 << i); } } } } } }根据这个代码修改

给以下代码添加注释#include <reg52.h> #include <intrins.h> #define u8 unsigned char #define u16 unsigned int #define DECODE_MODE 0x09 #define INTENSITY 0x0A #define SCAN_LIMIT 0x0B #define SHUT_DOWN 0x0C #define DISPLAY_TEST 0x0F #define BLOCKS 4 sbit MAX7219_CLK = P2^2; sbit MAX7219_CS = P2^1; sbit MAX7219_DIN = P2^0; u8 code bytes[] = { 0x3e,0x63,0x63,0x7f,0x63,0x63,0x63,0x63, //A 0x7e,0x63,0x63,0x7e,0x63,0x63,0x63,0x7e, //B 0x3e,0x63,0x63,0x60,0x60,0x63,0x63,0x3e, //C }; u8 val[BLOCKS]; u8 character_len = sizeof(bytes) / 8; void delay(u16 x) { u16 i,j; for(i = 0; i < x; i++) for(j = 0;j < 112; j++); } void Max7219_writeByte(u8 dat) { u8 i; MAX7219_CS = 0; for(i = 8; i >= 1; i--) { MAX7219_CLK = 0; MAX7219_DIN = dat & 0x80; // &10000000, 取最高位 dat = dat << 1; MAX7219_CLK = 1; } } void Max7219_singeWrite(u8 index, u8 addr, u8 dat) { MAX7219_CS = 0; Max7219_writeByte(addr); Max7219_writeByte(dat); while(index--) { Max7219_writeByte(0x00); Max7219_writeByte(0x00); } MAX7219_CS = 1; } void Max7219_multiWrite(u8 addr, u8 len, u8* dat) { MAX7219_CS = 0; while(len--) { Max7219_writeByte(addr); Max7219_writeByte(*dat++); } MAX7219_CS = 1; } void Max7219_init(void) { u8 i; for (i = 0; i < BLOCKS; i++) { Max7219_singeWrite(i, SHUT_DOWN, 0x01); // 0x00:shutdown, 0x01:normal Max7219_singeWrite(i, DECODE_MODE, 0x00); // No decode Max7219_singeWrite(i, INTENSITY, 0x03); // 0x00:min, 0x0F:max Max7219_singeWrite(i, SCAN_LIMIT, 0x07); // Display 8 digits Max7219_singeWrite(i, DISPLAY_TEST, 0x00); // 0x00:normal, 0x01:test mode } }

#include <STC89C5xRC.H> #include "I2C.h" #define AT24C02_ADDRESS void AT24c02_WriteByte(unsigned char WordAddress,Date) { unsigned char Ack; I2C_Start(); I2C_SendByte(AT24C02_ADDRESS); Ack=I2C_ReceiveAck(); if(Ack==0)P2=0X00; } void AT24c02_ReadByte(unsigned char WordAddress) { unsigned char Date; return Date; } #ifndef __AT24C02_H__ #define __AT24C02_H__ void AT24c02_WriteByte(unsigned char WordAddress,Date); void AT24c02_ReadByte(unsigned char WordAddress); #endif #include <STC89C5xRC.H> #include "Delay.h" #include "Key.h" #include "LCD1602.h" #include "AT24C02.h" void main() { LCD1602_Init(); LCD1602_ShowString(1,1,"Hello"); AT24c02_WriteByte(0,0); while(1) { } } #include <STC89C5xRC.H> sbit I2C_SCL=P2^1; sbit I2C_SDA=P2^0; void I2C_Start(void) { I2C_SDA=1; I2C_SCL=1; I2C_SDA=0; I2C_SCL=0; } void I2C_Stop(void) { I2C_SDA=0; I2C_SCL=1; I2C_SDA=1; } void I2C_SendByte(unsigned char Byte) { unsigned char i; for(i=0;i<8;i++) { I2C_SendByte=Byte&(0X80>>i); I2C_SCL=1; I2C_SCL=0; } } unsigned char I2C_ReceiveByte(void) { unsigned char i, Byte=0x00; I2C_SDA=1; for(i=0;i<8;i++) { I2C_SCL=1; if(I2C_SDA){Byte|0X80>>i;} I2C_SCL=0; } return Byte; } void I2C_SendAck(unsigned char I2C_bit) { I2C_SDA=I2C_bit; I2C_SCL=1; I2C_SCL=0; } unsigned char I2C_ReceiveAck(void) { unsigned char Byte,I2C_bit; I2C_SDA=1; I2C_SCL=1; I2C_bit=I2C_SDA; return Byte; } #ifndef __I2C_H__ #define __I2C_H__ void I2C_Start(void); void I2C_Stop(void); void I2C_SendByte(unsigned char Byte); unsigned char I2C_ReceiveByte(void); void I2C_SendAck(bit); unsigned char I2C_ReceiveAck(void); #endifBuild started: Project: AT24C02 Build target 'Target 1' compiling AT24c02.c... linking... *** WARNING L16: UNCALLED SEGMENT, IGNORED FOR OVERLAY PROCESS SEGMENT: ?PR?KEYNUMBER?KEY *** WARNING L16: UNCALLED SEGMENT, IGNORED FOR OVERLAY PROCESS SEGMENT: ?PR?_LCD1602_SHOWCHAR?LCD1602 *** WARNING L16: UNCALLED SEGMENT, IGNORED FOR OVERLAY PROCESS SEGMENT: ?PR?_LCD1602_SHOWNUM?LCD1602 *** WARNING L16: UNCALLED SEGMENT, IGNORED FOR OVERLAY PROCESS SEGMENT: ?PR?_LCD1602_SHOWSIGNEDNUM?LCD1602 *** WARNING L16: UNCALLED SEGMENT, IGNORED FOR OVERLAY PROCESS SEGMENT: ?PR?_LCD1602_SHOWHEXNUM?LCD1602 *** WARNING L16: UNCALLED SEGMENT, IGNORED FOR OVERLAY PROCESS SEGMENT: ?PR?_LCD1602_SHOWBINNUM?LCD1602 *** WARNING L16: UNCALLED SEGMENT, IGNORED FOR OVERLAY PROCESS SEGMENT: ?PR?_AT24C02_READBYTE?AT24C02 *** WARNING L16: UNCALLED SEGMENT, IGNORED FOR OVERLAY PROCESS SEGMENT: ?PR?_I2C_SENDACK?I2C *** WARNING L1: UNRESOLVED EXTERNAL SYMBOL SYMBOL: I2C_SENDACK MODULE: .\Objects\AT24c02.obj (AT24C02) *** WARNING L1: UNRESOLVED EXTERNAL SYMBOL SYMBOL: ?I2C_SENDACK?BIT MODULE: .\Objects\AT24c02.obj (AT24C02) *** WARNING L2: REFERENCE MADE TO UNRESOLVED EXTERNAL SYMBOL: ?I2C_SENDACK?BIT MODULE: .\Objects\AT24c02.obj (AT24C02) ADDRESS: 01FCH *** WARNING L2: REFERENCE MADE TO UNRESOLVED EXTERNAL SYMBOL: I2C_SENDACK MODULE: .\Objects\AT24c02.obj (AT24C02) ADDRESS: 01FEH Program Size: data=42.1 xdata=0 code=897 ".\Objects\AT24C02" - 0 Error(s), 12 Warning(s). Build Time Elapsed: 00:00:01

Main主程序 #include "sys.h" #include "delay.h" #include "adc.h" #include "gpio.h" #include "OLED_I2C.h" #include "stmflash.h" #include "ds18b20.h" #include "timer.h" #include "usart1.h" #include <stdio.h> #include <stdlib.h> #include <string.h> #include <stdbool.h> #define FLASH_SAVE_ADDR ((u32)0x0800F000) //设置FLASH 保存地址(必须为偶数) #define STM32_RX1_BUF Usart1RecBuf #define STM32_Rx1Counter RxCounter #define STM32_RX1BUFF_SIZE USART1_RXBUFF_SIZE extern const unsigned char BMP1[]; extern const unsigned char BMP2[]; char display[16]; //显示缓存区 short temperature=0; //温度 u8 setTempValue = 40; //温度上限 u8 setSmokeValue = 60; //烟雾上限 u8 sendSmsFlag = 0; //发送短信标志 u8 alarmFlag = 0x00; //蜂鸣器报警标志 u16 smoke=0; //烟雾 u8 setn=0; //记录设置按键按下的次数 bool shanshuo=0; bool shuaxin=0; bool fangdao=0; char PhoneNumber[11];//手机号码 char uniPhoneNum[44];//手机号码转码后存放数组 void UsartRx1BufClear(void) { memset(STM32_RX1_BUF, 0, STM32_RX1BUFF_SIZE);//清除缓存 STM32_Rx1Counter = 0; } /***********************************************************************************************/ /****************************** 以下为SIM800相关部分 ****************************************/ /***********************************************************************************************/ void PhoneNumChangeUnicode(char *str1,char *str2) //手机号转码 { u8 i; char *buf = str1; for(i = 0;i < 11;i ++) //发送中文短信手机号必须转码,前面需要加上003 { str2[i*4+0] = '0'; str2[i*4+1] = '0'; str2[i*4+2] = '3'; str2[i*4+3] = buf[i]; } } void gsm_atcmd_send(char *at)//GSM AT指令发送函数 { unsigned short waittry;//延时变量 do { gsm_rev_start = 0;//接收开始标志清零 gsm_rev_okflag = 0;//接收完成标志清零 waittry = 0;//延时变量清零 uart1_send((unsigned char *)at,0xFF);//串口发送内容 while(waittry ++ < 3000)//进入while延时 { if (gsm_rev_okflag == 1)//等待GSM返回OK { return;//返回出去 } delay_ms(1); } } while(gsm_rev_okflag == 0); } void gsm_init(void)//gsm初始化 { gsm_atcmd_send("AT\r\n");//测试用的 delay_ms(1000); gsm_atcmd_send("AT+CSCS=\"UCS2\"\r\n");//设置为unicode编码 delay_ms(1000); gsm_atcmd_send("AT+CMGF=1\r\n");//设置为文本模式 delay_ms(1000); gsm_atcmd_send("AT+CNMI=2,1\r\n");//来短信时提示,并存储到模块内存 delay_ms(1000); gsm_atcmd_send("AT+CMGD=1,4\r\n");//清除短信 delay_ms(1000); gsm_atcmd_send("AT+CSMP=17,0,2,25\r\n");//设置短信保留为5分钟,发送中文 } /* *number 为对方手机号 */ void gsm_send_msg(const char*number,char * content) { u8 len; unsigned char gsm_at_txbuf[60];//GSM AT命令缓存区 memset(gsm_at_txbuf, 0, 60);//缓存清零 strncpy((char *)gsm_at_txbuf,"AT+CMGS=\"",9);//将AT+CMGS=\",复制到gsm_at_txbuf memcpy(gsm_at_txbuf + 9, number, 44);//将手机号码复制到AT+CMGS=\"之后 len = strlen((char *)gsm_at_txbuf);//获取gsm_at_txbuf字符串长度 gsm_at_txbuf[len] = '"'; // AT+CMGS=\"12345678901\" gsm_at_txbuf[len + 1] = '\r'; gsm_at_txbuf[len + 2] = '\n';//gsm_at_txbuf最终的格式"AT+CMGS=\"手机号码\"\r\n" uart1_send(gsm_at_txbuf,0xFF);//发送需要接受短信的手机号码 delay_ms(1000); uart1_send((unsigned char *)content,0xFF); //发短信内容 delay_ms(100); printf("%c",0x1a); //发送结束符号 delay_ms(10); } /* *content 为短信内容 */ void sim800_send_sms(char *content) { bool send_error = 0; u16 send_count = 0; gsm_rev_okflag = 0; OLED_ShowStr(32,2,"Send Sms... ",2,0); PhoneNumChangeUnicode(PhoneNumber,uniPhoneNum); //在发送短信前,先将手机号转码 gsm_send_msg(uniPhoneNum,content);//发送短信 delay_ms(1000);//延时1秒 while(gsm_rev_okflag == 0)//等待返回OK指令 { if(send_count++ > 8000) { send_count = 0; send_error = 1; break; } delay_ms(1); }; gsm_rev_okflag = 0; if(send_error == 1) OLED_ShowStr(32,2,"Send Fail! ",2,0);//显示发送超时 else OLED_ShowStr(32,2," Send OK! ",2,0); UsartRx1BufClear(); delay_ms(1000);//延时1秒 OLED_ShowStr(32,2," ",2,0); } /***********************************************************************************************/ /****************************** end ****************************************/ /***********************************************************************************************/ void STM32_FlashCheck(void) // 检查是否是新的单片机,是的话清空存储区,否则保留 { u8 comper_str[6],i; STMFLASH_Read(FLASH_SAVE_ADDR + 0x10,(u16*)comper_str,5); comper_str[5] = '\0'; if(strstr((char *)comper_str,"FDYDZ") == NULL) //新的单片机 { STMFLASH_Write(FLASH_SAVE_ADDR + 0x10,(u16*)"FDYDZ",5); //写入“FDYDZ”,方便下次校验 delay_ms(50); STMFLASH_Write(FLASH_SAVE_ADDR + 0x40,(u16*)"12345678910",11);//存入初始手机号 delay_ms(50); } STMFLASH_Read(FLASH_SAVE_ADDR + 0x40,(u16*)PhoneNumber,11); //读出手机号 for(i = 0; i < 11 ; i++) { if(PhoneNumber[i]<'0' || PhoneNumber[i]>'9') { break; } } if(i != 11) { memset(PhoneNumber, 0 , 11); //清除缓存 sprintf(PhoneNumber,"12345678910"); } delay_ms(100); } void display_mode(void) { u8 i; //显示中文: 防盗模式 if(fangdao==1){for(i = 0;i < 4;i ++)OLED_ShowCN(i*16+32,0,i+0,1);}else {for(i = 0;i < 4;i ++)OLED_ShowCN(i*16+32,0,i+24,1);} } void displayInitInterface(void) //显示初始页面 { u8 i; for(i = 0;i < 2;i ++)OLED_ShowCN(i*16,4,i+6,0); //显示中文: 温度 for(i = 0;i < 2;i ++)OLED_ShowCN(i*16,6,i+8,0); //显示中文: 烟雾 OLED_ShowChar(32,0,':',2,0); OLED_ShowChar(32,4,':',2,0); OLED_ShowChar(32,6,':',2,0); display_mode(); } void Get_Temperature(void) //获取温度 { temperature=ReadTemperature(); if(temperature>=setTempValue) { if(!(alarmFlag&0x01)) { alarmFlag|=0x01; shanshuo = 0; sendSmsFlag = 2; //发送短信标志 } } else { alarmFlag&=0xFE; } if(temperature>=setTempValue && shanshuo) { OLED_ShowStr(40, 4," ", 2,0); } else { sprintf(display," %d",temperature); OLED_ShowStr(40, 4, (u8*)display, 2,0);//显示温度 OLED_ShowCentigrade(68, 4); //显示摄氏度 } } void Get_Smoke(void) //获取烟雾浓度 { u16 test_adc=0; /* 获取烟雾浓度 */ test_adc = Get_Adc_Average(ADC_Channel_9,10);//读取通道9的10次AD平均值 smoke = test_adc*99/4096;//转换成0-99百分比 if(smoke>=setSmokeValue) { if(!(alarmFlag&0x02)) { alarmFlag|=0x02; shanshuo = 0; sendSmsFlag = 3; //发送短信标志 } } else { alarmFlag&=0xFD; } if(smoke>=setSmokeValue && shanshuo) { OLED_ShowStr(40, 6," ", 2,0); } else { sprintf(display," %02d %%",smoke); OLED_ShowStr(40, 6, (u8*)display, 2,0);//显示温度 } } void displaySetValue(void) //显示设置值 { u8 add=2,i; if(setn==1) { OLED_ShowChar(56,4,setTempValue%100/10+'0',2,0);//显示 OLED_ShowChar(64,4,setTempValue%10+'0',2,0);//显示 } if(setn==2) { OLED_ShowChar(56,4,setSmokeValue%100/10+'0',2,0);//显示 OLED_ShowChar(64,4,setSmokeValue%10+'0',2,0);//显示 OLED_ShowChar(72,4,'%',2,0); } if(setn>=3) { for(i = 0;i < 11;i ++) { OLED_ShowChar((add++)*8,4,PhoneNumber[i],2,(setn+1)-(3+i));//显示手机号码 } } } void keyscan(void) //按键扫描 { u8 i; if(KEY1 == 0) //设置键 { delay_ms(20); if(KEY1 == 0) { while(KEY1 == 0); setn ++; if(setn == 1) { OLED_CLS();//清屏 for(i = 0;i < 4;i ++)OLED_ShowCN(i*16+32,0,i+10,0);//显示中文:设置温度 OLED_ShowCentigrade(75, 4); //显示摄氏度 } if(setn == 2) { for(i = 0;i < 4;i ++)OLED_ShowCN(i*16+32,0,i+14,0);//显示中文:设置烟雾 OLED_ShowChar(80,4,' ',2,0); } if(setn == 3) { for(i = 0;i < 6;i ++)OLED_ShowCN(i*16+16,0,i+18,0);//显示中文:设置手机号码 } if(setn >= 14) { setn = 0; OLED_CLS();//清屏 displayInitInterface(); STMFLASH_Write(FLASH_SAVE_ADDR + 0x40,(u16*)PhoneNumber,11); //退出设置,存入设置的手机号 } displaySetValue(); } } if(KEY2 == 0) //加键 { delay_ms(80); if(KEY2 == 0) { if(setTempValue < 99 && setn==1)setTempValue++; if(setSmokeValue < 99 && setn==2)setSmokeValue++; if(setn>=3) { PhoneNumber[setn-3]++; if(PhoneNumber[setn-3]>'9')PhoneNumber[setn-3]='0'; } displaySetValue(); } } if(KEY3 == 0) //减键 { delay_ms(80); if(KEY3 == 0) { if(setn==0) { fangdao=!fangdao; display_mode(); } if(setTempValue > 0 && setn==1)setTempValue--; if(setSmokeValue > 0 && setn==2)setSmokeValue--; if(setn>=3) { PhoneNumber[setn-3]--; if(PhoneNumber[setn-3]<'0')PhoneNumber[setn-3]='9'; } displaySetValue(); } } } int main(void) { bool flameFlag=0; bool SomebodyFlag=0; char SEND_BUF[400]; //发送短信缓存 delay_init(); //延时函数初始化 NVIC_Configuration(); //中断优先级配置 I2C_Configuration(); //IIC初始化 STM32_FlashCheck(); //FLASH初始化 delay_ms(200); OLED_Init(); //OLED液晶初始化 OLED_CLS(); //清屏 OLED_ShowStr(0,2," GSM Init... ",2,0); uart1_Init(9600); gsm_init();//gsm初始化 OLED_CLS();//清屏 Adc_Init(); //adc初始化 KEY_GPIO_Init(); //按键引脚初始化 SR501_GPIO_Init(); //人体红外初始化 DS18B20_GPIO_Init(); //温度初始化 DS18B20_Init(); //初始化显示 displayInitInterface(); //显示初始界面 TIM3_Init(99,719); //定时器初始化,定时1ms //Tout = ((arr+1)*(psc+1))/Tclk ; //Tclk:定时器输入频率(单位MHZ) //Tout:定时器溢出时间(单位us) while(1) { keyscan(); //按键扫描 if(setn == 0) { if(shuaxin == 1) //大概300ms刷新一次数据 { Get_Temperature(); //获取温度 Get_Smoke(); //获取烟雾 shuaxin = 0; } if(FLAME == 0) //检测到火焰 { delay_ms(10); if(FLAME == 0) { if(flameFlag == 0) { OLED_DrawBMP(88,4,120,8,(unsigned char *)BMP1); //显示火焰图片 sendSmsFlag = 1; //发送短信标志 } flameFlag = 1; } } else { if(flameFlag == 1) { OLED_ShowStr(88, 4, " ", 2,0); OLED_ShowStr(88, 6, " ", 2,0); } flameFlag = 0; } if(fangdao==1&&SR501==1) //在防盗模式下检测到有人 { if(SomebodyFlag==0) { OLED_DrawBMP(0,0,32,4,(unsigned char *)BMP2); //图显示 sendSmsFlag = 4; //发送短信标志 SomebodyFlag = 1; } } else { if(SomebodyFlag==1) { OLED_ShowStr(0, 0, " ", 2,0); OLED_ShowStr(0, 2, " ", 2,0); SomebodyFlag = 0; } } if(temperature>=setTempValue || smoke>=setSmokeValue || flameFlag || SomebodyFlag)BEEP=1;else BEEP=0; //检测到温度烟雾超标火焰蜂鸣器报警 if(temperature>=setTempValue)FAN=1;else FAN=0; //温度超标都开启风扇 if(smoke>=setSmokeValue || flameFlag)RELAY = 1; else RELAY = 0; //检测到有火或者烟雾超标,开启水泵 if(sendSmsFlag != 0) //发送短信 { char TEMP_BUF[100]; /*******************************************************************************************/ /*******************以下为短信内容处理部分,发送中文短信必须转换为Unicode码**************/ /******************************************************************************************/ memset(SEND_BUF,0,sizeof(SEND_BUF)); //清空缓冲区 switch(sendSmsFlag) { case(1): strcat(SEND_BUF,"8B66544AFF0168C06D4B5230706B7130FF01"); break; //警告!检测到火焰! case(2): strcat(SEND_BUF,"8B66544AFF016E295EA68FC79AD8FF01"); break; //警告!温度过高! case(3): strcat(SEND_BUF,"8B66544AFF0170DF96FE6D535EA68FC79AD8FF01"); break; //警告!烟雾浓度过高! case(4): strcat(SEND_BUF,"8B66544AFF0168C06D4B523067094EBAFF01"); break; //警告!检测到有人! default: break; } if(sendSmsFlag!=4) { memset(TEMP_BUF,0,sizeof(TEMP_BUF)); //清空缓冲区 sprintf(TEMP_BUF,"6E295EA6003A003%1d003%1d2103FF0C70DF96FE003A003%1d003%1d0025",temperature/10,temperature%10,smoke/10,smoke%10); strcat(SEND_BUF,TEMP_BUF); } sim800_send_sms((char *)SEND_BUF);//发送短信 sendSmsFlag = 0; /*******************************************************************************************/ /*************************** end *****************************/ /******************************************************************************************/ } } delay_ms(10); } } void TIM3_IRQHandler(void)//定时器3中断服务程序,用于记录时间 { static u16 timeCount1 = 0; if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET) //检查指定的TIM中断发生与否:TIM 中断源 { TIM_ClearITPendingBit(TIM3, TIM_IT_Update); //清除中断标志位 timeCount1++; if(timeCount1 >= 300) //300ms { timeCount1 = 0; shanshuo = !shanshuo; shuaxin = 1; } } } ADC初始化程序 #include "adc.h" #include "delay.h" //初始化ADC //这里我们仅以规则通道为例 //我们默认将开启通道0~3 void Adc_Init(void) { ADC_InitTypeDef ADC_InitStructure; GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB |RCC_APB2Periph_ADC1 , ENABLE ); //使能ADC1通道时钟 RCC_ADCCLKConfig(RCC_PCLK2_Div8); //设置ADC分频因子6 72M/8=9,ADC最大时间不能超过14M GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1|GPIO_Pin_0; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入引脚 GPIO_Init(GPIOB, &GPIO_InitStructure); ADC_DeInit(ADC1); //复位ADC1,将外设 ADC1 的全部寄存器重设为缺省值 ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //ADC工作模式:ADC1和ADC2工作在独立模式 ADC_InitStructure.ADC_ScanConvMode = DISABLE; //模数转换工作在单通道模式 ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; //模数转换工作在单次转换模式 ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //转换由软件而不是外部触发启动 ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //ADC数据右对齐 ADC_InitStructure.ADC_NbrOfChannel = 1; //顺序进行规则转换的ADC通道的数目 ADC_Init(ADC1, &ADC_InitStructure); //根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器 ADC_Cmd(ADC1, ENABLE); //使能指定的ADC1 ADC_ResetCalibration(ADC1); //使能复位校准 while(ADC_GetResetCalibrationStatus(ADC1)); //等待复位校准结束 ADC_StartCalibration(ADC1); //开启AD校准 while(ADC_GetCalibrationStatus(ADC1)); //等待校准结束 } //获得ADC值 //ch:通道值 9 u16 Get_Adc(u8 ch) { //设置指定ADC的规则组通道,一个序列,采样时间 ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_71Cycles5 ); //ADC1,ADC通道,采样时间为13.5周期 ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能指定的ADC1的软件转换启动功能 while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));//等待转换结束 return ADC_GetConversionValue(ADC1); //返回最近一次ADC1规则组的转换结果 } u16 Get_Adc_Average(u8 ch,u8 times) { u32 temp_val=0; u8 t; for(t=0;t<times;t++) { temp_val+=Get_Adc(ch); delay_ms(5); } return temp_val/times; } #ifndef __ADC_H #define __ADC_H #include "sys.h" void Adc_Init(void); u16 Get_Adc(u8 ch); u16 Get_Adc_Average(u8 ch,u8 times); #endif GPIO引脚初始化程序 #include "gpio.h" ////////////////////////////////////////////////////////////////////////////////// //按键舵机的GPIO设置 ////////////////////////////////////////////////////////////////////////////////// void KEY_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOB|RCC_APB2Periph_GPIOC|RCC_APB2Periph_AFIO, ENABLE); //使能PABC端口时钟 GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable,ENABLE); //关闭JTAG模式 使PB3,PB4变成普通IO口 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12|GPIO_Pin_13|GPIO_Pin_14|GPIO_Pin_15; // 端口配置 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; //上拉输入 GPIO_Init(GPIOB, &GPIO_InitStructure); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13|GPIO_Pin_14|GPIO_Pin_15; // 端口配置 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //IO口速度为50MHz GPIO_Init(GPIOC, &GPIO_InitStructure); GPIO_ResetBits(GPIOC,GPIO_Pin_13|GPIO_Pin_14|GPIO_Pin_15); //输出0 } void SR501_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //使能PA端口时钟 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0; // 端口配置 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD; //下拉输入 GPIO_Init(GPIOA, &GPIO_InitStructure); } #ifndef __GPIO_H #define __GPIO_H #include "sys.h" #define KEY1 PBin(12) #define KEY2 PBin(13) #define KEY3 PBin(14) #define FLAME PBin(15) #define BEEP PCout(13) #define FAN PCout(14) #define RELAY PCout(15) #define SR501 PAin(0) void KEY_GPIO_Init(void);//引脚初始化 void SR501_GPIO_Init(void); #endif OLED驱动程序 /************************************************************************************ * * Description:128*64点阵的OLED显示屏驱动文件SD1306驱动IIC通信方式显示屏 * * Others: none; * * Function List: * 1. void I2C_Configuration(void) -- 配置CPU的硬件I2C * 2. void I2C_WriteByte(uint8_t addr,uint8_t data) -- 向寄存器地址写一个byte的数据 * 3. void WriteCmd(unsigned char I2C_Command) -- 写命令 * 4. void WriteDat(unsigned char I2C_Data) -- 写数据 * 5. void OLED_Init(void) -- OLED屏初始化 * 6. void OLED_SetPos(unsigned char x, unsigned char y) -- 设置起始点坐标 * 7. void OLED_Fill(unsigned char fill_Data) -- 全屏填充 * 8. void OLED_CLS(void) -- 清屏 * 9. void OLED_ON(void) -- 唤醒 * 10. void OLED_OFF(void) -- 睡眠 * 11. void OLED_ShowStr(unsigned char x, unsigned char y, unsigned char ch[], unsigned char TextSize) -- 显示字符串(字体大小有6*8和8*16两种) * 12. void OLED_ShowCN(unsigned char x, unsigned char y, unsigned char N) -- 显示中文(中文需要先取模,然后放到codetab.h中) * 13. void OLED_DrawBMP(unsigned char x0,unsigned char y0,unsigned char x1,unsigned char y1,unsigned char BMP[]) -- BMP图片 * * History: none; * *************************************************************************************/ #include "OLED_I2C.h" #include "delay.h" #include "codetab.h" /* 定义I2C总线连接的GPIO端口, 用户只需要修改下面4行代码即可任意改变SCL和SDA的引脚 */ #define RCC_I2C_PORT RCC_APB2Periph_GPIOB /* GPIO端口时钟 */ #define PORT_I2C_SCL GPIOB /* GPIO端口 */ #define PIN_I2C_SCL GPIO_Pin_6 /* GPIO引脚 */ #define PORT_I2C_SDA GPIOB /* GPIO端口 */ #define PIN_I2C_SDA GPIO_Pin_7 /* GPIO引脚 */ #define I2C_SCL_PIN GPIO_Pin_6 /* 连接到SCL时钟线的GPIO */ #define I2C_SDA_PIN GPIO_Pin_7 /* 连接到SDA数据线的GPIO */ /* 定义读写SCL和SDA的宏 */ #define I2C_SCL_1() PORT_I2C_SCL->BSRR = I2C_SCL_PIN /* SCL = 1 */ #define I2C_SCL_0() PORT_I2C_SCL->BRR = I2C_SCL_PIN /* SCL = 0 */ #define I2C_SDA_1() PORT_I2C_SDA->BSRR = I2C_SDA_PIN /* SDA = 1 */ #define I2C_SDA_0() PORT_I2C_SDA->BRR = I2C_SDA_PIN /* SDA = 0 */ #define I2C_SDA_READ() ((PORT_I2C_SDA->IDR & I2C_SDA_PIN) != 0) /* 读SDA口线状态 */ #define I2C_SCL_READ() ((PORT_I2C_SCL->IDR & I2C_SCL_PIN) != 0) /* 读SCL口线状态 */ /* ********************************************************************************************************* * 函 数 名: bsp_InitI2C * 功能说明: 配置I2C总线的GPIO,采用模拟IO的方式实现 * 形 参: 无 * 返 回 值: 无 ********************************************************************************************************* */ void bsp_InitI2C_2(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_I2C_PORT, ENABLE); /* 打开GPIO时钟 */ GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD; /* 开漏输出模式 */ GPIO_InitStructure.GPIO_Pin = PIN_I2C_SCL; GPIO_Init(PORT_I2C_SCL, &GPIO_InitStructure); GPIO_InitStructure.GPIO_Pin = PIN_I2C_SDA; GPIO_Init(PORT_I2C_SDA, &GPIO_InitStructure); /* 给一个停止信号, 复位I2C总线上的所有设备到待机模式 */ i2c_Stop_2(); } /* ********************************************************************************************************* * 函 数 名: i2c_Start * 功能说明: CPU发起I2C总线启动信号 * 形 参: 无 * 返 回 值: 无 ********************************************************************************************************* */ void i2c_Start_2(void) { /* 当SCL高电平时,SDA出现一个下跳沿表示I2C总线启动信号 */ I2C_SDA_1(); I2C_SCL_1(); delay_us(4); I2C_SDA_0(); delay_us(4); I2C_SCL_0(); } /* ********************************************************************************************************* * 函 数 名: i2c_Start * 功能说明: CPU发起I2C总线停止信号 * 形 参: 无 * 返 回 值: 无 ********************************************************************************************************* */ void i2c_Stop_2(void) { /* 当SCL高电平时,SDA出现一个上跳沿表示I2C总线停止信号 */ I2C_SDA_0(); I2C_SCL_1(); delay_us(4); I2C_SDA_1(); delay_us(4); } /* ********************************************************************************************************* * 函 数 名: i2c_SendByte * 功能说明: CPU向I2C总线设备发送8bit数据 * 形 参: _ucByte : 等待发送的字节 * 返 回 值: 无 ********************************************************************************************************* */ void i2c_SendByte_2(uint8_t _ucByte) { uint8_t i; /* 先发送字节的高位bit7 */ for (i = 0; i < 8; i++) { if (_ucByte & 0x80) { I2C_SDA_1(); } else { I2C_SDA_0(); } delay_us(2); I2C_SCL_1(); delay_us(2); I2C_SCL_0(); _ucByte <<= 1; /* 左移一个bit */ delay_us(2); } } /* ********************************************************************************************************* * 函 数 名: i2c_ReadByte * 功能说明: CPU从I2C总线设备读取8bit数据 * 形 参: 无 * 返 回 值: 读到的数据 ********************************************************************************************************* */ uint8_t i2c_ReadByte_2(void) { uint8_t i; uint8_t value; /* 读到第1个bit为数据的bit7 */ value = 0; for (i = 0; i < 8; i++) { value <<= 1; I2C_SCL_1();//SCL在高电平期间,数据必须保持稳定 delay_us(2); if (I2C_SDA_READ()) { value++; } I2C_SCL_0(); delay_us(1); } return value; } /* ********************************************************************************************************* * 函 数 名: i2c_WaitAck * 功能说明: CPU产生一个时钟,并读取器件的ACK应答信号 * 形 参: 无 * 返 回 值: 返回0表示正确应答,1表示无器件响应 ********************************************************************************************************* */ uint8_t i2c_WaitAck_2(void) { uint8_t tempTime; I2C_SDA_1(); /* CPU释放SDA总线 */ delay_us(1); I2C_SCL_1(); /* CPU驱动SCL = 1, 此时器件会返回ACK应答 */ delay_us(1); while(I2C_SDA_READ()) { tempTime++; if(tempTime>250) { i2c_Stop_2(); return 1; } } I2C_SCL_0(); return 0; } /* ********************************************************************************************************* * 函 数 名: i2c_Ack * 功能说明: CPU产生一个ACK信号 * 形 参: 无 * 返 回 值: 无 ********************************************************************************************************* */ void i2c_Ack_2(void) { I2C_SDA_0(); /* CPU驱动SDA = 0 */ delay_us(5); I2C_SCL_1(); /* CPU产生1个时钟 */ delay_us(5); I2C_SCL_0(); delay_us(5); I2C_SDA_1(); /* CPU释放SDA总线 */ } /* ********************************************************************************************************* * 函 数 名: i2c_NAck * 功能说明: CPU产生1个NACK信号 * 形 参: 无 * 返 回 值: 无 ********************************************************************************************************* */ void i2c_NAck_2(void) { I2C_SDA_1(); /* CPU驱动SDA = 1 */ delay_us(5); I2C_SCL_1(); /* CPU产生1个时钟 */ delay_us(5); I2C_SCL_0(); delay_us(5); } void I2C_Configuration(void) { bsp_InitI2C_2(); } void I2C_WriteByte(uint8_t addr,uint8_t data) { /* 第1步:发起I2C总线启动信号 */ i2c_Start_2(); /* 第2步:发起控制字节,高7bit是地址,bit0是读写控制位,0表示写,1表示读 */ i2c_SendByte_2(OLED_ADDRESS | I2C_WR); /* 此处是写指令 */ /* 第3步:发送ACK */ i2c_WaitAck_2(); /* 第4步:发送字节地址 */ i2c_SendByte_2(addr); i2c_WaitAck_2(); /* 第5步:开始写入数据 */ i2c_SendByte_2(data); /* 第6步:发送ACK */ i2c_WaitAck_2(); /* 发送I2C总线停止信号 */ i2c_Stop_2(); } void WriteCmd(unsigned char I2C_Command)//写命令 { I2C_WriteByte(0x00, I2C_Command); } void WriteDat(unsigned char I2C_Data)//写数据 { I2C_WriteByte(0x40, I2C_Data); } void OLED_Init(void) { delay_ms(100); //这里的延时很重要 WriteCmd(0xAE); //display off WriteCmd(0x20); //Set Memory Addressing Mode WriteCmd(0x10); //00,Horizontal Addressing Mode;01,Vertical Addressing Mode;10,Page Addressing Mode (RESET);11,Invalid WriteCmd(0xb0); //Set Page Start Address for Page Addressing Mode,0-7 WriteCmd(0xc8); //Set COM Output Scan Direction WriteCmd(0x00); //---set low column address WriteCmd(0x10); //---set high column address WriteCmd(0x40); //--set start line address WriteCmd(0x81); //--set contrast control register WriteCmd(0xff); //亮度调节 0x00~0xff WriteCmd(0xa1); //--set segment re-map 0 to 127 WriteCmd(0xa6); //--set normal display WriteCmd(0xa8); //--set multiplex ratio(1 to 64) WriteCmd(0x3F); // WriteCmd(0xa4); //0xa4,Output follows RAM content;0xa5,Output ignores RAM content WriteCmd(0xd3); //-set display offset WriteCmd(0x00); //-not offset WriteCmd(0xd5); //--set display clock divide ratio/oscillator frequency WriteCmd(0xf0); //--set divide ratio WriteCmd(0xd9); //--set pre-charge period WriteCmd(0x22); // WriteCmd(0xda); //--set com pins hardware configuration WriteCmd(0x12); WriteCmd(0xdb); //--set vcomh WriteCmd(0x20); //0x20,0.77xVcc WriteCmd(0x8d); //--set DC-DC enable WriteCmd(0x14); // WriteCmd(0xaf); //--turn on oled panel } void OLED_SetPos(unsigned char x, unsigned char y) //设置起始点坐标 { WriteCmd(0xb0+y); WriteCmd(((x&0xf0)>>4)|0x10); WriteCmd((x&0x0f)|0x01); } void OLED_Fill(unsigned char fill_Data)//全屏填充 { unsigned char m,n; for(m=0;m<8;m++) { WriteCmd(0xb0+m); //page0-page1 WriteCmd(0x00); //low column start address WriteCmd(0x10); //high column start address for(n=0;n<128;n++) { WriteDat(fill_Data); } } } void OLED_CLS(void)//清屏 { OLED_Fill(0x00); } //-------------------------------------------------------------- // Prototype : void OLED_ON(void) // Calls : // Parameters : none // Description : 将OLED从休眠中唤醒 //-------------------------------------------------------------- void OLED_ON(void) { WriteCmd(0X8D); //设置电荷泵 WriteCmd(0X14); //开启电荷泵 WriteCmd(0XAF); //OLED唤醒 } //-------------------------------------------------------------- // Prototype : void OLED_OFF(void) // Calls : // Parameters : none // Description : 让OLED休眠 -- 休眠模式下,OLED功耗不到10uA //-------------------------------------------------------------- void OLED_OFF(void) { WriteCmd(0X8D); //设置电荷泵 WriteCmd(0X10); //关闭电荷泵 WriteCmd(0XAE); //OLED休眠 } //在指定位置显示一个字符,包括部分字符 //x:0~127 //y:0~7 //chr:显示的字符 //TextSize:字符大小(1:6*8 ; 2:8*16) //mode:1,反白显示;0,正常显示 void OLED_ShowChar(unsigned char x,unsigned char y,unsigned char chr,unsigned char TextSize,u8 mode) { unsigned char c=0,i=0; c=chr-' ';//得到偏移后的值 if(TextSize == 2) { if(x>120){x=0;y++;} OLED_SetPos(x,y); for(i=0;i<8;i++) if(mode==1)WriteDat(~(F8X16[c*16+i]));else WriteDat(F8X16[c*16+i]); OLED_SetPos(x,y+1); for(i=0;i<8;i++) if(mode==1)WriteDat(~(F8X16[c*16+i+8]));else WriteDat(F8X16[c*16+i+8]); } else { if(x>126){x=0;y++;} OLED_SetPos(x,y); for(i=0;i<6;i++) if(mode==1)WriteDat(~(F6x8[c][i])); else WriteDat(F6x8[c][i]); } } //-------------------------------------------------------------- // Prototype : void OLED_ShowChar(unsigned char x, unsigned char y, unsigned char ch[], unsigned char TextSize) // Calls : // Parameters : x,y -- 起始点坐标(x:0~127, y:0~7); ch[] -- 要显示的字符串; TextSize -- 字符大小(1:6*8 ; 2:8*16) // Description : 显示codetab.h中的ASCII字符,有6*8和8*16可选择 //-------------------------------------------------------------- void OLED_ShowStr(unsigned char x, unsigned char y, unsigned char ch[], unsigned char TextSize,u8 mode) { unsigned char c = 0,i = 0,j = 0; switch(TextSize) { case 1: { while(ch[j] != '\0') { c = ch[j] - 32; if(x > 126) { x = 0; y++; } OLED_SetPos(x,y); for(i=0;i<6;i++) if(mode==1)WriteDat(~(F6x8[c][i])); else WriteDat(F6x8[c][i]); x += 6; j++; } }break; case 2: { while(ch[j] != '\0') { c = ch[j] - 32; if(x > 120) { x = 0; y++; } OLED_SetPos(x,y); for(i=0;i<8;i++) if(mode==1)WriteDat(~(F8X16[c*16+i])); else WriteDat(F8X16[c*16+i]); OLED_SetPos(x,y+1); for(i=0;i<8;i++) if(mode==1)WriteDat(~(F8X16[c*16+i+8])); else WriteDat(F8X16[c*16+i+8]); x += 8; j++; } }break; } } //-------------------------------------------------------------- // Prototype : void OLED_ShowCN(unsigned char x, unsigned char y, unsigned char N) // Calls : // Parameters : x,y -- 起始点坐标(x:0~127, y:0~7); N:汉字在codetab.h中的索引 // Description : 显示codetab.h中的汉字,16*16点阵 //mode:1,反白显示;0,正常显示 //-------------------------------------------------------------- void OLED_ShowCN(unsigned char x, unsigned char y, unsigned char N,u8 mode) { unsigned char wm=0; unsigned int adder=32*N; OLED_SetPos(x , y); for(wm = 0;wm < 16;wm++) { if(mode==1)WriteDat(~(F16x16[adder]));else WriteDat(F16x16[adder]); adder += 1; } OLED_SetPos(x,y + 1); for(wm = 0;wm < 16;wm++) { if(mode==1)WriteDat(~(F16x16[adder]));else WriteDat(F16x16[adder]); adder += 1; } } //-------------------------------------------------------------- // Prototype : void OLED_DrawBMP(unsigned char x0,unsigned char y0,unsigned char x1,unsigned char y1,unsigned char BMP[]); // Calls : // Parameters : x0,y0 -- 起始点坐标(x0:0~127, y0:0~7); x1,y1 -- 起点对角线(结束点)的坐标(x1:1~128,y1:1~8) // Description : 显示BMP位图 //-------------------------------------------------------------- void OLED_DrawBMP(unsigned char x0,unsigned char y0,unsigned char x1,unsigned char y1,unsigned char BMP[]) { unsigned int j=0; unsigned char x,y; if(y1%8==0) y = y1/8; else y = y1/8 + 1; for(y=y0;y<y1;y++) { OLED_SetPos(x0,y); for(x=x0;x<x1;x++) { WriteDat(BMP[j++]); } } } void OLED_ShowCentigrade(unsigned char x, unsigned char y)//显示℃ { unsigned char wm=0; unsigned char BUF[]={ 0x10,0x28,0x10,0xC0,0x20,0x10,0x10,0x10,0x20,0x70,0x00,0x00,0x00,0x00,0x00,0x07, 0x08,0x10,0x10,0x10,0x10,0x08,0x04,0x00,/*"℃"*/ }; OLED_SetPos(x , y); for(wm = 0;wm < 12;wm++) { WriteDat(BUF[wm]); } OLED_SetPos(x,y + 1); for(wm = 0;wm < 12;wm++) { WriteDat(BUF[wm+12]); } } #ifndef __OLED_I2C_H #define __OLED_I2C_H #include "stm32f10x.h" #define OLED_ADDRESS 0x78 //通过调整0R电阻,屏可以0x78和0x7A两个地址 -- 默认0x78 #define I2C_WR 0 /* 写控制bit */ #define I2C_RD 1 /* 读控制bit */ void bsp_InitI2C_2(void); void i2c_Start_2(void); void i2c_Stop_2(void); void i2c_SendByte_2(uint8_t _ucByte); uint8_t i2c_ReadByte_2(void); uint8_t i2c_WaitAck_2(void); void i2c_Ack_2(void); void i2c_NAck_2(void); void I2C_Configuration(void); void I2C_WriteByte(uint8_t addr,uint8_t data); void WriteCmd(unsigned char I2C_Command); void WriteDat(unsigned char I2C_Data); void OLED_Init(void); void OLED_SetPos(unsigned char x, unsigned char y); void OLED_Fill(unsigned char fill_Data); void OLED_CLS(void); void OLED_ON(void); void OLED_OFF(void); void OLED_ShowChar(unsigned char x,unsigned char y,unsigned char chr,unsigned char TextSize,u8 mode); void OLED_ShowStr(unsigned char x, unsigned char y, unsigned char ch[], unsigned char TextSize,u8 mode); void OLED_ShowCN(unsigned char x, unsigned char y, unsigned char N,u8 mode); void OLED_DrawBMP(unsigned char x0,unsigned char y0,unsigned char x1,unsigned char y1,unsigned char BMP[]); void OLED_ShowCentigrade(unsigned char x, unsigned char y); #endif TIME定时器程序 #include "timer.h" //通用定时器中断初始化 //这里时钟选择为APB1的2倍,而APB1为36M //arr:自动重装值。 //psc:时钟预分频数 //这里使用的是定时器3! void TIM3_Init(u16 arr,u16 psc) { TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; NVIC_InitTypeDef NVIC_InitStructure; RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); //时钟使能 TIM_TimeBaseStructure.TIM_Period = arr; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值 计数到5000为500ms TIM_TimeBaseStructure.TIM_Prescaler =psc; //设置用来作为TIMx时钟频率除数的预分频值 10Khz的计数频率 TIM_TimeBaseStructure.TIM_ClockDivision = 0; //设置时钟分割:TDTS = Tck_tim TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM向上计数模式 TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位 TIM_ITConfig( //使能或者失能指定的TIM中断 TIM3, //TIM3 TIM_IT_Update , ENABLE //使能 ); NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn; //TIM3中断 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1; //先占优先级0级 NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1; //从优先级2级 NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道被使能 NVIC_Init(&NVIC_InitStructure); //根据NVIC_InitStruct中指定的参数初始化外设NVIC寄存器 TIM_Cmd(TIM3, ENABLE); //使能定时器3 } #ifndef __TIMER_H #define __TIMER_H #include "sys.h" void TIM3_Init(u16 arr,u16 psc); #endif DS18B20温度程序 #include "ds18b20.h" #include "delay.h" /******************************************************************************* 函数名:DS18B20_GPIO_Init 功能:初始化DS18B20引脚 输入: 输出: 返回值: 备注: *******************************************************************************/ void DS18B20_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_DS18B20_PORT, ENABLE); //使能PORTA口时钟 GPIO_InitStructure.GPIO_Pin = DS18B20_GPIO_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(DS18B20_GPIO_PORT, &GPIO_InitStructure); GPIO_SetBits(DS18B20_GPIO_PORT,DS18B20_GPIO_PIN); //输出1 } /******************************************************************************* 函数名:DS18B20_Init 功能:初始化DS18B20 输入: 输出: 返回值:初始化成功为0,不成功为1 备注: *******************************************************************************/ u8 DS18B20_Init(void) { unsigned char wait=0; DS18B20_IO_OUT(); //输出模式 DS18B20_OUT_0; //拉低 delay_us(750); //至少延时480us DS18B20_OUT_1; //拉高 delay_us(15); //15us DS18B20_IO_IN(); //输入模式 while(READ_DS18B20_IO && wait++<200)delay_us(1);//等待高电平结束 if(wait>=200)return 1; else wait=0; while(!READ_DS18B20_IO && wait++<240)delay_us(1);//等待低电平结束 if(wait>=240)return 1; else return 0; } /******************************************************************************* 函数名:DS18B20_ReadByte 功能:从DS18B20读一个字节 输入: 输出: 返回值:读取到的字节 备注: *******************************************************************************/ unsigned char DS18B20_ReadByte(void) { unsigned char i; unsigned char dat = 0; for (i=0; i<8; i++) //8位计数器 { dat >>= 1; DS18B20_IO_OUT(); //输出模式 DS18B20_OUT_0; //开始时间片 delay_us(2); //延时等待 DS18B20_OUT_1; //准备接收 DS18B20_IO_IN(); //输入模式 delay_us(12); //接收延时 if(READ_DS18B20_IO) dat |= 0x80; //读取数据 delay_us(60); //等待时间片结束 } return dat; } /******************************************************************************* 函数名:DS18B20_WriteByte 功能:写一个字节 输入:unsigned char dat 输出: 返回值: 备注: *******************************************************************************/ void DS18B20_WriteByte(unsigned char dat) { unsigned char i; unsigned char temp; DS18B20_IO_OUT();//输出模式 for (i=1; i<=8; i++) { temp = dat & 0x01; dat = dat >> 1; if (temp) { DS18B20_OUT_0; delay_us(2); DS18B20_OUT_1; //写1 delay_us(60); } else { DS18B20_OUT_0;//写0 delay_us(60); DS18B20_OUT_1; delay_us(2); } } } /************************************** 从DS18B20中获取温度值得浮点值 参数: 空 返回值: 读取到的温度值(有效范围-55.0~125.0) **************************************/ float ReadTemperature(void) { unsigned char TPH; //存放温度值的高字节 unsigned char TPL; //存放温度值的低字节 short i16=0; float f32=0; DS18B20_Init(); DS18B20_WriteByte(0xCC); //跳过ROM命令 DS18B20_WriteByte(0x44); //开始转换命令 DS18B20_Init(); DS18B20_WriteByte(0xCC); //跳过ROM命令 DS18B20_WriteByte(0xBE); //读暂存存储器命令 TPL = DS18B20_ReadByte(); //读温度低字节 TPH = DS18B20_ReadByte(); //读温度高字节 i16 = 0; i16 = (TPH<<8) |TPL; // 将高位(MSB)与低位(LSB)合并 f32 = i16 * 0.0625; // 12bit精度时温度值计算 return(f32); // 返回读取到的温度数值(float型) } #ifndef __DS18B20_H #define __DS18B20_H #include "sys.h" //如果想要修改引脚,只需修改下面的宏 #define RCC_DS18B20_PORT RCC_APB2Periph_GPIOA /* GPIO端口时钟 */ #define DS18B20_GPIO_PIN GPIO_Pin_11 #define DS18B20_GPIO_PORT GPIOA //IO方向设置(CRL寄存器对应引脚0~7,CRH寄存器对应引脚8~15) //DS18B20_GPIO_PORT->CRH&=0xFFFFFFF0为PA8引脚输出模式对应的寄存器清空 //DS18B20_GPIO_PORT->CRH|=0x00000008将(CNF8[1:0]设置为10:上拉/下拉输入模式,MODE8[1;0]设置为00:输入模式) //DS18B20_GPIO_PORT->CRH|=0x00000003将(CNF8[1:0]设置为00:通用推挽输出模式 ,MODE8[1;0]设置为11:最大50MHZ) #define DS18B20_IO_IN() {DS18B20_GPIO_PORT->CRH&=0xFFFF0FFF;DS18B20_GPIO_PORT->CRH|=0x00008000;} #define DS18B20_IO_OUT() {DS18B20_GPIO_PORT->CRH&=0xFFFF0FFF;DS18B20_GPIO_PORT->CRH|=0x00003000;} #define DS18B20_OUT_0 GPIO_ResetBits(DS18B20_GPIO_PORT, DS18B20_GPIO_PIN)//IO为低电平 #define DS18B20_OUT_1 GPIO_SetBits(DS18B20_GPIO_PORT, DS18B20_GPIO_PIN)//IO为高电平 #define READ_DS18B20_IO GPIO_ReadInputDataBit(DS18B20_GPIO_PORT, DS18B20_GPIO_PIN)//读取IO电平 void DS18B20_GPIO_Init(void); u8 DS18B20_Init(void); //初始化DS18B20 float ReadTemperature(void); //获取温度值 #endif 修改代码中使用OLED的部分,改为使用LCD1602实现

/*** ************************************************************************************************ * @version V1.0 * @author ¹С°à¿Æ¼¼ * @brief SCCB ½Ó¿ÚÏà¹Øº¯Êý£¨Èí¼þÄ£ÄâIICµÄÐÎʽ£© ************************************************************************************************* * @description * * ʵÑéÆ½Ì¨£ºÂ¹Ð¡°àSTM32H723ZGT6ºËÐÄ°å £¨ÐͺţºLXB723ZG-P1£© * ¿Í·þ΢ÐÅ£º19949278543 * >>>>> Îļþ˵Ã÷£º * * 1.SCCBÏà¹Øº¯Êý£¨SCCBºÍIICÒ»Ñù£© * 2.ʹÓÃÄ£Äâ½Ó¿ÚµÄÐÎʽ * 3.ͨÐÅËÙ¶ÈĬÈÏΪ 300KHz ×óÓÒ£¬×î´ó²»Äܳ¬¹ý400K * ************************************************************************************************************************ ***/ #include "sccb.h" /***************************************************************************************** * º¯ Êý Ãû: SCCB_GPIO_Config * Èë¿Ú²ÎÊý: ÎÞ * ·µ »Ø Öµ: ÎÞ * º¯Êý¹¦ÄÜ: ³õʼ»¯IICµÄGPIO¿Ú,ÍÆÍìÊä³ö * ˵ Ã÷: ÓÉÓÚIICͨÐÅËٶȲ»¸ß£¬ÕâÀïµÄIO¿ÚËÙ¶ÈÅäÖÃΪ2M¼´¿É ******************************************************************************************/ void SCCB_GPIO_Config (void) { GPIO_InitTypeDef GPIO_InitStruct = {0}; SCCB_SCL_CLK_ENABLE; //³õʼ»¯IO¿ÚʱÖÓ SCCB_SDA_CLK_ENABLE; GPIO_InitStruct.Pin = SCCB_SCL_PIN; // SCLÒý½Å GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_OD; // ¿ªÂ©Êä³ö GPIO_InitStruct.Pull = GPIO_NOPULL; // ²»´øÉÏÏÂÀ­ GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; // Ëٶȵȼ¶ HAL_GPIO_Init(SCCB_SCL_PORT, &GPIO_InitStruct); GPIO_InitStruct.Pin = SCCB_SDA_PIN; // SDAÒý½Å HAL_GPIO_Init(SCCB_SDA_PORT, &GPIO_InitStruct); GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; // ÍÆÍìÊä³ö GPIO_InitStruct.Pull = GPIO_PULLUP; // ÉÏÀ­ HAL_GPIO_WritePin(SCCB_SCL_PORT, SCCB_SCL_PIN, GPIO_PIN_SET); // SCLÊä³ö¸ßµçƽ HAL_GPIO_WritePin(SCCB_SDA_PORT, SCCB_SDA_PIN, GPIO_PIN_SET); // SDAÊä³ö¸ßµçƽ } /***************************************************************************************** * º¯ Êý Ãû: SCCB_Delay * Èë¿Ú²ÎÊý: a - ÑÓʱʱ¼ä * ·µ »Ø Öµ: ÎÞ * º¯Êý¹¦ÄÜ: ¼òµ¥ÑÓʱº¯Êý * ˵ Ã÷: ΪÁËÒÆÖ²µÄ¼ò±ãÐÔÇÒ¶ÔÑÓʱ¾«¶ÈÒªÇ󲻸ߣ¬ËùÒÔ²»ÐèҪʹÓö¨Ê±Æ÷×öÑÓʱ ******************************************************************************************/ void SCCB_Delay(uint32_t a) { volatile uint16_t i; while (a --) { for (i = 0; i < 3; i++); } } /***************************************************************************************** * º¯ Êý Ãû: SCCB_Start * Èë¿Ú²ÎÊý: ÎÞ * ·µ »Ø Öµ: ÎÞ * º¯Êý¹¦ÄÜ: IICÆðʼÐźŠ* ˵ Ã÷: ÔÚSCL´¦ÓÚ¸ßµçÆ½ÆÚ¼ä£¬SDAÓɸߵ½µÍÌø±äΪÆðʼÐźŠ******************************************************************************************/ void SCCB_Start(void) { SCCB_SDA(1); SCCB_SCL(1); SCCB_Delay(SCCB_DelayVaule); SCCB_SDA(0); SCCB_Delay(SCCB_DelayVaule); SCCB_SCL(0); SCCB_Delay(SCCB_DelayVaule); } /***************************************************************************************** * º¯ Êý Ãû: SCCB_Stop * Èë¿Ú²ÎÊý: ÎÞ * ·µ »Ø Öµ: ÎÞ * º¯Êý¹¦ÄÜ: IICÍ£Ö¹ÐźŠ* ˵ Ã÷: ÔÚSCL´¦ÓÚ¸ßµçÆ½ÆÚ¼ä£¬SDAÓɵ͵½¸ßÌø±äΪÆðʼÐźŠ******************************************************************************************/ void SCCB_Stop(void) { SCCB_SCL(0); SCCB_Delay(SCCB_DelayVaule); SCCB_SDA(0); SCCB_Delay(SCCB_DelayVaule); SCCB_SCL(1); SCCB_Delay(SCCB_DelayVaule); SCCB_SDA(1); SCCB_Delay(SCCB_DelayVaule); } /***************************************************************************************** * º¯ Êý Ãû: SCCB_ACK * Èë¿Ú²ÎÊý: ÎÞ * ·µ »Ø Öµ: ÎÞ * º¯Êý¹¦ÄÜ: IICÓ¦´ðÐźŠ* ˵ Ã÷: ÔÚSCLΪ¸ßµçƽÆÚ¼ä£¬SDAÒý½ÅÊä³öΪµÍµçƽ£¬²úÉúÓ¦´ðÐźŠ******************************************************************************************/ void SCCB_ACK(void) { SCCB_SCL(0); SCCB_Delay(SCCB_DelayVaule); SCCB_SDA(0); SCCB_Delay(SCCB_DelayVaule); SCCB_SCL(1); SCCB_Delay(SCCB_DelayVaule); SCCB_SCL(0); // SCLÊä³öµÍʱ£¬SDAÓ¦Á¢¼´À­¸ß£¬ÊÍ·Å×ÜÏß SCCB_SDA(1); SCCB_Delay(SCCB_DelayVaule); } /***************************************************************************************** * º¯ Êý Ãû: SCCB_NoACK * Èë¿Ú²ÎÊý: ÎÞ * ·µ »Ø Öµ: ÎÞ * º¯Êý¹¦ÄÜ: IIC·ÇÓ¦´ðÐźŠ* ˵ Ã÷: ÔÚSCLΪ¸ßµçƽÆÚ¼ä£¬ÈôSDAÒý½ÅΪ¸ßµçƽ£¬²úÉú·ÇÓ¦´ðÐźŠ******************************************************************************************/ void SCCB_NoACK(void) { SCCB_SCL(0); SCCB_Delay(SCCB_DelayVaule); SCCB_SDA(1); SCCB_Delay(SCCB_DelayVaule); SCCB_SCL(1); SCCB_Delay(SCCB_DelayVaule); SCCB_SCL(0); SCCB_Delay(SCCB_DelayVaule); } /***************************************************************************************** * º¯ Êý Ãû: SCCB_WaitACK * Èë¿Ú²ÎÊý: ÎÞ * ·µ »Ø Öµ: ÎÞ * º¯Êý¹¦ÄÜ: µÈ´ý½ÓÊÕÉ豸·¢³öÓ¦´ðÐźŠ* ˵ Ã÷: ÔÚSCLΪ¸ßµçƽÆÚ¼ä£¬Èô¼ì²âµ½SDAÒý½ÅΪµÍµçƽ£¬Ôò½ÓÊÕÉ豸ÏìÓ¦Õý³£ ******************************************************************************************/ uint8_t SCCB_WaitACK(void) { SCCB_SDA(1); SCCB_Delay(SCCB_DelayVaule); SCCB_SCL(1); SCCB_Delay(SCCB_DelayVaule); if( HAL_GPIO_ReadPin(SCCB_SDA_PORT,SCCB_SDA_PIN) != 0) //ÅжÏÉ豸ÊÇ·ñÓÐ×ö³öÏìÓ¦ { SCCB_SCL(0); SCCB_Delay( SCCB_DelayVaule ); return ACK_ERR; //ÎÞÓ¦´ð } else { SCCB_SCL(0); SCCB_Delay( SCCB_DelayVaule ); return ACK_OK; //Ó¦´ðÕý³£ } } /***************************************************************************************** * º¯ Êý Ãû: SCCB_WriteByte * Èë¿Ú²ÎÊý: IIC_Data - ҪдÈëµÄ8λÊý¾Ý * ·µ »Ø Öµ: ACK_OK - É豸ÏìÓ¦Õý³£ * ACK_ERR - É豸ÏìÓ¦´íÎó * º¯Êý¹¦ÄÜ: дһ×Ö½ÚÊý¾Ý * ˵ Ã÷: ¸ßλÔÚǰ ******************************************************************************************/ uint8_t SCCB_WriteByte(uint8_t IIC_Data) { uint8_t i; for (i = 0; i < 8; i++) { SCCB_SDA(IIC_Data & 0x80); SCCB_Delay( SCCB_DelayVaule ); SCCB_SCL(1); SCCB_Delay( SCCB_DelayVaule ); SCCB_SCL(0); if(i == 7) { SCCB_SDA(1); } IIC_Data <<= 1; } return SCCB_WaitACK(); //µÈ´ýÉ豸ÏìÓ¦ } /***************************************************************************************** * º¯ Êý Ãû: SCCB_ReadByte * Èë¿Ú²ÎÊý: ACK_Mode - ÏìӦģʽ£¬ÊäÈë1Ôò·¢³öÓ¦´ðÐźţ¬ÊäÈë0·¢³ö·ÇÓ¦´ðÐźŠ* ·µ »Ø Öµ: ACK_OK - É豸ÏìÓ¦Õý³£ * ACK_ERR - É豸ÏìÓ¦´íÎó * º¯Êý¹¦ÄÜ: ¶ÁÒ»×Ö½ÚÊý¾Ý * ˵ Ã÷: 1.¸ßλÔÚǰ * 2.Ó¦ÔÚÖ÷»ú½ÓÊÕ×îºóÒ»×Ö½ÚÊý¾Ýʱ·¢ËÍ·ÇÓ¦´ðÐźŠ******************************************************************************************/ uint8_t SCCB_ReadByte(uint8_t ACK_Mode) { uint8_t IIC_Data = 0; uint8_t i = 0; for (i = 0; i < 8; i++) { IIC_Data <<= 1; SCCB_SCL(1); SCCB_Delay( SCCB_DelayVaule ); IIC_Data |= (HAL_GPIO_ReadPin(SCCB_SDA_PORT,SCCB_SDA_PIN) & 0x01); SCCB_SCL(0); SCCB_Delay( SCCB_DelayVaule ); } if ( ACK_Mode == 1 ) // Ó¦´ðÐźŠSCCB_ACK(); else SCCB_NoACK(); // ·ÇÓ¦´ðÐźŠreturn IIC_Data; } /************************************************************************************************************************************* * º¯ Êý Ãû: SCCB_WriteHandle * * Èë¿Ú²ÎÊý: addr - Òª½øÐвÙ×÷µÄ¼Ä´æÆ÷(8λµØÖ·) * * ·µ »Ø Öµ: SUCCESS - ²Ù×÷³É¹¦£¬ERROR - ²Ù×÷ʧ°Ü * * º¯Êý¹¦ÄÜ: ¶ÔÖ¸¶¨µÄ¼Ä´æÆ÷(8λµØÖ·)Ö´ÐÐд²Ù×÷£¬OV2640Óõ½ ************************************************************************************************************************************/ uint8_t SCCB_WriteHandle (uint8_t addr) { uint8_t status; // ״̬±ê־λ SCCB_Start(); // Æô¶¯IICͨÐÅ if( SCCB_WriteByte(OV2640_DEVICE_ADDRESS) == ACK_OK ) //дÊý¾ÝÖ¸Áî { if( SCCB_WriteByte((uint8_t)(addr)) != ACK_OK ) { status = ERROR; // ²Ù×÷ʧ°Ü } } status = SUCCESS; // ²Ù×÷³É¹¦ return status; } /************************************************************************************************************************************* * º¯ Êý Ãû: SCCB_WriteReg * * Èë¿Ú²ÎÊý: addr - ҪдÈëµÄ¼Ä´æÆ÷(8λµØÖ·)£¬value - ҪдÈëµÄÊý¾Ý * * ·µ »Ø Öµ: SUCCESS - ²Ù×÷³É¹¦£¬ ERROR - ²Ù×÷ʧ°Ü * * º¯Êý¹¦ÄÜ: ¶ÔÖ¸¶¨µÄ¼Ä´æÆ÷(8λµØÖ·)дһ×Ö½ÚÊý¾Ý£¬OV2640Óõ½ ************************************************************************************************************************************/ uint8_t SCCB_WriteReg (uint8_t addr,uint8_t value) { uint8_t status; SCCB_Start(); //Æô¶¯IICͨѶ if( SCCB_WriteHandle(addr) == SUCCESS) //дÈëÒª²Ù×÷µÄ¼Ä´æÆ÷ { if (SCCB_WriteByte(value) != ACK_OK) //дÊý¾Ý { status = ERROR; } } SCCB_Stop(); // ֹͣͨѶ status = SUCCESS; // дÈë³É¹¦ return status; } /************************************************************************************************************************************* * º¯ Êý Ãû: SCCB_ReadReg * * Èë¿Ú²ÎÊý: addr - Òª¶ÁÈ¡µÄ¼Ä´æÆ÷(8λµØÖ·) * * ·µ »Ø Öµ: ¶Áµ½µÄÊý¾Ý * * º¯Êý¹¦ÄÜ: ¶ÔÖ¸¶¨µÄ¼Ä´æÆ÷(8λµØÖ·)¶Áȡһ×Ö½ÚÊý¾Ý£¬OV2640Óõ½ ************************************************************************************************************************************/ uint8_t SCCB_ReadReg (uint8_t addr) { uint8_t value = 0; SCCB_Start(); // Æô¶¯IICͨÐÅ if( SCCB_WriteHandle(addr) == SUCCESS) //дÈëÒª²Ù×÷µÄ¼Ä´æÆ÷ { SCCB_Stop(); // Í£Ö¹IICͨÐÅ SCCB_Start(); //ÖØÐÂÆô¶¯IICͨѶ if (SCCB_WriteByte(OV2640_DEVICE_ADDRESS|0X01) == ACK_OK) // ·¢ËͶÁÃüÁî { value = SCCB_ReadByte(0); // ¶Áµ½×îºóÒ»¸öÊý¾Ýʱ·¢ËÍ ·ÇÓ¦´ðÐźŠ} SCCB_Stop(); // Í£Ö¹IICͨÐÅ } return value; } /************************************************************************************************************************************* * º¯ Êý Ãû: SCCB_WriteHandle_16Bit * * Èë¿Ú²ÎÊý: addr - Òª½øÐвÙ×÷µÄ¼Ä´æÆ÷(16λµØÖ·) * * ·µ »Ø Öµ: SUCCESS - ²Ù×÷³É¹¦£¬ERROR - ²Ù×÷ʧ°Ü * * º¯Êý¹¦ÄÜ: ¶ÔÖ¸¶¨µÄ¼Ä´æÆ÷(16λµØÖ·)Ö´ÐÐд²Ù×÷£¬OV5640Óõ½ ************************************************************************************************************************************/ uint8_t SCCB_WriteHandle_16Bit (uint16_t addr) { uint8_t status; // ״̬±ê־λ SCCB_Start(); // Æô¶¯IICͨÐÅ if( SCCB_WriteByte(OV5640_DEVICE_ADDRESS) == ACK_OK ) //дÊý¾ÝÖ¸Áî { if( SCCB_WriteByte((uint8_t)(addr >> 8)) == ACK_OK ) //дÈë16λµØÖ· { if( SCCB_WriteByte((uint8_t)(addr)) != ACK_OK ) { status = ERROR; // ²Ù×÷ʧ°Ü } } } status = SUCCESS; // ²Ù×÷³É¹¦ return status; } /************************************************************************************************************************************* * º¯ Êý Ãû: SCCB_WriteReg_16Bit * * Èë¿Ú²ÎÊý: addr - ҪдÈëµÄ¼Ä´æÆ÷(16λµØÖ·) value - ҪдÈëµÄÊý¾Ý * * ·µ »Ø Öµ: SUCCESS - ²Ù×÷³É¹¦£¬ERROR - ²Ù×÷ʧ°Ü * * º¯Êý¹¦ÄÜ: ¶ÔÖ¸¶¨µÄ¼Ä´æÆ÷(16λµØÖ·)дһ×Ö½ÚÊý¾Ý£¬OV5640Óõ½ ************************************************************************************************************************************/ uint8_t SCCB_WriteReg_16Bit(uint16_t addr,uint8_t value) { uint8_t status; SCCB_Start(); //Æô¶¯IICͨѶ if( SCCB_WriteHandle_16Bit(addr) == SUCCESS) //дÈëÒª²Ù×÷µÄ¼Ä´æÆ÷ { if (SCCB_WriteByte(value) != ACK_OK) //дÊý¾Ý { status = ERROR; } } SCCB_Stop(); // ֹͣͨѶ status = SUCCESS; // дÈë³É¹¦ return status; } /************************************************************************************************************************************* * º¯ Êý Ãû: SCCB_ReadReg_16Bit * * Èë¿Ú²ÎÊý: addr - Òª¶ÁÈ¡µÄ¼Ä´æÆ÷(16λµØÖ·) * * ·µ »Ø Öµ: ¶Áµ½µÄÊý¾Ý * * º¯Êý¹¦ÄÜ: ¶ÔÖ¸¶¨µÄ¼Ä´æÆ÷(16λµØÖ·)¶Áȡһ×Ö½ÚÊý¾Ý£¬OV5640Óõ½ ************************************************************************************************************************************/ uint8_t SCCB_ReadReg_16Bit (uint16_t addr) { uint8_t value = 0; SCCB_Start(); // Æô¶¯IICͨÐÅ if( SCCB_WriteHandle_16Bit(addr) == SUCCESS) //дÈëÒª²Ù×÷µÄ¼Ä´æÆ÷ { SCCB_Stop(); // Í£Ö¹IICͨÐÅ SCCB_Start(); //ÖØÐÂÆô¶¯IICͨѶ if (SCCB_WriteByte(OV5640_DEVICE_ADDRESS|0X01) == ACK_OK) // ·¢ËͶÁÃüÁî { value = SCCB_ReadByte(0); // ¶Áµ½×îºóÒ»¸öÊý¾Ýʱ·¢ËÍ ·ÇÓ¦´ðÐźŠ} SCCB_Stop(); // Í£Ö¹IICͨÐÅ } return value; } /************************************************************************************************************************************* * º¯ Êý Ãû: SCCB_WriteBuffer_16Bit * * Èë¿Ú²ÎÊý: addr - ҪдÈëµÄ¼Ä´æÆ÷(16λµØÖ·) *pData - Êý¾ÝÇø size - Òª´«ÊäÊý¾ÝµÄ´óС * * ·µ »Ø Öµ: SUCCESS - ²Ù×÷³É¹¦£¬ERROR - ²Ù×÷ʧ°Ü * * º¯Êý¹¦ÄÜ: ¶ÔÖ¸¶¨µÄ¼Ä´æÆ÷(16λµØÖ·)ÅúÁ¿Ð´Êý¾Ý£¬OV5640 дÈë×Ô¶¯¶Ô½¹¹Ì¼þʱÓõ½ ************************************************************************************************************************************/ uint8_t SCCB_WriteBuffer_16Bit(uint16_t addr,uint8_t *pData, uint32_t size) { uint8_t status; uint32_t i; SCCB_Start(); //Æô¶¯IICͨѶ if( SCCB_WriteHandle_16Bit(addr) == SUCCESS) //дÈëÒª²Ù×÷µÄ¼Ä´æÆ÷ { for(i=0;i<size;i++) { SCCB_WriteByte(*pData);//дÊý¾Ý pData++; } } SCCB_Stop(); // ֹͣͨѶ status = SUCCESS; // дÈë³É¹¦ return status; } /********************************************************************************************/ 分析代码作用

找不到错在哪,但是温度就是不变 /* USER CODE BEGIN Header */ /** ****************************************************************************** * @file : main.c * @brief : Main program body ****************************************************************************** * @attention * * Copyright (c) 2025 STMicroelectronics. * All rights reserved. * * This software is licensed under terms that can be found in the LICENSE file * in the root directory of this software component. * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ /* USER CODE END Header */ /* Includes ------------------------------------------------------------------*/ #include "main.h" /* Private includes ----------------------------------------------------------*/ /* USER CODE BEGIN Includes */ #include "LCD.h" #include "DS18B20.h" /* USER CODE END Includes */ /* Private typedef -----------------------------------------------------------*/ /* USER CODE BEGIN PTD */ /* USER CODE END PTD */ /* Private define ------------------------------------------------------------*/ /* USER CODE BEGIN PD */ /* USER CODE END PD */ /* Private macro -------------------------------------------------------------*/ /* USER CODE BEGIN PM */ /* USER CODE END PM */ /* Private variables ---------------------------------------------------------*/ /* USER CODE BEGIN PV */ /* USER CODE END PV */ /* Private function prototypes -----------------------------------------------*/ void SystemClock_Config(void); static void MX_GPIO_Init(void); /* USER CODE BEGIN PFP */ /* USER CODE END PFP */ /* Private user code ---------------------------------------------------------*/ /* USER CODE BEGIN 0 */ /* USER CODE END 0 */ /** * @brief The application entry point. * @retval int */ int main(void) { /* USER CODE BEGIN 1 */ float T; /* USER CODE END 1 */ /* MCU Configuration--------------------------------------------------------*/ /* Reset of all peripherals, Initializes the Flash interface and the Systick. */ HAL_Init(); /* USER CODE BEGIN Init */ /* USER CODE END Init */ /* Configure the system clock */ SystemClock_Config(); /* USER CODE BEGIN SysInit */ /* USER CODE END SysInit */ /* Initialize all configured peripherals */ MX_GPIO_Init(); /* USER CODE BEGIN 2 */ lcd9648_init(); DS18B20_ConvertT(); lcd9648_clear(); lcd_show_string(0,0,96,48,16,"Temperature:"); lcd_reflash_gram(); // lcd9648_init(); // lcd9648_clear(); // lcd_show_string(0,0,96,48,12,"Hello World!"); // lcd_show_string(0,14,96,48,16,"Temperature:"); // //lcd_show_fontHZ(0, 30,"���пƼ�"); // lcd_reflash_gram(); /* USER CODE END 2 */ /* Infinite loop */ /* USER CODE BEGIN WHILE */ while (1) { DS18B20_ConvertT(); T=DS18B20_ReadT(); if(T<0){ lcd_show_char(0,1*16,'-',16,0); lcd_reflash_gram(); T=-T; } else{ lcd_show_char(0,1*16,'+',16,0); lcd_reflash_gram(); } lcd_show_num(1*8,1*16,(int)T,3,16,0); lcd_show_char(4*8,1*16,'.',16,0); lcd_show_num(5*8,1*16,(unsigned long)(T*10000)%10000,4,16,0); lcd_reflash_gram(); /* USER CODE END WHILE */ /* USER CODE BEGIN 3 */ HAL_Delay(500); } /* USER CODE END 3 */ } /** * @brief System Clock Configuration * @retval None */ void SystemClock_Config(void) { RCC_OscInitTypeDef RCC_OscInitStruct = {0}; RCC_ClkInitTypeDef RCC_ClkInitStruct = {0}; /** Initializes the RCC Oscillators according to the specified parameters * in the RCC_OscInitTypeDef structure. */ RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSI; RCC_OscInitStruct.HSIState = RCC_HSI_ON; RCC_OscInitStruct.HSICalibrationValue = RCC_HSICALIBRATION_DEFAULT; RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE; if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK) { Error_Handler(); } /** Initializes the CPU, AHB and APB buses clocks */ RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2; RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_HSI; RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1; RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV1; RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1; if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_0) != HAL_OK) { Error_Handler(); } } /** * @brief GPIO Initialization Function * @param None * @retval None */ static void MX_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStruct = {0}; /* USER CODE BEGIN MX_GPIO_Init_1 */ /* USER CODE END MX_GPIO_Init_1 */ /* GPIO Ports Clock Enable */ __HAL_RCC_GPIOC_CLK_ENABLE(); __HAL_RCC_GPIOB_CLK_ENABLE(); __HAL_RCC_GPIOA_CLK_ENABLE(); /*Configure GPIO pin Output Level */ HAL_GPIO_WritePin(GPIOC, SCL_Pin|SDA_Pin|RS_Pin|CS_Pin |RST_Pin, GPIO_PIN_RESET); /*Configure GPIO pin Output Level */ HAL_GPIO_WritePin(DQ_GPIO_Port, DQ_Pin, GPIO_PIN_RESET); /*Configure GPIO pins : SCL_Pin SDA_Pin RS_Pin CS_Pin RST_Pin */ GPIO_InitStruct.Pin = SCL_Pin|SDA_Pin|RS_Pin|CS_Pin |RST_Pin; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; HAL_GPIO_Init(GPIOC, &GPIO_InitStruct); /*Configure GPIO pin : DQ_Pin */ GPIO_InitStruct.Pin = DQ_Pin; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_OD; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; HAL_GPIO_Init(DQ_GPIO_Port, &GPIO_InitStruct); /* USER CODE BEGIN MX_GPIO_Init_2 */ HAL_GPIO_WritePin(DQ_GPIO_Port,DQ_Pin,GPIO_PIN_SET); /* USER CODE END MX_GPIO_Init_2 */ } /* USER CODE BEGIN 4 */ /* USER CODE END 4 */ /** * @brief This function is executed in case of error occurrence. * @retval None */ void Error_Handler(void) { /* USER CODE BEGIN Error_Handler_Debug */ /* User can add his own implementation to report the HAL error return state */ __disable_irq(); while (1) { } /* USER CODE END Error_Handler_Debug */ } #ifdef USE_FULL_ASSERT /** * @brief Reports the name of the source file and the source line number * where the assert_param error has occurred. * @param file: pointer to the source file name * @param line: assert_param error line source number * @retval None */ void assert_failed(uint8_t *file, uint32_t line) { /* USER CODE BEGIN 6 */ /* User can add his own implementation to report the file name and line number, ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */ /* USER CODE END 6 */ } #endif /* USE_FULL_ASSERT */

大家在看

recommend-type

mssdk10130048en MsSDK u14

mssdk101400111en.msi是microstation开发的sdk,版本是U14。文件解压密码是:https://blog.csdn.net/csdn_wuwt
recommend-type

matlab 伪距单点定位

本人利用matlab 编写的伪距单点定位,包含rinex 导航文件和观测文件读取的新方法,独立与定位程序之外。在单点定位的同时还进行了地球自转改正,卫星钟误差改正,接收机钟误差改正,地球自转改正,相对效应改正,电离层改正和对流层改正、额外部分还进行了简单的卡尔曼滤波处理定位结果。
recommend-type

libssl-1_1-x64.zip

如题,放到软件安装目录bin文件夹下即可。 在anaconda更新包时遇到了这个问题,找了了X64位可以用的,之后成功更新
recommend-type

Aptra NDC Reference manual

ATM 行业, 国外常用的Aptra NDC协议
recommend-type

的表中所-数据结构中文版

是用于对以位单位访问的点数进行指定的数据,在3.3.1项(1)的表中所 示的1次通信中可处理的点数以内进行指定。 a) 通过 ASCII代码进行数据通信时 将点数转换为 ASCII 代码 2位(16 进制数)后使用,从各高位进行 发送。 (示例) 5 点的情况 下 : 变为“05”,从“0”开始按顺序进行发送。 20 点的情 况下 : 变为“14”,从“1”开始按顺序进行发送。 b) 通过二进制代码进行数据通信时 使用表示点数的 1字节的数值进行发送。 (示例) 5 点的情况 下 : 发送 05H。 20 点的情 况下 : 发送 14H。 7) 设置/复位 是用于指定写入到位软元件中的数据的数据,以如下所示的值进行指 定。 写入数据 ON OFF 备注 ASCII 代码 “01” “00” 从“0”开始按顺序发送 2 字符 二进制代码 01H 00H 发送如左所示的 1 字节的数值

最新推荐

recommend-type

关于STM32的flash读写数据和HardFault_Handler的问题

void FLASH_WriteByte(u32 addr, u16 flashdata1) { FLASH_Status FLASHstatus = FLASH_COMPLETE; FLASH_Unlock(); // 解锁Flash接口 FLASH_ClearFlag(FLASH_FLAG_EOP | FLASH_FLAG_PGERR | FLASH_FLAG_WRPRTERR)...
recommend-type

《门户网站对比》.ppt

《门户网站对比》.ppt
recommend-type

C++实现的DecompressLibrary库解压缩GZ文件

根据提供的文件信息,我们可以深入探讨C++语言中关于解压缩库(Decompress Library)的使用,特别是针对.gz文件格式的解压过程。这里的“lib”通常指的是库(Library),是软件开发中用于提供特定功能的代码集合。在本例中,我们关注的库是用于处理.gz文件压缩包的解压库。 首先,我们要明确一个概念:.gz文件是一种基于GNU zip压缩算法的压缩文件格式,广泛用于Unix、Linux等操作系统上,对文件进行压缩以节省存储空间或网络传输时间。要解压.gz文件,开发者需要使用到支持gzip格式的解压缩库。 在C++中,处理.gz文件通常依赖于第三方库,如zlib或者Boost.IoStreams。codeproject.com是一个提供编程资源和示例代码的网站,程序员可以在该网站上找到现成的C++解压lib代码,来实现.gz文件的解压功能。 解压库(Decompress Library)提供的主要功能是读取.gz文件,执行解压缩算法,并将解压缩后的数据写入到指定的输出位置。在使用这些库时,我们通常需要链接相应的库文件,这样编译器在编译程序时能够找到并使用这些库中定义好的函数和类。 下面是使用C++解压.gz文件时,可能涉及的关键知识点: 1. Zlib库 - zlib是一个用于数据压缩的软件库,提供了许多用于压缩和解压缩数据的函数。 - zlib库支持.gz文件格式,并且在多数Linux发行版中都预装了zlib库。 - 在C++中使用zlib库,需要包含zlib.h头文件,同时链接z库文件。 2. Boost.IoStreams - Boost是一个提供大量可复用C++库的组织,其中的Boost.IoStreams库提供了对.gz文件的压缩和解压缩支持。 - Boost库的使用需要下载Boost源码包,配置好编译环境,并在编译时链接相应的Boost库。 3. C++ I/O操作 - 解压.gz文件需要使用C++的I/O流操作,比如使用ifstream读取.gz文件,使用ofstream输出解压后的文件。 - 对于流操作,我们常用的是std::ifstream和std::ofstream类。 4. 错误处理 - 解压缩过程中可能会遇到各种问题,如文件损坏、磁盘空间不足等,因此进行适当的错误处理是必不可少的。 - 正确地捕获异常,并提供清晰的错误信息,对于调试和用户反馈都非常重要。 5. 代码示例 - 从codeproject找到的C++解压lib很可能包含一个或多个源代码文件,这些文件会包含解压.gz文件所需的函数或类。 - 示例代码可能会展示如何初始化库、如何打开.gz文件、如何读取并处理压缩数据,以及如何释放资源等。 6. 库文件的链接 - 编译使用解压库的程序时,需要指定链接到的库文件,这在不同的编译器和操作系统中可能略有不同。 - 通常,在编译命令中加入-l参数,比如使用g++的话可能是`g++ -o DecompressLibrary DecompressLibrary.cpp -lz`,其中`-lz`表示链接zlib库。 7. 平台兼容性 - 在不同平台上使用解压库可能需要考虑平台兼容性问题。 - Windows系统可能需要额外的配置和库文件,因为zlib或其他库可能不是默认预装的。 根据以上知识点,我们可以得出,在C++中解压.gz文件主要涉及到对zlib或类似库的使用,以及熟悉C++的I/O操作。正确使用这些库,能够有效地对压缩文件进行解压,并处理可能出现的错误情况。如果从codeproject获取到的C++解压lib确实是针对.gz文件格式的,那么它很可能已经封装好了大部分的操作细节,让开发者能够以更简单的方式实现解压功能。
recommend-type

【数据融合技术】:甘肃土壤类型空间分析中的专业性应用

# 摘要 数据融合技术作为一种集成多源数据信息的方法,在土壤类型空间分析中发挥着关键作用。本文介绍了数据融合技术的基本概念及其理论基础,阐述了数据预处理、同步整合及冲突解决等关键技术,并详细描述了甘肃土壤类型数据准备的流程,包括数据采集、质量评估、空间化处理及融合实践准备。通过具体案例分析,展示了数据融合在土壤类型空间分布分析、土壤质量评估及土壤保护规划中的应用。同时,文
recommend-type

VM ware如何查看软件版本信息

<think>首先,用户的问题是:“我想查找VMware软件版本信息的查看方法请问如何在VMware中查看软件版本信息”。用户参考了之前的引用,但那些引用主要是关于查看VMware版本的一般方法,现在用户具体想知道在VMware软件中如何查看版本信息。回顾之前的对话:-用户第一次问的是VSCode的版本查看方法。-现在用户问的是VMware的版本查看方法。-系统级指令要求使用中文回答,正确格式化数学表达式(如果需要),但这里可能不需要数学表达式。-指令还要求生成相关问题,并在回答中引用段落时添加引用标识。用户提供的引用[1]到[5]是关于VMware版本的查看方法、下载等,但用户特别强调“参考
recommend-type

数据库课程设计报告:常用数据库综述

数据库是现代信息管理的基础,其技术广泛应用于各个领域。在高等教育中,数据库课程设计是一个重要环节,它不仅是学习理论知识的实践,也是培养学生综合运用数据库技术解决问题能力的平台。本知识点将围绕“经典数据库课程设计报告”展开,详细阐述数据库的基本概念、课程设计的目的和内容,以及在设计报告中常用的数据库技术。 ### 1. 数据库基本概念 #### 1.1 数据库定义 数据库(Database)是存储在计算机存储设备中的数据集合,这些数据集合是经过组织的、可共享的,并且可以被多个应用程序或用户共享访问。数据库管理系统(DBMS)提供了数据的定义、创建、维护和控制功能。 #### 1.2 数据库类型 数据库按照数据模型可以分为关系型数据库(如MySQL、Oracle)、层次型数据库、网状型数据库、面向对象型数据库等。其中,关系型数据库因其简单性和强大的操作能力而广泛使用。 #### 1.3 数据库特性 数据库具备安全性、完整性、一致性和可靠性等重要特性。安全性指的是防止数据被未授权访问和破坏。完整性指的是数据和数据库的结构必须符合既定规则。一致性保证了事务的执行使数据库从一个一致性状态转换到另一个一致性状态。可靠性则保证了系统发生故障时数据不会丢失。 ### 2. 课程设计目的 #### 2.1 理论与实践结合 数据库课程设计旨在将学生在课堂上学习的数据库理论知识与实际操作相结合,通过完成具体的数据库设计任务,加深对数据库知识的理解。 #### 2.2 培养实践能力 通过课程设计,学生能够提升分析问题、设计解决方案以及使用数据库技术实现这些方案的能力。这包括需求分析、概念设计、逻辑设计、物理设计、数据库实现、测试和维护等整个数据库开发周期。 ### 3. 课程设计内容 #### 3.1 需求分析 在设计报告的开始,需要对项目的目标和需求进行深入分析。这涉及到确定数据存储需求、数据处理需求、数据安全和隐私保护要求等。 #### 3.2 概念设计 概念设计阶段要制定出数据库的E-R模型(实体-关系模型),明确实体之间的关系。E-R模型的目的是确定数据库结构并形成数据库的全局视图。 #### 3.3 逻辑设计 基于概念设计,逻辑设计阶段将E-R模型转换成特定数据库系统的逻辑结构,通常是关系型数据库的表结构。在此阶段,设计者需要确定各个表的属性、数据类型、主键、外键以及索引等。 #### 3.4 物理设计 在物理设计阶段,针对特定的数据库系统,设计者需确定数据的存储方式、索引的具体实现方法、存储过程、触发器等数据库对象的创建。 #### 3.5 数据库实现 根据物理设计,实际创建数据库、表、视图、索引、触发器和存储过程等。同时,还需要编写用于数据录入、查询、更新和删除的SQL语句。 #### 3.6 测试与维护 设计完成之后,需要对数据库进行测试,确保其满足需求分析阶段确定的各项要求。测试过程包括单元测试、集成测试和系统测试。测试无误后,数据库还需要进行持续的维护和优化。 ### 4. 常用数据库技术 #### 4.1 SQL语言 SQL(结构化查询语言)是数据库管理的国际标准语言。它包括数据查询、数据操作、数据定义和数据控制四大功能。SQL语言是数据库课程设计中必备的技能。 #### 4.2 数据库设计工具 常用的数据库设计工具包括ER/Studio、Microsoft Visio、MySQL Workbench等。这些工具可以帮助设计者可视化地设计数据库结构,提高设计效率和准确性。 #### 4.3 数据库管理系统 数据库管理系统(DBMS)是用于创建和管理数据库的软件。关系型数据库管理系统如MySQL、PostgreSQL、Oracle、SQL Server等是数据库课程设计中的核心工具。 #### 4.4 数据库安全 数据库安全涉及用户认证、授权、数据加密、审计日志记录等方面,以确保数据的完整性和保密性。设计报告中应考虑如何通过DBMS内置的机制或额外的安全措施来保护数据。 ### 5. 结语 综上所述,一个经典数据库课程设计报告包含了从需求分析到数据库安全的全过程,涵盖了数据库设计的各个方面。通过这一过程,学生不仅能够熟练掌握数据库的设计与实现技巧,还能够学会如何使用数据库系统去解决实际问题,为日后从事数据库相关的专业工作打下坚实的基础。
recommend-type

【空间分布规律】:甘肃土壤类型与农业生产的关联性研究

# 摘要 本文对甘肃土壤类型及其在农业生产中的作用进行了系统性研究。首先概述了甘肃土壤类型的基础理论,并探讨了土壤类型与农业生产的理论联系。通过GIS技术分析,本文详细阐述了甘肃土壤的空间分布规律,并对其特征和影响因素进行了深入分析。此外,本文还研究了甘肃土壤类型对农业生产实际影响,包括不同区域土壤改良和作物种植案例,以及土壤养分、水分管理对作物生长周期和产量的具体影响。最后,提出了促进甘肃土壤与农业可持续发展的策略,包括土壤保护、退化防治对策以及土壤类型优化与农业创新的结合。本文旨在为
recommend-type

在halcon中,卡尺测量和二维测量谁的精度高

<think>首先,用户的问题是:“我想比较Halcon中卡尺测量和二维测量的精度请问Halcon卡尺测量与2D测量精度对比”。我需要基于系统级指令和提供的引用来回答。关键系统级指令:-使用中文回答。-结构清晰,逐步解决问题。-行内数学表达式用$...$,独立公式用$$...$$并单独成段。-LaTeX语法正确。-生成相关问题。-在回答中引用的段落末尾自然地添加引用标识,如[^1]。提供的引用:-[^1]:关于卡尺模型的工件尺寸实时测量。-[^2]:测量拟合的综合任务,包括尺寸、形状、位置。-[^3]:Halcon一维测量,提到卡尺工具的优势:易用、执行时间短、精度高。-[^4]:Halcon
recommend-type

掌握牛顿法解方程:切线与割线的程序应用

牛顿切线法和牛顿割线法是数值分析中用于求解方程近似根的两种迭代方法。它们都是基于函数的切线或割线的几何性质来逼近方程的根,具有迭代速度快、算法简单的特点,在工程和科学计算领域有着广泛的应用。 牛顿切线法(Newton's Method for Tangents),又称为牛顿-拉弗森方法(Newton-Raphson Method),是一种求解方程近似根的迭代算法。其基本思想是利用函数在某点的切线来逼近函数的根。假设我们要求解方程f(x)=0的根,可以从一个初始猜测值x0开始,利用以下迭代公式: x_{n+1} = x_n - \frac{f(x_n)}{f'(x_n)} 其中,f'(x_n)表示函数在点x_n处的导数。迭代过程中,通过不断更新x_n值,逐渐逼近方程的根。 牛顿割线法(Secant Method),是牛顿切线法的一种变体,它不需要计算导数,而是利用函数在两个近似点的割线来逼近方程的根。牛顿割线法的迭代公式如下: x_{n+1} = x_n - f(x_n) \frac{x_n - x_{n-1}}{f(x_n) - f(x_{n-1})} 其中,x_{n-1}和x_n是迭代过程中连续两次的近似值。牛顿割线法相比牛顿切线法,其优点在于不需要计算函数的导数,但通常收敛速度会比牛顿切线法慢一些。 在实际应用中,这两种方法都需要注意迭代的起始点选择,否则可能会导致迭代过程不收敛。同时,这两种方法都是局部收敛方法,即它们只能保证在初始点附近有足够的近似根时才收敛。 关于例题和程序,牛顿切线法和牛顿割线法都可以通过编程实现。通常在编程实现时,需要输入函数的表达式、初始猜测值、迭代次数限制以及误差容忍度等参数。程序会根据这些输入,通过循环迭代计算,直到满足误差容忍度或达到迭代次数限制为止。 在编程实现过程中,需要注意以下几点: 1. 初始猜测值的选择对迭代的收敛性有较大影响,需要根据具体问题来合理选择。 2. 当迭代过程中遇到函数值或导数值过大、过小,或者分母趋近于零时,需要进行适当的数值处理,以避免数值不稳定或除以零的错误。 3. 对于不同的方程和函数,可能需要选择不同的迭代终止条件,如设定一个误差值或迭代次数上限。 牛顿法(包含牛顿切线法和牛顿割线法)是一类非常强大的数值解法,尤其适用于求解非线性方程,其基本原理和程序实现的知识点在理工科的许多领域都有着广泛的应用,是数值分析领域的一个基石。 请注意,本知识点仅涵盖标题和描述中提到的内容,压缩包子文件列表中的信息并未提供,因此无法提供相关内容的知识点。
recommend-type

【制图技术】:甘肃高质量土壤分布TIF图件的成图策略

# 摘要 本文针对甘肃土壤分布数据的TIF图件制作进行了系统研究。首先概述了甘肃土壤的分布情况,接着介绍了TIF图件的基础知识,包括其格式特点、空间数据表达以及质量控制方法。随后,文中构建了成图策略的理论框架,分析了土壤分布图的信息需求与数据处理流程,并探讨了成图原则与标准。在实践操作部分,详细阐述了制图软