via_i2c.c revision 90b17f1b
17e6fb56fSmrg/*
27e6fb56fSmrg * Copyright 2004 The Unichrome Project  [unichrome.sf.net]
37e6fb56fSmrg * Copyright 1998-2003 VIA Technologies, Inc. All Rights Reserved.
47e6fb56fSmrg * Copyright 2001-2003 S3 Graphics, Inc. All Rights Reserved.
57e6fb56fSmrg *
67e6fb56fSmrg * Permission is hereby granted, free of charge, to any person obtaining a
77e6fb56fSmrg * copy of this software and associated documentation files (the "Software"),
87e6fb56fSmrg * to deal in the Software without restriction, including without limitation
97e6fb56fSmrg * the rights to use, copy, modify, merge, publish, distribute, sub license,
107e6fb56fSmrg * and/or sell copies of the Software, and to permit persons to whom the
117e6fb56fSmrg * Software is furnished to do so, subject to the following conditions:
127e6fb56fSmrg *
137e6fb56fSmrg * The above copyright notice and this permission notice (including the
147e6fb56fSmrg * next paragraph) shall be included in all copies or substantial portions
157e6fb56fSmrg * of the Software.
167e6fb56fSmrg *
177e6fb56fSmrg * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
187e6fb56fSmrg * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
197e6fb56fSmrg * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL
207e6fb56fSmrg * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
217e6fb56fSmrg * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
227e6fb56fSmrg * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
237e6fb56fSmrg * DEALINGS IN THE SOFTWARE.
247e6fb56fSmrg */
257e6fb56fSmrg
267e6fb56fSmrg/*
277e6fb56fSmrg * Implements three i2c busses through registers SR26, SR2c, and SR31.
287e6fb56fSmrg */
297e6fb56fSmrg
307e6fb56fSmrg#ifdef HAVE_CONFIG_H
317e6fb56fSmrg#include "config.h"
327e6fb56fSmrg#endif
337e6fb56fSmrg
347e6fb56fSmrg#include "via_driver.h"
357e6fb56fSmrg
367e6fb56fSmrg#define SDA_READ  0x04
377e6fb56fSmrg#define SCL_READ  0x08
387e6fb56fSmrg#define SDA_WRITE 0x10
397e6fb56fSmrg#define SCL_WRITE 0x20
407e6fb56fSmrg
417e6fb56fSmrg/*
427e6fb56fSmrg * CRT I2C
437e6fb56fSmrg */
447e6fb56fSmrgstatic void
457e6fb56fSmrgViaI2C1PutBits(I2CBusPtr Bus, int clock, int data)
467e6fb56fSmrg{
4790b17f1bSmrg    vgaHWPtr hwp = Bus->DriverPrivate.ptr;
487e6fb56fSmrg    CARD8 value = 0x01; /* Enable */
497e6fb56fSmrg
507e6fb56fSmrg    if (clock)
517e6fb56fSmrg        value |= SCL_WRITE;
527e6fb56fSmrg
537e6fb56fSmrg    if (data)
547e6fb56fSmrg        value |= SDA_WRITE;
557e6fb56fSmrg
567e6fb56fSmrg    ViaSeqMask(hwp, 0x26, value, 0x01 | SCL_WRITE | SDA_WRITE);
577e6fb56fSmrg}
587e6fb56fSmrg
597e6fb56fSmrgstatic void
607e6fb56fSmrgViaI2C1GetBits(I2CBusPtr Bus, int *clock, int *data)
617e6fb56fSmrg{
6290b17f1bSmrg    vgaHWPtr hwp = Bus->DriverPrivate.ptr;
637e6fb56fSmrg    CARD8 value = hwp->readSeq(hwp, 0x26);
647e6fb56fSmrg
657e6fb56fSmrg    *clock = (value & SCL_READ) != 0;
667e6fb56fSmrg    *data = (value & SDA_READ) != 0;
677e6fb56fSmrg}
687e6fb56fSmrg
697e6fb56fSmrgstatic I2CBusPtr
7090b17f1bSmrgViaI2CBus1Init(ScrnInfoPtr pScrn)
717e6fb56fSmrg{
727e6fb56fSmrg    I2CBusPtr pI2CBus = xf86CreateI2CBusRec();
7390b17f1bSmrg    vgaHWPtr hwp = VGAHWPTR(pScrn);
747e6fb56fSmrg
7590b17f1bSmrg    DEBUG(xf86DrvMsg(pScrn->scrnIndex, X_INFO, "ViaI2CBus1Init\n"));
767e6fb56fSmrg
777e6fb56fSmrg    if (!pI2CBus)
787e6fb56fSmrg        return NULL;
797e6fb56fSmrg
807e6fb56fSmrg    pI2CBus->BusName = "I2C bus 1";
8190b17f1bSmrg    pI2CBus->scrnIndex = pScrn->scrnIndex;
827e6fb56fSmrg    pI2CBus->I2CPutBits = ViaI2C1PutBits;
837e6fb56fSmrg    pI2CBus->I2CGetBits = ViaI2C1GetBits;
8490b17f1bSmrg    pI2CBus->DriverPrivate.ptr = hwp;
8590b17f1bSmrg    pI2CBus->ByteTimeout = 2200;
8690b17f1bSmrg    pI2CBus->StartTimeout = 550;
8790b17f1bSmrg    pI2CBus->HoldTime = 40;
8890b17f1bSmrg    pI2CBus->BitTimeout = 40;
897e6fb56fSmrg
907e6fb56fSmrg    if (!xf86I2CBusInit(pI2CBus)) {
917e6fb56fSmrg        xf86DestroyI2CBusRec(pI2CBus, TRUE, FALSE);
927e6fb56fSmrg        return NULL;
937e6fb56fSmrg    }
947e6fb56fSmrg    return pI2CBus;
957e6fb56fSmrg}
967e6fb56fSmrg
977e6fb56fSmrg/*
987e6fb56fSmrg * First data bus I2C: tends to have TV-encoders.
997e6fb56fSmrg */
1007e6fb56fSmrgstatic void
1017e6fb56fSmrgViaI2C2PutBits(I2CBusPtr Bus, int clock, int data)
1027e6fb56fSmrg{
10390b17f1bSmrg    vgaHWPtr hwp = Bus->DriverPrivate.ptr;
1047e6fb56fSmrg    CARD8 value = 0x01; /* Enable */
1057e6fb56fSmrg
1067e6fb56fSmrg    if (clock)
1077e6fb56fSmrg        value |= SCL_WRITE;
1087e6fb56fSmrg
1097e6fb56fSmrg    if (data)
1107e6fb56fSmrg        value |= SDA_WRITE;
1117e6fb56fSmrg
1127e6fb56fSmrg    ViaSeqMask(hwp, 0x31, value, 0x01 | SCL_WRITE | SDA_WRITE);
1137e6fb56fSmrg}
1147e6fb56fSmrg
1157e6fb56fSmrgstatic void
1167e6fb56fSmrgViaI2C2GetBits(I2CBusPtr Bus, int *clock, int *data)
1177e6fb56fSmrg{
11890b17f1bSmrg    vgaHWPtr hwp = Bus->DriverPrivate.ptr;
1197e6fb56fSmrg    CARD8 value = hwp->readSeq(hwp, 0x31);
1207e6fb56fSmrg
1217e6fb56fSmrg    *clock = (value & SCL_READ) != 0;
1227e6fb56fSmrg    *data = (value & SDA_READ) != 0;
1237e6fb56fSmrg}
1247e6fb56fSmrg
1257e6fb56fSmrgstatic I2CBusPtr
12690b17f1bSmrgViaI2CBus2Init(ScrnInfoPtr pScrn)
1277e6fb56fSmrg{
1287e6fb56fSmrg    I2CBusPtr pI2CBus = xf86CreateI2CBusRec();
12990b17f1bSmrg    vgaHWPtr hwp = VGAHWPTR(pScrn);
1307e6fb56fSmrg
13190b17f1bSmrg    DEBUG(xf86DrvMsg(pScrn->scrnIndex, X_INFO, "ViaI2cBus2Init\n"));
1327e6fb56fSmrg
1337e6fb56fSmrg    if (!pI2CBus)
1347e6fb56fSmrg        return NULL;
1357e6fb56fSmrg
1367e6fb56fSmrg    pI2CBus->BusName = "I2C bus 2";
13790b17f1bSmrg    pI2CBus->scrnIndex = pScrn->scrnIndex;
1387e6fb56fSmrg    pI2CBus->I2CPutBits = ViaI2C2PutBits;
1397e6fb56fSmrg    pI2CBus->I2CGetBits = ViaI2C2GetBits;
14090b17f1bSmrg    pI2CBus->DriverPrivate.ptr = hwp;
1417e6fb56fSmrg
1427e6fb56fSmrg    if (!xf86I2CBusInit(pI2CBus)) {
1437e6fb56fSmrg        xf86DestroyI2CBusRec(pI2CBus, TRUE, FALSE);
1447e6fb56fSmrg        return NULL;
1457e6fb56fSmrg    }
1467e6fb56fSmrg    return pI2CBus;
1477e6fb56fSmrg}
1487e6fb56fSmrg
1497e6fb56fSmrg/*
1507e6fb56fSmrg * A third I2C bus is implemented by a few IO pins.
1517e6fb56fSmrg * Requires higher level functions to be used properly.
1527e6fb56fSmrg */
1537e6fb56fSmrgstatic Bool
1547e6fb56fSmrgViaI2C3Start(I2CBusPtr b, int timeout)
1557e6fb56fSmrg{
15690b17f1bSmrg    vgaHWPtr hwp = b->DriverPrivate.ptr;
1577e6fb56fSmrg
1587e6fb56fSmrg    ViaSeqMask(hwp, 0x2C, 0xF0, 0xF0);
1597e6fb56fSmrg    b->I2CUDelay(b, b->RiseFallTime);
1607e6fb56fSmrg
1617e6fb56fSmrg    ViaSeqMask(hwp, 0x2C, 0x00, 0x10);
1627e6fb56fSmrg    b->I2CUDelay(b, b->HoldTime);
1637e6fb56fSmrg    ViaSeqMask(hwp, 0x2C, 0x00, 0x20);
1647e6fb56fSmrg    b->I2CUDelay(b, b->HoldTime);
1657e6fb56fSmrg
1667e6fb56fSmrg    return TRUE;
1677e6fb56fSmrg}
1687e6fb56fSmrg
1697e6fb56fSmrgstatic Bool
1707e6fb56fSmrgViaI2C3Address(I2CDevPtr d, I2CSlaveAddr addr)
1717e6fb56fSmrg{
1727e6fb56fSmrg    I2CBusPtr b = d->pI2CBus;
1737e6fb56fSmrg
1747e6fb56fSmrg#ifdef X_NEED_I2CSTART
1757e6fb56fSmrg    if (b->I2CStart(d->pI2CBus, d->StartTimeout)) {
1767e6fb56fSmrg#else
1777e6fb56fSmrg    if (ViaI2C3Start(d->pI2CBus, d->StartTimeout)) {
1787e6fb56fSmrg#endif
1797e6fb56fSmrg        if (b->I2CPutByte(d, addr & 0xFF)) {
1807e6fb56fSmrg            if ((addr & 0xF8) != 0xF0 && (addr & 0xFE) != 0x00)
1817e6fb56fSmrg                return TRUE;
1827e6fb56fSmrg
1837e6fb56fSmrg            if (b->I2CPutByte(d, (addr >> 8) & 0xFF))
1847e6fb56fSmrg                return TRUE;
1857e6fb56fSmrg        }
1867e6fb56fSmrg
1877e6fb56fSmrg        b->I2CStop(d);
1887e6fb56fSmrg    }
1897e6fb56fSmrg    return FALSE;
1907e6fb56fSmrg}
1917e6fb56fSmrg
1927e6fb56fSmrgstatic void
1937e6fb56fSmrgViaI2C3Stop(I2CDevPtr d)
1947e6fb56fSmrg{
1957e6fb56fSmrg    I2CBusPtr b = d->pI2CBus;
19690b17f1bSmrg    vgaHWPtr hwp = b->DriverPrivate.ptr;
1977e6fb56fSmrg
1987e6fb56fSmrg    ViaSeqMask(hwp, 0x2C, 0xC0, 0xF0);
1997e6fb56fSmrg    b->I2CUDelay(b, b->RiseFallTime);
2007e6fb56fSmrg
2017e6fb56fSmrg    ViaSeqMask(hwp, 0x2C, 0x20, 0x20);
2027e6fb56fSmrg    b->I2CUDelay(b, b->HoldTime);
2037e6fb56fSmrg
2047e6fb56fSmrg    ViaSeqMask(hwp, 0x2C, 0x10, 0x10);
2057e6fb56fSmrg    b->I2CUDelay(b, b->HoldTime);
2067e6fb56fSmrg
2077e6fb56fSmrg    ViaSeqMask(hwp, 0x2C, 0x00, 0x20);
2087e6fb56fSmrg    b->I2CUDelay(b, b->HoldTime);
2097e6fb56fSmrg}
2107e6fb56fSmrg
2117e6fb56fSmrgstatic void
2127e6fb56fSmrgViaI2C3PutBit(I2CBusPtr b, Bool sda, int timeout)
2137e6fb56fSmrg{
21490b17f1bSmrg    vgaHWPtr hwp = b->DriverPrivate.ptr;
2157e6fb56fSmrg
2167e6fb56fSmrg    if (sda)
2177e6fb56fSmrg        ViaSeqMask(hwp, 0x2C, 0x50, 0x50);
2187e6fb56fSmrg    else
2197e6fb56fSmrg        ViaSeqMask(hwp, 0x2C, 0x40, 0x50);
2207e6fb56fSmrg    b->I2CUDelay(b, b->RiseFallTime / 5);
2217e6fb56fSmrg
2227e6fb56fSmrg    ViaSeqMask(hwp, 0x2C, 0xA0, 0xA0);
2237e6fb56fSmrg    b->I2CUDelay(b, b->HoldTime);
2247e6fb56fSmrg    b->I2CUDelay(b, timeout);
2257e6fb56fSmrg
2267e6fb56fSmrg    ViaSeqMask(hwp, 0x2C, 0x80, 0xA0);
2277e6fb56fSmrg    b->I2CUDelay(b, b->RiseFallTime / 5);
2287e6fb56fSmrg}
2297e6fb56fSmrg
2307e6fb56fSmrgstatic Bool
2317e6fb56fSmrgViaI2C3PutByte(I2CDevPtr d, I2CByte data)
2327e6fb56fSmrg{
2337e6fb56fSmrg    I2CBusPtr b = d->pI2CBus;
23490b17f1bSmrg    vgaHWPtr hwp = b->DriverPrivate.ptr;
2357e6fb56fSmrg    Bool ret;
2367e6fb56fSmrg    int i;
2377e6fb56fSmrg
2387e6fb56fSmrg    for (i = 7; i >= 0; i--)
2397e6fb56fSmrg        ViaI2C3PutBit(b, (data >> i) & 0x01, b->BitTimeout);
2407e6fb56fSmrg
2417e6fb56fSmrg    /* Raise first to avoid false positives. */
2427e6fb56fSmrg    ViaSeqMask(hwp, 0x2C, 0x50, 0x50);
2437e6fb56fSmrg    ViaSeqMask(hwp, 0x2C, 0x00, 0x40);
2447e6fb56fSmrg    b->I2CUDelay(b, b->RiseFallTime);
2457e6fb56fSmrg    ViaSeqMask(hwp, 0x2C, 0xA0, 0xA0);
2467e6fb56fSmrg
2477e6fb56fSmrg    if (hwp->readSeq(hwp, 0x2C) & 0x04)
2487e6fb56fSmrg        ret = FALSE;
2497e6fb56fSmrg    else
2507e6fb56fSmrg        ret = TRUE;
2517e6fb56fSmrg
2527e6fb56fSmrg    ViaSeqMask(hwp, 0x2C, 0x80, 0xA0);
2537e6fb56fSmrg    b->I2CUDelay(b, b->RiseFallTime);
2547e6fb56fSmrg
2557e6fb56fSmrg    return ret;
2567e6fb56fSmrg}
2577e6fb56fSmrg
2587e6fb56fSmrgstatic Bool
2597e6fb56fSmrgViaI2C3GetBit(I2CBusPtr b, int timeout)
2607e6fb56fSmrg{
26190b17f1bSmrg    vgaHWPtr hwp = b->DriverPrivate.ptr;
2627e6fb56fSmrg    Bool ret;
2637e6fb56fSmrg
2647e6fb56fSmrg    ViaSeqMask(hwp, 0x2c, 0x80, 0xC0);
2657e6fb56fSmrg    b->I2CUDelay(b, b->RiseFallTime / 5);
2667e6fb56fSmrg    ViaSeqMask(hwp, 0x2c, 0xA0, 0xA0);
2677e6fb56fSmrg    b->I2CUDelay(b, 3 * b->HoldTime);
2687e6fb56fSmrg    b->I2CUDelay(b, timeout);
2697e6fb56fSmrg
2707e6fb56fSmrg    if (hwp->readSeq(hwp, 0x2C) & 0x04)
2717e6fb56fSmrg        ret = TRUE;
2727e6fb56fSmrg    else
2737e6fb56fSmrg        ret = FALSE;
2747e6fb56fSmrg
2757e6fb56fSmrg    ViaSeqMask(hwp, 0x2C, 0x80, 0xA0);
2767e6fb56fSmrg    b->I2CUDelay(b, b->HoldTime);
2777e6fb56fSmrg    b->I2CUDelay(b, b->RiseFallTime / 5);
2787e6fb56fSmrg
2797e6fb56fSmrg    return ret;
2807e6fb56fSmrg}
2817e6fb56fSmrg
2827e6fb56fSmrgstatic Bool
2837e6fb56fSmrgViaI2C3GetByte(I2CDevPtr d, I2CByte * data, Bool last)
2847e6fb56fSmrg{
2857e6fb56fSmrg    I2CBusPtr b = d->pI2CBus;
28690b17f1bSmrg    vgaHWPtr hwp = b->DriverPrivate.ptr;
2877e6fb56fSmrg    int i;
2887e6fb56fSmrg
2897e6fb56fSmrg    *data = 0x00;
2907e6fb56fSmrg
2917e6fb56fSmrg    for (i = 7; i >= 0; i--)
2927e6fb56fSmrg        if (ViaI2C3GetBit(b, b->BitTimeout))
2937e6fb56fSmrg            *data |= 0x01 << i;
2947e6fb56fSmrg
2957e6fb56fSmrg    if (last)   /* send NACK */
2967e6fb56fSmrg        ViaSeqMask(hwp, 0x2C, 0x50, 0x50);
2977e6fb56fSmrg    else        /* send ACK */
2987e6fb56fSmrg        ViaSeqMask(hwp, 0x2C, 0x40, 0x50);
2997e6fb56fSmrg
3007e6fb56fSmrg    ViaSeqMask(hwp, 0x2C, 0xA0, 0xA0);
3017e6fb56fSmrg    b->I2CUDelay(b, b->HoldTime);
3027e6fb56fSmrg
3037e6fb56fSmrg    ViaSeqMask(hwp, 0x2C, 0x80, 0xA0);
3047e6fb56fSmrg
3057e6fb56fSmrg    return TRUE;
3067e6fb56fSmrg}
3077e6fb56fSmrg
3087e6fb56fSmrgstatic I2CBusPtr
30990b17f1bSmrgViaI2CBus3Init(ScrnInfoPtr pScrn)
3107e6fb56fSmrg{
3117e6fb56fSmrg    I2CBusPtr pI2CBus = xf86CreateI2CBusRec();
31290b17f1bSmrg    vgaHWPtr hwp = VGAHWPTR(pScrn);
3137e6fb56fSmrg
31490b17f1bSmrg    DEBUG(xf86DrvMsg(pScrn->scrnIndex, X_INFO, "ViaI2CBus3Init\n"));
3157e6fb56fSmrg
3167e6fb56fSmrg    if (!pI2CBus)
3177e6fb56fSmrg        return NULL;
3187e6fb56fSmrg
3197e6fb56fSmrg    pI2CBus->BusName = "I2C bus 3";
32090b17f1bSmrg    pI2CBus->scrnIndex = pScrn->scrnIndex;
3217e6fb56fSmrg    pI2CBus->I2CAddress = ViaI2C3Address;
3227e6fb56fSmrg#ifdef X_NEED_I2CSTART
3237e6fb56fSmrg    pI2CBus->I2CStart = ViaI2C3Start;
3247e6fb56fSmrg#endif
3257e6fb56fSmrg    pI2CBus->I2CStop = ViaI2C3Stop;
3267e6fb56fSmrg    pI2CBus->I2CPutByte = ViaI2C3PutByte;
3277e6fb56fSmrg    pI2CBus->I2CGetByte = ViaI2C3GetByte;
32890b17f1bSmrg    pI2CBus->DriverPrivate.ptr = hwp;
3297e6fb56fSmrg
3307e6fb56fSmrg    pI2CBus->HoldTime = 10;
3317e6fb56fSmrg    pI2CBus->BitTimeout = 10;
3327e6fb56fSmrg    pI2CBus->ByteTimeout = 10;
3337e6fb56fSmrg    pI2CBus->StartTimeout = 10;
3347e6fb56fSmrg
3357e6fb56fSmrg    if (!xf86I2CBusInit(pI2CBus)) {
3367e6fb56fSmrg        xf86DestroyI2CBusRec(pI2CBus, TRUE, FALSE);
3377e6fb56fSmrg        return NULL;
3387e6fb56fSmrg    }
3397e6fb56fSmrg    return pI2CBus;
3407e6fb56fSmrg}
3417e6fb56fSmrg
3427e6fb56fSmrg#ifdef HAVE_DEBUG
3437e6fb56fSmrgstatic void
3447e6fb56fSmrgViaI2CScan(I2CBusPtr Bus)
3457e6fb56fSmrg{
3467e6fb56fSmrg    CARD8 i;
3477e6fb56fSmrg
3487e6fb56fSmrg    xf86DrvMsg(Bus->scrnIndex, X_INFO, "ViaI2CScan: Scanning %s\n",
3497e6fb56fSmrg               Bus->BusName);
3507e6fb56fSmrg
3517e6fb56fSmrg    for (i = 0x10; i < 0xF0; i += 2)
3527e6fb56fSmrg        if (xf86I2CProbeAddress(Bus, i))
3537e6fb56fSmrg            xf86DrvMsg(Bus->scrnIndex, X_PROBED, "Found slave on %s "
3547e6fb56fSmrg                       "- 0x%02X\n", Bus->BusName, i);
3557e6fb56fSmrg}
3567e6fb56fSmrg#endif /* HAVE_DEBUG */
3577e6fb56fSmrg
3587e6fb56fSmrgvoid
3597e6fb56fSmrgViaI2CInit(ScrnInfoPtr pScrn)
3607e6fb56fSmrg{
3617e6fb56fSmrg    VIAPtr pVia = VIAPTR(pScrn);
3627e6fb56fSmrg
3637e6fb56fSmrg    DEBUG(xf86DrvMsg(pScrn->scrnIndex, X_INFO, "ViaI2CInit\n"));
3647e6fb56fSmrg
36590b17f1bSmrg    if (pVia->I2CDevices & VIA_I2C_BUS1)
36690b17f1bSmrg        pVia->pI2CBus1 = ViaI2CBus1Init(pScrn);
36790b17f1bSmrg    if (pVia->I2CDevices & VIA_I2C_BUS2)
36890b17f1bSmrg        pVia->pI2CBus2 = ViaI2CBus2Init(pScrn);
36990b17f1bSmrg    if (pVia->I2CDevices & VIA_I2C_BUS3)
37090b17f1bSmrg        pVia->pI2CBus3 = ViaI2CBus3Init(pScrn);
3717e6fb56fSmrg
3727e6fb56fSmrg#ifdef HAVE_DEBUG
3737e6fb56fSmrg    if (pVia->I2CScan) {
3747e6fb56fSmrg        if (pVia->pI2CBus2)
3757e6fb56fSmrg            ViaI2CScan(pVia->pI2CBus2);
3767e6fb56fSmrg        if (pVia->pI2CBus3)
3777e6fb56fSmrg            ViaI2CScan(pVia->pI2CBus3);
3787e6fb56fSmrg    }
3797e6fb56fSmrg#endif
3807e6fb56fSmrg}
381