📄 zlib.c
字号:
RETURN_FALSE; } break; default: WRONG_PARAM_COUNT; } convert_to_string_ex(data); stream.data_type = Z_ASCII; stream.zalloc = (alloc_func) Z_NULL; stream.zfree = (free_func) Z_NULL; stream.opaque = (voidpf) Z_NULL; stream.next_in = (Bytef*) Z_STRVAL_PP(data); stream.avail_in = Z_STRLEN_PP(data); stream.avail_out = stream.avail_in + (stream.avail_in/1000) + 15 + 1; /* room for \0 */ s2 = (char *) emalloc(stream.avail_out); if(!s2) RETURN_FALSE; stream.next_out = s2; /* init with -MAX_WBITS disables the zlib internal headers */ status = deflateInit2(&stream, level, Z_DEFLATED, -MAX_WBITS, MAX_MEM_LEVEL, 0); if (status == Z_OK) { status = deflate(&stream, Z_FINISH); if (status != Z_STREAM_END) { deflateEnd(&stream); if (status == Z_OK) { status = Z_BUF_ERROR; } } else { status = deflateEnd(&stream); } } if (status==Z_OK) { s2 = erealloc(s2,stream.total_out+1); /* resize to buffer to the "right" size */ s2[ stream.total_out ] = '\0'; RETURN_STRINGL(s2, stream.total_out, 0); } else { efree(s2); php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s", zError(status)); RETURN_FALSE; }}/* }}} *//* {{{ proto string gzinflate(string data [, int length]) Unzip a gzip-compressed string */PHP_FUNCTION(gzinflate){ zval **data, **zlimit = NULL; int status,factor=1,maxfactor=16; unsigned long plength=0,length; char *s1=NULL,*s2=NULL; z_stream stream; switch (ZEND_NUM_ARGS()) { case 1: if (zend_get_parameters_ex(1, &data) == FAILURE) WRONG_PARAM_COUNT; length=0; break; case 2: if (zend_get_parameters_ex(2, &data, &zlimit) == FAILURE) WRONG_PARAM_COUNT; convert_to_long_ex(zlimit); if(Z_LVAL_PP(zlimit)<=0) { php_error_docref(NULL TSRMLS_CC, E_WARNING, "length must be greater zero"); RETURN_FALSE; } plength = Z_LVAL_PP(zlimit); break; default: WRONG_PARAM_COUNT; } convert_to_string_ex(data); /* stream.avail_out wants to know the output data length if none was given as a parameter we try from input length * 2 up to input length * 2^16 doubling it whenever it wasn't big enough that should be enaugh for all real life cases */ stream.zalloc = (alloc_func) Z_NULL; stream.zfree = (free_func) Z_NULL; do { length=plength?plength:Z_STRLEN_PP(data)*(1<<factor++); s2 = (char *) erealloc(s1,length); if(! s2) { if(s1) efree(s1); RETURN_FALSE; } stream.next_in = (Bytef*) Z_STRVAL_PP(data); stream.avail_in = (uInt) Z_STRLEN_PP(data) + 1; /* there is room for \0 */ stream.next_out = s2; stream.avail_out = (uInt) length; /* init with -MAX_WBITS disables the zlib internal headers */ status = inflateInit2(&stream, -MAX_WBITS); if (status == Z_OK) { status = inflate(&stream, Z_FINISH); if (status != Z_STREAM_END) { inflateEnd(&stream); if (status == Z_OK) { status = Z_BUF_ERROR; } } else { status = inflateEnd(&stream); } } s1=s2; } while((status==Z_BUF_ERROR)&&(!plength)&&(factor<maxfactor)); if(status==Z_OK) { s2 = erealloc(s2, stream.total_out + 1); /* room for \0 */ s2[ stream.total_out ] = '\0'; RETURN_STRINGL(s2, stream.total_out, 0); } else { efree(s2); php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s", zError(status)); RETURN_FALSE; }}/* }}} *//*`{{{ proto zlib_get_coding_type() Returns the coding type used for output compression */PHP_FUNCTION(zlib_get_coding_type){ switch (ZLIBG(ob_gzip_coding)) { case CODING_GZIP: RETURN_STRINGL("gzip", sizeof("gzip") - 1, 1); case CODING_DEFLATE: RETURN_STRINGL("deflate", sizeof("deflate") - 1, 1); } RETURN_FALSE;}/* {{{ php_do_deflate */static int php_do_deflate(uint str_length, Bytef **p_buffer, uint *p_buffer_len, zend_bool do_start, zend_bool do_end TSRMLS_DC){ Bytef *buffer; uInt prev_outlen, outlen; int err; int start_offset = ((do_start && ZLIBG(compression_coding) == CODING_GZIP) ? 10 : 0); int end_offset = (do_end?8:0); outlen = (uint)(sizeof(char) * (str_length * 1.001f + 12) + 1); /* leave some room for a trailing \0 */ if ((outlen+start_offset+end_offset) > *p_buffer_len) { buffer = (Bytef *) emalloc(outlen+start_offset+end_offset); } else { buffer = *p_buffer; } ZLIBG(stream).next_out = buffer+start_offset; ZLIBG(stream).avail_out = outlen; err = deflate(&ZLIBG(stream), Z_SYNC_FLUSH); while (err == Z_OK && !ZLIBG(stream).avail_out) { prev_outlen = outlen; outlen *= 3; if ((outlen+start_offset+end_offset) > *p_buffer_len) { buffer = erealloc(buffer, outlen+start_offset+end_offset); } ZLIBG(stream).next_out = buffer+start_offset + prev_outlen; ZLIBG(stream).avail_out = prev_outlen * 2; err = deflate(&ZLIBG(stream), Z_SYNC_FLUSH); } if (do_end) { err = deflate(&ZLIBG(stream), Z_FINISH); buffer[outlen + start_offset - ZLIBG(stream).avail_out] = '\0'; } *p_buffer = buffer; *p_buffer_len = outlen - ZLIBG(stream).avail_out; return err;}/* }}} *//* {{{ php_deflate_string */int php_deflate_string(const char *str, uint str_length, char **newstr, uint *new_length, int coding, zend_bool do_start, zend_bool do_end, int compression_level TSRMLS_DC){ int err; ZLIBG(compression_coding) = coding; if (do_start) { ZLIBG(stream).zalloc = Z_NULL; ZLIBG(stream).zfree = Z_NULL; ZLIBG(stream).opaque = Z_NULL; switch (coding) { case CODING_GZIP: /* windowBits is passed < 0 to suppress zlib header & trailer */ if (deflateInit2(&ZLIBG(stream), compression_level, Z_DEFLATED, -MAX_WBITS, MAX_MEM_LEVEL, Z_DEFAULT_STRATEGY) != Z_OK) { /* TODO: print out error */ return FAILURE; } ZLIBG(crc) = crc32(0L, Z_NULL, 0); break; case CODING_DEFLATE: if (deflateInit(&ZLIBG(stream), compression_level) != Z_OK) { /* TODO: print out error */ return FAILURE; } break; } } ZLIBG(stream).next_in = (Bytef*) str; ZLIBG(stream).avail_in = (uInt) str_length; if (ZLIBG(compression_coding) == CODING_GZIP) { ZLIBG(crc) = crc32(ZLIBG(crc), (const Bytef *) str, str_length); } err = php_do_deflate(str_length, (Bytef **) newstr, new_length, do_start, do_end TSRMLS_CC); /* TODO: error handling (err may be Z_STREAM_ERROR, Z_BUF_ERROR, ?) */ if (do_start && ZLIBG(compression_coding) == CODING_GZIP) { /* Write a very simple .gz header: */ (*newstr)[0] = gz_magic[0]; (*newstr)[1] = gz_magic[1]; (*newstr)[2] = Z_DEFLATED; (*newstr)[3] = (*newstr)[4] = (*newstr)[5] = (*newstr)[6] = (*newstr)[7] = (*newstr)[8] = 0; (*newstr)[9] = OS_CODE; *new_length += 10; } if (do_end) { if (ZLIBG(compression_coding) == CODING_GZIP) { char *trailer = (*newstr)+(*new_length); /* write crc & stream.total_in in LSB order */ trailer[0] = (char) ZLIBG(crc) & 0xFF; trailer[1] = (char) (ZLIBG(crc) >> 8) & 0xFF; trailer[2] = (char) (ZLIBG(crc) >> 16) & 0xFF; trailer[3] = (char) (ZLIBG(crc) >> 24) & 0xFF; trailer[4] = (char) ZLIBG(stream).total_in & 0xFF; trailer[5] = (char) (ZLIBG(stream).total_in >> 8) & 0xFF; trailer[6] = (char) (ZLIBG(stream).total_in >> 16) & 0xFF; trailer[7] = (char) (ZLIBG(stream).total_in >> 24) & 0xFF; trailer[8] = '\0'; *new_length += 8; } deflateEnd(&ZLIBG(stream)); } return SUCCESS;}/* }}} *//* {{{ proto string gzencode(string data [, int level [, int encoding_mode]]) GZ encode a string */PHP_FUNCTION(gzencode){ char *data, *s2; int data_len; long level = Z_DEFAULT_COMPRESSION, coding = CODING_GZIP; int status; z_stream stream; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s|ll", &data, &data_len, &level, &coding) == FAILURE) { return; } if((level<-1)||(level>9)) { php_error_docref(NULL TSRMLS_CC, E_WARNING, "compression level(%ld) must be within -1..9", level); RETURN_FALSE; } if((coding!=CODING_GZIP)&&(coding!=CODING_DEFLATE)) { php_error_docref(NULL TSRMLS_CC, E_WARNING, "encoding mode must be FORCE_GZIP or FORCE_DEFLATE"); RETURN_FALSE; } stream.zalloc = Z_NULL; stream.zfree = Z_NULL; stream.opaque = Z_NULL; stream.next_in = (Bytef*) data; stream.avail_in = data_len; stream.avail_out = stream.avail_in + (stream.avail_in/1000) + 15 + 1; /* room for \0 */ s2 = (char *) emalloc(stream.avail_out+GZIP_HEADER_LENGTH+(coding==CODING_GZIP?GZIP_FOOTER_LENGTH:0)); if(!s2) RETURN_FALSE; /* add gzip file header */ s2[0] = gz_magic[0]; s2[1] = gz_magic[1]; s2[2] = Z_DEFLATED; s2[3] = s2[4] = s2[5] = s2[6] = s2[7] = s2[8] = 0; /* time set to 0 */ s2[9] = OS_CODE; stream.next_out = &(s2[GZIP_HEADER_LENGTH]); switch (coding) { case CODING_GZIP: /* windowBits is passed < 0 to suppress zlib header & trailer */ if ((status = deflateInit2(&stream, level, Z_DEFLATED, -MAX_WBITS, MAX_MEM_LEVEL, Z_DEFAULT_STRATEGY)) != Z_OK) { php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s", zError(status)); RETURN_FALSE; } break; case CODING_DEFLATE: if ((status = deflateInit(&stream, level)) != Z_OK) { php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s", zError(status)); RETURN_FALSE; } break; } status = deflate(&stream, Z_FINISH); if (status != Z_STREAM_END) { deflateEnd(&stream); if (status == Z_OK) { status = Z_BUF_ERROR; } } else { status = deflateEnd(&stream); } if (status==Z_OK) { s2 = erealloc(s2,stream.total_out+GZIP_HEADER_LENGTH+(coding==CODING_GZIP?GZIP_FOOTER_LENGTH:0)+1); /* resize to buffer to the "right" size */ if (coding == CODING_GZIP) { char *trailer = s2+(stream.total_out+GZIP_HEADER_LENGTH); uLong crc = crc32(0L, Z_NULL, 0); crc = crc32(crc, (const Bytef *) data, data_len); /* write crc & stream.total_in in LSB order */ trailer[0] = (char) crc & 0xFF; trailer[1] = (char) (crc >> 8) & 0xFF; trailer[2] = (char) (crc >> 16) & 0xFF; trailer[3] = (char) (crc >> 24) & 0xFF; trailer[4] = (char) stream.total_in & 0xFF; trailer[5] = (char) (stream.total_in >> 8) & 0xFF; trailer[6] = (char) (stream.total_in >> 16) & 0xFF; trailer[7] = (char) (stream.total_in >> 24) & 0xFF; trailer[8] = '\0'; } RETURN_STRINGL(s2, stream.total_out+GZIP_HEADER_LENGTH+(coding==CODING_GZIP?GZIP_FOOTER_LENGTH:0), 0); } else { efree(s2); php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s", zError(status)); RETURN_FALSE; }}/* }}} *//* {{{ php_ob_gzhandler_check */int php_ob_gzhandler_check(TSRMLS_D){ /* check for wrong usages */ if (OG(ob_nesting_level>0)) { if (php_ob_handler_used("ob_gzhandler" TSRMLS_CC)) { php_error_docref("ref.outcontrol" TSRMLS_CC, E_WARNING, "output handler 'ob_gzhandler' cannot be used twice"); return FAILURE; } if (php_ob_handler_used("mb_output_handler" TSRMLS_CC)) { php_error_docref("ref.outcontrol" TSRMLS_CC, E_WARNING, "output handler 'ob_gzhandler' cannot be used after 'mb_output_handler'"); return FAILURE; } if (php_ob_handler_used("URL-Rewriter" TSRMLS_CC)) { php_error_docref("ref.outcontrol" TSRMLS_CC, E_WARNING, "output handler 'ob_gzhandler' cannot be used after 'URL-Rewriter'"); return FAILURE; } if (php_ob_init_conflict("ob_gzhandler", "zlib output compression" TSRMLS_CC)) { return FAILURE; } } return SUCCESS;}/* }}} *//* {{{ proto string ob_gzhandler(string str, int mode) Encode str based on accept-encoding setting - designed to be called from ob_start() */PHP_FUNCTION(ob_gzhandler){ int coding; zval **zv_string, **zv_mode; zval **data, **a_encoding; zend_bool return_original=0; zend_bool do_start, do_end; if (ZEND_NUM_ARGS()!=2 || zend_get_parameters_ex(2, &zv_string, &zv_mode)==FAILURE) { ZEND_WRONG_PARAM_COUNT(); } if (ZLIBG(ob_gzhandler_status)==-1 || zend_hash_find(&EG(symbol_table), "HTTP_SERVER_VARS", sizeof("HTTP_SERVER_VARS"), (void **) &data)==FAILURE || Z_TYPE_PP(data)!=IS_ARRAY || zend_hash_find(Z_ARRVAL_PP(data), "HTTP_ACCEPT_ENCODING", sizeof("HTTP_ACCEPT_ENCODING"), (void **) &a_encoding)==FAILURE) { ZLIBG(ob_gzhandler_status)=-1; RETURN_FALSE; } convert_to_string_ex(a_encoding); if (php_memnstr(Z_STRVAL_PP(a_encoding), "gzip", 4, Z_STRVAL_PP(a_encoding) + Z_STRLEN_PP(a_encoding))) { coding = CODING_GZIP; } else if(php_memnstr(Z_STRVAL_PP(a_encoding), "deflate", 7, Z_STRVAL_PP(a_encoding) + Z_STRLEN_PP(a_encoding))) { coding = CODING_DEFLATE; } else { ZLIBG(ob_gzhandler_status)=-1; RETURN_FALSE; } convert_to_long_ex(zv_mode); do_start = ((Z_LVAL_PP(zv_mode) & PHP_OUTPUT_HANDLER_START) ? 1 : 0); do_end = ((Z_LVAL_PP(zv_mode) & PHP_OUTPUT_HANDLER_END) ? 1 : 0); Z_STRVAL_P(return_value) = NULL; Z_STRLEN_P(return_value) = 0; if (php_deflate_string(Z_STRVAL_PP(zv_string), Z_STRLEN_PP(zv_string), &Z_STRVAL_P(return_value), &Z_STRLEN_P(return_value), coding, do_start, do_end, ZLIBG(output_compression_level) TSRMLS_CC)==SUCCESS) { Z_TYPE_P(return_value) = IS_STRING; if (do_start) { switch (coding) { case CODING_GZIP: if (sapi_add_header("Content-Encoding: gzip", sizeof("Content-Encoding: gzip") - 1, 1)==FAILURE) { return_original = 1; } if (sapi_add_header_ex("Vary: Accept-Encoding", sizeof("Vary: Accept-Encoding") - 1, 1, 0 TSRMLS_CC)==FAILURE) { return_original = 1; } break; case CODING_DEFLATE: if (sapi_add_header("Content-Encoding: deflate", sizeof("Content-Encoding: deflate") - 1, 1)==FAILURE) { return_original = 1; } if (sapi_add_header_ex("Vary: Accept-Encoding", sizeof("Vary: Accept-Encoding") - 1, 1, 0 TSRMLS_CC)==FAILURE) { return_original = 1; } break; default: return_original = 1; break; } } if (return_original) { zval_dtor(return_value); } } else { return_original = 1; } if (return_original) { /* return the original string */ *return_value = **zv_string; zval_copy_ctor(return_value); }}/* }}} *//* {{{ php_gzip_output_handler */static void php_gzip_output_handler(char *output, uint output_len, char **handled_output, uint *handled_output_len, int mode TSRMLS_DC){ zend_bool do_start, do_end; if (!ZLIBG(output_compression)) { *handled_output = NULL; } else { do_start = (mode & PHP_OUTPUT_HANDLER_START ? 1 : 0); do_end = (mode & PHP_OUTPUT_HANDLER_END ? 1 : 0); if (php_deflate_string(output, output_len, handled_output, handled_output_len, ZLIBG(ob_gzip_coding), do_start, do_end, ZLIBG(output_compression_level) TSRMLS_CC)!=SUCCESS) { zend_error(E_ERROR, "Compression failed"); } }}/* }}} *//* {{{ php_enable_output_compression */int php_enable_output_compression(int buffer_size TSRMLS_DC){ zval **a_encoding, **data; if (zend_hash_find(&EG(symbol_table), "HTTP_SERVER_VARS", sizeof("HTTP_SERVER_VARS"), (void **) &data)==FAILURE || Z_TYPE_PP(data)!=IS_ARRAY || zend_hash_find(Z_ARRVAL_PP(data), "HTTP_ACCEPT_ENCODING", sizeof("HTTP_ACCEPT_ENCODING"), (void **) &a_encoding)==FAILURE) { return FAILURE; } convert_to_string_ex(a_encoding); if (php_memnstr(Z_STRVAL_PP(a_encoding), "gzip", 4, Z_STRVAL_PP(a_encoding) + Z_STRLEN_PP(a_encoding))) { ZLIBG(ob_gzip_coding) = CODING_GZIP; } else if(php_memnstr(Z_STRVAL_PP(a_encoding), "deflate", 7, Z_STRVAL_PP(a_encoding) + Z_STRLEN_PP(a_encoding))) { ZLIBG(ob_gzip_coding) = CODING_DEFLATE; } else { return FAILURE; } php_ob_set_internal_handler(php_gzip_output_handler, (uint)buffer_size, "zlib output compression", 0 TSRMLS_CC); if (ZLIBG(output_handler) && strlen(ZLIBG(output_handler))) { php_start_ob_buffer_named(ZLIBG(output_handler), 0, 1 TSRMLS_CC); } return SUCCESS;}/* }}} *//* * Local variables: * tab-width: 4 * c-basic-offset: 4 * End: * vim600: sw=4 ts=4 fdm=marker * vim<600: sw=4 ts=4 */
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -