#include "uhci-hcd.h"
 
+#define EXTRA_SPACE    1024
+
 static struct dentry *uhci_debugfs_root;
 
 #ifdef DEBUG
        char *spid;
        u32 status, token;
 
-       /* Try to make sure there's enough memory */
-       if (len < 160)
-               return 0;
-
        status = td_status(uhci, td);
        out += sprintf(out, "%*s[%p] link (%08x) ", space, "", td,
                hc32_to_cpu(uhci, td->link));
                (status & TD_CTRL_CRCTIMEO) ? "CRC/Timeo " : "",
                (status & TD_CTRL_BITSTUFF) ? "BitStuff " : "",
                status & 0x7ff);
+       if (out - buf > len)
+               goto done;
 
        token = td_token(uhci, td);
        switch (uhci_packetid(token)) {
                spid);
        out += sprintf(out, "(buf=%08x)\n", hc32_to_cpu(uhci, td->buffer));
 
+done:
+       if (out - buf > len)
+               out += sprintf(out, " ...\n");
        return out - buf;
 }
 
        int i, nactive, ninactive;
        char *ptype;
 
-       if (len < 200)
-               return 0;
 
        out += sprintf(out, "urb_priv [%p] ", urbp);
        out += sprintf(out, "urb [%p] ", urbp->urb);
        out += sprintf(out, "Dev=%d ", usb_pipedevice(urbp->urb->pipe));
        out += sprintf(out, "EP=%x(%s) ", usb_pipeendpoint(urbp->urb->pipe),
                        (usb_pipein(urbp->urb->pipe) ? "IN" : "OUT"));
+       if (out - buf > len)
+               goto done;
 
        switch (usb_pipetype(urbp->urb->pipe)) {
        case PIPE_ISOCHRONOUS: ptype = "ISO"; break;
                out += sprintf(out, " Unlinked=%d", urbp->urb->unlinked);
        out += sprintf(out, "\n");
 
+       if (out - buf > len)
+               goto done;
+
        i = nactive = ninactive = 0;
        list_for_each_entry(td, &urbp->td_list, list) {
                if (urbp->qh->type != USB_ENDPOINT_XFER_ISOC &&
                        out += sprintf(out, "%*s%d: ", space + 2, "", i);
                        out += uhci_show_td(uhci, td, out,
                                        len - (out - buf), 0);
+                       if (out - buf > len)
+                               goto tail;
                } else {
                        if (td_status(uhci, td) & TD_CTRL_ACTIVE)
                                ++nactive;
                out += sprintf(out, "%*s[skipped %d inactive and %d active "
                                "TDs]\n",
                                space, "", ninactive, nactive);
-
+done:
+       if (out - buf > len)
+               out += sprintf(out, " ...\n");
+tail:
        return out - buf;
 }
 
        __hc32 element = qh_element(qh);
        char *qtype;
 
-       /* Try to make sure there's enough memory */
-       if (len < 80 * 7)
-               return 0;
-
        switch (qh->type) {
        case USB_ENDPOINT_XFER_ISOC: qtype = "ISO"; break;
        case USB_ENDPOINT_XFER_INT: qtype = "INT"; break;
        else if (qh->type == USB_ENDPOINT_XFER_INT)
                out += sprintf(out, "%*s    period %d phase %d load %d us\n",
                                space, "", qh->period, qh->phase, qh->load);
+       if (out - buf > len)
+               goto done;
 
        if (element & UHCI_PTR_QH(uhci))
                out += sprintf(out, "%*s  Element points to QH (bug?)\n", space, "");
        if (!(element & ~(UHCI_PTR_QH(uhci) | UHCI_PTR_DEPTH(uhci))))
                out += sprintf(out, "%*s  Element is NULL (bug?)\n", space, "");
 
+       if (out - buf > len)
+               goto done;
+
        if (list_empty(&qh->queue)) {
                out += sprintf(out, "%*s  queue is empty\n", space, "");
-               if (qh == uhci->skel_async_qh)
+               if (qh == uhci->skel_async_qh) {
                        out += uhci_show_td(uhci, uhci->term_td, out,
                                        len - (out - buf), 0);
+                       if (out - buf > len)
+                               goto tail;
+               }
        } else {
                struct urb_priv *urbp = list_entry(qh->queue.next,
                                struct urb_priv, node);
                                        space, "");
                i = nurbs = 0;
                list_for_each_entry(urbp, &qh->queue, node) {
-                       if (++i <= 10)
+                       if (++i <= 10) {
                                out += uhci_show_urbp(uhci, urbp, out,
                                                len - (out - buf), space + 2);
+                               if (out - buf > len)
+                                       goto tail;
+                       }
                        else
                                ++nurbs;
                }
                                        space, "", nurbs);
        }
 
+       if (out - buf > len)
+               goto done;
+
        if (qh->dummy_td) {
                out += sprintf(out, "%*s  Dummy TD\n", space, "");
                out += uhci_show_td(uhci, qh->dummy_td, out,
                                len - (out - buf), 0);
+               if (out - buf > len)
+                       goto tail;
        }
 
+done:
+       if (out - buf > len)
+               out += sprintf(out, " ...\n");
+tail:
        return out - buf;
 }
 
