Chinaunix首页 | 论坛 | 博客
  • 博客访问: 41630
  • 博文数量: 14
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 165
  • 用 户 组: 普通用户
  • 注册时间: 2022-11-22 23:41
个人简介

将分享技术博文作为一种快乐,提升自己帮助他人

文章分类

全部博文(14)

文章存档

2023年(9)

2022年(5)

我的朋友

分类: LINUX

2022-11-30 07:27:08

   DPDK网络功能中使用的rte_mbuf作用类似于内核态网络中的sk_buff,它是对接网络驱动和协议栈的接口。rte_mbuf的内存是应用在申请mbuf_pool时创建的,在《DPDK rte_mempool创建与使用》一文中有介绍,在此无需再细说其过程。申请pktmbuf pool对外开放的API如下:

点击(此处)折叠或打开

  1. struct rte_mempool *
  2. rte_pktmbuf_pool_create_by_ops(const char *name, unsigned int n,
  3.                                unsigned int cache_size, uint16_t priv_size,
  4.                                uint16_t data_room_size, int socket_id,
  5.                                const char *ops_name)
  6. struct rte_mempool *
  7. rte_pktmbuf_pool_create(const char *name, unsigned int nunsigned int cache_size,
  8.                         uint16_t priv_size, uint16_t data_room_sizeint socket_id)
rte_pktmbuf_pool_create_by_ops()和rte_pktmbuf_pool_create()的差异在于前者指定了rte_mempool_ops的名字。如果上层应用自己没有实现rte_mempool_ops,或者在eal层初始化时,没有通过使--mbuf-pool-ops-name来指定,则用rte_pktmbuf_pool_create()创建时,默认会使用ring_mp_mc(支持多生产者多消费者)。从这两个接口,可以看出,创建pktmbuf pool时,需要指定:mempool的name,mbuf个数,核的local cache大小,mbuf中私有数据的大小,mbuf中data_room的大小以及从哪个socket_id上申请。其中mbuf的个数,上层应用在进行初始化时,需根据模型进行估算,确保网络设备驱动和应用中均够用,不然可能会出现申请mbuf失败的问题。
从rte_pktmbuf_pool_create_by_ops()的接口中的如下代码,可以看出rte_mbuf的内存结构主要由三部分构成:rte_mbuf结构体,私有数据和data room。其中data room包括headroom和报文数据区域构成。headroom大小由RTE_PKTMBUF_HEADROOM宏控制,默认为128,可根据需要修改,该内存可供上层应用进行报文处理使用。

点击(此处)折叠或打开

  1. elt_size = sizeof(struct rte_mbuf) + (unsigned)priv_size (unsigned)data_room_size;
  2. memset(&mbp_priv, 0, sizeof(mbp_priv));
  3. mbp_priv.mbuf_data_room_size = data_room_size;
  4. mbp_priv.mbuf_priv_size = priv_size;
再根据pktmbuf初始化函数rte_pktmbuf_init()中的如下代码,即可得到rte_mbuf的内存结构。

点击(此处)折叠或打开

  1. priv_size = rte_pktmbuf_priv_size(mp);
  2. mbuf_size = sizeof(struct rte_mbuf) + priv_size;
  3. buf_len = rte_pktmbuf_data_room_size(mp);

  4. memset(m, 0, mbuf_size);
  5. /* start of buffer is after mbuf structure and priv data */
  6. m->priv_size = priv_size;
  7. m->buf_addr = (char *)m + mbuf_size;
  8. m->buf_iova = rte_mempool_virt2iova(m) + mbuf_size;
  9. m->buf_len = (uint16_t)buf_len;

  10. /* keep some headroom between start of buffer and data */
  11. m->data_off = RTE_MIN(RTE_PKTMBUF_HEADROOM, (uint16_t)m->buf_len)
rte_mbuf的内存结构如下图:


注意:rte_mbuf结构体由两个cache line构成,其中有很多成员,在此不展开细说,部分域段下文也有提到。rte_mbuf中有个next域段,如果一个报文只有一个mbuf,则mbuf中的next为NULL;如果一个报文由多个mbuf构成,则mbuf的next被用来指向下一个mbuf,结构如下图所示:


发送和接收报文时报文的一些信息一般填写在{BANNED}中国{BANNED}中国第一个mbuf中。


