我在高职教STM32——LCD液晶显示(3)

        大家好,我是老耿,高职青椒一枚,一直从事单片机、嵌入式、物联网等课程的教学。对于高职的学生层次,同行应该都懂的,老师在课堂上教学几乎是没什么成就感的。正因如此,才有了借助 CSDN 平台寻求认同感和成就感的想法。在这里,我准备陆续把自己花了很多心思的教学设计分享出来,主要面向广大师生朋友,单片机老鸟就略过吧。欢迎点赞+关注,各位的支持是本人持续输出的动力,多谢多谢!

        前边我们讲解了LED、按键和蜂鸣器的应用,这三类器件本身工作原理十分简单,因此我们的重点是放在STM32的GPIO上面。这一章我们来学习一下开发板配套的那块厚厚的液晶屏——LCD1602,聚焦的是这个器件本身的特点和工作时序。因此,我们需要熟读它的数据手册,因为手册里告诉了编程的要点、参数、时序等。阅读器件手册是做单片机和嵌入式开发必备的基本能力,我们就从这一章开始锻炼起来吧。为了不让篇幅太长,本章打算分四个部分来讲解,本文是第三部分。

【学习目标】

  1. 了解LCD1602的工作原理
  2. 掌握LCD1602的工作时序
  3. 领悟软件模拟时序的思路和方法

三、液晶静态显示实验

        本章的前两个部分花了不少篇幅,全方面的介绍了LCD1602以及与开发板之间的联系,传递出来的无非就是一个意思——吃透数据手册这别无他法,结合参考程序反复阅读手册,慢慢感悟,开发经验就是这么积累起来的。学完这个入门的液晶屏,后面还有更复杂的彩屏和触摸屏等着我们去学习,依然是“啃”数据手册。好了,下面我们就动手来写一个程序,把手册里的内容转换成代码,驱动LCD1602去显示我们想要的效果。

3.1 任务描述

        编写LCD1602驱动代码,上电之后可以在指定位置显示字符串信息,实验效果如图13所示。

图13 LCD1602静态显示实验效果

3.2 工程文件清单

        与之前的工程一样,控制一类新的硬件就增加一对与之匹配的驱动文件,即图14中的 lcd1602.clcd1602.h

图14 LCD1602工程文件

3.3 工程源码剖析

        这里为了突出源码的功能细节和排版之需,对源码进行了必要的分割处理。

3.3.1 lcd1602.h源码剖析

        该文件源码见代码清单4,主要是LCD1602端口操作的宏定义和驱动函数的声明,每个函数的功能和参数将在下面剖析 lcd1602.c 源码时解读。

//---------------------------------------------------------
// 代码清单4:lcd1602.h
//---------------------------------------------------------

#ifndef _LCD1602_H_
#define _LCD1602_H_
#include "stm32f10x.h"

//---------------------------------------------------------
// 端口操作宏定义
//---------------------------------------------------------
#define	 RS_H	GPIO_SetBits(GPIOC, GPIO_Pin_6)
#define  RS_L	GPIO_ResetBits(GPIOC, GPIO_Pin_6)
#define  RW_H	GPIO_SetBits(GPIOA, GPIO_Pin_11)
#define  RW_L	GPIO_ResetBits(GPIOA, GPIO_Pin_11)
#define  EN_H	GPIO_SetBits(GPIOB, GPIO_Pin_4)
#define  EN_L	GPIO_ResetBits(GPIOB, GPIO_Pin_4)
#define  READ_BUSY()	GPIO_ReadInputDataBit(GPIOC, GPIO_Pin_2)

//通过直接配置寄存器来改变PC2是输入还是输出
//读液晶状态时是输入,写命令和写数据时是输出
//GPIOx->CRL寄存器描述见手册8.2.1小节(P113)
#define PC2_OUT()	{GPIOC->CRL&=0xFFFFF0FF; GPIOC->CRL|=0x00000300;}
#define PC2_IN()	{GPIOC->CRL&=0xFFFFF0FF; GPIOC->CRL|=0x00000800;}

