<?php
/**
 * Intermediate procedural page parsing structure.
 * This structure parses defines, functions, and global variables by file,
 * and then iterates over the elements to document conflicts.
 * 
 * phpDocumentor :: automatic documentation generator
 * 
 * PHP versions 4 and 5
 *
 * Copyright (c) 2002-2008 Gregory Beaver
 * 
 * LICENSE:
 * 
 * This library is free software; you can redistribute it
 * and/or modify it under the terms of the GNU Lesser General
 * Public License as published by the Free Software Foundation;
 * either version 2.1 of the License, or (at your option) any
 * later version.
 * 
 * This library 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 GNU
 * Lesser General Public License for more details.
 * 
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
 *
 * @category  ToolsAndUtilities
 * @package   phpDocumentor
 * @author    Gregory Beaver <cellog@php.net>
 * @copyright 2002-2008 Gregory Beaver
 * @license   http://www.opensource.org/licenses/lgpl-license.php LGPL
 * @version   CVS: $Id: ProceduralPages.inc 253641 2008-02-24 02:35:44Z ashnazg $
 * @link      http://www.phpdoc.org
 * @link      http://pear.php.net/PhpDocumentor
 * @since     1.1
 * @todo      CS cleanup - change package to PhpDocumentor
 */

/**
 * Intermediate procedural page parsing structure.
 * This structure parses defines, functions, and global variables by file,
 * and then iterates over the elements to document conflicts.
 *
 * @category  ToolsAndUtilities
 * @package   phpDocumentor
 * @author    Greg Beaver <cellog@php.net>
 * @copyright 2002-2008 Gregory Beaver
 * @license   http://www.opensource.org/licenses/lgpl-license.php LGPL
 * @version   Release: 1.4.3
 * @link      http://www.phpdoc.org
 * @link      http://pear.php.net/PhpDocumentor
 * @since     1.1
 * @todo      CS cleanup - change package to PhpDocumentor
 */
