【STM32】文件系统FATFS与Flash的初步使用

文件系统简介

简介可以不看,直接看移植步骤
在这里插入图片描述
文件系统是介于应用层和底层间的模糊层。底层提供API,比如说使用SDIO或者SPI等读写一个字节。文件系统把这些API组合包装起来,并且提供一些列函数,我们可以使用这些函数进行更进一步的对存储设备的操作。

底层:操作单片机外设,读写。需要我们进行配置。
中层:中间层 FATFS 模块,实现了 FAT 文件读/写协议。一般不管
顶层:文件系统提供给我们的函数库,我们就是要使用这些。
FATFS的实现过程(白色框中的内容是需要我们自己实现的也就是底层设备的输入输出以及最高层的用户应用程序,蓝色框由fatfs提供)
在这里插入图片描述

源码获取

FATFS 的源码及英文详述,大家可以在:http://elm-chan.org/fsw/ff/00index_e.html 这个网站下载。下载解压之后的根目录下有这么两个文件夹,源码和帮助文档。
在这里插入图片描述
DOC文件夹下有Fatfs提供的具体函数的使用方法
SRC文件的构成如下
在这里插入图片描述

在这里插入图片描述
Fatfs的源码阅读可以参考《零死角玩转 STM32F103—指南者》中的阅读提示,如果只是想使用,那么只需要看如何移植

移植FATFS 准备工作

背景:两个文件 ffconf.h diskio.c

FATFS 模块在移植的时候,我们一般只需要修改 2 个文件,即 ffconf.hdiskio.c
ffconf.h :FATFS模块的所有配置项都是存放在 ffconf.h 里面,我们可以通过配置里面的一些选项,来满足自己的需求.FATFS 的说明文档里面有很详细的介绍
在这里插入图片描述

以spi读写flash为例,在实现了spi往flash中读写单个字节的底层代码之后,就可以进行fatfs的移植了。

0.创建一个新的工程,并且实现简单的点灯,spi操作等功能。

1.准备一份工程源码称为SPI—FatFs。将。将 FatFs 源码中的 src 文件夹整个文件夹拷贝一份至“SPI—FatFs 文件系统\USER\”文件夹下并修改名为“FATFS”。文件夹名字没有固定名称,这里只是方便书写。

2.打开工程文件,并将 FatFs 组件文件添加到工程中,需要添加有 ff.c、diskio.c 和cc936.c 三个文件.
在这里插入图片描述
3.添加 FATFS 文件夹到工程的 include 选项中。打开工程选项对话框,选择“C/C++”选项下的“Include Paths”项目,在弹出路径设置对话框中选择添加“FATFS”文件夹
在这里插入图片描述
4.修改 diskio.c 文件和 ffconf.h 文件,diskio.c 文件内容是与底层设备输入输出接口函数文件,不同硬件设计驱动就不同,需要的文件也不同;FatFs默认使用日语,我们想要支持简体中文需要
修改 FatFs 的配置,即修改 ffconf.h 文件。

FatFs 底层设备驱动函数

文件系统实现中间模糊层,提供上层函数给用户使用。需要用户实现底层接口。diskio.c 文件内容就是与底层接口相关的。

移植需要用户支持函数,一般只有前六个需要使用
在这里插入图片描述
disk_status,disk_initialize,disk_read是必要配置的。
disk_write,get_fattime,disk_ioctl (CTRL_SYNC)是实现创建文件、修改文件需要的。
为支持简体中文长文件名称需要添加 ff_convert 和 ff_wtoupper 函数,实际这两个已经在 cc936.c 文件中实现,我们只要直接把 cc936.c 文件添加到工程中就可以。

移植FATFS 主要步骤

1.配置数据类型:在 integer.h 里面去定义好数据的类型。这里需要了解你用的编译器的数据类型,并根据编译器定义好数据类型。

2.配置:通过 ffconf.h 配置 FATFS 的相关功能,以满足你的需要。

3.函数编写:打开 diskio.c,进行底层驱动编写,需要编写 5 个接口函数。
在这里插入图片描述

配置integer.h 以定义数据类型(一般不需要)

我们使用的是 MDK5.34 编译器,数据类型和 integer.h 里面定义的一致,所以此步,我们不需要做任何改动。

