PHP Internals personal scratch book
This is a list of things I've kept within reach to refer back or ponder while working with or around php-src, or potential projects for it.
This gets edited when I need to throw something at it. It might also not be accurate anymore.
php-src TODOs and Projects
Feel free to get inspired or tackle one of those projects.
My php-src TODO list: Girgias/todo-php-src
My PHP RFC drafts and ideas: Girgias/php-rfcs
php-src tools and explanations
Websites
LXR: https://heap.space
PHP Internals Book: http://www.phpinternalsbook.com/
Book by Zend on writing PHP extensions: https://www.zend.com/resources/writing-php-extensions
Incomplete PHP 7.2 engine docs + articles: https://phpinternals.net/
Check usage across open source repos: https://sourcegraph.com/
Compilation
To use CLANG instead: ./configure CC=clang ...
Add Compiler flags: ./configure CFLAGS="-Wflag" ...
Configure options for enabling ASAN/UBSAN: --enable-address-sanitizer --enable-undefined-sanitizer
Configure options for enabling Clang MSAN (PCRE JIT needs to be disabled): ./configure CC=clang --enable-memory-sanitizer --without-pcre-jit
Compiling with CFLAGS="-ggdb3"
allows GDB to have access to macros and preprocessor constants.
A reasonable basic configure for minimal testing:
./configure -C CFLAGS="-DPROFITABILITY_CHECKS=0 -DZEND_RC_DEBUG=1 -DZEND_VERIFY_FUNC_INFO=1 -DZEND_TRACK_ARENA_ALLOC=1 -ggdb3" \
--enable-address-sanitizer --enable-undefined-sanitizer --disable-all --enable-debug \
--enable-tokenizer --enable-opcache --enable-zend-test --enable-dl-test=shared
Tests need to run with the --asan flag if using the Address sanitizer.
Larger MSAN build:
MSAN needs all libraries instrumented, therefore limit ourselves to stuff which relies on libc.
./configure -C CC=clang --enable-memory-sanitizer --without-pcre-jit --without-sqlite3 --without-pdo-sqlite \
--without-libxml --disable-dom --disable-simplexml --disable-xml --disable-xmlreader --disable-xmlwriter \
--without-pcre-jit --disable-opcache-jit --enable-phpdbg --enable-zend-test --enable-dl-test=shared \
--with-pdo-mysql=mysqlnd --with-mysqli=mysqlnd --disable-mysqlnd-compression-support --without-pear \
--enable-exif --enable-sysvsem --enable-sysvshm --enable-shmop --enable-pcntl --enable-sockets \
--enable-mbstring --disable-mbregex --enable-bcmath --enable-calendar --enable-ftp
Large Build:
./configure -C CFLAGS="-DPROFITABILITY_CHECKS=0 -DZEND_RC_DEBUG=1 -DZEND_VERIFY_FUNC_INFO=1 -DZEND_TRACK_ARENA_ALLOC=1 -ggdb3" \
--enable-address-sanitizer --enable-undefined-sanitizer --enable-debug --enable-tokenizer --enable-opcache --enable-zend-test --enable-dl-test=shared \
--enable-pcntl --enable-mbstring --enable-fpm --enable-posix --enable-bcmath --enable-calendar --enable-ctype --enable-exif \
--enable-fileinfo --enable-filter --enable-ftp --enable-gd --enable-session --enable-sockets \
--enable-sysvmsg --enable-shmop --enable-sysvsem --enable-sysvshm \
--enable-dba --with-gdbm --with-cdb --enable-flatfile --enable-inifile --with-lmdb --with-tcadb \
--with-zip --with-zlib --with-bz2 \
--with-curl --with-ffi --with-gmp --with-tidy --with-enchant --with-openssl --with-sodium \
--with-libxml --enable-dom --enable-simplexml --enable-xml --enable-xmlreader --enable-xmlwriter --with-xsl --enable-soap \
--with-sqlite3 --with-mysqli --with-unixODBC --enable-pdo --with-pdo-firebird --with-pdo-mysql --with-pdo-pgsql --with-pdo-sqlite --with-pgsql \
--with-iconv --enable-intl --enable-phar --with-readline
Regenerating the PHP VM:
php Zend/zend_vm_gen.php
Testing php-src
General test command:
make test TEST_PHP_ARGS="--asan -q -j22 -d opcache.jit_buffer_size=16M -d opcache.jit=disable -d opcache.enable=1 -d opcache.enable_cli=1 -d opcache.file_update_protection=0 -d opcache.protect_memory=1 -x" TESTS="Zend/ ext/
Debugging a failed PHPT test with GDB:
failed_test.sh gdb
Testing with Valgrind:
ZEND_DONT_UNLOAD_MODULES=1 USE_ZEND_ALLOC=0 valgrind --leak-check=full --track-origins=yes make test
Testing PDO MySQL:
PDO_MYSQL_TEST_DSN="mysql:dbname=test;host=127.0.0.1;" PDO_MYSQL_TEST_DB="test" PDO_MYSQL_TEST_USER="pdo" PDO_MYSQL_TEST_PASS="password" make test TESTS="ext/pdo_mysql"
Testing the JIT:
First need to compile with CFLAGS="-DPROFITABILITY_CHECKS=0"
.
Then run:
make test TEST_PHP_ARGS="-q -j22 -d opcache.jit=function -d opcache.enable=1 -d opcache.enable_cli=1 -d opcache.file_update_protection=0 -d opcache.jit_buffer_size=1M" TESTS="Zend ext/opcache"
Potential way to debugging JIT Assembly:
make test TEST_PHP_ARGS="-dopcache.enable_cli=1 -dopcache.jit_buffer_size=16M -d opcache.jit_debug='1<<8'" TESTS="failing_test_file.phpt" &> debug-jit.txt
Out of tree PHP extensions
Testing with Valgrind:
ZEND_DONT_UNLOAD_MODULES=1 USE_ZEND_ALLOC=0 valgrind --leak-check=full --track-origins=yes path/to/php run-tests.php -P -d extension=./modules/ext.so
Generate ARGINFO:
path/to/php-src/scripts/dev/gen_stub.php ext.stub.php
Custom php-src builds for working with different versions/types (Debug/NTS/TS)
Use a MAKE INSTALL build by using the following command for php-src:
./configure --disable-all --enable-debug --prefix /path/to/custom-php/
The extension is now ready to be compiled. To do so, first move into the extension directory. Then do the following steps:
./configure --with-php-config=path/to/custom-php//bin/php-config CFLAGS="-Wfatal-errors -Wall -Wpedantic -Wextra -Wno-unused-parameter -Werror"
Test against chosen custom debug build:
/path/to/custom-php/bin/php run-tests.php -P -d extension=./modules/ext.so
Releasing on PECL
PECL's package.xml DTD location: https://pear.php.net/dtd/package-2.0.xsd
Generate test file list for package.xml:
find tests -name "*.phpt" | awk '{ print " <file role=\"test\" name=\""$0"\" />" }'
To generate the tar ball: pear package
Extension ideas:
Rational Number bata type: https://en.wikipedia.org/wiki/Rational_data_type
- Complex numbers (maybe better as a new zval type?)
- Matrices
C related knowledge:
The Lost Art of Structure Packing: http://www.catb.org/esr/structure-packing/
C99 restrict keyword
Compiler warnings to add/ideas
Sources:
https://kristerw.blogspot.com/2017/09/useful-gcc-warning-options-not-enabled.html
https://interrupt.memfault.com/blog/best-and-worst-gcc-clang-compiler-flags
https://blogs.oracle.com/linux/making-code-more-secure-with-gcc-part-1
-Wshadow: Impossible php-src macro usage leads to variable shadowing
-Wdouble-promotion: Nice to have
-Wformat=2 & -Wformat-truncation: seems useful to be able to enable
-Wundef: W.I.P.
-Wconversion: Huge amount of work
-fstack-usage & -Wstack-usage=: To check stack usage, useful for performance optimization?
Git commands and stuff
A list of links to various useful tutorials about git:
Slides of useful stuff: https://greg0ire.fr/git-gud
Splitting up one commit into more: https://embeddedartistry.com/blog/2018/02/19/code-cleanup-splitting-up-git-commits-in-the-middle-of-a-branch/
Cherry-picking multiple commits: https://stackoverflow.com/questions/1670970/how-to-cherry-pick-multiple-commits
Git reflog: https://gitready.com/intermediate/2009/02/09/reflog-your-safety-net.html
Fetching remote GitHub PRs locally
git fetch <remote> pull/$ID/head:$BRANCHNAME
Where $ID is the PR number and $BRANCHNAME the name given locally.
Followed by:
git checkout $BRANCHNAME
Fetching patch file from GitHub PR and applying it
Take the GitHub PR URL, if one add .patch
to the URL
it now points to a redirects to the git email patch file for this
pull request.
It is then easy to apply it locally with the following command:
curl -L $URL | git am --signoff
Rebase onto:
git rebase [--onto <new base>] [<upstream> [<branch>]]
Refresh commit date:
git commit --amend --date=now --no-edit
Refresh commit date and author name:
git commit --amend --date=now --reset-author --no-edit
Push all branches after doing a merge-up of a commit:
git push --atomic php PHP-8.1 PHP-8.2 master
Pretty git log graph
Someone sent me this command to generate a pretty git log graph:
git log --graph --pretty='\''%Cred%h%Creset -%C(auto)%d%Creset %s %Cgreen(%cr) %C(bold blue)<%an>%Creset'
Fixup branches when you forget to change feature branches before committing:
git checkout -b new-branch # switch to a new branch
git branch -f old-branch HEAD~3 # make old-branch point to some older commit
Deleting local branches after removing them from remote
Discovered here: https://medium.com/@kcmueller/delete-local-git-branches-that-were-deleted-on-remote-repository-b596b71b530c
git branch -vv | grep ': gone]'| grep -v "\*" | awk '{ print $1; }' | xargs -r git branch -d
And that's all for now.