123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233 |
- #include <math.h>
- #include "php.h"
- #include "php_uuencode.h"
- #define PHP_UU_ENC(c) ((c) ? ((c) & 077) + ' ' : '`')
- #define PHP_UU_ENC_C2(c) PHP_UU_ENC(((*(c) << 4) & 060) | ((*((c) + 1) >> 4) & 017))
- #define PHP_UU_ENC_C3(c) PHP_UU_ENC(((*(c + 1) << 2) & 074) | ((*((c) + 2) >> 6) & 03))
- #define PHP_UU_DEC(c) (((c) - ' ') & 077)
- PHPAPI zend_string *php_uuencode(const char *src, size_t src_len)
- {
- size_t len = 45;
- unsigned char *p;
- const unsigned char *s, *e, *ee;
- zend_string *dest;
-
- dest = zend_string_safe_alloc(src_len/2, 3, 46, 0);
- p = (unsigned char *) ZSTR_VAL(dest);
- s = (unsigned char *) src;
- e = s + src_len;
- while ((s + 3) < e) {
- ee = s + len;
- if (ee > e) {
- ee = e;
- len = ee - s;
- if (len % 3) {
- ee = s + (int) (floor((double)len / 3) * 3);
- }
- }
- *p++ = PHP_UU_ENC(len);
- while (s < ee) {
- *p++ = PHP_UU_ENC(*s >> 2);
- *p++ = PHP_UU_ENC_C2(s);
- *p++ = PHP_UU_ENC_C3(s);
- *p++ = PHP_UU_ENC(*(s + 2) & 077);
- s += 3;
- }
- if (len == 45) {
- *p++ = '\n';
- }
- }
- if (s < e) {
- if (len == 45) {
- *p++ = PHP_UU_ENC(e - s);
- len = 0;
- }
- *p++ = PHP_UU_ENC(*s >> 2);
- *p++ = PHP_UU_ENC_C2(s);
- *p++ = ((e - s) > 1) ? PHP_UU_ENC_C3(s) : PHP_UU_ENC('\0');
- *p++ = ((e - s) > 2) ? PHP_UU_ENC(*(s + 2) & 077) : PHP_UU_ENC('\0');
- }
- if (len < 45) {
- *p++ = '\n';
- }
- *p++ = PHP_UU_ENC('\0');
- *p++ = '\n';
- *p = '\0';
- dest = zend_string_truncate(dest, (char *) p - ZSTR_VAL(dest), 0);
- return dest;
- }
- PHPAPI zend_string *php_uudecode(const char *src, size_t src_len)
- {
- size_t len, total_len=0;
- char *p;
- const char *s, *e, *ee;
- zend_string *dest;
- if (src_len == 0) {
- return NULL;
- }
- dest = zend_string_alloc((size_t) ceil(src_len * 0.75), 0);
- p = ZSTR_VAL(dest);
- s = src;
- e = src + src_len;
- while (s < e) {
- if ((len = PHP_UU_DEC(*s++)) == 0) {
- break;
- }
-
- if (len > src_len) {
- goto err;
- }
- total_len += len;
- ee = s + (len == 45 ? 60 : (int) floor(len * 1.33));
-
- if (ee > e) {
- goto err;
- }
- while (s < ee) {
- if(s+4 > e) {
- goto err;
- }
- *p++ = PHP_UU_DEC(*s) << 2 | PHP_UU_DEC(*(s + 1)) >> 4;
- *p++ = PHP_UU_DEC(*(s + 1)) << 4 | PHP_UU_DEC(*(s + 2)) >> 2;
- *p++ = PHP_UU_DEC(*(s + 2)) << 6 | PHP_UU_DEC(*(s + 3));
- s += 4;
- }
- if (len < 45) {
- break;
- }
-
- s++;
- }
- assert(p >= ZSTR_VAL(dest));
- if ((len = total_len) > (size_t)(p - ZSTR_VAL(dest))) {
- *p++ = PHP_UU_DEC(*s) << 2 | PHP_UU_DEC(*(s + 1)) >> 4;
- if (len > 1) {
- *p++ = PHP_UU_DEC(*(s + 1)) << 4 | PHP_UU_DEC(*(s + 2)) >> 2;
- if (len > 2) {
- *p++ = PHP_UU_DEC(*(s + 2)) << 6 | PHP_UU_DEC(*(s + 3));
- }
- }
- }
- ZSTR_LEN(dest) = total_len;
- ZSTR_VAL(dest)[ZSTR_LEN(dest)] = '\0';
- return dest;
- err:
- zend_string_efree(dest);
- return NULL;
- }
- PHP_FUNCTION(convert_uuencode)
- {
- zend_string *src;
- ZEND_PARSE_PARAMETERS_START(1, 1)
- Z_PARAM_STR(src)
- ZEND_PARSE_PARAMETERS_END();
- RETURN_STR(php_uuencode(ZSTR_VAL(src), ZSTR_LEN(src)));
- }
- PHP_FUNCTION(convert_uudecode)
- {
- zend_string *src;
- zend_string *dest;
- ZEND_PARSE_PARAMETERS_START(1, 1)
- Z_PARAM_STR(src)
- ZEND_PARSE_PARAMETERS_END();
- if ((dest = php_uudecode(ZSTR_VAL(src), ZSTR_LEN(src))) == NULL) {
- php_error_docref(NULL, E_WARNING, "Argument #1 ($data) is not a valid uuencoded string");
- RETURN_FALSE;
- }
- RETURN_STR(dest);
- }
|