class ProceduralPages
{
    /**
     * file being parsed, used in every add function 
     * to match up elements with the file that contains them
     *
     * @see addClass(), addMethod(), addVar(), nextFile()
     * @var string
     */
    var $curfile;
    /**
     * array of all procedural pages ordered by name
     * Format:
     * <pre>
     *     array(
     *         name => array(
     *             fullpath => parserPage,
     *             fullpath => parserPage2 [if there are name conflicts],
     *             ...
     *         )
     *     )
     * </pre>
     *
     * @var array
     */
    var $pages = array();
    /**
     * array of all procedural pages ordered by name 
     * that have been ignored via -po or @access private or @ignore
     * Format:
     * <pre>
     *     array(
     *         name => array(
     *             fullpath => parserPage,
     *             fullpath => parserPage2 [if there are name conflicts],
     *             ...
     *         )
     *     )
     * </pre>
     *
     * @var array
     */
    var $ignorepages = array();
    /**
     * array of all procedural page names ordered by full path to the file
     * Format:
     * <pre>
     *     array(
     *         fullpath => name
     *     )
     * </pre>
     *
     * @var array
     */
    var $pathpages = array();
    /**
     * array of parsed includes organized by the full path 
     * of the file that contains the include.
     * Format:
     * <pre>
     *     array(
     *         full path => array(
     *             includename => {@link parserInclude}
     *         )
     *     )
     * </pre>
     *
     * @var array
     */
    var $includesbyfile = array();
    /**
     * array of parsed functions organized by the full path 
     * of the file that contains the function.
     * Format:
     * <pre>
     *     array(
     *         full path => array(
     *             functionname => {@link parserFunction}
     *         )
     *     )
     * </pre>
     *
     * @var array
     */
    var $functionsbyfile = array();
    /**
     * array of parsed defines organized by the full path 
     * of the file that contains the define.
     * Format:
     * <pre>
     *     array(
     *         full path => array(
     *             definename => {@link parserDefine}
     *         )
     *     )
     * </pre>
     *
     * @var array
     */
    var $definesbyfile = array();
    /**
     * array of parsed global variables organized by the full path 
     * of the file that contains the global variable definition.
     * Format:
     * <pre>
     *     array(
     *         full path => array(
     *             globalname => {@link parserGlobal}
     *         )
     *     )
     * </pre>
     *
     * @var array
     */
    var $globalsbyfile = array();
    /**
     * array of file names organized by functions that are in the file.
     *
     * This structure is designed to handle name conflicts.  Two files can contain
     * functions with the same name, and this array will record both filenames to
     * help control namespace errors
     * Format:
     * <pre>
     *     array(
     *         functionname => array(
     *             full path of file containing functionname,
     *             full path of file 2 containing functionname,
     *             ...
     *         )
     *     )
     * </pre>
     *
     * @var array
     */
    var $functionsbynamefile = array();
    /**
     * array of file names organized by defines that are in the file.
     * This structure is designed to handle name conflicts.  Two files 
     * can contain defines with the same name, and this array will
     * record both filenames to help control namespace errors
     * Format:
     * <pre>
     *     array(
     *         definename => array(
     *             full path of file containing definename,
     *             full path of file 2 containing definename,
     *             ...
     *         )
     *     )
     * </pre>
     *
     * @var array
     */
    var $definesbynamefile = array();
    /**
     * array of file names organized by global variables that are in the file.
     *
     * This structure is designed to handle name conflicts.  Two files can 
     * contain global variables with the same name, and this array will
     * record both filenames to help control namespace errors
     * Format:
     * <pre>
     *     array(
     *         global variablename => array(
     *             full path of file containing global variablename,
     *             full path of file 2 containing global variablename,
     *             ...
     *         )
     *     )
     * </pre>
     *
     * @var array
     */
    var $globalsbynamefile = array();
    /**
     * array of packages ordered by full path
     * Format:
     * <pre>
     *     array(
     *         fullpath => array(
     *             packagename,
     *             subpackagename
     *         )
     *     )
     * </pre>
     *
     * @var array
     */
    var $pagepackages = array();
    /**
     * array of packages assigned to classes in a file, ordered by fullpath
     * Format:
     * <pre>
     *     array(
     *         fullpath => array(
     *             packagename => array(
     *                 subpackagename => 1,
     *                 subpackagename => 1,
     *                 ..
     *             ),
     *             packagename2 => array(...
     *             )
     *         )
     *     )
     * </pre>
     *
     * @var array
     */
    var $pageclasspackages = array();
    /**
     * Namespace conflicts within all documented packages of functions
     * Format:
     * <pre>
     *     array(
     *         functionname => array(
     *             full path,
     *             full path,
     *             ...
     *         )
     *     )
     * </pre>
     *
     * @var array
     */
    var $functionconflicts = array();
    /**
     * Namespace conflicts within all documented pages
     * Format:
     * <pre>
     *     array(
     *         pagename => array(
     *             fullpath,
     *             fullpath,
     *             ...
     *         )
     *     )
     * </pre>
     *
     * @var array
     */
    var $pageconflicts = array();
    /**
     * Namespace conflicts within all documented packages of functions
     * Format:
     * <pre>
     *     array(
     *         functionname => array(
     *             full path,
     *             full path,
     *             ...
     *         )
     *     )
     * </pre>
     *
     * @var array
     */
    var $defineconflicts = array();
    /**
     * Namespace conflicts within all documented packages of functions
     * Format:
     * <pre>
     *     array(
     *         functionname => array(
     *             full path,
     *             full path,
     *             ...
     *         )
     *     )
     * </pre>
     *
     * @var array
     */
    var $globalconflicts = array();
    /**
     * @access private
     * @var array
     */
    var $revcpbf = array();
    /**
     * @access private
     * @var boolean
     */
    var $packagesetup = false;

    /**
     * sets up the {@link $pages} array
     *
     * @param parserPage &$element the parser page element
     *
     * @return void
     */
    function addPage(&$element)
    {
        $this->curfile
            = $element->getPath();
        $this->pages[$element->getFile()][$element->getPath()]
            = $element;
        $this->pathpages[$this->curfile]
            = $element->getFile();
        $this->addPagePackage($this->curfile, 
            $element->package, $element->subpackage);
    }
    
    /**
     * moves a page from the {@link $pages} array to the {@link $ignorepages} array
     *
     * @param parserPage &$element the parser page element
     *
     * @return void
     */
    function ignorePage(&$element)
    {
        $this->ignorepages[$element->getFile()][$element->getPath()]
            = $this->pages[$element->getFile()][$element->getPath()];
        unset($this->pages[$element->getFile()][$element->getPath()]);
    }

