Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion .github/matrix.php
Original file line number Diff line number Diff line change
Expand Up @@ -73,7 +73,7 @@ function select_jobs($repository, $trigger, $nightly, $labels, $php_version, $re
&& ($all_jobs || !$no_jobs || $test_benchmarking)
// push trigger is restricted to official repository.
&& ($repository === 'php/php-src' || $trigger === 'pull_request')) {
$jobs['BENCHMARKING'] = true;
$jobs['BENCHMARKING']['config']['integrated_opcache'] = version_compare($php_version, '8.5', '>=');
}
if ($all_jobs || $test_community) {
$jobs['COMMUNITY']['matrix'] = version_compare($php_version, '8.4', '>=')
Expand Down
1 change: 1 addition & 0 deletions .github/workflows/test-suite.yml
Original file line number Diff line number Diff line change
Expand Up @@ -1036,6 +1036,7 @@ jobs:
sudo mkdir -p /etc/php.d
sudo chmod 777 /etc/php.d
echo mysqli.default_socket=/var/run/mysqld/mysqld.sock > /etc/php.d/mysqli.ini
${{ !fromJson(inputs.branch).jobs.BENCHMARKING.config.integrated_opcache && 'echo zend_extension=opcache.so >> /etc/php.d/opcache.ini' || '' }}
echo opcache.enable=1 >> /etc/php.d/opcache.ini
echo opcache.enable_cli=1 >> /etc/php.d/opcache.ini
- name: Setup
Expand Down
3 changes: 3 additions & 0 deletions NEWS
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,7 @@ PHP NEWS
(BogdanUngureanu)
. Fixed bug GH-20426 (Spoofchecker::setRestrictionLevel() error message
suggests missing constants). (DanielEScherzer)
. Added grapheme_strrev (Yuya Hamada)

- JSON:
. Enriched JSON last error / exception message with error location.
Expand Down Expand Up @@ -123,6 +124,8 @@ PHP NEWS
defaulted to 0. (Jorg Sowa)
. Fixed bug GH-21058 (error_log() crashes with message_type 3 and
null destination). (David Carlier)
. Fixed bug GH-13204 (glob() fails if square bracket is in current directory).
(ndossche)

- Streams:
. Added so_keepalive, tcp_keepidle, tcp_keepintvl and tcp_keepcnt stream
Expand Down
4 changes: 4 additions & 0 deletions UPGRADING
Original file line number Diff line number Diff line change
Expand Up @@ -137,6 +137,10 @@ PHP 8.6 UPGRADE NOTES
. Added ReflectionProperty::isReadable() and ReflectionProperty::isWritable().
RFC: https://wiki.php.net/rfc/isreadable-iswriteable

- Intl:
. `grapheme_strrev()` returns strrev for grapheme cluster unit.
RFC: https://wiki.php.net/rfc/grapheme_strrev

- Standard:
. `clamp()` returns the given value if in range, else return the nearest
bound.
Expand Down
59 changes: 59 additions & 0 deletions ext/intl/grapheme/grapheme_string.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1135,4 +1135,63 @@ U_CFUNC PHP_FUNCTION(grapheme_levenshtein)
efree(ustring1);
}

U_CFUNC PHP_FUNCTION(grapheme_strrev)
{
zend_string *string;
UText *ut = nullptr;
UErrorCode ustatus = U_ZERO_ERROR;
UBreakIterator *bi;
char *pstr, *end, *p;
zend_string *ret;
int32_t pos = 0, current = 0, end_len = 0;
unsigned char u_break_iterator_buffer[U_BRK_SAFECLONE_BUFFERSIZE];

ZEND_PARSE_PARAMETERS_START(1, 1)
Z_PARAM_STR(string)
ZEND_PARSE_PARAMETERS_END();

if (ZSTR_LEN(string) == 0) {
RETURN_EMPTY_STRING();
}

pstr = ZSTR_VAL(string);
ut = utext_openUTF8(ut, pstr, ZSTR_LEN(string), &ustatus);

if (U_FAILURE(ustatus)) {
intl_error_set_code(nullptr, ustatus);
intl_error_set_custom_msg(nullptr, "Error opening UTF-8 text");

RETVAL_FALSE;
goto close;
}

bi = nullptr;
ustatus = U_ZERO_ERROR;

bi = grapheme_get_break_iterator((void*)u_break_iterator_buffer, &ustatus );
ret = zend_string_alloc(ZSTR_LEN(string), 0);
p = ZSTR_VAL(ret);

ubrk_setUText(bi, ut, &ustatus);
pos = ubrk_last(bi);
if (pos == UBRK_DONE) {
goto ubrk_end;
}

current = ZSTR_LEN(string);
for (end = pstr; pos != UBRK_DONE; ) {
pos = ubrk_previous(bi);
end_len = current - pos;
for (int32_t j = 0; j < end_len; j++) {
*p++ = *(pstr + pos + j);
}
current = pos;
}
ubrk_end:
RETVAL_NEW_STR(ret);
ubrk_close(bi);
close:
utext_close(ut);
}

