Skip to content
jackywei edited this page Jul 11, 2012 · 52 revisions

Note: These code examples assume the HipHop compiler is fully built.

Setting Up Your Environment

To get started, you need to configure two environment variables.

cd .. # into the root of the hphp checkout
export HPHP_HOME=`pwd`
export HPHP_LIB=`pwd`/bin
# if you followed the Ubuntu 9.10 instructions, you also need
export CMAKE_PREFIX_PATH=`/bin/pwd`/../

Choosing which Mode to Run HipHop

You can run HipHop in 5 different modes. These Hello World examples demonstrate each one. All commands are run from the src/ directory in these examples.

First, create a file called test.php. Populate it with some text, like, “echo Hello, World! > test.php”. Then choose from the following modes:

Mode 1: Compiling HipHop and running it directly.

$HPHP_HOME/src/hphp/hphp test.php

Mode 2: Compiling HipHop in a temporary directory and running the compiled program from the command line.

$HPHP_HOME/src/hphp/hphp test.php --keep-tempdir=1 --log=3
/tmp/hphp_p6vSsP/program    (use your own temporary directory name from output)

--keep-tempdir=1 can also be specified with -k 1. Note it’s single dash and there is a space, not “=” between “k” and “1”. This is something to watch out when working with boost command line options.

--log=3 outputs some verbose information, so you can find out which temporary directory it created. You may always specify your own output directory with --output-dir=mypath or -o mypath.

Mode 3: Compiling HipHop in a temporary directory and running the compiled program as a web server.

$HPHP_HOME/src/hphp/hphp test.php --keep-tempdir=1 --log=3
sudo /tmp/hphp_p6vSsP/program -m server

Then, from another window, run:

curl localhost/test.php

If you don’t want to use sudo, you can run HipHop on port 8080.

$HPHP_HOME/src/hphp/hphp test.php --keep-tempdir=1 --log=3
/tmp/hphp_p6vSsP/program -m server -p 8080
curl http://localhost:8080/test.php

Run this command to administer your server:

curl http://localhost:8080

You can also run the server as a daemon:

sudo /tmp/hphp_p6vSsP/program -m daemon

Mode 4: Interpreting HipHop directly.

$HPHP_HOME/src/hphpi/hphpi -f test.php  (note the "-f" flag)

Mode 5: Starting a Web server or daemon and interpreting HipHop on the fly.

sudo $HPHP_HOME/src/hphpi/hphpi -m server (or daemon)
curl localhost/test.php
curl localhost:8088

Compiling a Large Codebase

First, familiarize yourself with the various of switches of the compiler:

$HPHP_HOME/src/hphp/hphp --help

There are 3 ways to specify some flags. (1) by a configuration file in HDF format. Please read doc/hdf for more details with the format. Then use --config to specify the config file. (2) For almost every option in HDF file, you can list it directly in its dot notation format. For example, -v "node.subnode=value". (3) We created some shortcuts for most frequently used ones. They will look like --force.

The most important flags to learn are the ones for including or excluding files and directories. They were not designed cleanly and we may have to improve the way how they work. When in doubt, simply use the --input-list switch to take a list of file names prepared in a separate file.

You can get all the possible flags here: Runtime options

Using Parse-on-demand Mode (optional)

You can include files that are not specified from the command line into the compilation only if the compiler can determine where to find them. This means your include statements themselves are either:

  • Formed by simple literals; so the compiler can compute them during compilation time.
  • Written in simple form like "include_once $MY_ROOT.'/path/file.php';"
    Note: You can tell the compiler where to look for $MY_ROOT by creating a configuration file with content like this:
IncludeRoots {
  * {
    root = $MY_ROOT
    path = lib/my_code
  }
  * {
    root = $ANOTHER_ROOT
    path = anotherlib
  }
}

Use --config to include this configuration file. The compiler resolves the above include statement as “lib/my_code/path/file.php”.

Note: If you find parse-on-demand mode difficult to configure, try using --input-list to include every PHP file you want to compile.

Using distcc

For large compilations, we recommend setting up distcc.

Example: Compiling PHPUnit

1. Check out PHPUnit’s PHP files:

git clone git://github.com/sebastianbergmann/phpunit.git
cd phpunit
git checkout -b 3.4 origin/3.4

2. We will use the safest and the cleanest way to specify input files,

find . -name "*.php" > files.list

This prepares a list of all PHP files we want to compile.

3. Now we’re ready to compile the project.

$HPHP_HOME/src/hphp/hphp --input-list=files.list -k 1 --log=3 \
  --include-path="." --force=1 --cluster-count=50 \
  -v "AllDynamic=true" -v "AllVolatile=true" 

-k 1 or --keep-tempdir=1 so it creates a new temporary directory every time. This is convenient when you’re experimenting the compilation.

The --include-path is needed, because PHPUnit has file includes relative to root directory of phpunit. Without specifying this option, all includes in a format of “include ‘somepath/file.php’;” will be treated as relative path to the containing file.

--force=1 is needed to ignore warnings and errors HipHop found in the code. Without this option, the compiler will halt and dump out the errors on the screen, if any. With --force=1, those errors will mostly turn into run-time ones, and you may still find them in CodeError.js generated under the output directory.