mbuf分配

mbuf的分配接口有多个,作用各不一样。
  1. struct rte_mbuf *rte_mbuf_raw_alloc(struct rte_mempool *mp)
从mp内存池中申请一个未初始化的mbuf,它一般应用在网络驱动的Rx函数中,驱动负责初始化所有必须初始化的域段。如Kunpeng920 hns3网卡的Rx队列初始化时分配Rx队列中的mbuf的函数实现:

点击(此处)折叠或打开

  1. static int
  2. hns3_alloc_rx_queue_mbufs(struct hns3_hw *hw, struct hns3_rx_queue *rxq)
  3. {
  4.     struct rte_mbuf *mbuf;
  5.     uint64_t dma_addr;
  6.     uint16_t i;

  7.     for (i = 0; i < rxq->nb_rx_desc; i++) {
  8.         mbuf = rte_mbuf_raw_alloc(rxq->mb_pool);
  9.         if (unlikely(mbuf == NULL)) {
  10.             hns3_err(hw, "Failed to allocate RXD[%u] for rx queue!",
  11.                  i);
  12.             hns3_rx_queue_release_mbufs(rxq);
  13.             return -ENOMEM;
  14.         }

  15.         rte_mbuf_refcnt_set(mbuf, 1);
  16.         mbuf->next = NULL;
  17.         mbuf->data_off = RTE_PKTMBUF_HEADROOM;
  18.         mbuf->nb_segs = 1;
  19.         mbuf->port = rxq->port_id;

  20.         rxq->sw_ring[i].mbuf = mbuf;
  21.         dma_addr = rte_cpu_to_le_64(rte_mbuf_data_iova_default(mbuf));
  22.         rxq->rx_ring[i].addr = dma_addr;
  23.         rxq->rx_ring[i].rx.bd_base_info = 0;
  24.     }

  25.     return 0;
  26. }

2. struct rte_mbuf *rte_pktmbuf_alloc(struct rte_mempool *mp)
从mp内存池中分配一个新的mbuf,内部已初始化mbuf一些域段为默认值,如报文长度为0和nb_segs为1。该API一般供上层应用构造报文所用,申请到mbuf后,还需对mbuf进行操作,添加报文头和报文内容,并填充mbuf的data_len和pkt_len等域段。如examples/ptpclient/ptpclient.c中代码:

点击(此处)折叠或打开

  1. created_pkt = rte_pktmbuf_alloc(mbuf_pool);
  2. pkt_size = sizeof(struct rte_ether_hdr) +
  3.     sizeof(struct delay_req_msg);

  4. if (rte_pktmbuf_append(created_pkt, pkt_size) == NULL) {
  5.     rte_pktmbuf_free(created_pkt);
  6.     return;
  7. }
  8. created_pkt->data_len = pkt_size;
  9. created_pkt->pkt_len = pkt_size;
  10. eth_hdr = rte_pktmbuf_mtod(created_pkt, struct rte_ether_hdr *);
  11. rte_ether_addr_copy(&eth_addr, &eth_hdr->src_addr);

  12. /* Set multicast address 01-1B-19-00-00-00. */
  13. rte_ether_addr_copy(&eth_multicast, &eth_hdr->dst_addr);

  14. eth_hdr->ether_type = htons(PTP_PROTOCOL);
  15. req_msg = rte_pktmbuf_mtod_offset(created_pkt,
  16.     struct delay_req_msg *, sizeof(struct
  17.     rte_ether_hdr));

  18. req_msg->hdr.seq_id = htons(ptp_data->seqID_SYNC);
  19. req_msg->hdr.msg_type = DELAY_REQ;
  20. req_msg->hdr.ver = 2;
  21. req_msg->hdr.control = 1;
  22. req_msg->hdr.log_message_interval = 127;
  23. req_msg->hdr.message_length =
  24.     htons(sizeof(struct delay_req_msg));
  25. req_msg->hdr.domain_number = ptp_hdr->domain_number;

3. int rte_pktmbuf_alloc_bulk(struct rte_mempool *pool, struct rte_mbuf **mbufs, unsigned count)
出于性能优化的考虑,很多时候可能会用到批量申请mbuf,就会调用该批量申请接口。该接口从pool内存池中批量申请count个mbuf,申请的所有mbuf会被设置为默认值(与rte_pktmbuf_alloc申请的mbuf状态相同)。该接口在PMD驱动(ena驱动的ena_populate_rx_queue()函数中)和上层应用中都会用到。