配置 ffconf.h 选择模式

关于 ffconf.h 里面的相关配置,配置修改为我们需要的值即可,其他的配置用默认配置。

1 #define _USE_MKFS 1	//格式化功能选择,为使用 FatFs 格式化功能,需要把它设置为 1。
2 #define _CODE_PAGE 936	//语言功能选择,并要求把相关语言文件添加到工程宏。为支持简体中文文件名需要使用“936” 指的是把 cc936.c 文件添加到工程中
3 #define _USE_LFN 2	//长文件名支持,默认不支持长文件名,这里配置为 2,支持长文件名,并指定使用栈空间为缓冲区。
4 #define _VOLUMES 2	//指定物理设备数量
5 #define _MIN_SS 512	//指定扇区大小的最小值和最大值。SD 卡扇区大小一般都为 512字节,SPI Flash芯片扇区大小一般设置为 4096字节
6 #define _MAX_SS 4096	//指定扇区大小的最小值和最大值

在这里插入图片描述

为每个设备定义一个物理编号

 #define ATA 	0 	// 预留 SD 卡使用#define SPI_FLASH 	1 // 外部 SPI Flash

实现五个函数

设备状态获取(disk_status)、设备初始化(disk_initialize)、扇区读取(disk_read)、扇区写(disk_write)、其他控制(disk_ioctl)。

disk_initialize

函数名称disk_initialize
函数原型DSTATUS disk_initialize(BYTE Drive)
功能描述初始化磁盘驱动器
函数参数Drive:指定要初始化的逻辑驱动器号,即盘符,应当取值 0~9
返回值函数返回一个磁盘状态作为结果,对于磁盘状态的细节信息,请参考 disk_status函数
所在文件ff.c
实例disk_initialize(0); /* 初始化驱动器 0 */
注意事项disk_initialize 函数初始化一个逻辑驱动器为读/写做准备,函数成功时,返回值的 STA_NOINIT 标志被清零;应用程序不应调用此函数,否则卷上的 FAT 结构可能会损坏;如果需要重新初始化文件系统,可使用 f_mount 函数;在 FatFs 模块上卷注册处理时调用该函数可控制设备的改变;此函数在 FatFs 挂在卷时调用,应用程序不应该在 FatFs 活动时使用此函数
DSTATUS disk_status (BYTE pdrv /* 物理编号 */)
{
DSTATUS status = STA_NOINIT;
switch (pdrv) {case ATA: /* SD CARD 预留,也可以*/break;case SPI_FLASH: /* SPI Flash 状态检测:读取 SPI Flash 设备 ID */if (sFLASH_ID == SPI_FLASH_ReadID()) {/* 设备 ID 读取结果正确 */status &= ~STA_NOINIT;} else { /* 设备 ID 读取结果错误 */status = STA_NOINIT;}break;default:status = STA_NOINIT;}return status;}

注意点
SPI_FLASH_ReadID函数由用户自己实现,目的是检测设备是否已经就绪。

disk_status

函数名称disk_status
函数原型DRESULT disk_status (BYTE Drive)
功能描述返回当前磁盘驱动器的状态
函数参数Drive:指定要确认的逻辑驱动器号,即盘符,应当取值 0~9
返回值磁盘状态返回下列标志的组合,FatFs 只使用 STA_NOINIT 和 STA_PROTECTEDSTA_NOINIT: 表明磁盘驱动未初始化,下面列出了产生该标志置位或清零的原因:置位:系统复位,磁盘被移除和磁盘初始化函数失败。清零:磁盘初始化函数成功.STA_NODISK:表明驱动器中没有设备,安装磁盘驱动器后总为0 STA_PROTECTED:表明设备被写保护,不支持写保护的设备总为 0,当STA_NODISK 置位时非法
所在文件ff.c
实例disk_status(0); /* 获取驱动器 0 的状态 */
DSTATUS disk_initialize (BYTE pdrv /* 物理编号 */)
{	uint16_t i;DSTATUS status = STA_NOINIT;switch (pdrv) {case ATA: /* SD CARD */break;case SPI_FLASH: /* SPI Flash */SPI_FLASH_Init(); /* 初始化 SPI Flash */i=500;/* 延时一小段时间 */while (--i);SPI_Flash_WAKEUP();/* 唤醒 SPI Flash */status = disk_status(SPI_FLASH);/* 获取 SPI Flash 芯片状态 */break;default:status = STA_NOINIT;}return status;
}