-static int uhci_show_sc(int port, unsigned short status, char *buf, int len)
+static int uhci_show_sc(int port, unsigned short status, char *buf)
 {
-       char *out = buf;
-
-       /* Try to make sure there's enough memory */
-       if (len < 160)
-               return 0;
-
-       out += sprintf(out, "  stat%d     =     %04x  %s%s%s%s%s%s%s%s%s%s\n",
+       return sprintf(buf, "  stat%d     =     %04x  %s%s%s%s%s%s%s%s%s%s\n",
                port,
                status,
                (status & USBPORTSC_SUSP) ?     " Suspend" : "",
                (status & USBPORTSC_PE) ?       " Enabled" : "",
                (status & USBPORTSC_CSC) ?      " ConnectChange" : "",
                (status & USBPORTSC_CCS) ?      " Connected" : "");
-
-       return out - buf;
 }
 
-static int uhci_show_root_hub_state(struct uhci_hcd *uhci, char *buf, int len)
+static int uhci_show_root_hub_state(struct uhci_hcd *uhci, char *buf)
 {
-       char *out = buf;
        char *rh_state;
 
-       /* Try to make sure there's enough memory */
-       if (len < 60)
-               return 0;
-
        switch (uhci->rh_state) {
            case UHCI_RH_RESET:
                rh_state = "reset";             break;
            default:
                rh_state = "?";                 break;
        }
-       out += sprintf(out, "Root-hub state: %s   FSBR: %d\n",
+       return sprintf(buf, "Root-hub state: %s   FSBR: %d\n",
                        rh_state, uhci->fsbr_is_on);
-       return out - buf;
 }
 
 static int uhci_show_status(struct uhci_hcd *uhci, char *buf, int len)
        unsigned char sof;
        unsigned short portsc1, portsc2;
 
-       /* Try to make sure there's enough memory */
-       if (len < 80 * 9)
-               return 0;
 
        usbcmd    = uhci_readw(uhci, 0);
        usbstat   = uhci_readw(uhci, 2);
                (usbcmd & USBCMD_GRESET) ?  "GRESET " : "",
                (usbcmd & USBCMD_HCRESET) ? "HCRESET " : "",
                (usbcmd & USBCMD_RS) ?      "RS " : "");
+       if (out - buf > len)
+               goto done;
 
        out += sprintf(out, "  usbstat   =     %04x   %s%s%s%s%s%s\n",
                usbstat,
                (usbstat & USBSTS_RD) ?     "ResumeDetect " : "",
                (usbstat & USBSTS_ERROR) ?  "USBError " : "",
                (usbstat & USBSTS_USBINT) ? "USBINT " : "");
+       if (out - buf > len)
+               goto done;
 
        out += sprintf(out, "  usbint    =     %04x\n", usbint);
        out += sprintf(out, "  usbfrnum  =   (%d)%03x\n", (usbfrnum >> 10) & 1,
                0xfff & (4*(unsigned int)usbfrnum));
        out += sprintf(out, "  flbaseadd = %08x\n", flbaseadd);
        out += sprintf(out, "  sof       =       %02x\n", sof);
-       out += uhci_show_sc(1, portsc1, out, len - (out - buf));
-       out += uhci_show_sc(2, portsc2, out, len - (out - buf));
-       out += sprintf(out, "Most recent frame: %x (%d)   "
-                       "Last ISO frame: %x (%d)\n",
+       if (out - buf > len)
+               goto done;
+
+       out += uhci_show_sc(1, portsc1, out);
+       if (out - buf > len)
+               goto done;
+
+       out += uhci_show_sc(2, portsc2, out);
+       if (out - buf > len)
+               goto done;
+
+       out += sprintf(out,
+                       "Most recent frame: %x (%d)   Last ISO frame: %x (%d)\n",
                        uhci->frame_number, uhci->frame_number & 1023,
                        uhci->last_iso_frame, uhci->last_iso_frame & 1023);
 
+done:
+       if (out - buf > len)
+               out += sprintf(out, " ...\n");
        return out - buf;
 }
 
                "int8", "int4", "int2", "async", "term"
        };
 