4.int rte_mempool_get_bulk(struct rte_mempool *mp, void **obj_table, unsigned int n)
该接口是从mp内存池中申请n个mbuf对象,rte_pktmbuf_alloc_bulk()接口内部就调用了该接口。该接口申请的所有mbuf的状态与rte_mbuf_raw_alloc()申请的相同,都是原始的mbuf。也被PMD驱动和上层应用用于性能优化。如Kunpeng hns3 PMD收包函数中使用的批量申请处理:

点击(此处)折叠或打开

  1. static inline struct rte_mbuf *
  2. hns3_rx_alloc_buffer(struct hns3_rx_queue *rxq)
  3. {
  4.     int ret;

  5.     if (likely(rxq->bulk_mbuf_num > 0))
  6.         return rxq->bulk_mbuf[--rxq->bulk_mbuf_num];

  7.     ret = rte_mempool_get_bulk(rxq->mb_pool, (void **)rxq->bulk_mbuf,
  8.                  HNS3_BULK_ALLOC_MBUF_NUM);
  9.     if (likely(ret == 0)) {
  10.         rxq->bulk_mbuf_num = HNS3_BULK_ALLOC_MBUF_NUM;
  11.         return rxq->bulk_mbuf[--rxq->bulk_mbuf_num];
  12.     } else
  13.         return rte_mbuf_raw_alloc(rxq->mb_pool);
  14. }
再如testpmd中txonly.c用来发送报文的代码:

点击(此处)折叠或打开

  1. static void
  2. pkt_burst_transmit(struct fwd_stream *fs)
  3. {
  4.     struct rte_mbuf *pkts_burst[MAX_PKT_BURST];

  5.     ...
  6.     if (rte_mempool_get_bulk(mbp, (void **)pkts_burst,
  7.                 nb_pkt_per_burst) == 0) {
  8.         for (nb_pkt = 0; nb_pkt < nb_pkt_per_burst; nb_pkt++) {
  9.             if (unlikely(!pkt_burst_prepare(pkts_burst[nb_pkt], mbp,
  10.                             &eth_hdr, vlan_tci,
  11.                             vlan_tci_outer,
  12.                             ol_flags,
  13.                             nb_pkt, fs))) {
  14.                 rte_mempool_put_bulk(mbp,
  15.                         (void **)&pkts_burst[nb_pkt],
  16.                         nb_pkt_per_burst - nb_pkt);
  17.                 break;
  18.             }
  19.         }
  20.     } else {
  21.         for (nb_pkt = 0; nb_pkt < nb_pkt_per_burst; nb_pkt++) {
  22.             pkt = rte_mbuf_raw_alloc(mbp);
  23.             if (pkt == NULL)
  24.                 break;
  25.             if (unlikely(!pkt_burst_prepare(pkt, mbp, &eth_hdr,
  26.                             vlan_tci,
  27.                             vlan_tci_outer,
  28.                             ol_flags,
  29.                             nb_pkt, fs))) {
  30.                 rte_pktmbuf_free(pkt);
  31.                 break;
  32.             }
  33.             pkts_burst[nb_pkt] = pkt;
  34.         }
  35.     }

  36.     if (nb_pkt == 0)
  37.         return;

  38.     nb_tx = rte_eth_tx_burst(fs->tx_port, fs->tx_queue, pkts_burst, nb_pkt);
  39.     ...
  40. }

mbuf释放

涉及mbuf释放时,因为下文的mbuf克隆操作存在,会遇到两种mbuf:direct mbuf和indirect mbuf。他们的区别在于,direct mbuf是{BANNED}{BANNED}最佳佳原始的mbuf,indirect mbuf是从direct mbuf中克隆过来的,indirect mbuf的buf_iova和buf_addr均与direct mbuf的相同,即报文indirect mbuf的报文数据指向direct mbuf指向的报文数据。其余mbuf头的信息两者是一样的,indriect mbuf的引用计数refcnt为1