//---------------------------------------------------------
// 驱动函数声明
//---------------------------------------------------------
_Bool Lcd1602_WaitReady(void);
void Lcd1602_SendByte(u8 byte);
void Lcd1602_WriteCmd(u8 byte);
void Lcd1602_WriteData(u8 byte);
void Lcd1602_ShowChar(u8 x, u8 y, u8 ch);
void Lcd1602_ShowStr(u8 x, u8 y, u8 *str);
void Lcd1602_Clear(u8 pos);
void Lcd1602_Init(void);
void Lcd1602_Printf(u8 x, u8 y, char *fmt, ...);

#endif

3.3.2 lcd1602.c源码剖析

        该文件就是所有LCD1602驱动函数的定义,下面就逐个进行剖析。

        1) 头文件部分

        首先,把必要的头文件都加进来,如代码清单5所示。

/*
 ************************************************************************
 * 代码清单5:lcd1602.c的头文件
 * 描    述:LCD1602初始化、驱动
 * 平    台:麒麟座V3.2
 * 作    者:老耿
 * 日    期:2024-04-09
 * 固 件 库:ST3.5.0
 * 版    本:V1.0
 * 修改记录:无
 ************************************************************************
*/

//必要的头文件
#include "delay.h"
#include "lcd1602.h"

//C库
#include <stdarg.h>
#include <stdio.h

        2) Lcd1602_WaitReady()函数源码

        该函数就是用来检测液晶是否准备好,返回1表示“忙”,返回“0”表示“不忙”。详细源码见如下代码清单6。

/*
************************************************************
*	代码清单6:	Lcd1602_WaitReady()函数
*	函数功能:	等待液晶准备好
*	入口参数:	无
*	返回参数:	1:忙,0:不忙
*	说明:
************************************************************
*/
_Bool Lcd1602_WaitReady(void)
{
	PC2_IN();		//PC2输入模式
	RS_L;			//拉低RS
	RW_H;			//拉高RW
	EN_L;			//
	delay_us(1);	//EN高脉冲
	EN_H;			//
	return (_Bool)READ_BUSY();	//返回PC2状态
}

        2) Lcd1602_SendByte()函数源码

        该函数把一个字节(参数byte)送上液晶的8位数据端口,高3位送到PC2 ~ PC0,低5位送上PB9 ~ PB5。送数的过程如代码清单7所示,有一点曲折,但各位可以从中好好体会一下C语言位操作的严谨和奇妙。

/*
************************************************************
*	代码清单7:	Lcd1602_SendByte()函数
*	函数功能:	向LCD1602写一个字节
*	入口参数:	byte:需要写入的数据
*	返回参数:	无
*	说明:		
************************************************************
*/
void Lcd1602_SendByte(u8 byte)
{
	u16 value = 0;
	value = GPIO_ReadOutputData(GPIOB);		//读取GPIOB的数据
	value &= ~(0x001F << 5);				//清除bit5~8
	value |= ((u16)byte & 0x001F) << 5;		//将要写入的数据取低5位并左移5位
	GPIO_Write(GPIOB, value);				//写入GPIOB
	
	value = GPIO_ReadOutputData(GPIOC);		//读取GPIOC的数据
	value &= ~(0x0007 << 0);				//清除bit0~2
	value |= ((u16)byte & 0x00E0) >> 5;		//将要写入的数据取高3位并右移5位
	GPIO_Write(GPIOC, value);				//写入GPIOC
	
	delay_us(10);
}

        首先,我们得清楚,要改变的只有PC2 ~ PC0、PB9 ~ PB5这8位,而这两组I/O的其他位是不能变的,因为其它I/O还连着别的硬件呢。所以,才有了先保存这组I/O的值。接下来,低5位的操作过程可以用图15来表示,这几句很好的诠释了C语言常见的位操作在嵌入式层面是如何应用的,希望各位能好好领悟。同理,高3位送到PC2~PC0,各位可以自己琢磨和推导一下。

图15 PB9~PB5的送数过程

        3) Lcd1602_WriteCmd()函数源码

        该函数实现写一个命令(参数byte)到LCD1602,就是按照数据手册上写命令的时序编写的,大家可以对照手册来阅读,源码见如下代码清单8。