    /**
     * gathers path-related info about a given element
     *
     * @param string $path path to the element
     * @param mixed  &$c   ???
     *
     * @return array|bool an array of path info,
     *                    or FALSE 
     * @todo figure out what &$c is and update the param tag
     */
    function getPathInfo($path, &$c)
    {
        $path = str_replace('/', SMART_PATH_DELIMITER, $path);
        $info = array();
        if (!isset($this->pathpages[$path])) {
            return false;
        }

        $p = $this->pages[$this->pathpages[$path]][$path];
        // fixes [ 1391432 ] Too many underscores in include links.
        $p->name = $p->origName;
        $p->name = $c->getPageName($p);

        $info['package']    = $p->package;
        $info['subpackage'] = $p->subpackage;
        $info['name']       = $p->getFile();
        $info['source_loc'] = $p->getSourceLocation($c);

        $x = new pageLink;
        $x->addLink($p->path, $p->name, $p->file, $p->package, $p->subpackage);

        $info['docs'] = $c->returnSee($x);
        $p->name      = $p->origName;

        return $info;
    }
    
    /**
     * Change a page's name from its file to alias $name
     *
     * This function is used to handle a @name tag in a page-level DocBlock
     *
     * @param string $name the alias
     *
     * @return void
     */
    function setName($name)
    {
        if ($this->pages[$name][$this->curfile]->file == $name) {
            addWarning(PDERROR_NAME_ALIAS_SAME_AS_TARGET,'');

        } else {
            $this->pages[$name][$this->curfile]
                = $this->pages[$this->pathpages[$this->curfile]][$this->curfile];
            $this->pages[$name][$this->curfile]->file
                = $name;

            unset($this->pages[$this->pathpages[$this->curfile]][$this->curfile]);

            $this->pathpages[$this->curfile] = $name;
        }
    }
    
    /**
     * Changes the package of the page represented by $path
     *
     * changes package in both the {@link $pages} array 
     * and the {@link pagepackages} array
     *
     * @param string $path       full path
     * @param string $package    the package name
     * @param string $subpackage the subpackage name
     *
     * @return void
     */
    function addPagePackage($path, $package, $subpackage)
    {
        $this->pages[$this->pathpages[$path]][$path]->package
            = $package;
        $this->pages[$this->pathpages[$path]][$path]->subpackage
            = $subpackage;
        $this->pagepackages[$path]
            = array($package, $subpackage);

        if (isset($this->includesbyfile[$path])) {
            foreach ($this->includesbyfile[$path] as $i => $el) {
                $el->package                     = $package;
                $el->subpackage                  = $subpackage;
                $this->includesbyfile[$path][$i] = $el;
            }
        }
        if (isset($this->functionsbyfile[$path])) {
            foreach ($this->functionsbyfile[$path] as $i => $el) {
                $el->package                      = $package;
                $el->subpackage                   = $subpackage;
                $this->functionsbyfile[$path][$i] = $el;
            }
        }
        if (isset($this->definesbyfile[$path])) {
            foreach ($this->definesbyfile[$path] as $i => $el) {
                $el->package                    = $package;
                $el->subpackage                 = $subpackage;
                $this->definesbyfile[$path][$i] = $el;
            }
        }
        if (isset($this->globalsbyfile[$path])) {
            foreach ($this->globalsbyfile[$path] as $i => $el) {
                $el->package                    = $package;
                $el->subpackage                 = $subpackage;
                $this->globalsbyfile[$path][$i] = $el;
            }
        }
    }

    /**
     * sets up the {@link $includesbyfile} array using {@link $curfile}
     *
     * @param parserInclude &$element the "include" element object
     *
     * @return void
     */
    function addInclude(&$element)
    {
        $this->includesbyfile[$this->curfile][] = $element;
    }

    /**
     * sets up the {@link $functionsbyfile} array using {@link $curfile}
     *
     * @param parserFunction &$element the "function" object
     *
     * @return void
     */
    function addFunction(&$element)
    {
        if (isset($this->functionsbyfile[$this->curfile])) {
            foreach ($this->functionsbyfile[$this->curfile] as $i => $function) {
                if ($function->getName() == $element->getName()) {
                    addWarning(PDERROR_ELEMENT_IGNORED, 'function',
                        $element->getName(), $this->curfile);
                    return;
                }
            }
        }
        $this->functionsbyfile[$this->curfile][]          = $element;
        $this->functionsbynamefile[$element->getName()][] = $this->curfile;
    }

