root/drivers/isdn/mISDN/hwchannel.c
// SPDX-License-Identifier: GPL-2.0-only
/*
 *
 * Author       Karsten Keil <kkeil@novell.com>
 *
 * Copyright 2008  by Karsten Keil <kkeil@novell.com>
 */

#include <linux/gfp.h>
#include <linux/module.h>
#include <linux/mISDNhw.h>

static void
dchannel_bh(struct work_struct *ws)
{
        struct dchannel *dch  = container_of(ws, struct dchannel, workq);
        struct sk_buff  *skb;
        int             err;

        if (test_and_clear_bit(FLG_RECVQUEUE, &dch->Flags)) {
                while ((skb = skb_dequeue(&dch->rqueue))) {
                        if (likely(dch->dev.D.peer)) {
                                err = dch->dev.D.recv(dch->dev.D.peer, skb);
                                if (err)
                                        dev_kfree_skb(skb);
                        } else
                                dev_kfree_skb(skb);
                }
        }
        if (test_and_clear_bit(FLG_PHCHANGE, &dch->Flags)) {
                if (dch->phfunc)
                        dch->phfunc(dch);
        }
}

static void
bchannel_bh(struct work_struct *ws)
{
        struct bchannel *bch  = container_of(ws, struct bchannel, workq);
        struct sk_buff  *skb;
        int             err;

        if (test_and_clear_bit(FLG_RECVQUEUE, &bch->Flags)) {
                while ((skb = skb_dequeue(&bch->rqueue))) {
                        bch->rcount--;
                        if (likely(bch->ch.peer)) {
                                err = bch->ch.recv(bch->ch.peer, skb);
                                if (err)
                                        dev_kfree_skb(skb);
                        } else
                                dev_kfree_skb(skb);
                }
        }
}

int
mISDN_initdchannel(struct dchannel *ch, int maxlen, void *phf)
{
        test_and_set_bit(FLG_HDLC, &ch->Flags);
        ch->maxlen = maxlen;
        ch->hw = NULL;
        ch->rx_skb = NULL;
        ch->tx_skb = NULL;
        ch->tx_idx = 0;
        ch->phfunc = phf;
        skb_queue_head_init(&ch->squeue);
        skb_queue_head_init(&ch->rqueue);
        INIT_LIST_HEAD(&ch->dev.bchannels);
        INIT_WORK(&ch->workq, dchannel_bh);
        return 0;
}
EXPORT_SYMBOL(mISDN_initdchannel);

int
mISDN_initbchannel(struct bchannel *ch, unsigned short maxlen,
                   unsigned short minlen)
{
        ch->Flags = 0;
        ch->minlen = minlen;
        ch->next_minlen = minlen;
        ch->init_minlen = minlen;
        ch->maxlen = maxlen;
        ch->next_maxlen = maxlen;
        ch->init_maxlen = maxlen;
        ch->hw = NULL;
        ch->rx_skb = NULL;
        ch->tx_skb = NULL;
        ch->tx_idx = 0;
        skb_queue_head_init(&ch->rqueue);
        ch->rcount = 0;
        ch->next_skb = NULL;
        INIT_WORK(&ch->workq, bchannel_bh);
        return 0;
}
EXPORT_SYMBOL(mISDN_initbchannel);

int
mISDN_freedchannel(struct dchannel *ch)
{
        if (ch->tx_skb) {
                dev_kfree_skb(ch->tx_skb);
                ch->tx_skb = NULL;
        }
        if (ch->rx_skb) {
                dev_kfree_skb(ch->rx_skb);
                ch->rx_skb = NULL;
        }
        skb_queue_purge(&ch->squeue);
        skb_queue_purge(&ch->rqueue);
        flush_work(&ch->workq);
        return 0;
}
EXPORT_SYMBOL(mISDN_freedchannel);

void
mISDN_clear_bchannel(struct bchannel *ch)
{
        if (ch->tx_skb) {
                dev_kfree_skb(ch->tx_skb);
                ch->tx_skb = NULL;
        }
        ch->tx_idx = 0;
        if (ch->rx_skb) {
                dev_kfree_skb(ch->rx_skb);
                ch->rx_skb = NULL;
        }
        if (ch->next_skb) {
                dev_kfree_skb(ch->next_skb);
                ch->next_skb = NULL;
        }
        test_and_clear_bit(FLG_TX_BUSY, &ch->Flags);
        test_and_clear_bit(FLG_TX_NEXT, &ch->Flags);
        test_and_clear_bit(FLG_ACTIVE, &ch->Flags);
        test_and_clear_bit(FLG_FILLEMPTY, &ch->Flags);
        test_and_clear_bit(FLG_TX_EMPTY, &ch->Flags);
        test_and_clear_bit(FLG_RX_OFF, &ch->Flags);
        ch->dropcnt = 0;
        ch->minlen = ch->init_minlen;
        ch->next_minlen = ch->init_minlen;
        ch->maxlen = ch->init_maxlen;
        ch->next_maxlen = ch->init_maxlen;
        skb_queue_purge(&ch->rqueue);
        ch->rcount = 0;
}
EXPORT_SYMBOL(mISDN_clear_bchannel);

