diff --git a/.travis.yml b/.travis.yml
index cb81ea8340b0d463b4180f637aadd4f14cc57968..e55bf3ffd04abe0e7354501ea463e0ada25f2d84 100644
--- a/.travis.yml
+++ b/.travis.yml
@@ -38,12 +38,31 @@ matrix:
       language: perl
       perl: "5.10"
       env: BINDINGS=perl CC=clang
+    - compiler: gcc
+      language: php
+      php: "5.5"
+      env: BINDINGS=php
+    - compiler: clang
+      language: php
+      php: "5.5"
+      env: BINDINGS=php CC=clang
+    - compiler: gcc
+      language: php
+      php: "5.4"
+      env: BINDINGS=php
+    - compiler: clang
+      language: php
+      php: "5.4"
+      env: BINDINGS=php CC=clang
 before_install:
   - sudo apt-get update -qq
-  - if [ "$BINDINGS" != "none" ]; then sudo add-apt-repository ppa:dns/irc -y; sudo apt-get update -qq; sudo apt-get install -qq swig=2.0.8-1irc1~12.04; swig -version; fi
+  - if [ "$BINDINGS" != "none" ]; then sudo apt-get install -qq swig; fi
+  - if [ "$BINDINGS" == "perl" ]; then sudo add-apt-repository ppa:dns/irc -y; sudo apt-get update -qq; sudo apt-get install -qq swig=2.0.8-1irc1~12.04; fi
   - if [ "$BINDINGS" == "python" ]; then sudo apt-get install -qq python-dev; fi
 
 install: true
+before_script:
+  - if [ "$BINDINGS" == "php" ]; then phpenv config-add src/bindings/php/hammer.ini; fi
 script: 
   - scons bindings=$BINDINGS test
 notifications:
diff --git a/SConstruct b/SConstruct
index 872a9747f54086ceb5c4def9b92378facae16f38..a9f9c670cab5505c3d3043e0e7937c94c88f065c 100644
--- a/SConstruct
+++ b/SConstruct
@@ -7,7 +7,7 @@ import sys
 vars = Variables(None, ARGUMENTS)
 vars.Add(PathVariable('DESTDIR', "Root directory to install in (useful for packaging scripts)", None, PathVariable.PathIsDirCreate))
 vars.Add(PathVariable('prefix', "Where to install in the FHS", "/usr/local", PathVariable.PathAccept))
-vars.Add(ListVariable('bindings', 'Language bindings to build', 'none', ['python', 'perl']))
+vars.Add(ListVariable('bindings', 'Language bindings to build', 'none', ['python', 'perl', 'php']))
 
 env = Environment(ENV = {'PATH' : os.environ['PATH']}, variables = vars, tools=['default', 'scanreplace'], toolpath=['tools'])
 
@@ -69,7 +69,7 @@ dbg = env.Clone(VARIANT='debug')
 dbg.Append(CCFLAGS=['-g'])
 
 opt = env.Clone(VARIANT='opt')
-opt.Append(CCFLAGS="-O3")
+opt.Append(CCFLAGS=["-O3"])
 
 if GetOption("variant") == 'debug':
     env = dbg
@@ -95,6 +95,7 @@ env["ENV"].update(x for x in os.environ.items() if x[0].startswith("CCC_"))
 #env.Append(CPPPATH=os.path.join('#', "hammer"))
 
 testruns = []