    /**
     * sets up the {@link $globalsbyfile} array using {@link $curfile}
     *
     * @param parserGlobal &$element the "global" element
     *
     * @return void
     */
    function addGlobal(&$element)
    {
        if (isset($this->globalsbyfile[$this->curfile])) {
            foreach ($this->globalsbyfile[$this->curfile] as $i => $global) {
                if ($global->getName() == $element->getName()) {
                    addWarning(PDERROR_ELEMENT_IGNORED, 'global variable',
                        $element->getName(), $this->curfile);
                    return;
                }
            }
        }
        $this->globalsbyfile[$this->curfile][]          = $element;
        $this->globalsbynamefile[$element->getName()][] = $this->curfile;
    }

    /**
     * sets up the {@link $definesbyfile} array using {@link $curfile}
     *
     * @param parserDefine &$element the "define" element
     *
     * @return void
     */
    function addDefine(&$element)
    {
        if (isset($this->definesbyfile[$this->curfile])) {
            foreach ($this->definesbyfile[$this->curfile] as $i => $define) {
                if ($define->getName() == $element->getName()) {
                    addWarning(PDERROR_ELEMENT_IGNORED, 'define',
                        $element->getName(), $this->curfile);
                    return;
                }
            }
        }
        $this->definesbyfile[$this->curfile][]          = $element;
        $this->definesbynamefile[$element->getName()][] = $this->curfile;
    }
    
    /**
     * Used to align an element with the package of its parent page 
     * prior to Conversion.
     *
     * @param parserElement &$element the element to align
     *
     * @return void
     */
    function replaceElement(&$element)
    {
        if ($element->type == 'define') {
            foreach ($this->definesbyfile[$element->getPath()] as $i => $el) {
                if ($el->getName() == $element->getName()) {
                    $this->definesbyfile[$element->getPath()][$i] = &$element;
                }
            }
        } elseif ($element->type == 'global') {
            foreach ($this->globalsbyfile[$element->getPath()] as $i => $el) {
                if ($el->getName() == $element->getName()) {
                    $this->globalsbyfile[$element->getPath()][$i] = &$element;
                }
            }
        } elseif ($element->type == 'include') {
            foreach ($this->includesbyfile[$element->getPath()] as $i => $el) {
                if ($el->getName() == $element->getName()) {
                    $this->includesbyfile[$element->getPath()][$i] = &$element;
                }
            }
        } elseif ($element->type == 'function') {
            foreach ($this->functionsbyfile[$element->getPath()] as $i => $el) {
                if ($el->getName() == $element->getName()) {
                    $this->functionsbyfile[$element->getPath()][$i] = &$element;
                }
            }
        }
    }

    /**
     * adds a package from a class to the current file
     *
     * @param string $file       full path to the file that contains the class
     * @param string $package    package name
     * @param string $subpackage subpackage name
     *
     * @return void
     */
    function addClassPackageToFile($file, $package, $subpackage)
    {
        if (!isset($this->revcpbf[$file][$package][$subpackage])) {
            $this->pageclasspackages[$file][$package][$subpackage] = 1;
        }
        $this->revcpbf[$file][$package][$subpackage] = 1;
    }
    
    /**
     * if there is one class package in a file, 
     * the parent path inherits the package if its package is default.
     * helps with -po to avoid dumb bugs
     *
     * @return void
     */
    function setupPagePackages()
    {
        if ($this->packagesetup) {
            return;
        }
        foreach ($this->pageclasspackages as $fullpath => $packages) {
            if (isset($this->pagepackages[$fullpath])) {
                if ($this->pagepackages[$fullpath][0] 
                    == $GLOBALS['phpDocumentor_DefaultPackageName']
                ) {
                    if (count($packages) == 1) {
                        list($package, $subpackage) = each($packages);
                        if (count($subpackage) == 1) {
                            list($subpackage,) = each($subpackage);
                        } else {
                            $subpackage = '';
                        }
                        $this->addPagePackage($fullpath, $package, $subpackage);
                    }
                }
            }
        }
        $this->packagesetup = true;
    }
    
