+ url->path = estrndup("/", 1);
+ }
+ } else if (*url->path != '/') {
+ if (SG(request_info).request_uri && *SG(request_info).request_uri) {
+ const char *q = strchr(SG(request_info).request_uri, '?');
+ char *uri, *path;
+ size_t len;
+
+ if (q) {
+ uri = estrndup(SG(request_info).request_uri, len = q - SG(request_info).request_uri);
+ } else {
+ uri = estrndup(SG(request_info).request_uri, len = strlen(SG(request_info).request_uri));
+ }
+
+ php_dirname(uri, len);
+ spprintf(&path, 0, "%s/%s", uri, url->path);
+ efree(uri);
+ STR_SET(url->path, path);
+ } else {
+ char *uri;
+
+ spprintf(&uri, 0, "/%s", url->path);
+ STR_SET(url->path, uri);
+ }
+ }
+ if (url->path) {
+ char *ptr, *end = url->path + strlen(url->path) + 1;
+
+ for (ptr = strstr(url->path, "/."); ptr; ptr = strstr(ptr, "/.")) {
+ switch (ptr[2])
+ {
+ case '\0':
+ ptr[1] = '\0';
+ break;
+
+ case '/':
+ memmove(&ptr[1], &ptr[3], end - &ptr[3]);
+ break;
+
+ case '.':
+ if (ptr[3] == '/') {
+ char *pos = &ptr[4];
+ while (ptr != url->path) {
+ if (*--ptr == '/') {
+ break;
+ }
+ }
+ memmove(&ptr[1], pos, end - pos);
+ }
+ break;
+ }