+
 targets = ["$libpath",
            "$incpath",
            "$parsersincpath",
diff --git a/examples/SConscript b/examples/SConscript
index 94f32ac9ef2fe491c444089f39f2bc676fa1ffb6..0932bdacbbf51f4f2faaa73484313abd0eab9ad0 100644
--- a/examples/SConscript
+++ b/examples/SConscript
@@ -3,7 +3,8 @@ Import('env')
 example = env.Clone()
 example.Append(LIBS="hammer", LIBPATH="../src")
 
-example.Program('dns', ['dns.c', 'rr.c', 'dns_common.c'])
-example.Program('base64', 'base64.c')
-example.Program('base64_sem1', 'base64_sem1.c')
-example.Program('base64_sem2', 'base64_sem2.c')
+dns = example.Program('dns', ['dns.c', 'rr.c', 'dns_common.c'])
+base64 = example.Program('base64', 'base64.c')
+base64_sem1 = example.Program('base64_sem1', 'base64_sem1.c')
+base64_sem2 = example.Program('base64_sem2', 'base64_sem2.c')
+env.Alias("examples", [dns, base64, base64_sem1, base64_sem2])
\ No newline at end of file
diff --git a/src/bindings/php/README.md b/src/bindings/php/README.md
new file mode 100644
index 0000000000000000000000000000000000000000..5638c517a033054672bf8a185bf6e73504f67736
--- /dev/null
+++ b/src/bindings/php/README.md
@@ -0,0 +1,18 @@
+Building
+========
+Requirements:
+* SWIG 2.0
+* A properly configured [phpenv](https://github.com/CHH/phpenv)
+
+SCons finds your PHP include path from `php-config`, so if you don't have that working, you're going to have a bad time.
+
+If you want to run the tests, you will also need to install PHPUnit. Do this with pyrus and save yourself some hell. 
+
+    pyrus channel-discover pear.phpunit.de
+    pyrus channel-discover pear.symfony.com
+    pyrus channel-discover pear.symfony-project.com
+    pyrus install --optionaldeps phpunit/PHPUnit
+
+Installing
+==========
+We're not building a proper package yet, but you can copy `build/$VARIANT/src/bindings/php/hammer.so` to your PHP extension directory (`scons test` will do this for you if you're using phpenv; for a system-wide php you'll probably have to use sudo) and add "extension=hammer.so" to your php.ini. There is a "hammer.ini" in src/bindings/php for your convenience; you can put it in the `conf.d` directory where PHP expects to find its configuration. `scons test` will do this for you too. You'll also need to point your include_path to the location of hammer.php, which will be `build/$VARIANT/src/bindings/php/hammer.php` until you put it somewhere else.
\ No newline at end of file
diff --git a/src/bindings/php/SConscript b/src/bindings/php/SConscript
new file mode 100644
index 0000000000000000000000000000000000000000..34728af238c9a1b3ad478737e997921e8a0ff0b8
--- /dev/null
+++ b/src/bindings/php/SConscript
@@ -0,0 +1,31 @@
+# -*- python -*-
+import os, os.path
+Import('env libhammer_shared testruns')
+
+phpenv = env.Clone(IMPLICIT_COMMAND_DEPENDENCIES = 0)
+
+php_include = os.popen("php-config --include-dir").read().rstrip()
+phpenv.Append(CPPPATH = ['../../', php_include, os.path.join(php_include, 'main'), os.path.join(php_include, 'Zend'), os.path.join(php_include, 'TSRM')])
+phpenv.Append(CCFLAGS = ['-fpic', '-DSWIG', '-Wno-all', '-Wno-extra', '-Wno-error'])
+phpenv.Append(LIBS = ['hammer'])
+phpenv.Append(LIBPATH = ['../../']) 
+
+swig = ['hammer.i']
+bindings_src = phpenv.Command(['hammer.php', 'hammer_wrap.c', 'php_hammer.h'], swig, 'swig -php -DHAMMER_INTERNAL__NO_STDARG_H -Isrc/ $SOURCE')
+libhammer_php = phpenv.SharedLibrary('hammer', ['hammer_wrap.c'])
+Default(swig, bindings_src, libhammer_php)
+
+phptestenv = phpenv.Clone()
+phptestenv['ENV']['LD_LIBRARY_PATH'] = os.path.dirname(str(libhammer_shared[0]))
+phptests = ('Tests')
+phpextprefix = os.popen("php-config --extension-dir").read().rstrip()
+phplib = phptestenv.Command(os.path.join(phpextprefix, "hammer.so"), libhammer_php, Copy("$TARGET", "$SOURCE"))
+AlwaysBuild(phplib)
+phpprefix = os.popen("php-config --prefix").read().rstrip()
+phpincl = phptestenv.Command(os.path.join(os.path.join(phpprefix, "etc/conf.d"), "hammer.ini"), "hammer.ini", Copy("$TARGET", "$SOURCE"))
+phptestexec = phptestenv.Command(phptests, [phplib, phpincl], "phpenv exec phpunit --include-path " + os.path.dirname(libhammer_php[0].path) +" src/bindings/php/Tests")
+phptest = Alias("testphp", [phptestexec], phptestexec)
+AlwaysBuild(phptest)
+testruns.append(phptest)
+
+
diff --git a/src/bindings/php/Tests/ActionTest.php b/src/bindings/php/Tests/ActionTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..f4e6a5632cf463c4314e60719e6056e45eb380b9
--- /dev/null
+++ b/src/bindings/php/Tests/ActionTest.php
@@ -0,0 +1,40 @@
+<?php
+include_once 'hammer.php';
+
+function actTest($token) 
+{
+    if (is_array($token) === true) {
+        foreach($token as $chr) {
+            $ret[] = strtoupper($chr);
+        }
+        return $ret;
+    } else {
+        return strtoupper($token);
+    }
+}
+
+class ActionTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_action(hammer_sequence(hammer_choice(hammer_ch("a"), hammer_ch("A")), hammer_choice(hammer_ch("b"), hammer_ch("B"))), "actTest");
+    }
+    public function testSuccess()
+    {
+        $result1 = hammer_parse($this->parser, "ab");
+        $result2 = hammer_parse($this->parser, "AB");
+        $result3 = hammer_parse($this->parser, "aB");
+        $this->assertEquals(["A", "B"], $result1);
+        $this->assertEquals(["A", "B"], $result2);
+        $this->assertEquals(["A", "B"], $result3);
+    }
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "XX");
+        $this->assertEquals(NULL, $result);
+    }
+}
+
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/AndTest.php b/src/bindings/php/Tests/AndTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..91cb1156c2392ddf8aa14a3f1c2ef2ec4cff96c2
--- /dev/null
+++ b/src/bindings/php/Tests/AndTest.php
@@ -0,0 +1,35 @@
+<?php
+include_once 'hammer.php';
+
+class AndTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser1;
+    protected $parser2;
+    protected $parser3;
+
+    protected function setUp()
+    {
+        $this->parser1 = hammer_sequence(hammer_and(hammer_ch("0")), hammer_ch("0"));
+        $this->parser2 = hammer_sequence(hammer_and(hammer_ch("0")), hammer_ch("1"));
+        $this->parser3 = hammer_sequence(hammer_ch("1"), hammer_and(hammer_ch("2")));
+    }
+
+    public function testSuccess1()
+    {
+        $result = hammer_parse($this->parser1, "0");
+        $this->assertEquals(array("0"), $result);
+    }
+
+    public function testFailure2()
+    {
+        $result = hammer_parse($this->parser2, "0");
+        $this->assertEquals(NULL, $result);
+    }
+
+    public function testSuccess3()
+    {
+        $result = hammer_parse($this->parser3, "12");
+        $this->assertEquals(array("1"), $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/ButNotTest.php b/src/bindings/php/Tests/ButNotTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..f1af3dd74aade6ac44eb78a1bbbc6a8f86696d2b
--- /dev/null
+++ b/src/bindings/php/Tests/ButNotTest.php
@@ -0,0 +1,35 @@
+<?php
+include_once 'hammer.php';
+
+class ButNotTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser1;
+    protected $parser2;
+
+    protected function setUp()
+    {
+        $this->parser1 = hammer_butnot(hammer_ch("a"), hammer_token("ab"));
+        $this->parser2 = hammer_butnot(hammer_ch_range('0', '9'), hammer_ch('6'));
+    }
+
+    public function testSuccess1()
+    {
+        $result1 = hammer_parse($this->parser1, "a");
+        $result2 = hammer_parse($this->parser1, "aa");
+        $this->assertEquals("a", $result1);
+        $this->assertEquals("a", $result1);
+    }
+
+    public function testFailure1()
+    {
+        $result = hammer_parse($this->parser1, "ab");
+        $this->assertEquals(NULL, $result);
+    }
+
+    public function testFailure2()
+    {
+        $result = hammer_parse($this->parser2, "6");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/ChRangeTest.php b/src/bindings/php/Tests/ChRangeTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..52f3732ed5a0c70cff073f1f7900346701f9e302
--- /dev/null
+++ b/src/bindings/php/Tests/ChRangeTest.php
@@ -0,0 +1,24 @@
+<?php
+
+include_once 'hammer.php';
+
+class ChRangeTest extends PHPUnit_Framework_TestCase 
+{
+    protected $parser;
+
+    protected function setUp() 
+    {
+        $this->parser = hammer_ch_range("a", "c");
+    }
+    public function testSuccess() 
+    {
+        $result = hammer_parse($this->parser, "b");
+        $this->assertEquals("b", $result);
+    }     
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "d");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/ChTest.php b/src/bindings/php/Tests/ChTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..a38c655695c7523f28b4518d79e8d8e7ac0c6bdf
--- /dev/null
+++ b/src/bindings/php/Tests/ChTest.php
@@ -0,0 +1,24 @@
+<?php
+
+include_once 'hammer.php';
+
+class ChTest extends PHPUnit_Framework_TestCase 
+{
+    protected $parser;
+
+    protected function setUp() 
+    {
+        $this->parser = hammer_ch("\xa2");
+    }
+    public function testSuccess() 
+    {
+        $result = hammer_parse($this->parser, "\xa2");
+        $this->assertEquals("\xa2", $result);
+    }     
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "95");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/ChoiceTest.php b/src/bindings/php/Tests/ChoiceTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..cd9e4138b1475e075e82adcce8570d8077fa09ae
--- /dev/null
+++ b/src/bindings/php/Tests/ChoiceTest.php
@@ -0,0 +1,27 @@
+<?php
+include_once 'hammer.php';
+
+class ChoiceTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_choice(hammer_ch("a"), hammer_ch("b"));
+    }
+
+    public function testSuccess()
+    {
+        $result1 = hammer_parse($this->parser, "a");
+        $result2 = hammer_parse($this->parser, "b");
+        $this->assertEquals("a", $result1);
+        $this->assertEquals("b", $result2);
+    }
+
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "c");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/DifferenceTest.php b/src/bindings/php/Tests/DifferenceTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..297aa42a6cee8fb3b7662f1c0cf8056383ad0f58
--- /dev/null
+++ b/src/bindings/php/Tests/DifferenceTest.php
@@ -0,0 +1,25 @@
+<?php
+include_once 'hammer.php';
+
+class DifferenceTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_difference(hammer_token("ab"), hammer_ch("a"));
+    }
+
+    public function testSuccess()
+    {
+        $result = hammer_parse($this->parser, "ab");
+        $this->assertEquals("ab", $result);
+    }
+
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "a");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/EndTest.php b/src/bindings/php/Tests/EndTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..d6af2af89481f43ee9161353d1c3eb388a2331ab
--- /dev/null
+++ b/src/bindings/php/Tests/EndTest.php
@@ -0,0 +1,24 @@
+<?php
+include_once 'hammer.php';
+
+class EndPTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_sequence(hammer_ch("a"), hammer_end());
+    }
+
+    public function testSuccess()
+    {
+        $result = hammer_parse($this->parser, "a");
+        $this->assertEquals(array("a"), $result);
+    }
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "aa");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/EpsilonPTest.php b/src/bindings/php/Tests/EpsilonPTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..41100ccc0018d2a1f59c190818e906e3c12ebf45
--- /dev/null
+++ b/src/bindings/php/Tests/EpsilonPTest.php
@@ -0,0 +1,35 @@
+<?php
+include_once 'hammer.php';
+
+class EpsilonPTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser1;
+    protected $parser2;
+    protected $parser3;
+
+    protected function setUp()
+    {
+        $this->parser1 = hammer_sequence(hammer_ch("a"), hammer_epsilon(), hammer_ch("b"));
+        $this->parser2 = hammer_sequence(hammer_epsilon(), hammer_ch("a"));
+        $this->parser3 = hammer_sequence(hammer_ch("a"), hammer_epsilon());
+    }
+
+    public function testSuccess1()
+    {
+        $result = hammer_parse($this->parser1, "ab");
+        $this->assertEquals(array("a", "b"), $result);
+    }
+
+    public function testSuccess2()
+    {
+        $result = hammer_parse($this->parser2, "a");
+        $this->assertEquals(array("a"), $result);
+    }
+
+    public function testSuccess3()
+    {
+        $result = hammer_parse($this->parser3, "ab");
+        $this->assertEquals(array("a"), $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/IgnoreTest.php b/src/bindings/php/Tests/IgnoreTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..805ae08c0e88099d88a3d944c0b620b302b34e50
--- /dev/null
+++ b/src/bindings/php/Tests/IgnoreTest.php
@@ -0,0 +1,25 @@
+<?php
+include_once 'hammer.php';
+
+class IgnoreTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_sequence(hammer_ch("a"), hammer_ignore(hammer_ch("b")), hammer_ch("c"));
+    }
+
+    public function testSuccess()
+    {
+        $result = hammer_parse($this->parser, "abc");
+        $this->assertEquals(array("a", "c"), $result);
+    }
+
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "ac");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/InTest.php b/src/bindings/php/Tests/InTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..e209f1fc2b8e517bddd19c45a8df6b98e2c01d7c
--- /dev/null
+++ b/src/bindings/php/Tests/InTest.php
@@ -0,0 +1,23 @@
+<?php
+include_once 'hammer.php';
+
+class InTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_in("abc");
+    }
+    public function testSuccess()
+    {
+        $result = hammer_parse($this->parser, "b");
+        $this->assertEquals("b", $result);
+    }
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "d");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/Int16Test.php b/src/bindings/php/Tests/Int16Test.php
new file mode 100644
index 0000000000000000000000000000000000000000..f9f10cb0e6d47bae41eb723f24dae421c688d3f0
--- /dev/null
+++ b/src/bindings/php/Tests/Int16Test.php
@@ -0,0 +1,30 @@
+<?php
+
+include_once 'hammer.php';
+
+class Int16Test extends PHPUnit_Framework_TestCase 
+{
+    protected $parser;
+
+    protected function setUp() 
+    {
+        $this->parser = hammer_int16();
+    }
+    public function testNegative() 
+    {
+        $result = hammer_parse($this->parser, "\xfe\x00");
+        $this->assertEquals(-0x200, $result);
+    }     
+    public function testPositive() {
+        $result = hammer_parse($this->parser, "\x02\x00");
+        $this->assertEquals(0x200, $result);
+    }
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "\xfe");
+        $this->assertEquals(NULL, $result);
+        $result = hammer_parse($this->parser, "\x02");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/Int32Test.php b/src/bindings/php/Tests/Int32Test.php
new file mode 100644
index 0000000000000000000000000000000000000000..5798195abc8ccee4181c6e4b7483015bc7decb28
--- /dev/null
+++ b/src/bindings/php/Tests/Int32Test.php
@@ -0,0 +1,31 @@
+<?php
+
+include_once 'hammer.php';
+
+class Int32Test extends PHPUnit_Framework_TestCase 
+{
+    protected $parser;
+
+    protected function setUp() 
+    {
+        $this->parser = hammer_int32();
+    }
+    public function testNegative() 
+    {
+        $result = hammer_parse($this->parser, "\xff\xfe\x00\x00");
+        $this->assertEquals(-0x20000, $result);
+    }     
+    public function testPositive() 
+    {
+        $result = hammer_parse($this->parser, "\x00\x02\x00\x00");
+        $this->assertEquals(0x20000, $result);
+    }
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "\xff\xfe\x00");
+        $this->assertEquals(NULL, $result);
+        $result = hammer_parse($this->parser, "\x00\x02\x00");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/Int64Test.php b/src/bindings/php/Tests/Int64Test.php
new file mode 100644
index 0000000000000000000000000000000000000000..631a39f621e1ef3b2ebb8167b1b2021778c8debb
--- /dev/null
+++ b/src/bindings/php/Tests/Int64Test.php
@@ -0,0 +1,24 @@
+<?php
+
+include_once 'hammer.php';
+
+class Int64Test extends PHPUnit_Framework_TestCase 
+{
+    protected $parser;
+
+    protected function setUp() 
+    {
+        $this->parser = hammer_int64();
+    }
+    public function testSuccess() 
+    {
+        $result = hammer_parse($this->parser, "\xff\xff\xff\xfe\x00\x00\x00\x00");
+        $this->assertEquals(-0x200000000, $result);
+    }     
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "\xff\xff\xff\xfe\x00\x00\x00");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/Int8Test.php b/src/bindings/php/Tests/Int8Test.php
new file mode 100644
index 0000000000000000000000000000000000000000..b0769ecd4521c5fb055d252cd881591e01f727b7
--- /dev/null
+++ b/src/bindings/php/Tests/Int8Test.php
@@ -0,0 +1,24 @@
+<?php
+
+include_once 'hammer.php';
+
+class Int8Test extends PHPUnit_Framework_TestCase 
+{
+    protected $parser;
+
+    protected function setUp() 
+    {
+        $this->parser = hammer_int8();
+    }
+    public function testSuccess() 
+    {
+        $result = hammer_parse($this->parser, "\x88");
+        $this->assertEquals(-0x78, $result);
+    }     
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/IntRangeTest.php b/src/bindings/php/Tests/IntRangeTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..4d300e3ac39c280e8e1ce80d366b3f394914c39d
--- /dev/null
+++ b/src/bindings/php/Tests/IntRangeTest.php
@@ -0,0 +1,23 @@
+<?php
+include_once 'hammer.php';
+
+class IntRangeTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_int_range(hammer_uint8(), 3, 10);
+    }
+    public function testSuccess()
+    {
+        $result = hammer_parse($this->parser, "\x05");
+        $this->assertEquals(5, $result);
+    }
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "\xb");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/LeftTest.php b/src/bindings/php/Tests/LeftTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..e0ef01e4c51402dff6bdd97c187c17b9d25f31b2
--- /dev/null
+++ b/src/bindings/php/Tests/LeftTest.php
@@ -0,0 +1,27 @@
+<?php
+include_once 'hammer.php';
+
+class LeftTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_left(hammer_ch("a"), hammer_ch(" "));
+    }
+    public function testSuccess()
+    {
+        $result = hammer_parse($this->parser, "a ");
+        $this->assertEquals("a", $result);
+    }
+    public function testFailure()
+    {
+        $result1 = hammer_parse($this->parser, "a");
+        $result2 = hammer_parse($this->parser, " ");
+        $result3 = hammer_parse($this->parser, "ab");
+        $this->assertEquals(NULL, $result1);
+        $this->assertEquals(NULL, $result2);
+        $this->assertEquals(NULL, $result3);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/LeftrecTest.php b/src/bindings/php/Tests/LeftrecTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..e578a31513b0a499e6b6ab1f9704d0ee65ba9053
--- /dev/null
+++ b/src/bindings/php/Tests/LeftrecTest.php
@@ -0,0 +1,36 @@
+<?php
+include_once 'hammer.php';
+
+class LeftrecTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_indirect();
+        hammer_bind_indirect($this->parser, hammer_choice(hammer_sequence($this->parser, hammer_ch("a")), hammer_ch("a")));
+    }
+
+    public function testSuccess1()
+    {
+        $result = hammer_parse($this->parser, "a");
+        $this->assertEquals("a", $result);
+    }
+    /* These don't work in the underlying C so they won't work here either */
+/*
+    public function testSuccess2()
+    {
+        $result = hammer_parse($this->parser, "aa");
+        var_dump($result);
+        $this->assertEquals(array("a", "a"), $result);
+    }
+
+    public function testSuccess3()
+    {
+        $result = hammer_parse($this->parser, "aaa");
+        var_dump($result);
+        $this->assertEquals(array(array("a", "a"), "a"), $result);
+    }
+*/
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/Many1Test.php b/src/bindings/php/Tests/Many1Test.php
new file mode 100644
index 0000000000000000000000000000000000000000..50a6c416e96ef4af64a1ee0b66b42a28667ea80f
--- /dev/null
+++ b/src/bindings/php/Tests/Many1Test.php
@@ -0,0 +1,31 @@
+<?php
+include_once 'hammer.php';
+
+class Many1Test extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_many1(hammer_choice(hammer_ch("a"), hammer_ch("b")));
+    }
+
+    public function testSuccess()
+    {
+        $result1 = hammer_parse($this->parser, "a");
+        $result2 = hammer_parse($this->parser, "b");
+        $result3 = hammer_parse($this->parser, "aabbaba");
+        $this->assertEquals(array("a"), $result1);
+        $this->assertEquals(array("b"), $result2);
+        $this->assertEquals(array("a", "a", "b", "b", "a", "b", "a"), $result3);
+    }
+
+    public function testFailure()
+    {
+        $result1 = hammer_parse($this->parser, "");
+        $result2 = hammer_parse($this->parser, "daabbabadef");
+        $this->assertEquals(NULL, $result1);
+        $this->assertEquals(NULL, $result2);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/ManyTest.php b/src/bindings/php/Tests/ManyTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..b2abee297517250910e0759c39afca9b4c10a7f2
--- /dev/null
+++ b/src/bindings/php/Tests/ManyTest.php
@@ -0,0 +1,25 @@
+<?php
+include_once 'hammer.php';
+
+class ManyTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_many(hammer_choice(hammer_ch("a"), hammer_ch("b")));
+    }
+
+    public function testSuccess()
+    {
+        $result1 = hammer_parse($this->parser, "");
+        $result2 = hammer_parse($this->parser, "a");
+        $result3 = hammer_parse($this->parser, "b");
+        $result4 = hammer_parse($this->parser, "aabbaba");
+        $this->assertEquals(array(), $result1);
+        $this->assertEquals(array("a"), $result2);
+        $this->assertEquals(array("b"), $result3);
+        $this->assertEquals(array("a", "a", "b", "b", "a", "b", "a"), $result4);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/MiddleTest.php b/src/bindings/php/Tests/MiddleTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..93b6c48b68a944d672230304842d531c0c0acfe7
--- /dev/null
+++ b/src/bindings/php/Tests/MiddleTest.php
@@ -0,0 +1,35 @@
+<?php
+include_once 'hammer.php';
+
+class MiddleTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_middle(hammer_ch(" "), hammer_ch("a"), hammer_ch(" "));
+    }
+    public function testSuccess()
+    {
+        $result = hammer_parse($this->parser, " a ");
+        $this->assertEquals("a", $result);
+    }
+    public function testFailure()
+    {
+        $result1 = hammer_parse($this->parser, "a");
+        $result2 = hammer_parse($this->parser, " ");
+        $result3 = hammer_parse($this->parser, " a");
+        $result4 = hammer_parse($this->parser, "a ");
+        $result5 = hammer_parse($this->parser, " b ");
+        $result6 = hammer_parse($this->parser, "ba ");
+        $result7 = hammer_parse($this->parser, " ab");
+        $this->assertEquals(NULL, $result1);
+        $this->assertEquals(NULL, $result2);
+        $this->assertEquals(NULL, $result3);
+        $this->assertEquals(NULL, $result4);
+        $this->assertEquals(NULL, $result5);
+        $this->assertEquals(NULL, $result6);
+        $this->assertEquals(NULL, $result7);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/NotInTest.php b/src/bindings/php/Tests/NotInTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..c5a2fdfd73c29c856ff4e5ab468d40bd4895fed5
--- /dev/null
+++ b/src/bindings/php/Tests/NotInTest.php
@@ -0,0 +1,23 @@
+<?php
+include_once 'hammer.php';
+
+class NotInTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_not_in("abc");
+    }
+    public function testSuccess()
+    {
+        $result = hammer_parse($this->parser, "d");
+        $this->assertEquals("d", $result);
+    }
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "b");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/NotTest.php b/src/bindings/php/Tests/NotTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..15eb77a1173bb48ed3591d810cba9438c04c7e7f
--- /dev/null
+++ b/src/bindings/php/Tests/NotTest.php
@@ -0,0 +1,35 @@
+<?php
+include_once 'hammer.php';
+
+class NotTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser1;
+    protected $parser2;
+
+    protected function setUp()
+    {
+        $this->parser1 = hammer_sequence(hammer_ch("a"), hammer_choice(hammer_ch("+"), hammer_token("++")), hammer_ch("b"));
+        $this->parser2 = hammer_sequence(hammer_ch("a"), hammer_choice(hammer_sequence(hammer_ch("+"), hammer_not(hammer_ch("+"))), hammer_token("++")), hammer_ch("b"));
+    }
+
+    public function testSuccess1()
+    {
+        $result = hammer_parse($this->parser1, "a+b");
+        $this->assertEquals(array("a", "+", "b"), $result);
+    }
+
+    public function testFailure1()
+    {
+        $result = hammer_parse($this->parser1, "a++b");
+        $this->assertEquals(NULL, $result);
+    }
+
+    public function testSuccess2()
+    {
+        $result1 = hammer_parse($this->parser2, "a+b");
+        $result2 = hammer_parse($this->parser2, "a++b");
+        $this->assertEquals(array("a", array("+"), "b"), $result1);
+        $this->assertEquals(array("a", "++", "b"), $result2);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/NothingTest.php b/src/bindings/php/Tests/NothingTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..f827e47558e96698593f59ff091ee41893f20666
--- /dev/null
+++ b/src/bindings/php/Tests/NothingTest.php
@@ -0,0 +1,19 @@
+<?php
+include_once 'hammer.php';
+
+class NothingPTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_nothing();
+    }
+
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "a");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/OptionalTest.php b/src/bindings/php/Tests/OptionalTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..e150415f2a21d4a4ac4eb36f9b0323d05fe326a3
--- /dev/null
+++ b/src/bindings/php/Tests/OptionalTest.php
@@ -0,0 +1,33 @@
+<?php
+include_once 'hammer.php';
+
+class OptionalTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_sequence(hammer_ch("a"), hammer_optional(hammer_choice(hammer_ch("b"), hammer_ch("c"))), hammer_ch("d"));
+    }
+
+    public function testSuccess()
+    {
+        $result1 = hammer_parse($this->parser, "abd");
+        $result2 = hammer_parse($this->parser, "acd");
+        $result3 = hammer_parse($this->parser, "ad");
+        $this->assertEquals(array("a", "b", "d"), $result1);
+        $this->assertEquals(array("a", "c", "d"), $result2);
+        $this->assertEquals(array("a", NULL, "d"), $result3);
+    }
+
+    public function testFailure()
+    {
+        $result1 = hammer_parse($this->parser, "aed");
+        $result2 = hammer_parse($this->parser, "ab");
+        $result3 = hammer_parse($this->parser, "ac");
+        $this->assertEquals(NULL, $result1);
+        $this->assertEquals(NULL, $result2);
+        $this->assertEquals(NULL, $result3);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/PredicateTest.php b/src/bindings/php/Tests/PredicateTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..cc891d67c3ec2da74ca568b95a13fa85fc6f512b
--- /dev/null
+++ b/src/bindings/php/Tests/PredicateTest.php
@@ -0,0 +1,31 @@
+<?php
+include_once 'hammer.php';
+
+function predTest($token)
+{
+    return ($token[0] === $token[1]);
+}
+
+class PredicateTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_predicate(hammer_many1(hammer_choice(hammer_ch('a'), hammer_ch('b'))), "predTest");
+    }
+    public function testSuccess()
+    {
+        $result1 = hammer_parse($this->parser, "aa");
+        $result2 = hammer_parse($this->parser, "bb");
+        $this->assertEquals(["a", "a"], $result1);
+        $this->assertEquals(["b", "b"], $result2);
+    }
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "ab");
+        $this->assertEquals(NULL, $result);
+    }
+}
+
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/RepeatNTest.php b/src/bindings/php/Tests/RepeatNTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..5441ef64d2bcb0e08b1ab31a18b227962aa03da4
--- /dev/null
+++ b/src/bindings/php/Tests/RepeatNTest.php
@@ -0,0 +1,27 @@
+<?php
+include_once 'hammer.php';
+
+class RepeatNTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_repeat_n(hammer_choice(hammer_ch("a"), hammer_ch("b")), 2);
+    }
+
+    public function testSuccess()
+    {
+        $result = hammer_parse($this->parser, "abdef");
+        $this->assertEquals(array("a", "b"), $result);
+    }
+
+    public function testFailure()
+    {
+        $result1 = hammer_parse($this->parser, "adef");
+        $result2 = hammer_parse($this->parser, "dabdef");
+        $this->assertEquals(NULL, $result1);
+        $this->assertEquals(NULL, $result2);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/RightTest.php b/src/bindings/php/Tests/RightTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..215aacec6d7212ea2dc7e496f9cbc16dd59a2944
--- /dev/null
+++ b/src/bindings/php/Tests/RightTest.php
@@ -0,0 +1,27 @@
+<?php
+include_once 'hammer.php';
+
+class RightTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_right(hammer_ch(" "), hammer_ch("a"));
+    }
+    public function testSuccess()
+    {
+        $result = hammer_parse($this->parser, " a");
+        $this->assertEquals("a", $result);
+    }
+    public function testFailure()
+    {
+        $result1 = hammer_parse($this->parser, "a");
+        $result2 = hammer_parse($this->parser, " ");
+        $result3 = hammer_parse($this->parser, "ba");
+        $this->assertEquals(NULL, $result1);
+        $this->assertEquals(NULL, $result2);
+        $this->assertEquals(NULL, $result3);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/RightrecTest.php b/src/bindings/php/Tests/RightrecTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..ed21b353cc2f695bc49f4b1aca35a7152daceba5
--- /dev/null
+++ b/src/bindings/php/Tests/RightrecTest.php
@@ -0,0 +1,24 @@
+<?php
+include_once 'hammer.php';
+
+class RightrecTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_indirect();
+        hammer_bind_indirect($this->parser, hammer_choice(hammer_sequence(hammer_ch("a"), $this->parser), hammer_epsilon()));
+    }
+
+    public function testSuccess()
+    {
+        $result1 = hammer_parse($this->parser, "a");
+        $result2 = hammer_parse($this->parser, "aa");
+        $result3 = hammer_parse($this->parser, "aaa");
+        $this->assertEquals(array("a"), $result1);
+        $this->assertEquals(array("a", array("a")), $result2);
+        $this->assertEquals(array("a", array("a", array("a"))), $result3);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/SepBy1Test.php b/src/bindings/php/Tests/SepBy1Test.php
new file mode 100644
index 0000000000000000000000000000000000000000..57a5affb99bf0bd941d2b51372f0dfe1151e5093
--- /dev/null
+++ b/src/bindings/php/Tests/SepBy1Test.php
@@ -0,0 +1,31 @@
+<?php
+include_once 'hammer.php';
+
+class SepBy1Test extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_sep_by1(hammer_choice(hammer_ch("1"), hammer_ch("2"), hammer_ch("3")), hammer_ch(","));
+    }
+
+    public function testSuccess()
+    {
+        $result1 = hammer_parse($this->parser, "1,2,3");
+        $result2 = hammer_parse($this->parser, "1,3,2");
+        $result3 = hammer_parse($this->parser, "1,3");
+        $result4 = hammer_parse($this->parser, "3");
+        $this->assertEquals(array("1", "2", "3"), $result1);
+        $this->assertEquals(array("1", "3", "2"), $result2);
+        $this->assertEquals(array("1", "3"), $result3);
+        $this->assertEquals(array("3"), $result4);
+    }
+
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/SepByTest.php b/src/bindings/php/Tests/SepByTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..17c8e1656ed6200c122196c0fd9c400d5cd8b2ef
--- /dev/null
+++ b/src/bindings/php/Tests/SepByTest.php
@@ -0,0 +1,27 @@
+<?php
+include_once 'hammer.php';
+
+class SepByTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_sep_by(hammer_choice(hammer_ch("1"), hammer_ch("2"), hammer_ch("3")), hammer_ch(","));
+    }
+
+    public function testSuccess()
+    {
+        $result1 = hammer_parse($this->parser, "1,2,3");
+        $result2 = hammer_parse($this->parser, "1,3,2");
+        $result3 = hammer_parse($this->parser, "1,3");
+        $result4 = hammer_parse($this->parser, "3");
+        $result5 = hammer_parse($this->parser, "");
+        $this->assertEquals(array("1", "2", "3"), $result1);
+        $this->assertEquals(array("1", "3", "2"), $result2);
+        $this->assertEquals(array("1", "3"), $result3);
+        $this->assertEquals(array("3"), $result4);
+        $this->assertEquals(array(), $result5);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/SequenceTest.php b/src/bindings/php/Tests/SequenceTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..bafb4f1c9a74fff8f6471c2f7a9a2afe14a7b54d
--- /dev/null
+++ b/src/bindings/php/Tests/SequenceTest.php
@@ -0,0 +1,39 @@
+<?php
+include_once 'hammer.php';
+
+class SequenceTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser1;
+    protected $parser2;
+
+    protected function setUp()
+    {
+        $this->parser1 = hammer_sequence(hammer_ch("a"), hammer_ch("b"));
+        $this->parser2 = hammer_sequence(hammer_ch("a"), hammer_whitespace(hammer_ch("b")));
+    }
+
+    public function testSuccess1()
+    {
+        $result = hammer_parse($this->parser1, "ab");
+        $this->assertEquals(array("a", "b"), $result);
+    }
+    
+    public function testFailure1()
+    {
+        $result1 = hammer_parse($this->parser1, "a");
+        $result2 = hammer_parse($this->parser2, "b");
+        $this->assertEquals(NULL, $result1);
+        $this->assertEquals(NULL, $result2);
+    }
+
+    public function testSuccess2()
+    {
+        $result1 = hammer_parse($this->parser2, "ab");
+        $result2 = hammer_parse($this->parser2, "a b");
+        $result3 = hammer_parse($this->parser2, "a  b");
+        $this->assertEquals(array("a", "b"), $result1);
+        $this->assertEquals(array("a", "b"), $result2);
+        $this->assertEquals(array("a", "b"), $result3);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/TokenTest.php b/src/bindings/php/Tests/TokenTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..90c278d7cced74c3f79920d6c369800d047fb653
--- /dev/null
+++ b/src/bindings/php/Tests/TokenTest.php
@@ -0,0 +1,24 @@
+<?php
+
+include_once 'hammer.php';
+
+class TokenTest extends PHPUnit_Framework_TestCase 
+{
+    protected $parser;
+
+    protected function setUp() 
+    {
+        $this->parser = hammer_token("95\xa2");
+    }
+    public function testSuccess()
+    {
+        $result = hammer_parse($this->parser, "95\xa2");
+        $this->assertEquals("95\xa2", $result); 
+    }      
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "95");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/Uint16Test.php b/src/bindings/php/Tests/Uint16Test.php
new file mode 100644
index 0000000000000000000000000000000000000000..31c99cda7f33fc71ecc036519d59165df958528b
--- /dev/null
+++ b/src/bindings/php/Tests/Uint16Test.php
@@ -0,0 +1,24 @@
+<?php
+
+include_once 'hammer.php';
+
+class Uint16Test extends PHPUnit_Framework_TestCase 
+{
+    protected $parser;
+
+    protected function setUp() 
+    {
+        $this->parser = hammer_uint16();
+    }
+    public function testSuccess() 
+    {
+        $result = hammer_parse($this->parser, "\x02\x00");
+        $this->assertEquals(0x200, $result);
+    }     
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "\x02");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/Uint32Test.php b/src/bindings/php/Tests/Uint32Test.php
new file mode 100644
index 0000000000000000000000000000000000000000..593de6607ea8cadb6fddd9645b92c6ecad5354f5
--- /dev/null
+++ b/src/bindings/php/Tests/Uint32Test.php
@@ -0,0 +1,24 @@
+<?php
+
+include_once 'hammer.php';
+
+class Uint32Test extends PHPUnit_Framework_TestCase 
+{
+    protected $parser;
+
+    protected function setUp() 
+    {
+        $this->parser = hammer_uint32();
+    }
+    public function testSuccess() 
+    {
+        $result = hammer_parse($this->parser, "\x00\x02\x00\x00");
+        $this->assertEquals(0x20000, $result);
+    }     
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "\x00\x02\x00");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/Uint64Test.php b/src/bindings/php/Tests/Uint64Test.php
new file mode 100644
index 0000000000000000000000000000000000000000..e61edb635cf34acfd3223404d0c02ccca87bfeae
--- /dev/null
+++ b/src/bindings/php/Tests/Uint64Test.php
@@ -0,0 +1,24 @@
+<?php
+
+include_once 'hammer.php';
+
+class Uint64Test extends PHPUnit_Framework_TestCase 
+{
+    protected $parser;
+
+    protected function setUp() 
+    {
+        $this->parser = hammer_uint64();
+    }
+    public function testSuccess() 
+    {
+        $result = hammer_parse($this->parser, "\x00\x00\x00\x02\x00\x00\x00\x00");
+        $this->assertEquals(0x200000000, $result);
+    }     
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "\x00\x00\x00\x02\x00\x00\x00");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/Uint8Test.php b/src/bindings/php/Tests/Uint8Test.php
new file mode 100644
index 0000000000000000000000000000000000000000..6ebfa28bd468f521773a975aff28204b02ae2999
--- /dev/null
+++ b/src/bindings/php/Tests/Uint8Test.php
@@ -0,0 +1,24 @@
+<?php
+
+include_once 'hammer.php';
+
+class Uint8Test extends PHPUnit_Framework_TestCase 
+{
+    protected $parser;
+
+    protected function setUp() 
+    {
+        $this->parser = hammer_uint8();
+    }
+    public function testSuccess() 
+    {
+        $result = hammer_parse($this->parser, "\x78");
+        $this->assertEquals(0x78, $result);
+    }     
+    public function testFailure()
+    {
+        $result = hammer_parse($this->parser, "");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/WhitespaceTest.php b/src/bindings/php/Tests/WhitespaceTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..c5f6e900f3fa288eb5597989991e5f4ba6a3266c
--- /dev/null
+++ b/src/bindings/php/Tests/WhitespaceTest.php
@@ -0,0 +1,43 @@
+<?php
+include_once 'hammer.php';
+
+class WhitespaceTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser1;
+    protected $parser2;
+    
+    protected function setUp()
+    {
+        $this->parser1 = hammer_whitespace(hammer_ch("a"));
+        $this->parser2 = hammer_whitespace(hammer_end());
+    }
+    public function testSuccess1()
+    {
+        $result1 = hammer_parse($this->parser1, "a");
+        $result2 = hammer_parse($this->parser1, " a");
+        $result3 = hammer_parse($this->parser1, "  a");
+        $result4 = hammer_parse($this->parser1, "\ta");
+        $this->assertEquals("a", $result1);
+        $this->assertEquals("a", $result2);
+        $this->assertEquals("a", $result3);
+        $this->assertEquals("a", $result4);
+    }
+    public function testFailure1()
+    {
+        $result = hammer_parse($this->parser1, "_a");
+        $this->assertEquals(NULL, $result);
+    }
+    public function testSuccess2()
+    {
+        $result1 = hammer_parse($this->parser2, "");
+        $result2 = hammer_parse($this->parser2, "  ");
+        $this->assertEquals(NULL, $result1);
+        $this->assertEquals(NULL, $result2);
+    }
+    public function testFailure2()
+    {
+        $result = hammer_parse($this->parser2, "  x");
+        $this->assertEquals(NULL, $result);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/Tests/XorTest.php b/src/bindings/php/Tests/XorTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..1d20d1e9a4fc55017045a35e2baca7da4e670dba
--- /dev/null
+++ b/src/bindings/php/Tests/XorTest.php
@@ -0,0 +1,29 @@
+<?php
+include_once 'hammer.php';
+
+class XorTest extends PHPUnit_Framework_TestCase
+{
+    protected $parser;
+
+    protected function setUp()
+    {
+        $this->parser = hammer_xor(hammer_ch_range("0", "6"), hammer_ch_range("5", "9"));
+    }
+
+    public function testSuccess()
+    {
+        $result1 = hammer_parse($this->parser, "0");
+        $result2 = hammer_parse($this->parser, "9");
+        $this->assertEquals("0", $result1);
+        $this->assertEquals("9", $result2);
+    }
+
+    public function testFailure()
+    {
+        $result1 = hammer_parse($this->parser, "5");
+        $result2 = hammer_parse($this->parser, "a");
+        $this->assertEquals(NULL, $result1);
+        $this->assertEquals(NULL, $result2);
+    }
+}
+?>
\ No newline at end of file
diff --git a/src/bindings/php/hammer.i b/src/bindings/php/hammer.i
new file mode 100644
index 0000000000000000000000000000000000000000..11fb12ab7c5ccb11c60b663a5c64714c92d35e9d
--- /dev/null
+++ b/src/bindings/php/hammer.i
@@ -0,0 +1,255 @@
+%module hammer;
+%include "exception.i";
+%{
+#include "allocator.h"
+  %}
+
+%ignore HCountedArray_;
+
+%inline %{
+  static int h_tt_php;
+  %}
+
+%init %{
+  h_tt_php = h_allocate_token_type("com.upstandinghackers.hammer.php");
+  %}
+
+%inline {
+  struct HParsedToken_;
+  struct HParseResult_;
+  void hpt_to_php(const struct HParsedToken_ *token, zval *return_value);
+  
+  static struct HParsedToken_* call_action(const struct HParseResult_ *p, void* user_data);
+ }
+
+%typemap(in) (const uint8_t* str, const size_t len) {
+  $1 = (uint8_t*)(*$input)->value.str.val;
+  $2 = (*$input)->value.str.len;
+ }
+
+%apply (const uint8_t* str, const size_t len) { (const uint8_t* input, size_t length) }
+%apply (const uint8_t* str, const size_t len) { (const uint8_t* charset, size_t length) }
+
+%typemap(in) void*[] {
+  if (IS_ARRAY == Z_TYPE_PP($input)) {
+    HashTable *arr = Z_ARRVAL_PP($input);
+    HashPosition pointer;
+    int size = zend_hash_num_elements(arr);
+    int i = 0;
+    int res = 0;
+    $1 = (void**)malloc((size+1)*sizeof(HParser*));
+    for (i=0; i<size; i++) {
+      zval **data;
+      if (zend_hash_index_find(arr, i, (void**)&data) == FAILURE) {
+	$1 = NULL;
+	zend_throw_exception(zend_exception_get_default(TSRMLS_C), "index in parser array out of bounds", 0 TSRMLS_CC);
+      } else {
+	res = SWIG_ConvertPtr(*data, &($1[i]), SWIGTYPE_p_HParser_, 0 | 0);
+	if (!SWIG_IsOK(res)) {
+	  zend_throw_exception(zend_exception_get_default(TSRMLS_C), "that wasn't an HParser", 0 TSRMLS_CC);
+	}
+      }
+    } 
+  } else {
+    $1 = NULL;
+    zend_throw_exception(zend_exception_get_default(TSRMLS_C), "that wasn't an array of HParsers", 0 TSRMLS_CC);
+  }
+ }
+
+%typemap(in) uint8_t {
+  if (IS_LONG == Z_TYPE_PP($input)) {
+    $1 = Z_LVAL_PP($input);
+  } else if (IS_STRING != Z_TYPE_PP($input)) {
+    // FIXME raise some error
+  } else {
+    $1 = *(uint8_t*)Z_STRVAL_PP($input);
+  }
+ }
+
+%typemap(out) HBytes* {
+  RETVAL_STRINGL((char*)$1->token, $1->len, 1);
+ }
+
+%typemap(out) struct HParseResult_* {
+  if ($1 == NULL) {
+    /* If we want parse errors to be exceptions, this is the place to do it */
+    //SWIG_exception(SWIG_TypeError, "typemap: should have been an HParseResult*, was NULL");
+    RETVAL_NULL();
+  } else {
+    hpt_to_php($1->ast, $result);
+  }
+ }
+
+%rename("hammer_token") h_token;
+%rename("hammer_int_range") h_int_range;
+%rename("hammer_bits") h_bits;
+%rename("hammer_int64") h_int64;
+%rename("hammer_int32") h_int32;
+%rename("hammer_int16") h_int16;
+%rename("hammer_int8") h_int8;
+%rename("hammer_uint64") h_uint64;
+%rename("hammer_uint32") h_uint32;
+%rename("hammer_uint16") h_uint16;
+%rename("hammer_uint8") h_uint8;
+%rename("hammer_whitespace") h_whitespace;
+%rename("hammer_left") h_left;
+%rename("hammer_right") h_right;
+%rename("hammer_middle") h_middle;
+%rename("hammer_end") h_end_p;
+%rename("hammer_nothing") h_nothing_p;
+%rename("hammer_butnot") h_butnot;
+%rename("hammer_difference") h_difference;
+%rename("hammer_xor") h_xor;
+%rename("hammer_many") h_many;
+%rename("hammer_many1") h_many1;
+%rename("hammer_repeat_n") h_repeat_n;
+%rename("hammer_optional") h_optional;
+%rename("hammer_ignore") h_ignore;
+%rename("hammer_sep_by") h_sepBy;
+%rename("hammer_sep_by1") h_sepBy1;
+%rename("hammer_epsilon") h_epsilon_p;
+%rename("hammer_length_value") h_length_value;
+%rename("hammer_and") h_and;
+%rename("hammer_not") h_not;
+%rename("hammer_indirect") h_indirect;
+%rename("hammer_bind_indirect") h_bind_indirect;
+%rename("hammer_compile") h_compile;
+%rename("hammer_parse") h_parse;
+
+%include "../swig/hammer.i";
+
+%inline {
+  void hpt_to_php(const HParsedToken *token, zval *return_value) {
+    TSRMLS_FETCH();
+    if (!token) {
+      RETVAL_NULL();
+      return;
+    }
+    switch (token->token_type) {
+    case TT_NONE:
+      RETVAL_NULL();
+      break;
+    case TT_BYTES:
+      RETVAL_STRINGL((char*)token->token_data.bytes.token, token->token_data.bytes.len, 1);
+      break;
+    case TT_SINT:
+      RETVAL_LONG(token->token_data.sint);
+      break;
+    case TT_UINT:
+      RETVAL_LONG(token->token_data.uint);
+      break;
+    case TT_SEQUENCE:
+      array_init(return_value);
+      for (int i=0; i < token->token_data.seq->used; i++) {
+	zval *tmp;
+	ALLOC_INIT_ZVAL(tmp);
+	hpt_to_php(token->token_data.seq->elements[i], tmp);
+	add_next_index_zval(return_value, tmp);
+      }
+      break;
+    default:
+      if (token->token_type == h_tt_php) { 
+	zval *tmp;
+	tmp = (zval*)token->token_data.user;
+	RETVAL_ZVAL(tmp, 0, 0);
+      } else {
+	int res = 0;
+	res = SWIG_ConvertPtr(return_value, (void*)token, SWIGTYPE_p_HParsedToken_, 0 | 0);
+	if (!SWIG_IsOK(res)) {
+	  zend_throw_exception(zend_exception_get_default(TSRMLS_C), "hpt_to_php: that wasn't an HParsedToken", 0 TSRMLS_CC);
+	}
+	// TODO: support registry
+      }
+      break;
+    }
+  }
+
+  static HParsedToken* call_action(const HParseResult *p, void *user_data) {
+    zval **args;
+    zval func;
+    zval *ret;
+    TSRMLS_FETCH();
+    args = (zval**)h_arena_malloc(p->arena, sizeof(*args) * 1); // one-element array of pointers
+    MAKE_STD_ZVAL(args[0]);
+    ALLOC_INIT_ZVAL(ret);
+    ZVAL_STRING(&func, (const char*)user_data, 0);
+    hpt_to_php(p->ast, args[0]);
+    int ok = call_user_function(EG(function_table), NULL, &func, ret, 1, args TSRMLS_CC);
+    if (ok != SUCCESS) {
+      zend_throw_exception(zend_exception_get_default(TSRMLS_C), "call_action failed", 0 TSRMLS_CC);
+      return NULL;
+    }
+    // Whatever the zval is, stuff it into a token
+    HParsedToken *tok = h_make(p->arena, h_tt_php, ret);
+    return tok;
+  }
+
+  static int call_predicate(HParseResult *p, void *user_data) {
+    zval **args;
+    zval func;
+    zval *ret;
+    TSRMLS_FETCH();
+    args = (zval**)h_arena_malloc(p->arena, sizeof(*args) * 1); // one-element array of pointers
+    MAKE_STD_ZVAL(args[0]);
+    ALLOC_INIT_ZVAL(ret);
+    ZVAL_STRING(&func, (const char*)user_data, 0);
+    hpt_to_php(p->ast, args[0]);
+    int ok = call_user_function(EG(function_table), NULL, &func, ret, 1, args TSRMLS_CC);
+    if (ok != SUCCESS) {
+      zend_throw_exception(zend_exception_get_default(TSRMLS_C), "call_predicate failed", 0 TSRMLS_CC);
+      return 0;
+    }
+    return Z_LVAL_P(ret);
+  }
+
+  HParser* hammer_action(HParser *parser, const char *name) {
+    const char *fname = strdup(name);
+    return h_action(parser, call_action, (void*)fname);
+  }
+
+  HParser* hammer_predicate(HParser *parser, const char *name) {
+    const char *fname = strdup(name);
+    return h_attr_bool(parser, call_predicate, (void*)fname);
+  }
+ }
+
+%pragma(php) code="
+
+function hammer_ch($ch)
+{
+    if (is_string($ch)) 
+        return hammer_token($ch);
+    else
+        return h_ch($ch);
+}
+
+function hammer_choice()
+{
+    $arg_list = func_get_args();
+    $arg_list[] = NULL;
+    return h_choice__a($arg_list);    
+}
+
+function hammer_sequence()
+{
+    $arg_list = func_get_args();
+    $arg_list[] = NULL;
+    return h_sequence__a($arg_list);
+}
+
+function hammer_ch_range($low, $high)
+{
+    return hammer_action(h_ch_range($low, $high), \"chr\");
+}
+
+function hammer_in($charset)
+{
+    return hammer_action(h_in($charset), \"chr\");
+}
+
+function hammer_not_in($charset)
+{
+    return hammer_action(h_not_in($charset), \"chr\");
+}
+"
+
diff --git a/src/bindings/php/hammer.ini b/src/bindings/php/hammer.ini
new file mode 100644
index 0000000000000000000000000000000000000000..5ac194259c7c56a251f2cc0035d88ee16dfcb7d5
--- /dev/null
+++ b/src/bindings/php/hammer.ini
@@ -0,0 +1,2 @@
+enable_dl = On
+extension = "hammer.so"
diff --git a/src/bindings/swig/hammer.i b/src/bindings/swig/hammer.i
index 13c30a4dee457d6c3fdd2112bf92fb652d7c99d0..122ffe4012937add3bd71107aef0589980f3575d 100644
--- a/src/bindings/swig/hammer.i
+++ b/src/bindings/swig/hammer.i
@@ -344,3 +344,5 @@ def int64(): return _h_int64()
 %}
 
 #endif
+
+