    /**
     * extracts function, define, and global variable name conflicts within the 
     * same package and between different packages.  No two elements with the same
     * name are allowed in the same package, to keep automatic linking possible.
     *
     * @param mixed &$render the renderer object
     *
     * @return void
     * @access private
     * @todo functions, defines, and globals are coded,
     *       but pages section is empty... does it need to be coded?
     */
    function setupConflicts(&$render)
    {
        foreach ($this->functionsbynamefile as $function => $paths) {
            if (count($paths) - 1) {
                //conflict
                $package = array();
                foreach ($paths as $path) {
                    // create a list of conflicting functions in each package
                    $package[$this->pagepackages[$path][0]][] = $path;
                }
                foreach ($package as $pathpackages) {
                    // if at least 2 functions exist in the same package, 
                    // delete all but the first one and add warnings
                    if (count($pathpackages) - 1) {
                        for ($i=1; $i < count($pathpackages); $i++) {
                            addWarning(PDERROR_ELEMENT_IGNORED, 'function',
                                $function, $pathpackages[$i]);
                            foreach ($this->functionsbyfile[$pathpackages[$i]] 
                                as $j => $blah
                            ) {
                                if ($this->functionsbyfile[$pathpackages[$i]][$j]
                                    ->getName() == $function
                                ) {
                                    unset($this
                                        ->functionsbyfile[$pathpackages[$i]][$j]);
                                }
                            }
                            $oth = array_flip($paths);
                            unset($paths[$oth[$pathpackages[$i]]]);
                        }
                    }
                }
                $this->functionconflicts[$function] = $paths;
            }
        }

        foreach ($this->definesbynamefile as $define => $paths) {
            if (count($paths) - 1) { 
                //conflict
                $package = array();
                foreach ($paths as $path) {
                    // create a list of conflicting functions in each package
                    $package[$this->pagepackages[$path][0]][] = $path;
                }
                foreach ($package as $pathpackages) {
                    // if at least 2 functions exist in the same package, 
                    // delete all but the first one and add warnings
                    if (count($pathpackages) - 1) {
                        for ($i=1; $i < count($pathpackages); $i++) {
                            addWarning(PDERROR_ELEMENT_IGNORED, 'define',
                                $define, $pathpackages[$i]);
                            foreach ($this->definesbyfile[$pathpackages[$i]]
                                as $j => $blah
                            ) {
                                if ($this->definesbyfile[$pathpackages[$i]][$j]
                                    ->getName() == $define
                                ) {
                                    unset($this
                                        ->definesbyfile[$pathpackages[$i]][$j]);
                                }
                            }
                            $oth = array_flip($paths);
                            unset($paths[$oth[$pathpackages[$i]]]);
                        }
                    }
                }
                $this->defineconflicts[$define] = $paths;
            }
        }

        foreach ($this->globalsbynamefile as $global => $paths) {
            if (count($paths) - 1) { 
                //conflict
                $package = array();
                foreach ($paths as $path) {
                    // create a list of conflicting functions in each package
                    $package[$this->pagepackages[$path][0]][] = $path;
                }
                foreach ($package as $pathpackages) {
                    // if at least 2 functions exist in the same package, 
                    // delete all but the first one and add warnings
                    if (count($pathpackages) - 1) {
                        for ($i=1; $i < count($pathpackages); $i++) {
                            addWarning(PDERROR_ELEMENT_IGNORED, 'global variable',
                                $global, $pathpackages[$i]);
                            foreach ($this->globalsbyfile[$pathpackages[$i]] 
                                as $j => $blah
                            ) {
                                if ($this->globalsbyfile[$pathpackages[$i]][$j]
                                    ->getName() == $global
                                ) {
                                    unset($this
                                        ->globalsbyfile[$pathpackages[$i]][$j]);
                                }
                            }
                            $oth = array_flip($paths);
                            unset($paths[$oth[$pathpackages[$i]]]);
                        }
                    }
                }
                $this->globalconflicts[$global] = $paths;
            }
        }
        
        /*
         * @todo does this section still need to be coded???
         */
        foreach ($this->pages as $name => $pages) {
            if (count($pages) - 1) { 
                // possible conflict
            }
        }
    }
    