/*
************************************************************
*	代码清单8:	Lcd1602_WriteCmd()函数
*	函数功能:	向LCD1602写命令
*	入口参数:	byte:需要写入的命令
*	返回参数:	无
*	说明:
************************************************************
*/
void Lcd1602_WriteCmd(u8 byte)
{
	while(Lcd1602_WaitReady());		//等到不忙
	PC2_OUT();						//PC2输出模式
	RS_L;							//拉低RS
	RW_L;							//拉低RW
	Lcd1602_SendByte(byte);			//准备命令码
	EN_H;							//拉高使能
	delay_us(20);					//保持一定时间
	EN_L;							//拉低使能
	delay_us(5);
}

        4) Lcd1602_WriteData()函数源码

        该函数与写命令函数是一个套路,就是通过拉高RS改成了数据模式,源码见代码清单9。

/*
************************************************************
*	代码清单9:	Lcd1602_WriteData()
*	函数功能:	向LCD1602写数据
*	入口参数:	byte:需要写入的数据
*	返回参数:	无
*	说明:
************************************************************
*/
void Lcd1602_WriteData(u8 byte)
{
	while(Lcd1602_WaitReady());		//等到不忙
	PC2_OUT();						//PC2输出模式
	RS_H;							//拉高RS
	RW_L;							//拉低RW
	Lcd1602_SendByte(byte);			//准备数据
	EN_H;							//拉高使能
	delay_us(20);					//保持一定时间
	EN_L;							//拉低使能
	delay_us(5);
}

        5) Lcd1602_SetCursor()函数源码

        该函数用来设置光标的位置,参数x和y是位置坐标,x是行坐标(0表示第一行,1表示第二行),y是列坐标(0~15),源码见如下代码清单10。

/*
************************************************************
*	代码清单10:	Lcd1602_SetCursor()函数
*	函数功能:	设置显示RAM地址地址,即光标位置
*	入口参数:	x:行坐标(0第一行,1第二行)
*				y:列坐标(0~15)
*	返回参数:	无
*	说明:
************************************************************
*/
void Lcd1602_SetCursor(u8 x, u8 y)
{
	u8 addr;
	
	if(x==0)				//第一行
		addr = 0x00 + y;
	else					//第二行
		addr = 0x40 + y;

	Lcd1602_WriteCmd(addr|0x80);	//写入地址
}

        6) Lcd1602_ShowChar()函数源码

        该函数用来显示单个字符,参数x和y与上面一样,确定在哪个位置显示,ch为字符内容,源码见如下代码清单11。

/*
************************************************************
*	代码清单11:	Lcd1602_ShowChar()函数
*	函数功能:	在液晶上显示单个字符
*	入口参数:	x和y:显示的坐标(同上)
*				ch:待显示的字符
*	返回参数:	无
*	说明:
************************************************************
*/
void Lcd1602_ShowChar(u8 x, u8 y, u8 ch)
{
	Lcd1602_SetCursor(x, y);	//设置坐标
	Lcd1602_WriteData(ch);		//显示字符
}

        7) Lcd1602_ShowStr()函数源码

        该函数用来显示字符串信息,参数x和y与上面一样,确定从哪个位置开始显示,*str指向待显示的字符串空间,源码见如下代码清单12。

/*
************************************************************
*	代码清单12:	Lcd1602_ShowStr()函数
*	函数功能:	在液晶上显示字符串
*	入口参数:	x和y:显示的起始坐标(同上)
*				str:字符串指针
*	返回参数:	无
*	说明:
************************************************************
*/
void Lcd1602_ShowStr(u8 x, u8 y, u8 *str)
{
	Lcd1602_SetCursor(x, y);
	
	//每写完一个字符,光标会自动指向下一个位置
	while(*str)		//字符串没结束就不停
	{
		Lcd1602_WriteData(*str);	//写入当前字符
		str++;						//指向下一个字符
	}
}

        8) Lcd1602_Clear()函数源码

        该函数用来清屏,参数pos可取值为0、1、2,分别表示清除第一行、第二行和整屏,源码见如下代码清单13。