/* }}} */
2 changes: 2 additions & 0 deletions ext/intl/php_intl.stub.php
Original file line number Diff line number Diff line change
Expand Up @@ -445,6 +445,8 @@ function grapheme_str_split(string $string, int $length = 1): array|false {}

function grapheme_levenshtein(string $string1, string $string2, int $insertion_cost = 1, int $replacement_cost = 1, int $deletion_cost = 1, string $locale = ""): int|false {}

function grapheme_strrev(string $string): string|false {}

/** @param int $next */
function grapheme_extract(string $haystack, int $size, int $type = GRAPHEME_EXTR_COUNT, int $offset = 0, &$next = null): string|false {}

Expand Down
8 changes: 7 additions & 1 deletion ext/intl/php_intl_arginfo.h

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Binary file added ext/intl/tests/grapheme_strrev.phpt
Binary file not shown.
56 changes: 47 additions & 9 deletions ext/standard/dir.c
Original file line number Diff line number Diff line change
Expand Up @@ -400,13 +400,34 @@ PHP_FUNCTION(getcwd)
/* }}} */

/* {{{ Find pathnames matching a pattern */
#if defined(ZTS) && defined(PHP_GLOB_ALTDIRFUNC)
static void *php_glob_opendir_wrapper(const char *path)
{
return VCWD_OPENDIR(path);
}

static void php_glob_closedir_wrapper(void *dir)
{
(void) closedir(dir);
}

static int php_glob_lstat_wrapper(const char *buf, zend_stat_t *sb)
{
return VCWD_LSTAT(buf, sb);
}

static int php_glob_stat_wrapper(const char *buf, zend_stat_t *sb)
{
return VCWD_STAT(buf, sb);
}
#endif

PHP_FUNCTION(glob)
{
size_t cwd_skip = 0;
#ifdef ZTS
#if defined(ZTS) && !defined(PHP_GLOB_ALTDIRFUNC)
char cwd[MAXPATHLEN];
char work_pattern[MAXPATHLEN];
char *result;
#endif
char *pattern = NULL;
size_t pattern_len;
Expand All @@ -433,28 +454,45 @@ PHP_FUNCTION(glob)
RETURN_FALSE;
}

memset(&globbuf, 0, sizeof(globbuf));

int passed_glob_flags = flags & PHP_GLOB_FLAGMASK;

#ifdef ZTS
if (!IS_ABSOLUTE_PATH(pattern, pattern_len)) {
result = VCWD_GETCWD(cwd, MAXPATHLEN);
/* System glob uses the current work directory which is not thread safe.
* The first fix is to override the functions used to open/read/... paths
* with the VCWD ones used in PHP.
* If that functionality is unavailable for whatever reason, fall back
* to prepending the current working directory to the passed path.
* However, that comes with limitations regarding meta characters
* that is not solvable in general (GH-13204). */
# ifdef PHP_GLOB_ALTDIRFUNC
globbuf.gl_opendir = php_glob_opendir_wrapper;
globbuf.gl_readdir = (struct dirent *(*)(void *)) readdir;
globbuf.gl_closedir = php_glob_closedir_wrapper;
globbuf.gl_lstat = php_glob_lstat_wrapper;
globbuf.gl_stat = php_glob_stat_wrapper;
passed_glob_flags |= PHP_GLOB_ALTDIRFUNC;
# else
char *result = VCWD_GETCWD(cwd, MAXPATHLEN);
if (!result) {
cwd[0] = '\0';
}
#ifdef PHP_WIN32
# ifdef PHP_WIN32
if (IS_SLASH(*pattern)) {
cwd[2] = '\0';
}
#endif
# endif
cwd_skip = strlen(cwd)+1;

snprintf(work_pattern, MAXPATHLEN, "%s%c%s", cwd, DEFAULT_SLASH, pattern);
pattern = work_pattern;
# endif
}
#endif


memset(&globbuf, 0, sizeof(globbuf));
globbuf.gl_offs = 0;
if (0 != (ret = php_glob(pattern, flags & PHP_GLOB_FLAGMASK, NULL, &globbuf))) {
if (0 != (ret = php_glob(pattern, passed_glob_flags, NULL, &globbuf))) {
#ifdef PHP_GLOB_NOMATCH
if (PHP_GLOB_NOMATCH == ret) {
/* Some glob implementation simply return no data if no matches
Expand Down
12 changes: 12 additions & 0 deletions ext/standard/tests/file/gh13204.phpt
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
--TEST--
GH-13204 (glob() fails if square bracket is in current directory)
--FILE--
<?php
chdir(__DIR__ . '/gh13204[brackets]/');
var_dump(glob('./*'));
?>
--EXPECT--
array(1) {
[0]=>
string(11) "./empty.txt"
}
Empty file.