1. void rte_mbuf_raw_free(struct rte_mbuf *m)
释放一个mbuf到它对应的内存池中,调用者必须保证它的引用技术refcnt=1, refcnt=1, next=NULL, nb_segs=1。它不支持释放indirect mbuf,不支持有externel buffer的mbuf,不支持有pinned external buffer的mbuf。相关的宏定义如下:

点击(此处)折叠或打开

  1. #define RTE_MBUF_DIRECT(mb) \
  2.     (!((mb)->ol_flags & (RTE_MBUF_F_INDIRECT | RTE_MBUF_F_EXTERNAL)))
  3. #define RTE_MBUF_CLONED(mb) ((mb)->ol_flags & RTE_MBUF_F_INDIRECT)
  4. #define RTE_MBUF_HAS_EXTBUF(mb) ((mb)->ol_flags & RTE_MBUF_F_EXTERNAL)
  5. #define RTE_MBUF_HAS_PINNED_EXTBUF(mb) \
  6.     (rte_pktmbuf_priv_flags(mb->pool) & RTE_PKTMBUF_POOL_F_PINNED_EXT_BUF)
2. void rte_pktmbuf_free(struct rte_mbuf *m)
释放一个mbuf链到内存池,如果mbuf有多个段,则都会被放到对应的mempool中。支持释放indirect mbuf和indirect mbuf。

3. void rte_pktmbuf_free_seg(struct rte_mbuf *m)
释放一个mbuf,注意如果mbuf是多段时,则应该使用rte_pktmbuf_free去释放。

4.void rte_pktmbuf_free_bulk(struct rte_mbuf **mbufs, unsigned int count)
批量释放mbuf,若该mbuf是多段的,也均会释放到内存池。释放的mbuf必须时direct mbuf,如果释放indirect mbuf可能会导致业务异常。

5. void rte_mempool_put_bulk(struct rte_mempool *mp, void * const *obj_table, unsigned int n)
直接将多个mbuf批量释放到指定内存池,依赖使用者确保被释放的mbuf都是来自该内存池。释放的mbuf必须时direct mbuf,如果释放indirect mbuf可能会导致业务异常。

mbuf拷贝和克隆

mbuf克隆属于浅度拷贝,接口定义如下:
struct rte_mbuf* rte_pktmbuf_clone(struct rte_mbuf *md, struct rte_mempool *mp)
从mp内存池中申请mbuf来克隆目标mbuf,目标mbuf可以是indirect、direct mbuf,和有external buffer的mbuf该接口实现时,内部有一个实现mbuf克隆的关键接口:

void rte_pktmbuf_attach(struct rte_mbuf *mi, struct rte_mbuf *m)。该接口实现将mi的mbuf关联attach到m的mbuf中。mi的mbuf除了自身的引用计数为1外,其余都和md的mbuf的数据域段一致。mi的buff_addr和buf_iova均指向md的mbuf中装的报文数据域。如果该md的mbuf是indirect mbuf,则会通过rte_mbuf_from_indirect(m)对direct mbuf的引用计数+1。rte_pktmbuf_attach的实现如下:

点击(此处)折叠或打开

  1. static inline void rte_pktmbuf_attach(struct rte_mbuf *mi, struct rte_mbuf *m)
  2. {
  3.     RTE_ASSERT(RTE_MBUF_DIRECT(mi) &&
  4.      rte_mbuf_refcnt_read(mi) == 1);

  5.     if (RTE_MBUF_HAS_EXTBUF(m)) {
  6.         rte_mbuf_ext_refcnt_update(m->shinfo, 1);
  7.         mi->ol_flags = m->ol_flags;
  8.         mi->shinfo = m->shinfo;
  9.     } else {
  10.         /* if m is not direct, get the mbuf that embeds the data */
  11.         rte_mbuf_refcnt_update(rte_mbuf_from_indirect(m), 1);
  12.         mi->priv_size = m->priv_size;
  13.         mi->ol_flags = m->ol_flags | RTE_MBUF_F_INDIRECT;
  14.     }

  15.     __rte_pktmbuf_copy_hdr(mi, m);

  16.     mi->data_off = m->data_off;
  17.     mi->data_len = m->data_len;
  18.     mi->buf_iova = m->buf_iova;
  19.     mi->buf_addr = m->buf_addr;
  20.     mi->buf_len = m->buf_len;

  21.     mi->next = NULL;
  22.     mi->pkt_len = mi->data_len;
  23.     mi->nb_segs = 1;

  24.     __rte_mbuf_sanity_check(mi, 1);
  25.     __rte_mbuf_sanity_check(m, 0);
  26. }