/*
************************************************************
*	代码清单13:	Lcd1602_Clear()函数
*	函数功能:	LCD1602清除指定行
*	入口参数:	pos:指定的行
*	返回参数:	无
*	说明:		0-第一行		1-第二行		2-两行
************************************************************
*/
void Lcd1602_Clear(u8 pos)
{
	switch(pos)
	{
		case 0:
			Lcd1602_ShowStr(0, 0 , "                ");
			break;
		case 1:
			Lcd1602_ShowStr(1, 0 , "                ");
			break;		
		case 2:
			Lcd1602_WriteCmd(0x01);	//清屏命令
			break;
		default:
			break;
	}
}

        9) Lcd1602_Init()函数源码

        该函数完成LCD1602上电之后的初始化,一方面将所连接的I/O口全部初始化,另一方面按照数据手册交待的复位步骤对液晶进行初始化,源码见如下代码清单14。

/*
************************************************************
*	代码清单14:	Lcd1602_Init()函数
*	函数功能:	LCD1602初始化
*	入口参数:	无
*	返回参数:	无
*	说明:		RW-PA11		RS-PC6		EN-PC3
*				D7~D5 - PC2~PC0		D4~D0 - PB9~PB5
************************************************************
*/
void Lcd1602_Init(void)
{
	GPIO_InitTypeDef gpio_initstruct;
	
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | \
						   RCC_APB2Periph_GPIOB | \
						   RCC_APB2Periph_GPIOC, ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);
    GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable, ENABLE);	//禁止JTAG功能
	
	gpio_initstruct.GPIO_Mode = GPIO_Mode_Out_PP;
	gpio_initstruct.GPIO_Pin = GPIO_Pin_4 | GPIO_Pin_5 | \
							   GPIO_Pin_6 | GPIO_Pin_7 | \
							   GPIO_Pin_8 | GPIO_Pin_9;
	gpio_initstruct.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOB, &gpio_initstruct);
	
	gpio_initstruct.GPIO_Pin = GPIO_Pin_11;
	GPIO_Init(GPIOA, &gpio_initstruct);
	
	gpio_initstruct.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | \
							   GPIO_Pin_2 | GPIO_Pin_6;
	GPIO_Init(GPIOC, &gpio_initstruct);
			
    Lcd1602_WriteCmd(0x38);	//16*2显示,5*7点阵,8位数据接口
    Lcd1602_WriteCmd(0x0C);	//开显示,光标关闭
    Lcd1602_WriteCmd(0x06);	//字符不动,光标移动
    Lcd1602_WriteCmd(0x01);	//清屏
}

        LCD1602液晶手册提供了一个初始化过程,由于不检测“忙”位,所以程序比较复杂,如图16所示。而我们编写的程序已经将检测“忙”位的功能嵌入到写操作里面了,所以只用了最后4行语句就完成了同样效果,更加简易方便。手册上描述的那个,大家仅作了解即可。以后在别的资料里看到了与我们这类不一样的初始化也不要困惑,注意跟我们这里联系和对比。

图16 数据手册上的初始化过程

3.3.3 main.c源码剖析

        主程序比较简单,完成初始化之后就调用显示函数在屏上指定的位置显示指定的字符串,源码见如下代码清单15。

/**
 ******************************************************
 * 代码清单15:main.c
 * 项    目:LCD1602液晶显示
 * 任务描述:静态显示
 * 实验平台:OneNET STM32开发板V3.2
 * 作    者:老耿
 * 日    期:yyyy/mm/dd
 ******************************************************
**/
 
//-----------------------------------------------------
// 必要的头文件
//-----------------------------------------------------
#include "delay.h"
#include "lcd1602.h"

int main()
{
	delay_init();				//Systick初始化,用于普通的延时
	Lcd1602_Init();				//LCD1602初始化
	
	Lcd1602_ShowStr(0, 3, "KylinV3.2");		//第一行第4个字符开始显示字符串
	Lcd1602_ShowStr(1, 2, "STM32 Board");	//第二行第3个字符开始显示字符串

	while(1);
}

3.3.4 验证与测试

        程序下载前,接好液晶屏和电源适配器,并将电源拨动开关置于OFF处(如图17所示)。程序下载后,电源开关拨至ON,即可实现实验效果。

图17 实验效果与电源拨动开关

(第三部分完,共四部分) 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