void
mISDN_freebchannel(struct bchannel *ch)
{
        cancel_work_sync(&ch->workq);
        mISDN_clear_bchannel(ch);
}
EXPORT_SYMBOL(mISDN_freebchannel);

int
mISDN_ctrl_bchannel(struct bchannel *bch, struct mISDN_ctrl_req *cq)
{
        int ret = 0;

        switch (cq->op) {
        case MISDN_CTRL_GETOP:
                cq->op = MISDN_CTRL_RX_BUFFER | MISDN_CTRL_FILL_EMPTY |
                         MISDN_CTRL_RX_OFF;
                break;
        case MISDN_CTRL_FILL_EMPTY:
                if (cq->p1) {
                        memset(bch->fill, cq->p2 & 0xff, MISDN_BCH_FILL_SIZE);
                        test_and_set_bit(FLG_FILLEMPTY, &bch->Flags);
                } else {
                        test_and_clear_bit(FLG_FILLEMPTY, &bch->Flags);
                }
                break;
        case MISDN_CTRL_RX_OFF:
                /* read back dropped byte count */
                cq->p2 = bch->dropcnt;
                if (cq->p1)
                        test_and_set_bit(FLG_RX_OFF, &bch->Flags);
                else
                        test_and_clear_bit(FLG_RX_OFF, &bch->Flags);
                bch->dropcnt = 0;
                break;
        case MISDN_CTRL_RX_BUFFER:
                if (cq->p2 > MISDN_CTRL_RX_SIZE_IGNORE)
                        bch->next_maxlen = cq->p2;
                if (cq->p1 > MISDN_CTRL_RX_SIZE_IGNORE)
                        bch->next_minlen = cq->p1;
                /* we return the old values */
                cq->p1 = bch->minlen;
                cq->p2 = bch->maxlen;
                break;
        default:
                pr_info("mISDN unhandled control %x operation\n", cq->op);
                ret = -EINVAL;
                break;
        }
        return ret;
}
EXPORT_SYMBOL(mISDN_ctrl_bchannel);

static inline u_int
get_sapi_tei(u_char *p)
{
        u_int   sapi, tei;

        sapi = *p >> 2;
        tei = p[1] >> 1;
        return sapi | (tei << 8);
}

void
recv_Dchannel(struct dchannel *dch)
{
        struct mISDNhead *hh;

        if (dch->rx_skb->len < 2) { /* at least 2 for sapi / tei */
                dev_kfree_skb(dch->rx_skb);
                dch->rx_skb = NULL;
                return;
        }
        hh = mISDN_HEAD_P(dch->rx_skb);
        hh->prim = PH_DATA_IND;
        hh->id = get_sapi_tei(dch->rx_skb->data);
        skb_queue_tail(&dch->rqueue, dch->rx_skb);
        dch->rx_skb = NULL;
        schedule_event(dch, FLG_RECVQUEUE);
}
EXPORT_SYMBOL(recv_Dchannel);

void
recv_Echannel(struct dchannel *ech, struct dchannel *dch)
{
        struct mISDNhead *hh;

        if (ech->rx_skb->len < 2) { /* at least 2 for sapi / tei */
                dev_kfree_skb(ech->rx_skb);
                ech->rx_skb = NULL;
                return;
        }
        hh = mISDN_HEAD_P(ech->rx_skb);
        hh->prim = PH_DATA_E_IND;
        hh->id = get_sapi_tei(ech->rx_skb->data);
        skb_queue_tail(&dch->rqueue, ech->rx_skb);
        ech->rx_skb = NULL;
        schedule_event(dch, FLG_RECVQUEUE);
}
EXPORT_SYMBOL(recv_Echannel);