    /**
     * called by {@link parserFunction::getConflicts()} to get 
     * inter-package conflicts, should not be called directly
     *
     * @param string $name the function name to check
     *
     * @access private
     * @return array|bool Format: (package => {@link parserFunction} 
     *                    of conflicting function)
     *                    or FALSE if the function is not recorded as a conflict
     */
    function getFuncConflicts($name)
    {
        if (!isset($this->functionconflicts[$name])) {
            return false;
        }
        $a = array();
        foreach ($this->functionconflicts[$name] as $conflict) {
            foreach ($this->functionsbyfile[$conflict] as $i => $func) {
                if ($func->getName() == $name) {
                    $a[$this->functionsbyfile[$conflict][$i]->docblock->package] 
                        = $this->functionsbyfile[$conflict][$i];
                }
            }
        }
        return $a;
    }
    
    /**
     * called by {@link parserGlobal::getConflicts()} 
     * to get inter-package conflicts, should not be called directly
     *
     * @param string $name the global name to check
     *
     * @access private
     * @return array|bool Format: (package => {@link parserGlobal} 
     *                    of conflicting global variable)
     *                    or FALSE if the global is not recorded as a conflict
     */
    function getGlobalConflicts($name)
    {
        if (!isset($this->globalconflicts[$name])) {
            return false;
        }
        $a = array();
        foreach ($this->globalconflicts[$name] as $conflict) {
            foreach ($this->globalsbyfile[$conflict] as $i => $func) {
                if ($func->getName() == $name) {
                    $a[$this->globalsbyfile[$conflict][$i]->docblock->package] 
                        = $this->globalsbyfile[$conflict][$i];
                }
            }
        }
        return $a;
    }
    
    /**
     * called by {@link parserDefine::getConflicts()} 
     * to get inter-package conflicts, should not be called directly
     *
     * @param string $name the define name to check
     *
     * @access private
     * @return array|bool Format: (package => {@link parserDefine} 
     *                    of conflicting define)
     *                    or FALSE if the define is not recorded as a conflict
     */
    function getDefineConflicts($name)
    {
        if (!isset($this->defineconflicts[$name])) {
            return false;
        }
        $a = array();
        foreach ($this->defineconflicts[$name] as $conflict) {
            foreach ($this->definesbyfile[$conflict] as $i => $func) {
                if ($func->getName() == $name) {
                    $a[$this->definesbyfile[$conflict][$i]->docblock->package] 
                        = $this->definesbyfile[$conflict][$i];
                }
            }
        }
        return $a;
    }
    
    /**
     * Adjusts packages of all pages and removes name conflicts within a package
     *
     * Automatic linking requires that each linkable name have exactly one element 
     * associated with it.  In other words, there cannot be two functions named 
     * foo() in the same package.
     *
     * This also adheres to php rules with one exception:
     *
     * <code>
     * if ($test == 3) {
     *    define('whatever', 'this thing');
     * } else {
     *    define('whatever', 'this other thing');
     * }
     * </code>
     *
     * phpDocumentor is not aware of conditional control structures because it 
     * would slow things down considerably.  So, what phpDocumentor does is 
     * automatically ignore the second define and raise a warning.  The warning can
     * be eliminated with an @ignore tag on the second element like so:
     *
     * <code>
     * if ($test == 3) {
     *    define('whatever', 'this thing');
     * } else {
     *    /**
     *     * @ignore
     *     {@*}
     *    define('whatever', 'this other thing');
     * }
     * </code>
     *
     * If there are two files that contain the same procedural elements in the 
     * same package (for example, a common configuration file common.php), they 
     * will also be ignored as if they were in the same file.  The reasoning
     * behind this is simple.  A package is an indivisible set of files and 
     * classes that a user will include in their code.  Name conflicts must be 
     * avoided to allow successful execution.
     *
     * This function also plays the all-important role of calling 
     * {@link phpDocumentor_IntermediateParser::addElementToPage()} in order to add 
     * processed elements to their pages for Conversion.
     *
     * @param phpDocumentor_IntermediateParser &$render the parser
     *
     * @return void
     */
    function setupPages(&$render)
    {
        global $_phpDocumentor_setting;
        phpDocumentor_out("\nProcessing Procedural Page Element Name Conflicts\n\n");
        flush();
        $this->setupPagePackages();
        $this->setupConflicts($render);
        // phpDocumentor_out("\nProcessing Procedural Pages\n\n");
        foreach ($this->pathpages as $path => $name) {
            // phpDocumentor_out("Processing $path\n");
            $a = $this->pagepackages[$path];
            $b = &$this->pages[$name][$path];
            $render->addPage($b, $path);
            $render->addUses($b, $path);
            if (isset($this->includesbyfile[$path])) {
                foreach ($this->includesbyfile[$path] as $include) {
                    $include->docblock->package    = $a[0];
                    $include->docblock->subpackage = $a[1];
                    $render->addElementToPage($include, $path);
                }
            }
    
            if (isset($this->functionsbyfile[$path])) {
                foreach ($this->functionsbyfile[$path] as $function) {
                    $function->docblock->package    = $a[0];
                    $function->docblock->subpackage = $a[1];
                    $render->addElementToPage($function, $path);
                    $render->addUses($function, $path);
                }
            }
    
            if (isset($this->definesbyfile[$path])) {
                foreach ($this->definesbyfile[$path] as $define) {
                    $define->docblock->package    = $a[0];
                    $define->docblock->subpackage = $a[1];
                    $render->addElementToPage($define, $path);
                    $render->addUses($define, $path);
                }
            }
    
            if (isset($this->globalsbyfile[$path])) {
                foreach ($this->globalsbyfile[$path] as $global) {
                    $global->docblock->package    = $a[0];
                    $global->docblock->subpackage = $a[1];
                    $render->addElementToPage($global, $path);
                    $render->addUses($global, $path);
                }
            }
        }
    }
    