被attach的mbuf被打上RTE_MBUF_F_INDIRECT的标记。称为indirect mbuf。

mbuf拷贝是深度拷贝,接口如下:
struct rte_mbuf * rte_pktmbuf_copy(const struct rte_mbuf *m, struct rte_mempool *mp, uint32_t off, uint32_t len)
从mp内存池中申请mbuf来拷贝目标mbuf,目标mbuf可以是indirect、direct mbuf,和有external buffer的mbuf。off是前面拷贝的偏移,len是拷贝的长度。拷贝的长度超过mbuf的数据包长度时,函数内部会自动调整。rte_pktmbuf_copy支持拷贝多段的mbuf,但是mbuf的私有数据不会被拷贝,如果mbuf是一个indirect或external buffer的mbuf,这个特征会被从ol_flags中去掉。

mbuf解封操作

mbuf的解封操作一般供上层应用或者用户态协议栈处理报文使用。
得到mbuf中的报文数据起始位置:

点击(此处)折叠或打开

  1. #define rte_pktmbuf_mtod_offset(m, t, o) \
                    ((t)(void *)((char *)(m)->buf_addr + (m)->data_off + (o)))
  2. #define rte_pktmbuf_mtod(m, t) rte_pktmbuf_mtod_offset(m, t, 0)
m->data_off则是headroom的大小,m->buf_addr+m->data_off则表示mbuf中报文数据的开始位置。例如对于ipv4-udp报文得到ip头的地址:

点击(此处)折叠或打开

  1. ip_hdr = rte_pktmbuf_mtod_offset(pktstruct rte_ipv4_hdr *sizeof(struct rte_ether_hdr));
相对于报文起始地址偏移一个以太网头的大小,则得到ip头的首地址。
同样的得到udp头的地址:

点击(此处)折叠或打开

  1. udp_hdr = rte_pktmbuf_mtod_offset(pktstruct rte_udp_hdr *,
  2.                     sizeof(struct rte_ether_hdr) + sizeof(struct rte_ipv4_hdr));
得到mbuf中headroom的大小接口:uint16_t rte_pktmbuf_headroom(const struct rte_mbuf *m)。实际返回的就是m->data_off。
得到mbuf中tailroom的大小接口如下:

点击(此处)折叠或打开

  1. static inline uint16_t rte_pktmbuf_tailroom(const struct rte_mbuf *m)
  2. {
  3.     __rte_mbuf_sanity_check(m, 0);
  4.     return (uint16_t)(m->buf_len - rte_pktmbuf_headroom(m) -
  5.              m->data_len);
  6. }
即buf的总大小减去headroom和data_len大小,即为tailroom大小。
获取mbuf中报文的长度#define rte_pktmbuf_pkt_len(m) ((m)->pkt_len)
获取当前mbuf数据的长度 #define rte_pktmbuf_data_len(m) ((m)->data_len)
以上两个长度有差别:m->pkt_len是整个报文的长度,m->data_len是当前mbuf中报文数据的长度,当报文有单个mbuf构成时,他们是相等的。报文由多个mbuf构成时,pkt_len等于每个mbuf的data_len之和。

报文数据起始位置向headroom方向扩展len个字节:rte_pktmbuf_prepend(m, len)

点击(此处)折叠或打开

  1. static inline char *rte_pktmbuf_prepend(struct rte_mbuf *m,
  2.                     uint16_t len)
  3. {
  4.     __rte_mbuf_sanity_check(m, 1);

  5.     if (unlikely(len > rte_pktmbuf_headroom(m)))
  6.         return NULL;

  7.     /* NB: elaborating the subtraction like this instead of using
  8.      * -= allows us to ensure the result type is uint16_t
  9.      * avoiding compiler warnings on gcc 8.1 at least */
  10.     m->data_off = (uint16_t)(m->data_off - len);
  11.     m->data_len = (uint16_t)(m->data_len + len);
  12.     m->pkt_len = (m->pkt_len + len);

  13.     return (char *)m->buf_addr + m->data_off;
  14. }