void
recv_Bchannel(struct bchannel *bch, unsigned int id, bool force)
{
        struct mISDNhead *hh;

        /* if allocation did fail upper functions still may call us */
        if (unlikely(!bch->rx_skb))
                return;
        if (unlikely(!bch->rx_skb->len)) {
                /* we have no data to send - this may happen after recovery
                 * from overflow or too small allocation.
                 * We need to free the buffer here */
                dev_kfree_skb(bch->rx_skb);
                bch->rx_skb = NULL;
        } else {
                if (test_bit(FLG_TRANSPARENT, &bch->Flags) &&
                    (bch->rx_skb->len < bch->minlen) && !force)
                                return;
                hh = mISDN_HEAD_P(bch->rx_skb);
                hh->prim = PH_DATA_IND;
                hh->id = id;
                if (bch->rcount >= 64) {
                        printk(KERN_WARNING
                               "B%d receive queue overflow - flushing!\n",
                               bch->nr);
                        skb_queue_purge(&bch->rqueue);
                }
                bch->rcount++;
                skb_queue_tail(&bch->rqueue, bch->rx_skb);
                bch->rx_skb = NULL;
                schedule_event(bch, FLG_RECVQUEUE);
        }
}
EXPORT_SYMBOL(recv_Bchannel);

void
recv_Dchannel_skb(struct dchannel *dch, struct sk_buff *skb)
{
        skb_queue_tail(&dch->rqueue, skb);
        schedule_event(dch, FLG_RECVQUEUE);
}
EXPORT_SYMBOL(recv_Dchannel_skb);

void
recv_Bchannel_skb(struct bchannel *bch, struct sk_buff *skb)
{
        if (bch->rcount >= 64) {
                printk(KERN_WARNING "B-channel %p receive queue overflow, "
                       "flushing!\n", bch);
                skb_queue_purge(&bch->rqueue);
                bch->rcount = 0;
        }
        bch->rcount++;
        skb_queue_tail(&bch->rqueue, skb);
        schedule_event(bch, FLG_RECVQUEUE);
}
EXPORT_SYMBOL(recv_Bchannel_skb);

static void
confirm_Dsend(struct dchannel *dch)
{
        struct sk_buff  *skb;

        skb = _alloc_mISDN_skb(PH_DATA_CNF, mISDN_HEAD_ID(dch->tx_skb),
                               0, NULL, GFP_ATOMIC);
        if (!skb) {
                printk(KERN_ERR "%s: no skb id %x\n", __func__,
                       mISDN_HEAD_ID(dch->tx_skb));
                return;
        }
        skb_queue_tail(&dch->rqueue, skb);
        schedule_event(dch, FLG_RECVQUEUE);
}

int
get_next_dframe(struct dchannel *dch)
{
        dch->tx_idx = 0;
        dch->tx_skb = skb_dequeue(&dch->squeue);
        if (dch->tx_skb) {
                confirm_Dsend(dch);
                return 1;
        }
        dch->tx_skb = NULL;
        test_and_clear_bit(FLG_TX_BUSY, &dch->Flags);
        return 0;
}
EXPORT_SYMBOL(get_next_dframe);

static void
confirm_Bsend(struct bchannel *bch)
{
        struct sk_buff  *skb;

        if (bch->rcount >= 64) {
                printk(KERN_WARNING "B-channel %p receive queue overflow, "
                       "flushing!\n", bch);
                skb_queue_purge(&bch->rqueue);
                bch->rcount = 0;
        }
        skb = _alloc_mISDN_skb(PH_DATA_CNF, mISDN_HEAD_ID(bch->tx_skb),
                               0, NULL, GFP_ATOMIC);
        if (!skb) {
                printk(KERN_ERR "%s: no skb id %x\n", __func__,
                       mISDN_HEAD_ID(bch->tx_skb));
                return;
        }
        bch->rcount++;
        skb_queue_tail(&bch->rqueue, skb);
        schedule_event(bch, FLG_RECVQUEUE);
}

int
get_next_bframe(struct bchannel *bch)
{
        bch->tx_idx = 0;
        if (test_bit(FLG_TX_NEXT, &bch->Flags)) {
                bch->tx_skb = bch->next_skb;
                if (bch->tx_skb) {
                        bch->next_skb = NULL;
                        test_and_clear_bit(FLG_TX_NEXT, &bch->Flags);
                        /* confirm imediately to allow next data */
                        confirm_Bsend(bch);
                        return 1;
                } else {
                        test_and_clear_bit(FLG_TX_NEXT, &bch->Flags);
                        printk(KERN_WARNING "B TX_NEXT without skb\n");
                }
        }
        bch->tx_skb = NULL;
        test_and_clear_bit(FLG_TX_BUSY, &bch->Flags);
        return 0;
}
EXPORT_SYMBOL(get_next_bframe);

