v4l2_id &= V4L2_CTRL_ID_MASK;
 
        /* Find the control. */
-       __uvc_find_control(chain->processing, v4l2_id, mapping, &ctrl, next);
-       if (ctrl && !next)
-               return ctrl;
-
-       list_for_each_entry(entity, &chain->iterms, chain) {
-               __uvc_find_control(entity, v4l2_id, mapping, &ctrl, next);
-               if (ctrl && !next)
-                       return ctrl;
-       }
-
-       list_for_each_entry(entity, &chain->extensions, chain) {
+       list_for_each_entry(entity, &chain->entities, chain) {
                __uvc_find_control(entity, v4l2_id, mapping, &ctrl, next);
                if (ctrl && !next)
                        return ctrl;
        int ret = 0;
 
        /* Find the control. */
-       ret = uvc_ctrl_commit_entity(chain->dev, chain->processing, rollback);
-       if (ret < 0)
-               goto done;
-
-       list_for_each_entry(entity, &chain->iterms, chain) {
-               ret = uvc_ctrl_commit_entity(chain->dev, entity, rollback);
-               if (ret < 0)
-                       goto done;
-       }
-
-       list_for_each_entry(entity, &chain->extensions, chain) {
+       list_for_each_entry(entity, &chain->entities, chain) {
                ret = uvc_ctrl_commit_entity(chain->dev, entity, rollback);
                if (ret < 0)
                        goto done;
        int ret;
 
        /* Find the extension unit. */
-       list_for_each_entry(entity, &chain->extensions, chain) {
-               if (entity->id == xctrl->unit)
+       list_for_each_entry(entity, &chain->entities, chain) {
+               if (UVC_ENTITY_TYPE(entity) == UVC_VC_EXTENSION_UNIT &&
+                   entity->id == xctrl->unit)
                        break;
        }
 
 
                        return -1;
                }
 
-               list_add_tail(&entity->chain, &chain->extensions);
                break;
 
        case UVC_VC_PROCESSING_UNIT:
                if (uvc_trace_param & UVC_TRACE_PROBE)
                        printk(" <- IT %d\n", entity->id);
 
-               list_add_tail(&entity->chain, &chain->iterms);
                break;
 
        case UVC_TT_STREAMING:
                        return -1;
                }
 
-               list_add_tail(&entity->chain, &chain->iterms);
                break;
 
        default:
                return -1;
        }
 
+       list_add_tail(&entity->chain, &chain->entities);
        return 0;
 }
 
                                return -EINVAL;
                        }
 
-                       list_add_tail(&forward->chain, &chain->extensions);
+                       list_add_tail(&forward->chain, &chain->entities);
                        if (uvc_trace_param & UVC_TRACE_PROBE) {
                                if (!found)
                                        printk(" (->");
                                return -EINVAL;
                        }
 
-                       list_add_tail(&forward->chain, &chain->oterms);
+                       list_add_tail(&forward->chain, &chain->entities);
                        if (uvc_trace_param & UVC_TRACE_PROBE) {
                                if (!found)
                                        printk(" (->");
                        if (uvc_trace_param & UVC_TRACE_PROBE)
                                printk(" %d", term->id);
 
-                       list_add_tail(&term->chain, &chain->iterms);
+                       list_add_tail(&term->chain, &chain->entities);
                        uvc_scan_chain_forward(chain, term, entity);
                }
 
        int id;
 
        entity = oterm;
-       list_add_tail(&entity->chain, &chain->oterms);
+       list_add_tail(&entity->chain, &chain->entities);
        uvc_trace(UVC_TRACE_PROBE, "Scanning UVC chain: OT %d", entity->id);
 
        id = entity->output.bSourceID;
        return 0;
 }
 
-static unsigned int uvc_print_terms(struct list_head *terms, char *buffer)
+static unsigned int uvc_print_terms(struct list_head *terms, u16 dir,
+               char *buffer)
 {
        struct uvc_entity *term;
        unsigned int nterms = 0;
        char *p = buffer;
 
        list_for_each_entry(term, terms, chain) {
-               p += sprintf(p, "%u", term->id);
-               if (term->chain.next != terms) {
+               if (!UVC_ENTITY_IS_TERM(term) ||
+                   UVC_TERM_DIRECTION(term) != dir)
+                       continue;
+
+               if (nterms)
                        p += sprintf(p, ",");
-                       if (++nterms >= 4) {
-                               p += sprintf(p, "...");
-                               break;
-                       }
+               if (++nterms >= 4) {
+                       p += sprintf(p, "...");
+                       break;
                }
+               p += sprintf(p, "%u", term->id);
        }
 
        return p - buffer;
        static char buffer[43];
        char *p = buffer;
 
-       p += uvc_print_terms(&chain->iterms, p);
+       p += uvc_print_terms(&chain->entities, UVC_TERM_INPUT, p);
        p += sprintf(p, " -> ");
-       uvc_print_terms(&chain->oterms, p);
+       uvc_print_terms(&chain->entities, UVC_TERM_OUTPUT, p);
 
        return buffer;
 }
                if (chain == NULL)
                        return -ENOMEM;
 
-               INIT_LIST_HEAD(&chain->iterms);
-               INIT_LIST_HEAD(&chain->oterms);
-               INIT_LIST_HEAD(&chain->extensions);
+               INIT_LIST_HEAD(&chain->entities);
                mutex_init(&chain->ctrl_mutex);
                chain->dev = dev;
 
  * Register all video devices in all chains.
  */
 static int uvc_register_terms(struct uvc_device *dev,
-       struct uvc_video_chain *chain, struct list_head *terms)
+       struct uvc_video_chain *chain)
 {
        struct uvc_streaming *stream;
        struct uvc_entity *term;
        int ret;
 
-       list_for_each_entry(term, terms, chain) {
+       list_for_each_entry(term, &chain->entities, chain) {
                if (UVC_ENTITY_TYPE(term) != UVC_TT_STREAMING)
                        continue;
 
        int ret;
 
        list_for_each_entry(chain, &dev->chains, list) {
-               ret = uvc_register_terms(dev, chain, &chain->iterms);
-               if (ret < 0)
-                       return ret;
-
-               ret = uvc_register_terms(dev, chain, &chain->oterms);
+               ret = uvc_register_terms(dev, chain);
                if (ret < 0)
                        return ret;
        }