    /**
     * sets the parser base
     *
     * @param mixed $pbase the parser base
     *
     * @return void
     */
    function setParseBase($pbase)
    {
        $this->_parsedbase = $pbase;
    }
    
    /**
     * checks to see if the parsed file matches the given path
     *
     * @param string $path   the path to look for
     * @param string $infile the file to check
     *
     * @return parserPage|bool matched parserPage if found,
     *                         or FALSE if not found
     */
    function pathMatchesParsedFile($path, $infile)
    {
        $test = $this->getRealPath($path, $infile);
        if (is_string($test)) {
            if (isset($this->pathpages[$test])) {
                return $this->pages[$this->pathpages[$test]][$test];
            }
            if (PHPDOCUMENTOR_WINDOWS) {
                $test = str_replace('/', '\\', $test);
            }
            if (isset($this->pathpages[$test])) {
                $a = $this->pages[$this->pathpages[$test]][$test];
                if (is_array($a->packageOutput) 
                    && !in_array($a->package, $a->packageOutput)
                ) {
                    return false;
                }
                return $this->pages[$this->pathpages[$test]][$test];
            }
        } else {
            foreach ($test as $file) {
                if (isset($this->pathpages[$file])) {
                    return $this->pages[$this->pathpages[$file]][$file];
                }
                if (PHPDOCUMENTOR_WINDOWS) {
                    $file = str_replace('/', '\\', $file);
                }
                if (isset($this->pathpages[$file])) {
                    $a = $this->pages[$this->pathpages[$file]][$file];
                    if (is_array($a->packageOutput) 
                        && !in_array($a->package, $a->packageOutput)
                    ) {
                        return false;
                    }
                    return $this->pages[$this->pathpages[$file]][$file];
                }
            }
        }
        return false;
    }
    
    /**
     * Ensures the path to the file is an absolute path
     * 
     * @param string $path path to the file
     * @param string $file the file name
     *
     * @return array|string returns an array of possible file locations or
     *                      a string if there is an exact match
     */
    function getRealPath($path, $file)
    {
        $curdir = str_replace('\\', '/', dirname($file));
        $path   = str_replace('\\', '/', $path);
        if (strpos($path, ':') !== false) {
            // windows, and we have a drive letter
            return $path;
        } elseif (strpos($path, '/') === 0) {
            return $path;
        }
        // not an absolute path
        $path = explode('/', $path);
        if ($path[0] == '.') {
            $path[0] = dirname($file);
            return join($path, '/');
        } elseif ($path[0] == '..') {
            $dirfile = explode('/', dirname(str_replace('\\', '/', $file)));
            // remove the current directory
            array_pop($dirfile); 
            if (!count($dirfile)) {
                // we were at a top-level dir!
                return false;
            }
            // replace .. with parent dirname
            $path[0] = join($dirfile, '/');
            return join($path, '/');
        } else {
            $path = join($path, '/');
            return array($curdir . PATH_DELIMITER . $path,
                str_replace('\\', '/', PHPDOCUMENTOR_BASE)
                . PATH_DELIMITER . $path);
        }
    }
}
?>