| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597 |
- /*
- * Copyright : (C) 2022 Phytium Information Technology, Inc.
- * All Rights Reserved.
- *
- * This program is OPEN SOURCE software: you can redistribute it and/or modify it
- * under the terms of the Phytium Public License as published by the Phytium Technology Co.,Ltd,
- * either version 1.0 of the License, or (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,but WITHOUT ANY WARRANTY;
- * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the Phytium Public License for more details.
- *
- *
- * FilePath: fxhci_dev.c
- * Date: 2022-02-11 13:33:12
- * LastEditTime: 2022-02-18 09:12:46
- * Description: This files is for implementation of XHCI device
- *
- * Modify History:
- * Ver Who Date Changes
- * ----- ------ -------- --------------------------------------
- * 1.0 Zhugengyu 2022/2/7 init commit
- */
- #include <string.h>
- #include "fsleep.h"
- #include "fcache.h"
- #include "fdebug.h"
- #include "fxhci_private.h"
- #define FUSB_DEBUG_TAG "FXHCI_DEV"
- #define FUSB_ERROR(format, ...) FT_DEBUG_PRINT_E(FUSB_DEBUG_TAG, format, ##__VA_ARGS__)
- #define FUSB_WARN(format, ...) FT_DEBUG_PRINT_W(FUSB_DEBUG_TAG, format, ##__VA_ARGS__)
- #define FUSB_INFO(format, ...) FT_DEBUG_PRINT_I(FUSB_DEBUG_TAG, format, ##__VA_ARGS__)
- #define FUSB_DEBUG(format, ...) FT_DEBUG_PRINT_D(FUSB_DEBUG_TAG, format, ##__VA_ARGS__)
- static u32 FXhciGenRounte(FXhci *const xhci, const int hubport, const int hubaddr)
- {
- FASSERT(xhci);
- if (!hubaddr)
- return 0;
- u32 route_string = FXHCI_SC_GET(ROUTE, xhci->dev[hubaddr].ctx.slot);
- int i;
- for (i = 0; i < 20; i += 4)
- {
- if (!(route_string & (0xf << i)))
- {
- route_string |= (hubport & 0xf) << i;
- break;
- }
- }
- return route_string;
- }
- static int FXhciGetRoothubPort(FXhci *const xhci, const int hubport, const int hubaddr)
- {
- FASSERT(xhci);
- if (!hubaddr)
- return hubport;
- return FXHCI_SC_GET(RHPORT, xhci->dev[hubaddr].ctx.slot);
- }
- static int FXhciGetTT(FXhci *const xhci, const FUsbSpeed speed,
- const int hubport, const int hubaddr,
- int *const tt, int *const tt_port)
- {
- FASSERT(xhci);
- if (!hubaddr)
- return 0;
- const FXhciSlotCtx *const slot = xhci->dev[hubaddr].ctx.slot;
- if ((*tt = FXHCI_SC_GET(TTID, slot)))
- {
- *tt_port = FXHCI_SC_GET(TTPORT, slot);
- }
- else if (speed < FUSB_HIGH_SPEED &&
- FXHCI_SC_GET(SPEED1, slot) - 1 == FUSB_HIGH_SPEED)
- {
- *tt = hubaddr;
- *tt_port = hubport;
- }
- return *tt != 0;
- }
- static void FXhciReapSlots(FXhci *const xhci, int skip_slot)
- {
- FASSERT(xhci);
- int i;
- FUsb *instance = xhci->usb;
- FUSB_INFO("xHC resource shortage, trying to reap old slots... ");
- for (i = 1; i <= xhci->max_slots_en; i++)
- {
- if (i == skip_slot)
- continue; /* don't reap slot we were working on */
- if (xhci->dev[i].transfer_rings[1])
- continue; /* slot still in use */
- if (NULL == xhci->dev[i].ctx.raw)
- continue; /* slot already disabled */
- const FXhciTransCode cc = FXhciCmdDisableSlot(xhci, i);
- if (cc != FXHCI_CC_SUCCESS)
- FUSB_INFO("Failed to disable slot %d: %d ", i, cc);
- else
- FUSB_INFO("Successfully reaped slot %d ", i);
- xhci->dcbaa[i] = 0;
- FUSB_FREE(instance, xhci->dev[i].ctx.raw);
- xhci->dev[i].ctx.raw = NULL;
- }
- }
- static FXhciInputCtx *FXhciMakeInputCtx(FXhci *xhci, const size_t ctxsize)
- {
- int i;
- const size_t size = (1 + FXHCI_NUM_EPS) * ctxsize;
- FUsb *instance = xhci->usb;
- FXhciInputCtx *const ic = FUSB_ALLOCATE(instance, sizeof(*ic), FUSB_DEFAULT_ALIGN);
- void *dma_buffer = FUSB_ALLOCATE(instance, size, 64);
- if ((NULL == ic) || (NULL == dma_buffer))
- {
- FUSB_FREE(instance, ic);
- FUSB_FREE(instance, dma_buffer);
- return NULL;
- }
- memset(dma_buffer, 0, size);
- ic->drop = dma_buffer + 0;
- ic->add = dma_buffer + 4;
- dma_buffer += ctxsize;
- for (i = 0; i < FXHCI_NUM_EPS; i++, dma_buffer += ctxsize)
- ic->dev.ep[i] = dma_buffer;
- return ic;
- }
- /**
- * @name: FXhciSetAddress
- * @msg: 设备USB设备的地址
- * @return {FUsbDev *}, USB设备实例
- * @param {FUsbHc} *controller, USB控制器实例
- * @param {FUsbSpeed} speed, USB设备的速度类型
- * @param {int} hubport, USB设备连接的port号
- * @param {int} hubaddr, USB设备连接的hub地址
- */
- FUsbDev *FXhciSetAddress(FUsbHc *controller, FUsbSpeed speed, int hubport, int hubaddr)
- {
- FXhci *const xhci = FXHCI_INST_GET(controller);
- FASSERT(xhci);
- const size_t ctxsize = FXhciGetCtxSize(&xhci->mmio);
- FXhciDevInfo *di = NULL;
- FUsbDev *dev = NULL;
- FUsb *instance = controller->usb;
- int i;
- FXhciInputCtx *const ic = FXhciMakeInputCtx(xhci, ctxsize);
- FXhciTransRing *const tr = FUSB_ALLOCATE(instance, sizeof(*tr), FUSB_DEFAULT_ALIGN);
- if (NULL != tr)
- {
- FASSERT(NULL == tr->ring);
- tr->ring = FXHCI_ALIGN(xhci, 16, FXHCI_TRANSFER_RING_SIZE * sizeof(FXhciTrb));
- }
- if ((NULL == ic) || (NULL == tr) || (NULL == tr->ring))
- {
- FUSB_INFO("Out of memory ");
- goto _free_return;
- }
- int slot_id;
- FXhciTransCode cc = FXhciCmdEnableSlot(xhci, &slot_id);
- if (cc == FXHCI_CC_NO_SLOTS_AVAILABLE)
- {
- FXhciReapSlots(xhci, 0);
- cc = FXhciCmdEnableSlot(xhci, &slot_id);
- }
- if (cc != FXHCI_CC_SUCCESS)
- {
- FUSB_INFO("Enable slot failed: %d ", cc);
- goto _free_return;
- }
- else
- {
- FUSB_INFO("Enabled slot %d ", slot_id);
- }
- di = &xhci->dev[slot_id];
- void *dma_buffer = FUSB_ALLOCATE(instance, FXHCI_NUM_EPS * ctxsize, 64);
- if (NULL == dma_buffer)
- goto _disable_return;
- memset(dma_buffer, 0, FXHCI_NUM_EPS * ctxsize);
- for (i = 0; i < FXHCI_NUM_EPS; i++, dma_buffer += ctxsize)
- di->ctx.ep[i] = dma_buffer;
- *ic->add = (1 << 0) /* Slot Context */ | (1 << 1) /* EP0 Context */;
- FXHCI_SC_SET(ROUTE, ic->dev.slot, FXhciGenRounte(xhci, hubport, hubaddr));
- FXHCI_SC_SET(SPEED1, ic->dev.slot, speed + 1);
- FXHCI_SC_SET(CTXENT, ic->dev.slot, 1); /* the endpoint 0 context */
- FXHCI_SC_SET(RHPORT, ic->dev.slot, FXhciGetRoothubPort(xhci, hubport, hubaddr));
- int tt, tt_port;
- if (FXhciGetTT(xhci, speed, hubport, hubaddr, &tt, &tt_port))
- {
- FUSB_INFO("TT for %d: %d[%d] ", slot_id, tt, tt_port);
- FXHCI_SC_SET(MTT, ic->dev.slot, FXHCI_SC_GET(MTT, xhci->dev[tt].ctx.slot));
- FXHCI_SC_SET(TTID, ic->dev.slot, tt);
- FXHCI_SC_SET(TTPORT, ic->dev.slot, tt_port);
- }
- di->transfer_rings[1] = tr;
- FXhciInitCycleRing(tr, FXHCI_TRANSFER_RING_SIZE);
- ic->dev.ep0->tr_dq_low = (uintptr)(tr->ring);
- ic->dev.ep0->tr_dq_high = 0;
- FXHCI_EC_SET(TYPE, ic->dev.ep0, FXHCI_EP_CONTROL);
- FXHCI_EC_SET(AVRTRB, ic->dev.ep0, 8);
- FXHCI_EC_SET(MPS, ic->dev.ep0, FUsbSpeedtoDefaultMaxPacketSz(speed));
- FXHCI_EC_SET(CERR, ic->dev.ep0, 3);
- FXHCI_EC_SET(DCS, ic->dev.ep0, 1);
- xhci->dcbaa[slot_id] = (uintptr)(di->ctx.raw);
- FCacheDCacheInvalidateRange((uintptr)ic, sizeof(*ic)); /* flush cache of input address */
- cc = FXhciCmdAddressDevice(xhci, slot_id, ic);
- if (cc == FXHCI_CC_RESOURCE_ERROR)
- {
- FXhciReapSlots(xhci, slot_id);
- cc = FXhciCmdAddressDevice(xhci, slot_id, ic);
- }
- if (cc != FXHCI_CC_SUCCESS)
- {
- FUSB_INFO("Address device failed: %d ", cc);
- goto _disable_return;
- }
- else
- {
- FUSB_INFO("Addressed device %d (USB: %d) ",
- slot_id, FXHCI_SC_GET(UADDR, di->ctx.slot));
- }
- fsleep_millisec(FUSB_SET_ADDRESS_MDELAY);
- dev = FUsbInitDevEntry(controller, slot_id);
- if (!dev)
- goto _disable_return;
- dev->address = slot_id;
- dev->hub = hubaddr;
- dev->port = hubport;
- dev->speed = speed;
- dev->endpoints[0].dev = dev;
- dev->endpoints[0].endpoint = 0;
- dev->endpoints[0].toggle = 0;
- dev->endpoints[0].direction = FUSB_SETUP;
- dev->endpoints[0].type = FUSB_CONTROL_EP;
- u8 buf[8];
- if (FUsbGetDescriptor(dev, FUsbGenerateReqType(FUSB_REQ_DEVICE_TO_HOST, FUSB_REQ_TYPE_STANDARD, FUSB_REQ_RECP_DEV), FUSB_DESC_TYPE_DEVICE, 0, buf, sizeof(buf)) != sizeof(buf))
- {
- FUSB_INFO("first FUsbGetDescriptor(FUSB_DESC_TYPE_DEVICE) failed ");
- goto _disable_return;
- }
- dev->endpoints[0].maxpacketsize = FUsbDecodeMaxPacketSz0(speed, buf[7]);
- if (dev->endpoints[0].maxpacketsize != FUsbSpeedtoDefaultMaxPacketSz(speed))
- {
- memset((void *)ic->dev.ep0, 0x00, ctxsize);
- *ic->add = (1 << 1); /* EP0 Context */
- FXHCI_EC_SET(MPS, ic->dev.ep0, dev->endpoints[0].maxpacketsize);
- /* flush cache of input context before send command */
- FCacheDCacheInvalidateRange((uintptr)ic, sizeof(*ic));
- cc = FXhciCmdEvaluateCtx(xhci, slot_id, ic);
- if (cc == FXHCI_CC_RESOURCE_ERROR)
- {
- FXhciReapSlots(xhci, slot_id);
- cc = FXhciCmdEvaluateCtx(xhci, slot_id, ic);
- }
- if (cc != FXHCI_CC_SUCCESS)
- {
- FUSB_INFO("Context evaluation failed: %d ", cc);
- goto _disable_return;
- }
- }
- goto _free_ic_return;
- _disable_return:
- FXhciCmdDisableSlot(xhci, slot_id);
- xhci->dcbaa[slot_id] = 0;
- FUsbDetachDev(controller, slot_id);
- dev = NULL;
- _free_return:
- if (tr)
- FUSB_FREE(instance, (void *)tr->ring);
- FUSB_FREE(instance, tr);
- if (di)
- {
- FUSB_FREE(instance, di->ctx.raw);
- di->ctx.raw = 0;
- }
- _free_ic_return:
- if (ic)
- {
- FUSB_FREE(instance, ic->raw);
- FUSB_FREE(instance, ic);
- }
- return dev;
- }
- static int FXhciFinishHubConfig(FUsbDev *const dev, FXhciInputCtx *const ic)
- {
- int type = FUsbIsSuperSpeed(dev->speed) ? 0x2a : 0x29; /* similar enough */
- FUsbHubDescriptor desc;
- if (FUsbGetDescriptor(dev, FUsbGenerateReqType(FUSB_REQ_DEVICE_TO_HOST, FUSB_REQ_TYPE_CLASS, FUSB_REQ_RECP_DEV), type, 0, &desc, sizeof(desc)) != sizeof(desc))
- {
- FUSB_INFO("Failed to fetch hub descriptor ");
- return FXHCI_CC_COMMUNICATION_ERROR;
- }
- FXHCI_SC_SET(HUB, ic->dev.slot, 1);
- FXHCI_SC_SET(MTT, ic->dev.slot, 0); /* No support for Multi-TT */
- FXHCI_SC_SET(NPORTS, ic->dev.slot, desc.bNbrPorts);
- if (dev->speed == FUSB_HIGH_SPEED)
- FXHCI_SC_SET(TTT, ic->dev.slot,
- (desc.wHubCharacteristics >> 5) & 0x0003);
- return 0;
- }
- static size_t FXhciBoundInterval(const FUsbEndpoint *const ep)
- {
- if ((ep->dev->speed == FUSB_LOW_SPEED &&
- (ep->type == FUSB_ISOCHRONOUS_EP ||
- ep->type == FUSB_INTERRUPT_EP)) ||
- (ep->dev->speed == FUSB_FULL_SPEED &&
- ep->type == FUSB_INTERRUPT_EP))
- {
- if (ep->interval < 3)
- return 3;
- else if (ep->interval > 11)
- return 11;
- else
- return ep->interval;
- }
- else
- {
- if (ep->interval < 0)
- return 0;
- else if (ep->interval > 15)
- return 15;
- else
- return ep->interval;
- }
- }
- static int FXhciFinishEpConfig(const FUsbEndpoint *const ep, FXhciInputCtx *const ic)
- {
- FXhci *const xhci = FXHCI_INST_GET(ep->dev->controller);
- FASSERT(xhci);
- FUsb *instance = xhci->usb;
- const int ep_id = FXhciEpId(ep);
- FUSB_INFO("ep_id: %d ", ep_id);
- if (ep_id <= 1 || 32 <= ep_id)
- return FXHCI_CC_DRIVER_ERROR;
- FXhciTransRing *const tr = FUSB_ALLOCATE(instance, sizeof(*tr), FUSB_DEFAULT_ALIGN);
- if (NULL != tr)
- {
- FASSERT(NULL == tr->ring);
- tr->ring = FXHCI_ALIGN(xhci, 16, FXHCI_TRANSFER_RING_SIZE * sizeof(FXhciTrb));
- }
- if ((NULL == tr) || (NULL == tr->ring))
- {
- FUSB_FREE(instance, tr);
- FUSB_ERROR("Out of memory ");
- return FXHCI_CC_OUT_OF_MEMORY;
- }
- xhci->dev[ep->dev->address].transfer_rings[ep_id] = tr;
- FXhciInitCycleRing(tr, FXHCI_TRANSFER_RING_SIZE);
- *ic->add |= (1 << ep_id);
- if ((int)FXHCI_SC_GET(CTXENT, ic->dev.slot) < ep_id)
- FXHCI_SC_SET(CTXENT, ic->dev.slot, ep_id);
- FXhciEpCtx *const epctx = ic->dev.ep[ep_id];
- FUSB_DEBUG("Filling epctx (@%p) ", epctx);
- epctx->tr_dq_low = (uintptr)(tr->ring);
- epctx->tr_dq_high = 0;
- FXHCI_EC_SET(INTVAL, epctx, FXhciBoundInterval(ep));
- FXHCI_EC_SET(CERR, epctx, 3);
- FXHCI_EC_SET(TYPE, epctx, ep->type | ((ep->direction != FUSB_OUT) << 2));
- FXHCI_EC_SET(MPS, epctx, ep->maxpacketsize);
- FXHCI_EC_SET(DCS, epctx, 1);
- size_t avrtrb;
- switch (ep->type)
- {
- case FUSB_BULK_EP:
- case FUSB_ISOCHRONOUS_EP:
- avrtrb = 3 * 1024;
- break;
- case FUSB_INTERRUPT_EP:
- avrtrb = 1024;
- break;
- default:
- avrtrb = 8;
- break;
- }
- FXHCI_EC_SET(AVRTRB, epctx, avrtrb);
- FXHCI_EC_SET(MXESIT, epctx, FXHCI_EC_GET(MPS, epctx) * FXHCI_EC_GET(MBS, epctx));
- return 0;
- }
- /**
- * @name: FXhciFinishDevConfig
- * @msg: 完成USB设备配置
- * @return {FXhciTransCode} 传输返回码
- * @param {FUsbDev} *dev, USB设备实例
- */
- FXhciTransCode FXhciFinishDevConfig(FUsbDev *const dev)
- {
- FXhci *const xhci = FXHCI_INST_GET(dev->controller);
- FASSERT(xhci);
- FUsb *instance = xhci->usb;
- int slot_id = dev->address;
- FXhciDevInfo *const di = &xhci->dev[slot_id];
- int i;
- FXhciTransCode ret = FXHCI_CC_ZERO_BYTES;
- FXhciInputCtx *const ic = FXhciMakeInputCtx(xhci, FXhciGetCtxSize(&xhci->mmio));
- if (!ic)
- {
- FUSB_INFO("Out of memory ");
- return FXHCI_CC_OUT_OF_MEMORY;
- }
- *ic->add = (1 << 0); /* Slot Context */
- ic->dev.slot->f1 = di->ctx.slot->f1;
- ic->dev.slot->f2 = di->ctx.slot->f2;
- ic->dev.slot->f3 = di->ctx.slot->f3;
- /* f4 *must* be 0 in the Input Context... yeah, it's weird, I know. */
- FCacheDCacheInvalidateRange((uintptr)ic, sizeof(*ic));
- if (dev->descriptor->bDeviceClass == FUSB_HUB_DEVICE)
- {
- ret = FXhciFinishHubConfig(dev, ic);
- if (ret)
- goto _free_return;
- }
- for (i = 1; i < dev->num_endp; ++i)
- {
- ret = FXhciFinishEpConfig(&dev->endpoints[i], ic);
- if (ret)
- goto _free_ep_ctx_return;
- }
- const int config_id = dev->configuration->bConfigurationValue;
- FUSB_INFO("config_id: %d ", config_id);
- FXhciTransCode cc = FXhciCmdConfigureEp(xhci, slot_id, config_id, ic);
- if (cc == FXHCI_CC_RESOURCE_ERROR || cc == FXHCI_CC_BANDWIDTH_ERROR)
- {
- FXhciReapSlots(xhci, slot_id);
- cc = FXhciCmdConfigureEp(xhci, slot_id, config_id, ic);
- }
- if (cc != FXHCI_CC_SUCCESS)
- {
- FUSB_INFO("Configure endpoint failed: %d ", cc);
- ret = FXHCI_CC_CONTROLLER_ERROR;
- goto _free_ep_ctx_return;
- }
- else
- {
- FUSB_INFO("Endpoints configured ");
- }
- goto _free_return;
- _free_ep_ctx_return:
- for (i = 2; i < 31; ++i)
- {
- if (di->transfer_rings[i])
- {
- FUSB_FREE(instance, (void *)di->transfer_rings[i]->ring);
- FUSB_FREE(instance, di->transfer_rings[i]);
- }
- di->transfer_rings[i] = NULL;
- }
- _free_return:
- if (NULL != ic)
- {
- FUSB_FREE(instance, ic->raw);
- FUSB_FREE(instance, ic);
- }
- return ret;
- }
- /**
- * @name: FXhciDestoryDev
- * @msg: 删除指定USB设备实例
- * @return {*}
- * @param {FUsbHc} *controller, USB控制器实例
- * @param {int} slot_id, USB设备所在的slot号
- */
- void FXhciDestoryDev(FUsbHc *const controller, const int slot_id)
- {
- FXhci *const xhci = FXHCI_INST_GET(controller);
- FASSERT(xhci);
- FUsb *instance = xhci->usb;
- if (slot_id <= 0 || slot_id > xhci->max_slots_en)
- return;
- FXhciInputCtx *const ic = FXhciMakeInputCtx(xhci, FXhciGetCtxSize(&xhci->mmio));
- if (NULL == ic)
- {
- FUSB_WARN("Out of memory, leaking resources! ");
- return;
- }
- const int num_eps = controller->devices[slot_id]->num_endp;
- *ic->add = 0; /* Leave Slot/EP0 state as it is for now. */
- *ic->drop = (1 << num_eps) - 1; /* Drop all endpoints we can. */
- *ic->drop &= ~(1 << 1 | 1 << 0); /* Not allowed to drop EP0 or Slot. */
- FCacheDCacheInvalidateRange((uintptr)ic, sizeof(*ic));
- FXhciTransCode cc = FXhciCmdEvaluateCtx(xhci, slot_id, ic);
- if (NULL != ic)
- {
- FUSB_FREE(instance, ic->raw);
- FUSB_FREE(instance, ic);
- }
- if (cc != FXHCI_CC_SUCCESS)
- FUSB_INFO("Failed to quiesce slot %d: %d ", slot_id, cc);
- cc = FXhciCmdStopEp(xhci, slot_id, FXHCI_EP0_ID);
- if (cc != FXHCI_CC_SUCCESS)
- FUSB_INFO("Failed to stop EP0 on slot %d: %d ", slot_id, cc);
- int i;
- FXhciDevInfo *const di = &xhci->dev[slot_id];
- for (i = 1; i < /*num_eps*/FXHCI_NUM_EPS; ++i)
- {
- if (di->transfer_rings[i])
- {
- FUSB_FREE(instance, (void *)di->transfer_rings[i]->ring);
- FUSB_FREE(instance, (void *)di->transfer_rings[i]);
- }
- FUSB_FREE(instance, di->interrupt_queues[i]);
- }
- /* free device context */
- if (NULL != di->ctx.raw)
- {
- FUSB_FREE(instance, di->ctx.raw);
- di->ctx.raw = NULL;
- }
- FUSB_INFO("Stopped slot %d, but not disabling it yet. ", slot_id);
- di->transfer_rings[1] = NULL;
- return;
- }
|