-       out += uhci_show_root_hub_state(uhci, out, len - (out - buf));
+       out += uhci_show_root_hub_state(uhci, out);
+       if (out - buf > len)
+               goto done;
        out += sprintf(out, "HC status\n");
        out += uhci_show_status(uhci, out, len - (out - buf));
+       if (out - buf > len)
+               goto tail;
 
        out += sprintf(out, "Periodic load table\n");
        for (i = 0; i < MAX_PHASE; ++i) {
                        uhci_to_hcd(uhci)->self.bandwidth_int_reqs,
                        uhci_to_hcd(uhci)->self.bandwidth_isoc_reqs);
        if (debug <= 1)
-               return out - buf;
+               goto tail;
 
        out += sprintf(out, "Frame List\n");
        nframes = 10;
        for (i = 0; i < UHCI_NUMFRAMES; ++i) {
                __hc32 qh_dma;
 
+               if (out - buf > len)
+                       goto done;
                j = 0;
                td = uhci->frame_cpu[i];
                link = uhci->frame[i];
                        td = list_entry(tmp, struct uhci_td, fl_list);
                        tmp = tmp->next;
                        if (link != LINK_TO_TD(uhci, td)) {
-                               if (nframes > 0)
+                               if (nframes > 0) {
                                        out += sprintf(out, "    link does "
                                                "not match list entry!\n");
-                               else
+                                       if (out - buf > len)
+                                               goto done;
+                               } else
                                        ++nerrs;
                        }
-                       if (nframes > 0)
+                       if (nframes > 0) {
                                out += uhci_show_td(uhci, td, out,
                                                len - (out - buf), 4);
+                               if (out - buf > len)
+                                       goto tail;
+                       }
                        link = td->link;
                } while (tmp != head);
 
                                out += sprintf(out, "   link does not match "
                                        "QH (%08x)!\n",
                                        hc32_to_cpu(uhci, qh_dma));
+                               if (out - buf > len)
+                                       goto done;
                        } else
                                ++nerrs;
                }
 
        out += sprintf(out, "Skeleton QHs\n");
 
+       if (out - buf > len)
+               goto done;
+
        fsbr_link = 0;
        for (i = 0; i < UHCI_NUM_SKELQH; ++i) {
                int cnt = 0;
                qh = uhci->skelqh[i];
                out += sprintf(out, "- skel_%s_qh\n", qh_names[i]); \
                out += uhci_show_qh(uhci, qh, out, len - (out - buf), 4);
+               if (out - buf > len)
+                       goto tail;
 
                /* Last QH is the Terminating QH, it's different */
                if (i == SKEL_TERM) {
-                       if (qh_element(qh) != LINK_TO_TD(uhci, uhci->term_td))
+                       if (qh_element(qh) != LINK_TO_TD(uhci, uhci->term_td)) {
                                out += sprintf(out, "    skel_term_qh element is not set to term_td!\n");
+                               if (out - buf > len)
+                                       goto done;
+                       }
                        link = fsbr_link;
                        if (!link)
                                link = LINK_TO_QH(uhci, uhci->skel_term_qh);
                while (tmp != head) {
                        qh = list_entry(tmp, struct uhci_qh, node);
                        tmp = tmp->next;
-                       if (++cnt <= 10)
+                       if (++cnt <= 10) {
                                out += uhci_show_qh(uhci, qh, out,
                                                len - (out - buf), 4);
+                               if (out - buf > len)
+                                       goto tail;
+                       }
                        if (!fsbr_link && qh->skel >= SKEL_FSBR)
                                fsbr_link = LINK_TO_QH(uhci, qh);
                }
 check_qh_link:
                if (qh->link != link)
                        out += sprintf(out, "    last QH not linked to next skeleton!\n");
+
+               if (out - buf > len)
+                       goto done;
        }
 
+done:
+       if (out - buf > len)
+               out += sprintf(out, " ...\n");
+tail:
        return out - buf;
 }
 
        up->size = 0;
        spin_lock_irqsave(&uhci->lock, flags);
        if (uhci->is_initialized)
-               up->size = uhci_sprint_schedule(uhci, up->data, MAX_OUTPUT);
+               up->size = uhci_sprint_schedule(uhci, up->data,
+                                       MAX_OUTPUT - EXTRA_SPACE);
        spin_unlock_irqrestore(&uhci->lock, flags);
 
        file->private_data = up;