前端开发流程与技术选型

目录 一、简介 二、前端职责 三、开发步骤 四、技术选型 五、页面展示 一、简介 做一个网站时&#xff0c;能看到的一切都是前端程序员的工作&#xff0c;负责网页或者app的结构、样式、用户操作网站时的事件逻辑&#xff08;比如点击一个按钮&#xff09;。 二、前端职…

一、系统学习微服务遇到的问题集合

1、启动了nacos服务&#xff0c;没有在注册列表 应该是版本问题 Alibaba-nacos版本 nacos-文档 Spring Cloud Alibaba-中文 Spring-Cloud-Alibaba-英文 Spring-Cloud-Gateway 写的很好的一篇文章 在Spring initial上面配置 start.aliyun.com 重新下载 < 2、 No Feign…

嵌入式系统中的加解密签名

笔者来了解一下嵌入式系统中的加解密 1、背景与名词解释 笔者最近在做安全升级相关的模块&#xff0c;碰到了一些相关的概念和一些应用场景&#xff0c;特来学习记录一下。 1.1 名词解释 对称加密&#xff1a;对称加密是一种加密方法&#xff0c;使用相同的密钥&#xff08;…

力扣刷题 杨辉三角(使用c++ vector解法)

杨辉三角 题目描述示例1示例2提示:代码 题目描述 给定一个非负整数 numRows&#xff0c;生成「杨辉三角」的前 numRows 行。 在「杨辉三角」中&#xff0c;每个数是它左上方和右上方的数的和。 示例1 输入: numRows 5 输出: [[1],[1,1],[1,2,1],[1,3,3,1],[1,4,6,4,1]] 示例2 …

4、SpringMVC 实战小项目【加法计算器、用户登录、留言板、图书管理系统】

SpringMVC 实战小项目 3.1 加法计算器3.1.1 准备⼯作前端 3.1.2 约定前后端交互接⼝需求分析接⼝定义请求参数:响应数据: 3.1.3 服务器代码 3.2 ⽤⼾登录3.2.1 准备⼯作3.2.2 约定前后端交互接⼝3.2.3 实现服务器端代码 3.3 留⾔板实现服务器端代码 3.4 图书管理系统准备后端 3…

【内存管理】页面分配机制

前言 Linux内核中是如何分配出页面的&#xff0c;如果我们站在CPU的角度去看这个问题&#xff0c;CPU能分配出来的页面是以物理页面为单位的。也就是我们计算机中常讲的分页机制。本文就看下Linux内核是如何管理&#xff0c;释放和分配这些物理页面的。 伙伴算法 伙伴系统的…

Visual Studio开发环境搭建

原文&#xff1a;https://blog.c12th.cn/archives/25.html Visual Studio开发环境搭建 测试&#xff1a;笔记本原装操作系统&#xff1a;Windows 10 家庭中文版 资源分享链接&#xff1a;提取码&#xff1a;qbt2 注意事项&#xff1a;注意查看本地硬盘是否够用&#xff0c;建议…

在阿里云使用Docker部署MySQL服务,并且通过IDEA进行连接

阿里云使用Docker部署MySQL服务&#xff0c;并且通过IDEA进行连接 这里演示如何使用阿里云来进行MySQL的部署&#xff0c;系统使用的是Linux系统 (Ubuntu)。 为什么使用Docker? 首先是因为它的可移植性可以在任何有Docker环境的系统上运行应用&#xff0c;避免了在不通操作系…

SpringBoot+ENC实现密钥加密及使用原理

&#x1f60a; 作者&#xff1a; 一恍过去 &#x1f496; 主页&#xff1a; https://blog.csdn.net/zhuocailing3390 &#x1f38a; 社区&#xff1a; Java技术栈交流 &#x1f389; 主题&#xff1a; SpringBootENC实现密钥加密及使用原理 ⏱️ 创作时间&#xff1a; 202…

AtCoder Beginner Contest 359 A~C(D~F更新中...)

A.Count Takahashi 题意 给出 N N N个字符串&#xff0c;每个字符串为以下两种字符串之一&#xff1a; "Takahashi" "Aoki" 请你统计"Takahashi"出现了多少次。 分析 输入并统计即可。 代码 #include <bits/stdc.h>using namespa…