void
queue_ch_frame(struct mISDNchannel *ch, u_int pr, int id, struct sk_buff *skb)
{
        struct mISDNhead *hh;

        if (!skb) {
                _queue_data(ch, pr, id, 0, NULL, GFP_ATOMIC);
        } else {
                if (ch->peer) {
                        hh = mISDN_HEAD_P(skb);
                        hh->prim = pr;
                        hh->id = id;
                        if (!ch->recv(ch->peer, skb))
                                return;
                }
                dev_kfree_skb(skb);
        }
}
EXPORT_SYMBOL(queue_ch_frame);

int
dchannel_senddata(struct dchannel *ch, struct sk_buff *skb)
{
        /* check oversize */
        if (skb->len <= 0) {
                printk(KERN_WARNING "%s: skb too small\n", __func__);
                return -EINVAL;
        }
        if (skb->len > ch->maxlen) {
                printk(KERN_WARNING "%s: skb too large(%d/%d)\n",
                       __func__, skb->len, ch->maxlen);
                return -EINVAL;
        }
        /* HW lock must be obtained */
        if (test_and_set_bit(FLG_TX_BUSY, &ch->Flags)) {
                skb_queue_tail(&ch->squeue, skb);
                return 0;
        } else {
                /* write to fifo */
                ch->tx_skb = skb;
                ch->tx_idx = 0;
                return 1;
        }
}
EXPORT_SYMBOL(dchannel_senddata);

int
bchannel_senddata(struct bchannel *ch, struct sk_buff *skb)
{

        /* check oversize */
        if (skb->len <= 0) {
                printk(KERN_WARNING "%s: skb too small\n", __func__);
                return -EINVAL;
        }
        if (skb->len > ch->maxlen) {
                printk(KERN_WARNING "%s: skb too large(%d/%d)\n",
                       __func__, skb->len, ch->maxlen);
                return -EINVAL;
        }
        /* HW lock must be obtained */
        /* check for pending next_skb */
        if (ch->next_skb) {
                printk(KERN_WARNING
                       "%s: next_skb exist ERROR (skb->len=%d next_skb->len=%d)\n",
                       __func__, skb->len, ch->next_skb->len);
                return -EBUSY;
        }
        if (test_and_set_bit(FLG_TX_BUSY, &ch->Flags)) {
                test_and_set_bit(FLG_TX_NEXT, &ch->Flags);
                ch->next_skb = skb;
                return 0;
        } else {
                /* write to fifo */
                ch->tx_skb = skb;
                ch->tx_idx = 0;
                confirm_Bsend(ch);
                return 1;
        }
}
EXPORT_SYMBOL(bchannel_senddata);

/* The function allocates a new receive skb on demand with a size for the
 * requirements of the current protocol. It returns the tailroom of the
 * receive skb or an error.
 */
int
bchannel_get_rxbuf(struct bchannel *bch, int reqlen)
{
        int len;

        if (bch->rx_skb) {
                len = skb_tailroom(bch->rx_skb);
                if (len < reqlen) {
                        pr_warn("B%d no space for %d (only %d) bytes\n",
                                bch->nr, reqlen, len);
                        if (test_bit(FLG_TRANSPARENT, &bch->Flags)) {
                                /* send what we have now and try a new buffer */
                                recv_Bchannel(bch, 0, true);
                        } else {
                                /* on HDLC we have to drop too big frames */
                                return -EMSGSIZE;
                        }
                } else {
                        return len;
                }
        }
        /* update current min/max length first */
        if (unlikely(bch->maxlen != bch->next_maxlen))
                bch->maxlen = bch->next_maxlen;
        if (unlikely(bch->minlen != bch->next_minlen))
                bch->minlen = bch->next_minlen;
        if (unlikely(reqlen > bch->maxlen))
                return -EMSGSIZE;
        if (test_bit(FLG_TRANSPARENT, &bch->Flags)) {
                if (reqlen >= bch->minlen) {
                        len = reqlen;
                } else {
                        len = 2 * bch->minlen;
                        if (len > bch->maxlen)
                                len = bch->maxlen;
                }
        } else {
                /* with HDLC we do not know the length yet */
                len = bch->maxlen;
        }
        bch->rx_skb = mI_alloc_skb(len, GFP_ATOMIC);
        if (!bch->rx_skb) {
                pr_warn("B%d receive no memory for %d bytes\n", bch->nr, len);
                len = -ENOMEM;
        }
        return len;
}
EXPORT_SYMBOL(bchannel_get_rxbuf);