-
-
Notifications
You must be signed in to change notification settings - Fork 248
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat: add support for libphp and the embed SAPI #153
Conversation
It's a huge update. I tried to support embed sapi before but failed as you said (but it was a long time ago).
The reason why I don't use For macOS, I'm also learning some system library and executable things. |
I've found a hack to workaround the MacOS error:
Basically, I extract all .o files from the generated archive and then recreate it. Then, tweaking the LDFLAGS environment variable like this allows building: I'm not sure of what's going on exactly, but I suspect that the libtool version bundled with PHP doesn't pass the appropriate flags to We could include this hack in static-php-cli, but I'm not sure if it's a good idea. Maybe the PHP build system should be fixed instead. |
I think we can accept using some hacks to solve this first, and then submit the issue to other projects (if other projects are slower to update). BTW, tweaking the LDFLAGS, LIBS variables before compiling, is a typical problem. Someone mentioned before that the modification here is to manually inject variable values through pkg-config. The intention of this is to solve the dependency problem of those libraries that cannot automatically use pkg-config, and reduce a lot of manual replacement of |
Workaround implemented. The problem with This is ready to be merged on my side! |
Thanks, I'll review and run it on my local machine. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It works very well, I'm glad you can bring the new SAPI in.
By the way, I found that embed can also use hardcoded ini parameters: --with-hardcoded-ini
or -I
. (method in SourcePatcher::patchHardcodedINI
)
And there are some additional changes that need to be made:
If you have no idea or have no time to do, I can handle 1 and 2 after merging this pull request. 3 is not urgent now, sooner or later I will need to write some test extensions and SAPI scripts. |
Thanks for your quick review (and for this awesome project). I'm on vacation for 10 days, but I can work on both items when I'll be back (any help is also welcome of course!).
I started something here that can probably be adapted for inclusion in this project: https://github.com/dunglas/frankenphp/pull/198/files#diff-8affca5993d9a57b7337aa153dfbae8c6b8b2ffe6c01e1954415ffdd254e2a6bR1
That would be awesome and would allow us to improve the build times of FrankenPHP. Currently, it takes 1h of machine time to build libphp for Linux and Mac, but this could be done only 1 time in this project: https://github.com/dunglas/frankenphp/actions/runs/6030196478/ What could be awesome is to be able to build automatically the latest version of libphp for popular OS and architectures each time a new version of PHP is tagged, or when a dependency has a security issue.
This is touchy because libphp is used by the embed SAPI but also by other SAPIs. Some extensions may not work for embed, but work with other SAPIs. For instance, FrankenPHP and NGINX Unit provide their own SAPI that are supported by opcache (https://github.com/php/php-src/blob/dda6b8f682e6d81dce8f6ef37dd8351c85807a5c/ext/opcache/ZendAccelerator.c#L2819-L2820), and I tested that it works with this patch. However, embed itself isn't supported by opcache (on purpose, having compiled causes no issue but it isn't used). These extensions seem to work well: https://github.com/dunglas/frankenphp/pull/198/files#diff-03075e9917c9760de11a71ccf816416084b36ee75c60f595d865da73619a3258R6 Maybe could we add some automated tests to check extension support (this could also be done for CLI, FPM etc.)? |
Yeah, that's what I thought. So far I've written several tests:
|
I'm also writing documentation in static-php-cli-docs, including contributing guide. Actually, I only wrote Actions that need manually trigger compilation. I have no plan to build it automatically, because people need different extensions, not about security reason. (but I think we can change it. After all, security issues are also very important) And for my self-hosted server artifacts, I also trigger it manually (because my server has a strict whitelist and firewall). My self-hosted server artifacts are just example files and typical out-of-the-box files, I prefer people to use static-php-cli to manually build the binaries they need. |
How can I easily check if libphp.a is working as intended? |
The easiest way is probably to create a sample program using the embed SAPI like this one: https://github.com/php/php-src/tree/master/sapi/embed#basic-example Alternatively, linking FrankenPHP with libphp.a works well too. |
666 |
During the testing process, I found that if I want to quickly obtain the compilation parameters for using libphp, it is best to use pkg-config. For libraries that will generate pc files after compilation, they can be recorded in lib.json. For libraries that do not use pkg-config, we may need to manually record the .a file and headers of the corresponding library. What do you think about add the pkgconf file of each library and writing an output function that can directly obtain the LIBS parameters of the corresponding library? |
This reverts commit 6eadbca.
@crazywhalecc using That being said, one way or another it would be great to be able to retrieve the list of libs to pass to the compiler and linker for using static-php-cli in scripts! |
I had to disable the Opcache JIT because linking breaks on Linux when enabled:
Everything is green now, and I'm able to build static builds of FrankenPHP compiled with the following extensions: Do you think we can merge this PR @crazywhalecc? We'll be able to iterate on the other topics in subsequent PRs. |
Yeah of course. I'm working on exporting the libs and cflags, in order to make a easier sanity check for |
This patch enables the creation of static
libphp
builds thanks to a new--with-embed
flag.libphp
can be used to create static programs using the embed SAPI as well as programs using other SAPIs such as FrankenPHP (the reason why I'm working on this: dunglas/frankenphp#198), NGINX Unit and various other projects.Downloads
Currently, I have not updated GitHub Actions workflows to build
libphp
, however, this could be interesting to distribute a pre-compiled version oflibphp
to speed up builds of FrankenPHP and of other projects depending on it. This will require to zip thebuildroot/
directory, or at least all the.a
libraries and the headers.This can be done in a subsequent PR if relevant.
Known issue on macOS
The static library works perfectly well on Linux, but only a limited subset of extensions are currently supported on macOS. Basically, all extensions that depend on third-party static libraries (e.g. curl or pdo_sqlite) fail to be built properly.
Building
libphp.a
works but a weird linking error occurs when linking it with the final program (in my case, FrankenPHP):This may be related to the linker used to build .
a
libraries (the GNU linker must not be used) or to the use of the-mmacosx-version-min
flag, but I didn't manage to find the exact reason yet. If anyone with Apple-specific knowledge has a better clue of what's going on, I would be happy to get some help!