注意点
SPI_FLASH_Init();SPI_Flash_WAKEUP();都是由用户自己实现的底层功能,具体需要参照Flash对应的要求进行编写(一般也就是按着一定的规则发送数据字节过去,所以一般编程顺序是,线实现单字节的spi收发,再按照要求编写多字节的功能,再嵌入文件系统)

disk_read

函数名称disk_read
函数原型DRESULT disk_read (BYTE Drive, BYTE* Buffer, DWORD SectorNumber, BYTE SectorCount)
功能描述从磁盘驱动器上读取扇区
函数参数Drive:指定逻辑驱动器号,即盘符,应当取值 0~9 Buffer:指向存储读取数据字节数组的指针,需要为所读取字节数的大小,扇区统计的扇区大小是需要的(注:FaFts 指定的内存地址并不总是字对齐的,如果硬件不支持不对齐的数据传输,函数里需要进行处理)SectorNumber:指定起始扇区的逻辑块(LBA)上的地址SectorCount:指定要读取的扇区数,取值 1~128
返回值RES_OK(0):函数成功RES_ERROR:读操作期间产生了任何错误且不能恢复它RES_PARERR:非法参数RES_NOTRDY:磁盘驱动器没有初始化
所在文件ff.c
DRESULT disk_read (BYTE pdrv, /* 设备物理编号(0..) */BYTE *buff, /* 数据缓存区 */DWORD sector, /* 扇区首地址 */UINT count /* 扇区个数(1..128) */)
{DRESULT status = RES_PARERR;switch (pdrv) {case ATA: /* SD CARD */break;case SPI_FLASH:sector+=512;/* 扇区偏移 2MB,外部 Flash 文件系统空间放在 SPI Flash 后面 6MB 空间 */SPI_FLASH_BufferRead(buff, sector <<12, count<<12);status = RES_OK;break;default:status = RES_PARERR;}return status;}

注意点
参数pdrv 为设备物理编号,用户自定义,buff:BYTE 类型指针变量,buff指向用来存放读取到数据的存储区首地址,sector 是一个 DWORD 类型变量,指定要读取数据的扇区首地址。count 是一个 UINT 类型变量,指定扇区数量。
数据类型:BYTE 类型实际是 unsigned char 类型,DWORD 类型实际是 unsigned long 类型,UINT类型实际是 unsigned int 类型,类型定义在 integer.h 文件中。
偏移 :板使用的 SPI Flash 芯片型号为 W25Q64FV,每个扇区大小为 4096 个字节(4KB),总共有8M字节空间若。要从Flash的首地址开始存放文件系统则不需要偏移。 只将后部分6MB空间分配给FatFs使用,即 FatFs 是从 2MB 空间开始,为实现这个效果需要将所有的读写地址都偏移 512 个扇区空间
SPI_FLASH_BufferRead():读flash的底层程序,实现在指定地址读取指定长度的数据,示例

/**
* @brief 读取 FLASH 数据
* @param pBuffer,存储读出数据的指针
* @param ReadAddr,读取地址
* @param NumByteToRead,读取数据长度
* @retval 无
*/
void SPI_FLASH_BufferRead(u8* pBuffer, u32 ReadAddr, u16 NumByteToRead)
{SPI_FLASH_CS_LOW(); /* 选择 FLASH: CS 低电平 */SPI_FLASH_SendByte(W25X_ReadData); /* 发送 读 指令 */SPI_FLASH_SendByte((ReadAddr & 0xFF0000) >> 16) /* 发送 读 地址高位 */SPI_FLASH_SendByte((ReadAddr& 0xFF00) >> 8); /* 发送 读 地址中位 */SPI_FLASH_SendByte(ReadAddr & 0xFF); /* 发送 读 地址低位 */while (NumByteToRead--)/* 读取数据 */{*pBuffer = SPI_FLASH_SendByte(Dummy_Byte);/* 读取一个字节*/pBuffer++;/* 指向下一个字节缓冲区 */}SPI_FLASH_CS_HIGH();/* 停止信号 FLASH: CS 高电平 */
}