向tailroom方向扩展len个字节的报文数据:rte_pktmbuf_append(m, len)

点击(此处)折叠或打开

  1. static inline char *rte_pktmbuf_append(struct rte_mbuf *m, uint16_t len)
  2. {
  3.     void *tail;
  4.     struct rte_mbuf *m_last;

  5.     __rte_mbuf_sanity_check(m, 1);

  6.     m_last = rte_pktmbuf_lastseg(m);
  7.     if (unlikely(len > rte_pktmbuf_tailroom(m_last)))
  8.         return NULL;

  9.     tail = (char *)m_last->buf_addr + m_last->data_off + m_last->data_len;
  10.     m_last->data_len = (uint16_t)(m_last->data_len + len);
  11.     m->pkt_len = (m->pkt_len + len);
  12.     return (char*) tail;
  13. }
该方式在PMD中也会用到,例如当发送的报文长度太短,需要对进行将报文padding到支持的长度,否则可能会触发硬件异常。

上面是向headroom和tailroom扩展报文数据,rte_mbuf库中也对外开放了向这两个方向移除报文内容的接口。
从报文头移除len个字节的报文数据:rte_pktmbuf_adj(m, len)

点击(此处)折叠或打开

  1. static inline char *rte_pktmbuf_adj(struct rte_mbuf *m, uint16_t len)
  2. {
  3.     __rte_mbuf_sanity_check(m, 1);

  4.     if (unlikely(len > m->data_len))
  5.         return NULL;

  6.     /* NB: elaborating the addition like this instead of using
  7.      * += allows us to ensure the result type is uint16_t
  8.      * avoiding compiler warnings on gcc 8.1 at least */
  9.     m->data_len = (uint16_t)(m->data_len - len);
  10.     m->data_off = (uint16_t)(m->data_off + len);
  11.     m->pkt_len = (m->pkt_len - len);
  12.     return (char *)m->buf_addr + m->data_off;
  13. }

从报文尾移除len个字节的报文数据:rte_pktmbuf_trim(m, len)

点击(此处)折叠或打开

  1. static inline int rte_pktmbuf_trim(struct rte_mbuf *m, uint16_t len)
  2. {
  3.     struct rte_mbuf *m_last;

  4.     __rte_mbuf_sanity_check(m, 1);

  5.     m_last = rte_pktmbuf_lastseg(m);
  6.     if (unlikely(len > m_last->data_len))
  7.         return -1;

  8.     m_last->data_len = (uint16_t)(m_last->data_len - len);
  9.     m->pkt_len = (m->pkt_len - len);
  10.     return 0;
  11. }

链接一个mbuf到另一个mbuf上:rte_pktmbuf_chain(m1, m2)
支持链接多段的mbuf。

点击(此处)折叠或打开

  1. static inline int rte_pktmbuf_chain(struct rte_mbuf *head, struct rte_mbuf *tail)
  2. {
  3.     struct rte_mbuf *cur_tail;

  4.     /* Check for number-of-segments-overflow */
  5.     if (head->nb_segs + tail->nb_segs > RTE_MBUF_MAX_NB_SEGS)
  6.         return -EOVERFLOW;

  7.     /* Chain 'tail' onto the old tail */
  8.     cur_tail = rte_pktmbuf_lastseg(head);
  9.     cur_tail->next = tail;

  10.     /* accumulate number of segments and total length.
  11.      * NB: elaborating the addition like this instead of using
  12.      * -= allows us to ensure the result type is uint16_t
  13.      * avoiding compiler warnings on gcc 8.1 at least */
  14.     head->nb_segs = (uint16_t)(head->nb_segs + tail->nb_segs);
  15.     head->pkt_len += tail->pkt_len;

  16.     /* pkt_len is only set in the head */
  17.     tail->pkt_len = tail->data_len;

  18.     return 0;
  19. }

rte_mbuf读取操作

读rte_mbuf报文内容:
void *rte_pktmbuf_read(const struct rte_mbuf *m, uint32_t off, uint32_t len, void *buf)
从报文长度偏移off位置读取len个字节的报文内容到buf中。

支持dump mbuf的头信息和报文内容到文件:
void rte_pktmbuf_dump(FILE *f, const struct rte_mbuf *m, unsigned dump_len)
dump_len:表示要dump的报文长度。


阅读(3995) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~