--cluster-count=50 helps compilation, with or without distcc. Without this flag, each PHP file generates one .cpp file. When the number of PHP files is large, we may end up with too many .cpp files to compile. With clustering, no matter how many PHP files we have, HipHop will generate roughly the specified number of .cpp files, so it’s easier to feed them into distcc with fewer rounds. What we found is, cluster count should be slightly smaller than number of distcc workers. For example, if you have 20 machines each with 8 distcc workers, cluster count of 100 may be suitable. But one should change the numbers up and down to compare compilation time to find out the optimal value.

-v "AllDynamic=true" With this option, we can support dynamic function calls and dynamic method calls without any problems. Recommended to turn on, if coding has them. It will sacrifice performance a little bit, but it’s safe to have it.

-v "AllVolatile=true" With this option, we can support dynamic declarations of functions and classes without any problems. This is not recommended to turn on, unless your coding has crazy testing of function_exists() or class_exists() before or after declarations and the order is meaningful. PHPUnit happens to call get_declared_classes() before and after loading some class files and compare their returns to find new classes. Therefore, we need to add this switch to PHPUnit. Most likely, you don’t have to. It sacrifices performance in various degrees.

4. Now you should have a compiled PHPUnit binary. Report any problems to us, if you cannot reach this far. To run the binary,

php phpunit.php (in PHP)
/tmp/hphp_po33pK/program -f phpunit.php (in HipHop, note the -f flag)
php phpunit.php PHPUnit/Tests/Framework/SuiteTest.php
/tmp/hphp_po33pK/program -v "Server.SourceRoot=`pwd`" \
   -f phpunit.php PHPUnit/Tests/Framework/SuiteTest.php

Note that the compiled binary “program” has to run from the same directory you normally run phpunit.php, only because PHPUnit has file_exists() testing that goes to local disk to look for some .php files. There is a way to remove this disk location dependency by building a static file cache, but we will leave that to some more advanced instructions.

Also note that -v "Server.SourceRoot=`pwd`" normally is not needed. But PHPUnit has quite a few file location based operations that will try to compare source file name with what’s on local disk with realpath() calls. So we had to add this one to run through the tests.

5. Some useful tips:

(1) If you just created a binary with --keep-tempdir=1, but forgot to copy the name, a simple command can normally find it,

ls -altrd /tmp/hphp_* | tail -1

(2) You may run out of disk space with too many temporary directories. Just rm all HipHop temps like this,

rm -fR /tmp/hphp_*

Example: Running PHPUnit under HPHPi

$HPHP_HOME/src/hphpi/hphpi -f phpunit.php
$HPHP_HOME/src/hphpi/hphpi -f phpunit.php PHPUnit/Tests/Framework/SuiteTest.php

Haiping: We can pass all in SuiteTest.php, but we do have several other tests under PHPUnit/Tests/Framework that we’re not able to fully pass yet, due to some local disk assumption in PHPUnit and perhaps some minor bugs. Still debugging to see if we can fix all these issues.

Example: Compiling WordPress

1. Get a copy of WordPress. Please note, we identified 2 or 3 problems with WordPress that need to be fixed before HipHop can compile it. These have been fixed in trunk of the Wordpress SVN but not backported.

wget http://wordpress.org/wordpress-2.9.1.tar.gz
tar zxvf wordpress-2.9.1.tar.gz
cd wordpress
[patch to fix some PHP coding problems that will cause compilation errors]

2. Create a config.php, perhaps by copying config.sample.php and set up database information. This file needs to be prepared BEFORE the compilation, so it’s compiled into the final binary. Any changes of this file need a re-compilation of the whole package. NOTE: use the loopback interface (typically ‘127.0.0.1’) instead of ‘localhost’; see this thread on the mailing list for an explanation.

3. This prepares a list of all PHP files we want to compile:

find . -name "*.php" > files.list

4. Now we’re ready to compile the project.

$HPHP_HOME/src/hphp/hphp --input-list=files.list -k 1 --log=3 \
  --force=1 --cluster-count=50

This is simpler than PHPUnit, because WordPress doesn’t have as much dynamic coding as PHPUnit does.

5. Now you should have a compiled binary. To run it,

sudo /tmp/hphp_xpl7hT/program -m server -v "Server.SourceRoot=`pwd`" \
  -v "Server.DefaultDocument=index.php" -c $HPHP_HOME/bin/mime.hdf

sudo because we need to listen to port 80, the only port WordPress works on.

-m server runs the program in server mode. -m daemon is okay as well.

-v "Server.SourceRoot=`pwd`" We still need this to locate image and css files.

-v "Server.DefaultDocument=index.php", so http://server/ would work.

-c $HPHP_HOME/bin/mime.hdf has a list of static content file extensions that need to be loaded by the server to be able to serve those files with different MIME headers.

If you want to see verbose logging, add these flags,

-v "Log.Level=Verbose" This will output a lot more errors, warnings and information.
-v "Log.NoSilencer=on" This prints out errors from statements that have “@” operators, which WordPress code uses a lot.
-v "Log.Header=on" This will print a header for each line of logging. The most interesting in the header is a long string with hex-encoding. That’s hex-encoded stacktrace. To translate it into something readable, run this command,

/tmp/hphp_xpl7hT/program -m translate the-long-hex-string-without-brackets