web集群-nginx(nginx三种文件模块,nginx用户请求流程,域名访问网站,虚拟主机,搭建大型直播购物平台)

nginx文件模块 lineinfile未来修改配置文件使用&#xff0c;类似于sed -i ‘sg’ 和sed ‘cai’ 掌握file模块&#xff1a;创建文件&#xff0c;目录&#xff0c;创建软链接&#xff0c;修改权限和所有者&#xff0c;删除文件目录 服务管理systemd systemctl相当于linux sys…

[stm32]温湿度采集与OLED显示

一、I2C总线协议 I2C&#xff08;Inter-integrated circuit &#xff09;是一种允许从不同的芯片或电路与不同的主芯片通信的协议。它仅用于短距离通信&#xff0c;是一种用于两个或多个设备之间进行数据传输的串行总线技术&#xff0c;它可以让你在微处理器、传感器、存储器、…

UE5开发游戏Tutorial

文章目录 PlayerStart 初始化设置默认 LevelBP_Character 初始化BP_Character 添加动画BP_Character 攻击BP_Enemy 初始化 以及 AI 运动Camera Collision 相机碰撞BP_Character 生命以及伤害Wave Spawner 波生成UI 初始化以及 Damage Screen指定位置随机生成添加声音环境 Envir…

使用SpringCache实现Redis缓存

目录 一 什么是Spring Cache 二 Spring Cache 各注解作用 ①EnableCaching ②Cacheable ③CachePut ④CacheEvict 三实现步骤 ①导入spring cache依赖和Redis依赖 ②配置Redis连接信息 ③在启动类上加上开启spring cache的注解 ④ 在对应的方法上加上需要的注解 一 什么…

PINN解偏微分方程实例4

PINN解偏微分方程实例4 一、正问题1. Diffusion equation2. Burgers’ equation3. Allen–Cahn equation4. Wave equation 二、反问题1. Burgers’ equation3. 部分代码示例 本文使用 PINN解偏微分方程实例1中展示的代码求解了以四个具体的偏微分方程&#xff0c;包括Diffusio…

长亭谛听教程部署和详细教程

PPT 图片先挂着 挺概念的 谛听的能力 hw的时候可能会问你用过的安全产品能力能加分挺重要 溯源反制 反制很重要感觉很厉害 取证分析 诱捕牵制 其实就是蜜罐 有模板直接爬取某些网页模板进行伪装 部署要求 挺低的 对linux内核版本有要求 需要root 还有系统配置也要修改 …

论文阅读--Efficient Hybrid Zoom using Camera Fusion on Mobile Phones

这是谷歌影像团队 2023 年发表在 Siggraph Asia 上的一篇文章&#xff0c;主要介绍的是利用多摄融合的思路进行变焦。 单反相机因为卓越的硬件性能&#xff0c;可以非常方便的实现光学变焦。不过目前的智能手机&#xff0c;受制于物理空间的限制&#xff0c;还不能做到像单反一…

long long ago

一、long 众所周知&#xff0c;英文单词 long&#xff0c;表示长,长的。 但是&#xff0c;还有很多你不知道到的东西&#xff0c;根据英文单词首字母象形原则&#xff0c;我们可以做一下单词long结构分析&#xff1a; long l长 ong长 什么意思&#xff1f;就是说首字线 l…

Maven的依赖传递、依赖管理、依赖作用域

在Maven项目中通常会引入大量依赖&#xff0c;但依赖管理不当&#xff0c;会造成版本混乱冲突或者目标包臃肿。因此&#xff0c;我们以SpringBoot为例&#xff0c;从三方面探索依赖的使用规则。 1、 依赖传递 依赖是会传递的&#xff0c;依赖的依赖也会连带引入。例如在项目中…

AI大模型企业应用实战(14)-langchain的Embedding

1 安装依赖 ! pip install --upgrade langchain ! pip install --upgrade openai0.27.8 ! pip install -U langchain-openai ! pip show openai ! pip show langchain ! pip show langchain-openai 2 Embed_documents # 1. 导入所需的库 from langchain_openai import Open…