disk_write

函数名称disk_write
函数原型DRESULT disk_write (BYTE Drive, const BYTE* Buffer, DWORD SectorNumber, BYTE SectorCount)
功能描述向磁盘写入一个或多个扇区
函数参数Drive:指定逻辑驱动器号,即盘符,应当取值 0~9Buffer:指向要写入字节数组的指针(注:FaFts 指定的内存地址并不总是字对齐的,如果硬件不支持不对齐的数据传输,函数里需要进行处理)SectorNumber:指定起始扇区的逻辑块(LBA)上的地址SectorCount:指定要写入的扇区数,取值 1~128
返回值RES_OK(0):函数成功RES_ERROR:写操作期间产生了任何错误且不能恢复它RES_WRPER:媒体被写保护RES_PARERR:非法参数RES_NOTRDY:磁盘驱动器没有初始化
所在文件ff.c
注意事项只读配置中不需要此函数
DRESULT disk_write (BYTE pdrv, /* 设备物理编号(0..) */const BYTE *buff, /* 欲写入数据的缓存区 */DWORD sector, /* 扇区首地址 */UINT count /* 扇区个数(1..128) */
)
{uint32_t write_addr;DRESULT status = RES_PARERR;if (!count) {return RES_PARERR; /* Check parameter */}switch (pdrv) {case ATA: /* SD CARD */break;case SPI_FLASH:/* 扇区偏移 2MB,外部 Flash 文件系统空间放在 SPI Flash 后面 6MB 空间 */sector+=512;write_addr = sector<<12;SPI_FLASH_SectorErase(write_addr);//对于Flash 先擦除再写SPI_FLASH_BufferWrite((u8 *)buff,write_addr,count<<12);//写status = RES_OK;break;default:status = RES_PARERR;}return status;}

SPI_FLASH_SectorErase SPI_FLASH_BufferWrite用户实现

disk_ioctl

函数名称disk_ioctl
函数原型DRESULT disk_ioctl (BYTE Drive, BYTE Command, void* Buffer)
功能描述控制设备指定特性和除了读/写外的杂项功能
函数参数Drive:指定逻辑驱动器号,即盘符,应当取值 0~9Command:指定命令代码Buffer:指向参数缓冲区的指针,取决于命令代码,不使用时,指定一个 NULL指针
返回值RES_OK(0):函数成功RES_ERROR:写操作期间产生了任何错误且不能恢复它RES_PARERR:非法参数RES_NOTRDY:磁盘驱动器没有初始化
所在文件ff.c
注意事项CTRL_SYNC:确保磁盘驱动器已经完成了写处理,当磁盘 I/O 有一个写回缓存,立即刷新原扇区,只读配置下不适用此命令GET_SECTOR_SIZE:返回磁盘的扇区大小,只用于 f_mkfs()GET_SECTOR_COUNT:返回可利用的扇区数,_MAX_SS ≥ 1024 时可用GET_BLOCK_SIZE:获得擦除块大小,只用于 f_mkfs()CTRL_ERASE_SECTOR:强制擦除一块的扇区,_USE_ERASE > 0 时可用
DRESULT disk_ioctl (
BYTE pdrv, /* 物理编号 */
BYTE cmd, /* 控制指令 */
void *buff /* 写入或者读取数据地址指针 */
)
{DRESULT status = RES_PARERR;switch (pdrv) {case ATA: /* SD CARD */break;case SPI_FLASH:switch (cmd) {/* 扇区数量:1536*4096/1024/1024=6(MB) */case GET_SECTOR_COUNT:*(DWORD * )buff = 1536;break;/* 扇区大小 */case GET_SECTOR_SIZE :*(WORD * )buff = 4096;break;/* 同时擦除扇区个数 */case GET_BLOCK_SIZE :*(DWORD * )buff = 1;break;}status = RES_OK;break;default:status = RES_PARERR;}return status;}

参数pdrv 为设备物理编号,cmd 为控制指令,包括发出同步信号、获取扇区数目、获取扇区大小、获取擦除块数量等等指令,buff 为指令对应的数据指针。
cmd:对 于 SPI Flash 芯 片 , 为 支 持 FatFs 格 式 化 功 能 , 需 要 用 到 获 取 扇 区 数 量(GET_SECTOR_COUNT)指令和获取擦除块数量(GET_BLOCK_SIZE)。另外,SD 卡扇区大小为 512 字节,SPI Flash 芯片一般设置扇区大小为 4096 字节,所以需要用到获取扇区大小(GET_SECTOR_SIZE)指令。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

分享一个python基于数据可视化的智慧社区服务平台源码

&#x1f495;&#x1f495;作者&#xff1a;计算机源码社 &#x1f495;&#x1f495;个人简介&#xff1a;本人七年开发经验&#xff0c;擅长Java、Python、PHP、.NET、Node.js、微信小程序、爬虫、大数据等&#xff0c;大家有这一块的问题可以一起交流&#xff01; &#x1…

MySQL数据类型

目录 MySQL数据类型 数据类型分类 数值类型 tinyint类型 bit类型 float类型 decimal类型 字符串类型 char类型 varchar类型 char和varchar比较 时间日期类型 enum和set类型 MySQL数据类型 数据类型的作用&#xff1a; 决定了存储数据时应该开辟的空间大小。决定…

CocosCreator3.8研究笔记(十二)CocosCreator 字体资源理解

Cocos Creator 常用的字体资源有三种&#xff1a;系统字体、动态字体、位图字体。 一、系统字体 系统字体是调用运行平台自带的系统字体来渲染文字&#xff0c;不需要用户在项目中添加任何相关资源。 使用系统字体&#xff0c; Label 组件 Use System Font 属性需要勾选。 Fo…

手写签名到背景上合为1张图

手写签名到背景上合为1张图 package.json中 "signature_pad": "3.0.0-beta.3"<template><div class"home"><canvas id"canvas" width"500" height"300"></canvas><button click"…

【计算机基础知识4】网络通信协议:TCP、UDP、WebSockets

目录 一、TCP&#xff08;传输控制协议&#xff09; 1. TCP的特点 2. TCP的连接建立和终止 3. TCP的可靠性机制 4. TCP的流量控制 二、UDP&#xff08;用户数据报协议&#xff09; 1. UDP的特点 2. UDP的使用场景 三、WebSockets 1. WebSockets协议的特点 2. WebSock…

长安链BaaS服务平台调研

目录 一、菜单功能二、其他说明2.1、服务平台的部署方式2.2、链本身2.3、建链流程2.4、支持连接已部署的链2.5、链治理投票2.6、支持动态节点操作2.7、支持应用 长安链ChainMaker管理平台文档地址&#xff1a;https://docs.chainmaker.org.cn 一、菜单功能 菜单子菜单/功能点…

强大的JTAG边界扫描(1):基本原理介绍

文章目录 1. 什么是边界扫描&#xff1f;2. JTAG硬件接口3. 边界扫描相关的软硬件4. 学习资料5. 总结 我是怎么了解到边界扫描的呢&#xff1f; 这就要从我淘到一块FPGA板卡的事情说起了。 前段时间我在某二手平台上淘了一块FPGA板子&#xff0c;它长这样&#xff1a; 板子的…

【广州华锐互动】元宇宙技术如何赋能传统工业企业?

随着科技的飞速发展&#xff0c;我们正处于工业革命4.0的时代&#xff0c;数字化、网络化和智能化正在深刻地改变着我们的生活和工作方式。在这个变革的大潮中&#xff0c;工业元宇宙平台应运而生&#xff0c;为企业带来了前所未有的机遇和挑战。 广州华锐互动开发的工业元宇宙…

vite搭建vue3项目

参考视频 1.使用npm搭建vite项目,会自动搭建vue3项目 npm create vitelatest yarn create vite2.手动搭建vue3项目 创建一个项目名称的文件夹执行命令&#xff1a;npm init -y 快速的创建一个默认的包信息安装vite: npm i vite -D -D开发环境的依赖 安装vue,现在默认是vue3.…

nodejs下载指定版本

1.搜索nodejs打开官网nodejs官网&#xff08;除了去官网下载之外还可以使用nvm下载&#xff09; 2.点击downloads 3.往下滑点击Previous Releases(以前的版本) 4.找到你想下载的版本点开&#xff08;此处可能没你想要的具体版本&#xff0c;没关系找到大版本号相同的点开就行了…

【PowerQuery】Excel 的自动刷新功能-最低一分钟刷新

在Excel集成了PowerQuery之后,它提供了数据的手动刷新功能之外,也提供了数据的自动刷新功能。需要注意的是,PowerQuery提供的自动刷新功能是针对连接的,也就是说在PowerQuery自动刷新功能不是全局刷新功能,而是针对连接本身提供。接下来我们来看一下如何实现PowerQuery连接…

快速实现抖音上下滑动,你不知道的ViewPager2用法,信息量巨大,建议收藏点赞。老tier~

万能ViewPager2适配器–SmartViewPager2Adapter 特点功能 完全脱离xml&#xff0c;所有效果只需要通过api调用 具体功能&#xff1a;1. 两句代码实现抖音列表效果2. 无感且丝滑&#xff0c;动态从头部或者底部加载数据3. 设置上下加载监听&#xff0c;再达到预加载limit的时…

Python之面向对象(一)

目录 面向对象为什么要面向对象&#xff1f;要素 定义一些特性内存空间实例的创建与初始化创建实例方法__new__初始化实例方法__init__ 类的继承多态新式类与经典类 面向对象 为什么要面向对象&#xff1f; 方便版本更新迭代&#xff0c;程序结构清晰明了 要素 类&#xff…

Java(三)逻辑控制(if....else,循环语句)与方法

逻辑控制&#xff08;if....else&#xff0c;循环语句&#xff09;与方法 四、逻辑控制1.if...else(常用)1.1表达格式&#xff08;三种&#xff09; 2.switch...case(用的少)2.1表达式 3.while(常用)3.1语法格式3.2关键字beak&#xff1a;3.3关键字 continue&#xff1a; 4.for…

【web开发】6、Django(1)

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 一、Django是什么&#xff1f;二、使用步骤1.安装Django2.创建项目3.创建app4.快速上手 数据库操作1.安装第三方模块2.自己创建数据库3.DJango链接数据库4.DJango操…

vue 部署到本机IIS 部署 SPA 应用

安装 URL Rewrite Works With: IIS 7, IIS 7.5, IIS 8, IIS 8.5, IIS 10 URL Rewrite : The Official Microsoft IIS Site 目前电脑IIS是6版本的&#xff0c;以下的方法不太合适操作。目前用Nginx部署&#xff0c;够用了。 nginx配置参考&#xff1a; uni-app 前面项目&am…

2023年09月编程语言流行度排名

点击查看最新编程语言流行度排名&#xff08;每月更新&#xff09; 2023年09月编程语言流行度排名 编程语言流行度排名是通过分析在谷歌上搜索语言教程的频率而创建的 一门语言教程被搜索的次数越多&#xff0c;大家就会认为该语言越受欢迎。这是一个领先指标。原始数据来自…

无需麻烦验证,文字验证码一键通过

前言 文字验证码&#xff0c;简单易用&#xff0c;安全可靠&#xff01;不需要麻烦的图形识别。这种验证方式不仅方便快捷&#xff0c;而且能有效防止恶意攻击和机器人访问。无需担心复杂操作&#xff0c;只需几秒钟就能完成验证过程。保护您的个人信息和数据安全&#xff0c;…

QML与C++的交互操作

QML旨在通过C 代码轻松扩展。Qt QML模块中的类使QML对象能够从C 加载和操作&#xff0c;QML引擎与Qt元对象系统集成的本质使得C 功能可以直接从QML调用。这允许开发混合应用程序&#xff0c;这些应用程序是通过混合使用QML&#xff0c;JavaScript和C 代码实现的。除了从QML访问…

『SpringBoot 源码分析』run() 方法执行流程:(2)刷新应用上下文-准备阶段

『SpringBoot 源码分析』run() 方法执行流程&#xff1a;&#xff08;2&#xff09;刷新应用上下文-准备阶段 基于 2.2.9.RELEASE问题&#xff1a;当方法进行了注释标记之后&#xff0c;springboot 又是怎么注入到容器中并创建类呢&#xff1f; 首先创建测试主程序 package …