PHP  
 PHP: Test and Code Coverage Analysis
downloads | QA | documentation | faq | getting help | mailing lists | reporting bugs | php.net sites | links | my php.net 
 

LTP GCOV extension - code coverage report
Current view: directory - gmp - gmp.c
Test: PHP Code Coverage
Date: 2009-11-19 Instrumented lines: 505
Code covered: 96.0 % Executed lines: 485
Legend: not executed executed

       1                 : /*
       2                 :    +----------------------------------------------------------------------+
       3                 :    | PHP Version 5                                                        |
       4                 :    +----------------------------------------------------------------------+
       5                 :    | Copyright (c) 1997-2009 The PHP Group                                |
       6                 :    +----------------------------------------------------------------------+
       7                 :    | This source file is subject to version 3.01 of the PHP license,      |
       8                 :    | that is bundled with this package in the file LICENSE, and is        |
       9                 :    | available through the world-wide-web at the following url:           |
      10                 :    | http://www.php.net/license/3_01.txt                                  |
      11                 :    | If you did not receive a copy of the PHP license and are unable to   |
      12                 :    | obtain it through the world-wide-web, please send a note to          |
      13                 :    | license@php.net so we can mail you a copy immediately.               |
      14                 :    +----------------------------------------------------------------------+
      15                 :    | Author: Stanislav Malyshev <stas@php.net>                            |
      16                 :    +----------------------------------------------------------------------+
      17                 :  */
      18                 : 
      19                 : #ifdef HAVE_CONFIG_H
      20                 : #include "config.h"
      21                 : #endif
      22                 : 
      23                 : #include "php.h"
      24                 : #include "php_ini.h"
      25                 : #include "php_gmp.h"
      26                 : #include "ext/standard/info.h"
      27                 : 
      28                 : #if HAVE_GMP
      29                 : 
      30                 : #include <gmp.h>
      31                 : 
      32                 : /* Needed for gmp_random() */
      33                 : #include "ext/standard/php_rand.h"
      34                 : #include "ext/standard/php_lcg.h"
      35                 : #define GMP_ABS(x) ((x) >= 0 ? (x) : -(x))
      36                 : 
      37                 : /* True global resources - no need for thread safety here */
      38                 : static int le_gmp;
      39                 : 
      40                 : /* {{{ arginfo */
      41                 : static
      42                 : ZEND_BEGIN_ARG_INFO_EX(arginfo_gmp_init, 0, 0, 1)
      43                 :         ZEND_ARG_INFO(0, number)
      44                 :         ZEND_ARG_INFO(0, base)
      45                 : ZEND_END_ARG_INFO()
      46                 : 
      47                 : static
      48                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_intval, 0)
      49                 :         ZEND_ARG_INFO(0, gmpnumber)
      50                 : ZEND_END_ARG_INFO()
      51                 : 
      52                 : static
      53                 : ZEND_BEGIN_ARG_INFO_EX(arginfo_gmp_strval, 0, 0, 1)
      54                 :         ZEND_ARG_INFO(0, gmpnumber)
      55                 :         ZEND_ARG_INFO(0, base)
      56                 : ZEND_END_ARG_INFO()
      57                 : 
      58                 : static
      59                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_add, 0)
      60                 :         ZEND_ARG_INFO(0, a)
      61                 :         ZEND_ARG_INFO(0, b)
      62                 : ZEND_END_ARG_INFO()
      63                 : 
      64                 : static
      65                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_sub, 0)
      66                 :         ZEND_ARG_INFO(0, a)
      67                 :         ZEND_ARG_INFO(0, b)
      68                 : ZEND_END_ARG_INFO()
      69                 : 
      70                 : static
      71                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_mul, 0)
      72                 :         ZEND_ARG_INFO(0, a)
      73                 :         ZEND_ARG_INFO(0, b)
      74                 : ZEND_END_ARG_INFO()
      75                 : 
      76                 : static
      77                 : ZEND_BEGIN_ARG_INFO_EX(arginfo_gmp_div_qr, 0, 0, 2)
      78                 :         ZEND_ARG_INFO(0, a)
      79                 :         ZEND_ARG_INFO(0, b)
      80                 :         ZEND_ARG_INFO(0, round)
      81                 : ZEND_END_ARG_INFO()
      82                 : 
      83                 : static
      84                 : ZEND_BEGIN_ARG_INFO_EX(arginfo_gmp_div_r, 0, 0, 2)
      85                 :         ZEND_ARG_INFO(0, a)
      86                 :         ZEND_ARG_INFO(0, b)
      87                 :         ZEND_ARG_INFO(0, round)
      88                 : ZEND_END_ARG_INFO()
      89                 : 
      90                 : static
      91                 : ZEND_BEGIN_ARG_INFO_EX(arginfo_gmp_div_q, 0, 0, 2)
      92                 :         ZEND_ARG_INFO(0, a)
      93                 :         ZEND_ARG_INFO(0, b)
      94                 :         ZEND_ARG_INFO(0, round)
      95                 : ZEND_END_ARG_INFO()
      96                 : 
      97                 : static
      98                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_mod, 0)
      99                 :         ZEND_ARG_INFO(0, a)
     100                 :         ZEND_ARG_INFO(0, b)
     101                 : ZEND_END_ARG_INFO()
     102                 : 
     103                 : static
     104                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_divexact, 0)
     105                 :         ZEND_ARG_INFO(0, a)
     106                 :         ZEND_ARG_INFO(0, b)
     107                 : ZEND_END_ARG_INFO()
     108                 : 
     109                 : static
     110                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_neg, 0)
     111                 :         ZEND_ARG_INFO(0, a)
     112                 : ZEND_END_ARG_INFO()
     113                 : 
     114                 : static
     115                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_abs, 0)
     116                 :         ZEND_ARG_INFO(0, a)
     117                 : ZEND_END_ARG_INFO()
     118                 : 
     119                 : static
     120                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_fact, 0)
     121                 :         ZEND_ARG_INFO(0, a)
     122                 : ZEND_END_ARG_INFO()
     123                 : 
     124                 : static
     125                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_pow, 0)
     126                 :         ZEND_ARG_INFO(0, base)
     127                 :         ZEND_ARG_INFO(0, exp)
     128                 : ZEND_END_ARG_INFO()
     129                 : 
     130                 : static
     131                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_powm, 0)
     132                 :         ZEND_ARG_INFO(0, base)
     133                 :         ZEND_ARG_INFO(0, exp)
     134                 :         ZEND_ARG_INFO(0, mod)
     135                 : ZEND_END_ARG_INFO()
     136                 : 
     137                 : static
     138                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_sqrt, 0)
     139                 :         ZEND_ARG_INFO(0, a)
     140                 : ZEND_END_ARG_INFO()
     141                 : 
     142                 : static
     143                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_sqrtrem, 0)
     144                 :         ZEND_ARG_INFO(0, a)
     145                 : ZEND_END_ARG_INFO()
     146                 : 
     147                 : static
     148                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_perfect_square, 0)
     149                 :         ZEND_ARG_INFO(0, a)
     150                 : ZEND_END_ARG_INFO()
     151                 : 
     152                 : static
     153                 : ZEND_BEGIN_ARG_INFO_EX(arginfo_gmp_prob_prime, 0, 0, 1)
     154                 :         ZEND_ARG_INFO(0, a)
     155                 :         ZEND_ARG_INFO(0, reps)
     156                 : ZEND_END_ARG_INFO()
     157                 : 
     158                 : static
     159                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_gcd, 0)
     160                 :         ZEND_ARG_INFO(0, a)
     161                 :         ZEND_ARG_INFO(0, b)
     162                 : ZEND_END_ARG_INFO()
     163                 : 
     164                 : static
     165                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_gcdext, 0)
     166                 :         ZEND_ARG_INFO(0, a)
     167                 :         ZEND_ARG_INFO(0, b)
     168                 : ZEND_END_ARG_INFO()
     169                 : 
     170                 : static
     171                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_invert, 0)
     172                 :         ZEND_ARG_INFO(0, a)
     173                 :         ZEND_ARG_INFO(0, b)
     174                 : ZEND_END_ARG_INFO()
     175                 : 
     176                 : static
     177                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_jacobi, 0)
     178                 :         ZEND_ARG_INFO(0, a)
     179                 :         ZEND_ARG_INFO(0, b)
     180                 : ZEND_END_ARG_INFO()
     181                 : 
     182                 : static
     183                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_legendre, 0)
     184                 :         ZEND_ARG_INFO(0, a)
     185                 :         ZEND_ARG_INFO(0, b)
     186                 : ZEND_END_ARG_INFO()
     187                 : 
     188                 : static
     189                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_cmp, 0)
     190                 :         ZEND_ARG_INFO(0, a)
     191                 :         ZEND_ARG_INFO(0, b)
     192                 : ZEND_END_ARG_INFO()
     193                 : 
     194                 : static
     195                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_sign, 0)
     196                 :         ZEND_ARG_INFO(0, a)
     197                 : ZEND_END_ARG_INFO()
     198                 : 
     199                 : static
     200                 : ZEND_BEGIN_ARG_INFO_EX(arginfo_gmp_random, 0, 0, 0)
     201                 :         ZEND_ARG_INFO(0, limiter)
     202                 : ZEND_END_ARG_INFO()
     203                 : 
     204                 : static
     205                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_and, 0)
     206                 :         ZEND_ARG_INFO(0, a)
     207                 :         ZEND_ARG_INFO(0, b)
     208                 : ZEND_END_ARG_INFO()
     209                 : 
     210                 : static
     211                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_or, 0)
     212                 :         ZEND_ARG_INFO(0, a)
     213                 :         ZEND_ARG_INFO(0, b)
     214                 : ZEND_END_ARG_INFO()
     215                 : 
     216                 : static
     217                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_com, 0)
     218                 :         ZEND_ARG_INFO(0, a)
     219                 : ZEND_END_ARG_INFO()
     220                 : 
     221                 : static
     222                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_xor, 0)
     223                 :         ZEND_ARG_INFO(0, a)
     224                 :         ZEND_ARG_INFO(0, b)
     225                 : ZEND_END_ARG_INFO()
     226                 : 
     227                 : static
     228                 : ZEND_BEGIN_ARG_INFO_EX(arginfo_gmp_setbit, 0, 0, 2)
     229                 :         ZEND_ARG_INFO(1, a)
     230                 :         ZEND_ARG_INFO(0, index)
     231                 :         ZEND_ARG_INFO(0, set_clear)
     232                 : ZEND_END_ARG_INFO()
     233                 : 
     234                 : static
     235                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_clrbit, 0)
     236                 :         ZEND_ARG_INFO(1, a)
     237                 :         ZEND_ARG_INFO(0, index)
     238                 : ZEND_END_ARG_INFO()
     239                 : 
     240                 : static
     241                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_popcount, 0)
     242                 :         ZEND_ARG_INFO(0, a)
     243                 : ZEND_END_ARG_INFO()
     244                 : 
     245                 : static
     246                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_hamdist, 0)
     247                 :         ZEND_ARG_INFO(0, a)
     248                 :         ZEND_ARG_INFO(0, b)
     249                 : ZEND_END_ARG_INFO()
     250                 : 
     251                 : static
     252                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_scan0, 0)
     253                 :         ZEND_ARG_INFO(0, a)
     254                 :         ZEND_ARG_INFO(0, start)
     255                 : ZEND_END_ARG_INFO()
     256                 : 
     257                 : static
     258                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_scan1, 0)
     259                 :         ZEND_ARG_INFO(0, a)
     260                 :         ZEND_ARG_INFO(0, start)
     261                 : ZEND_END_ARG_INFO()
     262                 : 
     263                 : static
     264                 : ZEND_BEGIN_ARG_INFO(arginfo_gmp_nextprime, 0)
     265                 :         ZEND_ARG_INFO(0, a)
     266                 : ZEND_END_ARG_INFO()
     267                 : 
     268                 : /* }}} */
     269                 : 
     270                 : ZEND_DECLARE_MODULE_GLOBALS(gmp)
     271                 : static ZEND_GINIT_FUNCTION(gmp);
     272                 : 
     273                 : /* {{{ gmp_functions[]
     274                 :  */
     275                 : zend_function_entry gmp_functions[] = {
     276                 :         ZEND_FE(gmp_init,       arginfo_gmp_init)
     277                 :         ZEND_FE(gmp_intval,     arginfo_gmp_intval)
     278                 :         ZEND_FE(gmp_strval,     arginfo_gmp_strval)
     279                 :         ZEND_FE(gmp_add,            arginfo_gmp_add)
     280                 :         ZEND_FE(gmp_sub,            arginfo_gmp_sub)
     281                 :         ZEND_FE(gmp_mul,            arginfo_gmp_mul)
     282                 :         ZEND_FE(gmp_div_qr,     arginfo_gmp_div_qr)
     283                 :         ZEND_FE(gmp_div_q,      arginfo_gmp_div_q)
     284                 :         ZEND_FE(gmp_div_r,      arginfo_gmp_div_r)
     285                 :         ZEND_FALIAS(gmp_div,    gmp_div_q, arginfo_gmp_div_q)
     286                 :         ZEND_FE(gmp_mod,        arginfo_gmp_mod)
     287                 :         ZEND_FE(gmp_divexact,   arginfo_gmp_divexact)
     288                 :         ZEND_FE(gmp_neg,        arginfo_gmp_neg)
     289                 :         ZEND_FE(gmp_abs,        arginfo_gmp_abs)
     290                 :         ZEND_FE(gmp_fact,       arginfo_gmp_fact)
     291                 :         ZEND_FE(gmp_sqrt,       arginfo_gmp_sqrt)
     292                 :         ZEND_FE(gmp_sqrtrem,    arginfo_gmp_sqrtrem)
     293                 :         ZEND_FE(gmp_pow,        arginfo_gmp_pow)
     294                 :         ZEND_FE(gmp_powm,       arginfo_gmp_powm)
     295                 :         ZEND_FE(gmp_perfect_square,     arginfo_gmp_perfect_square)
     296                 :         ZEND_FE(gmp_prob_prime, arginfo_gmp_prob_prime)
     297                 :         ZEND_FE(gmp_gcd,        arginfo_gmp_gcd)
     298                 :         ZEND_FE(gmp_gcdext,     arginfo_gmp_gcdext)
     299                 :         ZEND_FE(gmp_invert,     arginfo_gmp_invert)
     300                 :         ZEND_FE(gmp_jacobi,     arginfo_gmp_jacobi)
     301                 :         ZEND_FE(gmp_legendre,   arginfo_gmp_legendre)
     302                 :         ZEND_FE(gmp_cmp,        arginfo_gmp_cmp)
     303                 :         ZEND_FE(gmp_sign,       arginfo_gmp_sign)
     304                 :         ZEND_FE(gmp_random,     arginfo_gmp_random)
     305                 :         ZEND_FE(gmp_and,        arginfo_gmp_and)
     306                 :         ZEND_FE(gmp_or, arginfo_gmp_or)
     307                 :         ZEND_FE(gmp_com,        arginfo_gmp_com)
     308                 :         ZEND_FE(gmp_xor,        arginfo_gmp_xor)
     309                 :         ZEND_FE(gmp_setbit,     arginfo_gmp_setbit)
     310                 :         ZEND_FE(gmp_clrbit,     arginfo_gmp_clrbit)
     311                 :         ZEND_FE(gmp_scan0, arginfo_gmp_scan0)
     312                 :         ZEND_FE(gmp_scan1, arginfo_gmp_scan1)
     313                 :         ZEND_FE(gmp_popcount, arginfo_gmp_popcount)
     314                 :         ZEND_FE(gmp_hamdist, arginfo_gmp_hamdist)
     315                 :         ZEND_FE(gmp_nextprime, arginfo_gmp_nextprime)
     316                 :         {NULL, NULL, NULL}      /* Must be the last line in gmp_functions[] */
     317                 : };
     318                 : /* }}} */
     319                 : 
     320                 : /* {{{ gmp_module_entry
     321                 :  */
     322                 : zend_module_entry gmp_module_entry = {
     323                 :         STANDARD_MODULE_HEADER,
     324                 :         "gmp",
     325                 :         gmp_functions,
     326                 :         ZEND_MODULE_STARTUP_N(gmp),
     327                 :         NULL,
     328                 :         NULL,
     329                 :         ZEND_MODULE_DEACTIVATE_N(gmp),
     330                 :         ZEND_MODULE_INFO_N(gmp),
     331                 :         NO_VERSION_YET,
     332                 :         ZEND_MODULE_GLOBALS(gmp),
     333                 :         ZEND_GINIT(gmp),
     334                 :         NULL,
     335                 :         NULL,
     336                 :         STANDARD_MODULE_PROPERTIES_EX
     337                 : };
     338                 : /* }}} */
     339                 : 
     340                 : #ifdef COMPILE_DL_GMP
     341                 : ZEND_GET_MODULE(gmp)
     342                 : #endif
     343                 : 
     344                 : static void _php_gmpnum_free(zend_rsrc_list_entry *rsrc TSRMLS_DC);
     345                 : 
     346                 : #define GMP_RESOURCE_NAME "GMP integer"
     347                 : 
     348                 : #define GMP_ROUND_ZERO      0
     349                 : #define GMP_ROUND_PLUSINF   1
     350                 : #define GMP_ROUND_MINUSINF  2
     351                 : 
     352                 : /* {{{ gmp_emalloc
     353                 :  */
     354                 : static void *gmp_emalloc(size_t size)
     355            2754 : {
     356            2754 :         return emalloc(size);
     357                 : }
     358                 : /* }}} */
     359                 : 
     360                 : /* {{{ gmp_erealloc
     361                 :  */
     362                 : static void *gmp_erealloc(void *ptr, size_t old_size, size_t new_size)
     363            1215 : {
     364            1215 :         return erealloc(ptr, new_size);
     365                 : }
     366                 : /* }}} */
     367                 : 
     368                 : /* {{{ gmp_efree
     369                 :  */
     370                 : static void gmp_efree(void *ptr, size_t size)
     371            2754 : {
     372            2754 :         efree(ptr);
     373            2754 : }
     374                 : /* }}} */
     375                 : 
     376                 : /* {{{ ZEND_GINIT_FUNCTION
     377                 :  */
     378                 : static ZEND_GINIT_FUNCTION(gmp)
     379           13565 : {
     380           13565 :         gmp_globals->rand_initialized = 0;
     381           13565 : }
     382                 : /* }}} */
     383                 : 
     384                 : /* {{{ ZEND_MINIT_FUNCTION
     385                 :  */
     386                 : ZEND_MODULE_STARTUP_D(gmp)
     387           13565 : {
     388           13565 :         le_gmp = zend_register_list_destructors_ex(_php_gmpnum_free, NULL, GMP_RESOURCE_NAME, module_number);
     389           13565 :         REGISTER_LONG_CONSTANT("GMP_ROUND_ZERO", GMP_ROUND_ZERO, CONST_CS | CONST_PERSISTENT);
     390           13565 :         REGISTER_LONG_CONSTANT("GMP_ROUND_PLUSINF", GMP_ROUND_PLUSINF, CONST_CS | CONST_PERSISTENT);
     391           13565 :         REGISTER_LONG_CONSTANT("GMP_ROUND_MINUSINF", GMP_ROUND_MINUSINF, CONST_CS | CONST_PERSISTENT);
     392           13565 :         REGISTER_STRING_CONSTANT("GMP_VERSION", (char *)gmp_version, CONST_CS | CONST_PERSISTENT);
     393                 : 
     394           13565 :         mp_set_memory_functions(gmp_emalloc, gmp_erealloc, gmp_efree);
     395                 : 
     396           13565 :         return SUCCESS;
     397                 : }
     398                 : /* }}} */
     399                 : 
     400                 : /* {{{ ZEND_RSHUTDOWN_FUNCTION
     401                 :  */
     402                 : ZEND_MODULE_DEACTIVATE_D(gmp)
     403           13584 : {
     404           13584 :         if (GMPG(rand_initialized)) {
     405               1 :                 gmp_randclear(GMPG(rand_state));
     406               1 :                 GMPG(rand_initialized) = 0;
     407                 :         }
     408                 : 
     409           13584 :         return SUCCESS;
     410                 : }
     411                 : /* }}} */
     412                 : 
     413                 : /* {{{ ZEND_MINFO_FUNCTION
     414                 :  */
     415                 : ZEND_MODULE_INFO_D(gmp)
     416               6 : {
     417               6 :         php_info_print_table_start();
     418               6 :         php_info_print_table_row(2, "gmp support", "enabled");
     419               6 :         php_info_print_table_row(2, "GMP version", gmp_version);
     420               6 :         php_info_print_table_end();
     421               6 : }
     422                 : /* }}} */
     423                 : 
     424                 : /* Fetch zval to be GMP number.
     425                 :    Initially, zval can be also number or string */
     426                 : #define FETCH_GMP_ZVAL(gmpnumber, zval) \
     427                 : if (Z_TYPE_PP(zval) == IS_RESOURCE) { \
     428                 :         ZEND_FETCH_RESOURCE(gmpnumber, mpz_t *, zval, -1, GMP_RESOURCE_NAME, le_gmp);\
     429                 : } else {\
     430                 :         if (convert_to_gmp(&gmpnumber, zval, 0 TSRMLS_CC) == FAILURE) {\
     431                 :                 RETURN_FALSE;\
     432                 :         }\
     433                 :         ZEND_REGISTER_RESOURCE(NULL, gmpnumber, le_gmp);\
     434                 : }
     435                 : 
     436                 : /* create a new initialized GMP number */
     437                 : #define INIT_GMP_NUM(gmpnumber) { gmpnumber=emalloc(sizeof(mpz_t)); mpz_init(*gmpnumber); }
     438                 : #define FREE_GMP_NUM(gmpnumber) { mpz_clear(*gmpnumber); efree(gmpnumber); }
     439                 : 
     440                 : /* {{{ convert_to_gmp
     441                 :  * Convert zval to be gmp number */
     442                 : static int convert_to_gmp(mpz_t * *gmpnumber, zval **val, int base TSRMLS_DC) 
     443            1569 : {
     444            1569 :         int ret = 0;
     445            1569 :         int skip_lead = 0;
     446                 : 
     447            1569 :         *gmpnumber = emalloc(sizeof(mpz_t));
     448                 : 
     449            1569 :         switch (Z_TYPE_PP(val)) {
     450                 :         case IS_LONG:
     451                 :         case IS_BOOL:
     452                 :         case IS_CONSTANT:
     453                 :                 {
     454            1337 :                         convert_to_long_ex(val);
     455            1337 :                         mpz_init_set_si(**gmpnumber, Z_LVAL_PP(val));
     456                 :                 }
     457            1337 :                 break;
     458                 :         case IS_STRING:
     459                 :                 {
     460             171 :                         char *numstr = Z_STRVAL_PP(val);
     461                 : 
     462             171 :                         if (Z_STRLEN_PP(val) > 2) {
     463             141 :                                 if (numstr[0] == '0') {
     464              11 :                                         if (numstr[1] == 'x' || numstr[1] == 'X') {
     465               2 :                                                 base = 16;
     466               2 :                                                 skip_lead = 1;
     467               7 :                                         } else if (base != 16 && (numstr[1] == 'b' || numstr[1] == 'B')) {
     468               2 :                                                 base = 2;
     469               2 :                                                 skip_lead = 1;
     470                 :                                         }
     471                 :                                 }
     472                 :                         }
     473             171 :                         ret = mpz_init_set_str(**gmpnumber, (skip_lead ? &numstr[2] : numstr), base);
     474                 :                 }
     475             171 :                 break;
     476                 :         default:
     477              61 :                 php_error_docref(NULL TSRMLS_CC, E_WARNING,"Unable to convert variable to GMP - wrong type");
     478              61 :                 efree(*gmpnumber);
     479              61 :                 return FAILURE;
     480                 :         }
     481                 : 
     482            1508 :         if (ret) {
     483              20 :                 FREE_GMP_NUM(*gmpnumber);
     484              20 :                 return FAILURE;
     485                 :         }
     486                 :         
     487            1488 :         return SUCCESS;
     488                 : }
     489                 : /* }}} */
     490                 : 
     491                 : /* {{{ typedefs
     492                 :  */
     493                 : typedef void (*gmp_unary_op_t)(mpz_ptr, mpz_srcptr);
     494                 : typedef int (*gmp_unary_opl_t)(mpz_srcptr);
     495                 : 
     496                 : typedef void (*gmp_unary_ui_op_t)(mpz_ptr, unsigned long);
     497                 : 
     498                 : typedef void (*gmp_binary_op_t)(mpz_ptr, mpz_srcptr, mpz_srcptr);
     499                 : typedef int (*gmp_binary_opl_t)(mpz_srcptr, mpz_srcptr);
     500                 : 
     501                 : typedef unsigned long (*gmp_binary_ui_op_t)(mpz_ptr, mpz_srcptr, unsigned long);
     502                 : typedef void (*gmp_binary_op2_t)(mpz_ptr, mpz_ptr, mpz_srcptr, mpz_srcptr);
     503                 : typedef unsigned long (*gmp_binary_ui_op2_t)(mpz_ptr, mpz_ptr, mpz_srcptr, unsigned long);
     504                 : /* }}} */
     505                 : 
     506                 : #define gmp_zval_binary_ui_op(r, a, b, o, u) gmp_zval_binary_ui_op_ex(r, a, b, o, u, 0, 0 TSRMLS_CC)
     507                 : #define gmp_zval_binary_ui_op2(r, a, b, o, u) gmp_zval_binary_ui_op2_ex(r, a, b, o, u, 0, 0 TSRMLS_CC)
     508                 : 
     509                 : #define gmp_binary_ui_op(op, uop) _gmp_binary_ui_op(INTERNAL_FUNCTION_PARAM_PASSTHRU, op, uop)
     510                 : #define gmp_binary_op(op)         _gmp_binary_ui_op(INTERNAL_FUNCTION_PARAM_PASSTHRU, op, NULL)
     511                 : #define gmp_binary_opl(op) _gmp_binary_opl(INTERNAL_FUNCTION_PARAM_PASSTHRU, op)
     512                 : 
     513                 : /* Unary operations */
     514                 : #define gmp_unary_op(op)         _gmp_unary_op(INTERNAL_FUNCTION_PARAM_PASSTHRU, op)
     515                 : #define gmp_unary_opl(op)         _gmp_unary_opl(INTERNAL_FUNCTION_PARAM_PASSTHRU, op)
     516                 : #define gmp_unary_ui_op(op)      _gmp_unary_ui_op(INTERNAL_FUNCTION_PARAM_PASSTHRU, op)
     517                 : 
     518                 : /* {{{ gmp_zval_binary_ui_op_ex
     519                 :    Execute GMP binary operation.
     520                 :    May return GMP resource or long if operation allows this
     521                 : */
     522                 : static inline void gmp_zval_binary_ui_op_ex(zval *return_value, zval **a_arg, zval **b_arg, gmp_binary_op_t gmp_op, gmp_binary_ui_op_t gmp_ui_op, int allow_ui_return, int check_b_zero TSRMLS_DC) 
     523            1082 : {
     524                 :         mpz_t *gmpnum_a, *gmpnum_b, *gmpnum_result;
     525            1082 :         unsigned long long_result = 0;
     526            1082 :         int use_ui = 0;
     527                 : 
     528            1082 :         FETCH_GMP_ZVAL(gmpnum_a, a_arg);
     529                 :         
     530            1108 :         if (gmp_ui_op && Z_TYPE_PP(b_arg) == IS_LONG && Z_LVAL_PP(b_arg) >= 0) {
     531              41 :                 use_ui = 1;
     532                 :         } else {
     533            1026 :                 FETCH_GMP_ZVAL(gmpnum_b, b_arg);
     534                 :         }
     535                 : 
     536            1064 :         if(check_b_zero) {
     537              23 :                 int b_is_zero = 0;
     538              23 :                 if(use_ui) {
     539              21 :                         b_is_zero = (Z_LVAL_PP(b_arg) == 0);
     540                 :                 } else {
     541               2 :                         b_is_zero = !mpz_cmp_ui(*gmpnum_b, 0);
     542                 :                 }
     543                 : 
     544              23 :                 if(b_is_zero) {
     545               4 :                         php_error_docref(NULL TSRMLS_CC, E_WARNING, "Zero operand not allowed");
     546               4 :                         RETURN_FALSE;
     547                 :                 }
     548                 :         }
     549                 : 
     550            1060 :         INIT_GMP_NUM(gmpnum_result);
     551                 : 
     552            1097 :         if (use_ui && gmp_ui_op) {
     553              37 :                 if (allow_ui_return) {
     554              15 :                         long_result = gmp_ui_op(*gmpnum_result, *gmpnum_a, (unsigned long)Z_LVAL_PP(b_arg));
     555                 :                 } else {
     556              22 :                         gmp_ui_op(*gmpnum_result, *gmpnum_a, (unsigned long)Z_LVAL_PP(b_arg));
     557                 :                 }
     558                 :         } else {
     559            1023 :                 gmp_op(*gmpnum_result, *gmpnum_a, *gmpnum_b);
     560                 :         }
     561                 : 
     562            1060 :         if (use_ui && allow_ui_return) {
     563              15 :                 FREE_GMP_NUM(gmpnum_result);
     564              15 :                 RETURN_LONG((long)long_result);
     565                 :         } else {
     566            1045 :                 ZEND_REGISTER_RESOURCE(return_value, gmpnum_result, le_gmp);
     567                 :         }
     568                 : }
     569                 : /* }}} */
     570                 : 
     571                 : /* {{{ gmp_zval_binary_ui_op2_ex
     572                 :    Execute GMP binary operation which returns 2 values.
     573                 :    May return GMP resources or longs if operation allows this.
     574                 : */
     575                 : static inline void gmp_zval_binary_ui_op2_ex(zval *return_value, zval **a_arg, zval **b_arg, gmp_binary_op2_t gmp_op, gmp_binary_ui_op2_t gmp_ui_op, int allow_ui_return, int check_b_zero TSRMLS_DC)
     576              12 : {
     577                 :         mpz_t *gmpnum_a, *gmpnum_b, *gmpnum_result1, *gmpnum_result2;
     578                 :         zval r;
     579              12 :         int use_ui = 0;
     580              12 :         unsigned long long_result = 0;
     581                 : 
     582              12 :         FETCH_GMP_ZVAL(gmpnum_a, a_arg);
     583                 : 
     584              20 :         if (gmp_ui_op && Z_TYPE_PP(b_arg) == IS_LONG && Z_LVAL_PP(b_arg) >= 0) {
     585                 :                 /* use _ui function */
     586              10 :                 use_ui = 1;
     587                 :         } else {
     588               0 :                 FETCH_GMP_ZVAL(gmpnum_b, b_arg);
     589                 :         }
     590                 : 
     591              10 :         if(check_b_zero) {
     592              10 :                 int b_is_zero = 0;
     593              10 :                 if(use_ui) {
     594              10 :                         b_is_zero = (Z_LVAL_PP(b_arg) == 0);
     595                 :                 } else {
     596               0 :                         b_is_zero = !mpz_cmp_ui(*gmpnum_b, 0);
     597                 :                 }
     598                 : 
     599              10 :                 if(b_is_zero) {
     600               2 :                         php_error_docref(NULL TSRMLS_CC, E_WARNING, "Zero operand not allowed");
     601               2 :                         RETURN_FALSE;
     602                 :                 }
     603                 :         }
     604                 : 
     605               8 :         INIT_GMP_NUM(gmpnum_result1);
     606               8 :         INIT_GMP_NUM(gmpnum_result2);
     607                 : 
     608              16 :         if (use_ui && gmp_ui_op) {
     609               8 :                 if (allow_ui_return) {
     610               0 :                         long_result = gmp_ui_op(*gmpnum_result1, *gmpnum_result2, *gmpnum_a, (unsigned long)Z_LVAL_PP(b_arg));
     611                 :                 } else {
     612               8 :                         gmp_ui_op(*gmpnum_result1, *gmpnum_result2, *gmpnum_a, (unsigned long)Z_LVAL_PP(b_arg));
     613                 :                 }
     614                 :         } else {
     615               0 :                 gmp_op(*gmpnum_result1, *gmpnum_result2, *gmpnum_a, *gmpnum_b);
     616                 :         }
     617                 : 
     618               8 :         array_init(return_value);
     619               8 :         ZEND_REGISTER_RESOURCE(&r, gmpnum_result1, le_gmp);
     620               8 :         add_index_resource(return_value, 0, Z_LVAL(r));
     621               8 :         if (use_ui && allow_ui_return) {
     622               0 :                 mpz_clear(*gmpnum_result2);
     623               0 :                 add_index_long(return_value, 1, long_result);
     624                 :         } else {
     625               8 :                 ZEND_REGISTER_RESOURCE(&r, gmpnum_result2, le_gmp);
     626               8 :                 add_index_resource(return_value, 1, Z_LVAL(r));
     627                 :         }
     628                 : }
     629                 : /* }}} */
     630                 : 
     631                 : /* {{{ _gmp_binary_ui_op
     632                 :  */
     633                 : static inline void _gmp_binary_ui_op(INTERNAL_FUNCTION_PARAMETERS, gmp_binary_op_t gmp_op, gmp_binary_ui_op_t gmp_ui_op)
     634            1049 : {
     635                 :         zval **a_arg, **b_arg;
     636                 : 
     637            1049 :         if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &a_arg, &b_arg) == FAILURE){
     638               7 :                 WRONG_PARAM_COUNT;
     639                 :         }
     640                 :         
     641            1042 :         gmp_zval_binary_ui_op(return_value, a_arg, b_arg, gmp_op, gmp_ui_op);
     642                 : }
     643                 : /* }}} */
     644                 : 
     645                 : /* Unary operations */
     646                 : 
     647                 : /* {{{ gmp_zval_unary_op
     648                 :  */
     649                 : static inline void gmp_zval_unary_op(zval *return_value, zval **a_arg, gmp_unary_op_t gmp_op TSRMLS_DC) 
     650              37 : {
     651                 :         mpz_t *gmpnum_a, *gmpnum_result;
     652                 :         
     653              37 :         FETCH_GMP_ZVAL(gmpnum_a, a_arg);
     654                 : 
     655              25 :         INIT_GMP_NUM(gmpnum_result);
     656              25 :         gmp_op(*gmpnum_result, *gmpnum_a);
     657                 : 
     658              25 :         ZEND_REGISTER_RESOURCE(return_value, gmpnum_result, le_gmp);
     659                 : }
     660                 : /* }}} */
     661                 : 
     662                 : /* {{{ gmp_zval_unary_ui_op
     663                 :  */
     664                 : static inline void gmp_zval_unary_ui_op(zval *return_value, zval **a_arg, gmp_unary_ui_op_t gmp_op)
     665              11 : {
     666                 :         mpz_t *gmpnum_result;
     667                 : 
     668              11 :         convert_to_long_ex(a_arg);
     669                 : 
     670              11 :         INIT_GMP_NUM(gmpnum_result);
     671              11 :         gmp_op(*gmpnum_result, Z_LVAL_PP(a_arg));
     672                 : 
     673              11 :         ZEND_REGISTER_RESOURCE(return_value, gmpnum_result, le_gmp);
     674              11 : }
     675                 : /* }}} */
     676                 : 
     677                 : /* {{{ _gmp_unary_ui_op
     678                 :    Execute GMP unary operation.
     679                 : */
     680                 : static inline void _gmp_unary_ui_op(INTERNAL_FUNCTION_PARAMETERS, gmp_unary_ui_op_t gmp_op)
     681               0 : {
     682                 :         zval **a_arg;
     683                 : 
     684               0 :         if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &a_arg) == FAILURE){
     685               0 :                 WRONG_PARAM_COUNT;
     686                 :         }
     687                 :         
     688               0 :         gmp_zval_unary_ui_op(return_value, a_arg, gmp_op);
     689                 : }
     690                 : /* }}} */
     691                 : 
     692                 : /* {{{ _gmp_unary_op
     693                 :  */
     694                 : static inline void _gmp_unary_op(INTERNAL_FUNCTION_PARAMETERS, gmp_unary_op_t gmp_op)
     695              42 : {
     696                 :         zval **a_arg;
     697                 : 
     698              42 :         if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &a_arg) == FAILURE){
     699               5 :                 WRONG_PARAM_COUNT;
     700                 :         }
     701                 :         
     702              37 :         gmp_zval_unary_op(return_value, a_arg, gmp_op TSRMLS_CC);
     703                 : }
     704                 : /* }}} */
     705                 : 
     706                 : /* {{{ _gmp_unary_opl
     707                 :  */
     708                 : static inline void _gmp_unary_opl(INTERNAL_FUNCTION_PARAMETERS, gmp_unary_opl_t gmp_op)
     709               0 : {
     710                 :         zval **a_arg;
     711                 :         mpz_t *gmpnum_a;
     712                 : 
     713               0 :         if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &a_arg) == FAILURE){
     714               0 :                 WRONG_PARAM_COUNT;
     715                 :         }
     716                 :         
     717               0 :         FETCH_GMP_ZVAL(gmpnum_a, a_arg);
     718                 : 
     719               0 :         RETURN_LONG(gmp_op(*gmpnum_a));
     720                 : }
     721                 : /* }}} */
     722                 : 
     723                 : /* {{{ _gmp_binary_opl
     724                 :  */
     725                 : static inline void _gmp_binary_opl(INTERNAL_FUNCTION_PARAMETERS, gmp_binary_opl_t gmp_op)
     726              46 : {
     727                 :         zval **a_arg, **b_arg;
     728                 :         mpz_t *gmpnum_a, *gmpnum_b;
     729                 : 
     730              46 :         if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &a_arg, &b_arg) == FAILURE){
     731               6 :                 WRONG_PARAM_COUNT;
     732                 :         }
     733                 : 
     734              40 :         FETCH_GMP_ZVAL(gmpnum_a, a_arg);
     735              36 :         FETCH_GMP_ZVAL(gmpnum_b, a_arg);
     736                 : 
     737              36 :         RETURN_LONG(gmp_op(*gmpnum_a, *gmpnum_b));
     738                 : }
     739                 : /* }}} */
     740                 : 
     741                 : /* {{{ proto resource gmp_init(mixed number [, int base])
     742                 :    Initializes GMP number */
     743                 : ZEND_FUNCTION(gmp_init)
     744              96 : {
     745                 :         zval **number_arg, **base_arg;
     746                 :         mpz_t * gmpnumber;
     747                 :         int argc;
     748              96 :         int base=0;
     749                 : 
     750              96 :         argc = ZEND_NUM_ARGS();
     751              96 :         if (argc < 1 || argc > 2 || zend_get_parameters_ex(argc, &number_arg, &base_arg) == FAILURE){
     752               3 :                 WRONG_PARAM_COUNT;
     753                 :         }
     754                 : 
     755              93 :         if (argc == 2) {
     756              11 :                         convert_to_long_ex(base_arg);
     757              11 :                         base = Z_LVAL_PP(base_arg);
     758              11 :                 if (base < 2 || base > 36) {
     759               1 :                         php_error_docref(NULL TSRMLS_CC, E_WARNING, "Bad base for conversion: %d (should be between 2 and 36)", base);
     760               1 :                         RETURN_FALSE;
     761                 :                 }
     762                 :         }
     763                 : 
     764              92 :         if (convert_to_gmp(&gmpnumber, number_arg, base TSRMLS_CC) == FAILURE) {
     765               5 :                 RETURN_FALSE;
     766                 :         }
     767                 : 
     768                 :         /* Write your own code here to handle argument number. */
     769              87 :         ZEND_REGISTER_RESOURCE(return_value, gmpnumber, le_gmp);
     770                 : }
     771                 : /* }}} */
     772                 : 
     773                 : /* {{{ proto int gmp_intval(resource gmpnumber)
     774                 :    Gets signed long value of GMP number */
     775                 : ZEND_FUNCTION(gmp_intval)
     776              19 : {
     777                 :         zval **gmpnumber_arg;
     778                 :         mpz_t * gmpnum;
     779                 : 
     780              19 :         if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &gmpnumber_arg) == FAILURE){
     781               1 :                 WRONG_PARAM_COUNT;
     782                 :         }
     783                 :         
     784              18 :         if (Z_TYPE_PP(gmpnumber_arg) == IS_RESOURCE) {
     785               8 :                 ZEND_FETCH_RESOURCE(gmpnum, mpz_t *, gmpnumber_arg, -1, GMP_RESOURCE_NAME, le_gmp);
     786               7 :                 RETVAL_LONG(mpz_get_si(*gmpnum));
     787                 :         } else {
     788              10 :                 convert_to_long_ex(gmpnumber_arg);
     789              10 :                 RETVAL_LONG(Z_LVAL_PP(gmpnumber_arg));
     790                 :         }
     791                 : }
     792                 : /* }}} */
     793                 : 
     794                 : /* {{{ proto string gmp_strval(resource gmpnumber [, int base])
     795                 :    Gets string representation of GMP number  */
     796                 : ZEND_FUNCTION(gmp_strval)
     797             297 : {
     798                 :         zval **gmpnumber_arg, **base_arg;
     799             297 :         int base=10, num_len, argc;
     800                 :         mpz_t * gmpnum;
     801                 :         char *out_string;
     802                 : 
     803             297 :         argc = ZEND_NUM_ARGS();
     804             297 :         if (argc < 1 || argc > 2 || zend_get_parameters_ex(argc, &gmpnumber_arg, &base_arg) == FAILURE){
     805               1 :                 WRONG_PARAM_COUNT;
     806                 :         }
     807                 : 
     808             296 :         FETCH_GMP_ZVAL(gmpnum, gmpnumber_arg);
     809                 : 
     810             280 :         switch (argc) {
     811                 :                 case 2:
     812               7 :                         convert_to_long_ex(base_arg);
     813               7 :                         base = Z_LVAL_PP(base_arg);
     814               7 :                         break;
     815                 :                 case 1:
     816             273 :                         base = 10;
     817                 :                         break;
     818                 :         }
     819                 : 
     820             280 :         if (base < 2 || base > 36) {
     821               5 :                 php_error_docref(NULL TSRMLS_CC, E_WARNING, "Bad base for conversion: %d", base);
     822               5 :                 RETURN_FALSE;
     823                 :         }
     824                 : 
     825             275 :         num_len = mpz_sizeinbase(*gmpnum, base);
     826             275 :         out_string = emalloc(num_len+2);
     827             275 :         if (mpz_sgn(*gmpnum) < 0) {
     828              31 :                 num_len++;
     829                 :         }
     830             275 :         mpz_get_str(out_string, base, *gmpnum);
     831                 : 
     832                 :         /* 
     833                 :         From GMP documentation for mpz_sizeinbase():
     834                 :         The returned value will be exact or 1 too big.  If base is a power of
     835                 :         2, the returned value will always be exact.
     836                 : 
     837                 :         So let's check to see if we already have a \0 byte...
     838                 :         */
     839                 : 
     840             275 :         if (out_string[num_len-1] == '\0') {
     841              35 :                 num_len--;
     842                 :         } else {
     843             240 :                 out_string[num_len] = '\0';
     844                 :         }
     845             275 :         RETVAL_STRINGL(out_string, num_len, 0);
     846                 : }
     847                 : /* }}} */
     848                 : 
     849                 : /* {{{ proto resource gmp_add(resource a, resource b)
     850                 :    Add a and b */
     851                 : ZEND_FUNCTION(gmp_add)
     852              15 : {
     853              15 :         gmp_binary_ui_op(mpz_add, (gmp_binary_ui_op_t)mpz_add_ui);
     854              15 : }
     855                 : /* }}} */
     856                 : 
     857                 : /* {{{ proto resource gmp_sub(resource a, resource b)
     858                 :    Subtract b from a */
     859                 : ZEND_FUNCTION(gmp_sub)
     860               9 : {
     861               9 :         gmp_binary_ui_op(mpz_sub, (gmp_binary_ui_op_t)mpz_sub_ui);
     862               9 : }
     863                 : /* }}} */
     864                 : 
     865                 : /* {{{ proto resource gmp_mul(resource a, resource b)
     866                 :    Multiply a and b */
     867                 : ZEND_FUNCTION(gmp_mul)
     868             999 : {
     869             999 :         gmp_binary_ui_op(mpz_mul, (gmp_binary_ui_op_t)mpz_mul_ui);
     870             999 : }
     871                 : /* }}} */
     872                 : 
     873                 : /* {{{ proto array gmp_div_qr(resource a, resource b [, int round])
     874                 :    Divide a by b, returns quotient and reminder */
     875                 : ZEND_FUNCTION(gmp_div_qr)
     876              15 : {
     877                 :         zval **a_arg, **b_arg, **round_arg;
     878              15 :         int round = GMP_ROUND_ZERO, argc;
     879                 : 
     880              15 :         argc = ZEND_NUM_ARGS();
     881              15 :         if (argc < 2 || argc > 3 || zend_get_parameters_ex(argc, &a_arg, &b_arg, &round_arg) == FAILURE){
     882               2 :                 WRONG_PARAM_COUNT;
     883                 :         }
     884                 : 
     885              13 :         switch (argc) {
     886                 :                 case 3:
     887               6 :                         convert_to_long_ex(round_arg);
     888               6 :                         round = Z_LVAL_PP(round_arg);
     889               6 :                         break;
     890                 :                 case 2:
     891               7 :                         round = GMP_ROUND_ZERO;
     892                 :                         break;
     893                 :         }
     894                 : 
     895              13 :         switch (round) {
     896                 :         case GMP_ROUND_ZERO:
     897               8 :                 gmp_zval_binary_ui_op2_ex(return_value, a_arg, b_arg, mpz_tdiv_qr, (gmp_binary_ui_op2_t)mpz_tdiv_qr_ui, 0, 1 TSRMLS_CC);
     898               8 :                 break;
     899                 :         case GMP_ROUND_PLUSINF:
     900               2 :                 gmp_zval_binary_ui_op2_ex(return_value, a_arg, b_arg, mpz_cdiv_qr, (gmp_binary_ui_op2_t)mpz_cdiv_qr_ui, 0, 1 TSRMLS_CC);
     901               2 :                 break;
     902                 :         case GMP_ROUND_MINUSINF:
     903               2 :                 gmp_zval_binary_ui_op2_ex(return_value, a_arg, b_arg, mpz_fdiv_qr, (gmp_binary_ui_op2_t)mpz_fdiv_qr_ui, 0, 1 TSRMLS_CC);
     904                 :                 break;
     905                 :         }
     906                 :                                                            
     907                 : }
     908                 : /* }}} */
     909                 : 
     910                 : /* {{{ proto resource gmp_div_r(resource a, resource b [, int round])
     911                 :    Divide a by b, returns reminder only */
     912                 : ZEND_FUNCTION(gmp_div_r)
     913              14 : {
     914                 :         zval **a_arg, **b_arg, **round_arg;
     915              14 :         int round = GMP_ROUND_ZERO, argc;
     916                 : 
     917              14 :         argc = ZEND_NUM_ARGS();
     918              14 :         if (argc < 2 || argc > 3 || zend_get_parameters_ex(argc, &a_arg, &b_arg, &round_arg) == FAILURE){
     919               2 :                 WRONG_PARAM_COUNT;
     920                 :         }
     921                 : 
     922              12 :         switch (argc) {
     923                 :                 case 3:
     924               6 :                         convert_to_long_ex(round_arg);
     925               6 :                         round = Z_LVAL_PP(round_arg);
     926               6 :                         break;
     927                 :                 case 2:
     928               6 :                         round = GMP_ROUND_ZERO;
     929                 :                         break;
     930                 :         }
     931                 : 
     932              12 :         switch (round) {
     933                 :         case GMP_ROUND_ZERO:
     934               7 :                 gmp_zval_binary_ui_op_ex(return_value, a_arg, b_arg, mpz_tdiv_r, (gmp_binary_ui_op_t)mpz_tdiv_r_ui, 1, 1 TSRMLS_CC);
     935               7 :                 break;
     936                 :         case GMP_ROUND_PLUSINF:
     937               2 :                 gmp_zval_binary_ui_op_ex(return_value, a_arg, b_arg, mpz_cdiv_r, (gmp_binary_ui_op_t)mpz_cdiv_r_ui, 1, 1 TSRMLS_CC);
     938               2 :                 break;
     939                 :         case GMP_ROUND_MINUSINF:
     940               2 :                 gmp_zval_binary_ui_op_ex(return_value, a_arg, b_arg, mpz_fdiv_r, (gmp_binary_ui_op_t)mpz_fdiv_r_ui, 1, 1 TSRMLS_CC);
     941                 :                 break;
     942                 :         }
     943                 : }
     944                 : /* }}} */
     945                 : 
     946                 : /* {{{ proto resource gmp_div_q(resource a, resource b [, int round])
     947                 :    Divide a by b, returns quotient only */
     948                 : ZEND_FUNCTION(gmp_div_q)
     949              15 : {
     950                 :         zval **a_arg, **b_arg, **round_arg;
     951              15 :         int round = GMP_ROUND_ZERO, argc;
     952                 : 
     953              15 :         argc = ZEND_NUM_ARGS();
     954              15 :         if (argc < 2 || argc > 3 || zend_get_parameters_ex(argc, &a_arg, &b_arg, &round_arg) == FAILURE){
     955               2 :                 WRONG_PARAM_COUNT;
     956                 :         }
     957                 : 
     958              13 :         switch (argc) {
     959                 :                 case 3:
     960               6 :                         convert_to_long_ex(round_arg);
     961               6 :                         round = Z_LVAL_PP(round_arg);
     962               6 :                         break;
     963                 :                 case 2:
     964               7 :                         round = GMP_ROUND_ZERO;
     965                 :                         break;
     966                 :         }
     967                 : 
     968              13 :         switch (round) {
     969                 :         case GMP_ROUND_ZERO:
     970               8 :                 gmp_zval_binary_ui_op_ex(return_value, a_arg, b_arg, mpz_tdiv_q, (gmp_binary_ui_op_t)mpz_tdiv_q_ui, 0, 1 TSRMLS_CC);
     971               8 :                 break;
     972                 :         case GMP_ROUND_PLUSINF:
     973               2 :                 gmp_zval_binary_ui_op_ex(return_value, a_arg, b_arg, mpz_cdiv_q, (gmp_binary_ui_op_t)mpz_cdiv_q_ui, 0, 1 TSRMLS_CC);
     974               2 :                 break;
     975                 :         case GMP_ROUND_MINUSINF:
     976               2 :                 gmp_zval_binary_ui_op_ex(return_value, a_arg, b_arg, mpz_fdiv_q, (gmp_binary_ui_op_t)mpz_fdiv_q_ui, 0, 1 TSRMLS_CC);
     977                 :                 break;
     978                 :         }
     979                 :                                                            
     980                 : }
     981                 : /* }}} */
     982                 : 
     983                 : /* {{{ proto resource gmp_mod(resource a, resource b)
     984                 :    Computes a modulo b */
     985                 : ZEND_FUNCTION(gmp_mod)
     986               8 : {
     987                 :         zval **a_arg, **b_arg;
     988                 : 
     989               8 :         if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &a_arg, &b_arg) == FAILURE){
     990               2 :                 WRONG_PARAM_COUNT;
     991                 :         }
     992                 : 
     993               6 :         gmp_zval_binary_ui_op_ex(return_value, a_arg, b_arg, mpz_mod, (gmp_binary_ui_op_t)mpz_mod_ui, 1, 1 TSRMLS_CC);
     994                 : }
     995                 : /* }}} */
     996                 : 
     997                 : /* {{{ proto resource gmp_divexact(resource a, resource b)
     998                 :    Divide a by b using exact division algorithm */
     999                 : ZEND_FUNCTION(gmp_divexact)
    1000               0 : {
    1001                 :         zval **a_arg, **b_arg;
    1002                 : 
    1003               0 :         if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &a_arg, &b_arg) == FAILURE){
    1004               0 :                 WRONG_PARAM_COUNT;
    1005                 :         }
    1006                 :         
    1007               0 :         gmp_zval_binary_ui_op_ex(return_value, a_arg, b_arg, mpz_divexact, NULL, 0, 1 TSRMLS_CC);
    1008                 : }
    1009                 : /* }}} */
    1010                 : 
    1011                 : /* {{{ proto resource gmp_neg(resource a)
    1012                 :    Negates a number */
    1013                 : ZEND_FUNCTION(gmp_neg)
    1014              11 : {
    1015              11 :         gmp_unary_op(mpz_neg);
    1016              11 : }
    1017                 : /* }}} */
    1018                 : 
    1019                 : /* {{{ proto resource gmp_abs(resource a)
    1020                 :    Calculates absolute value */
    1021                 : ZEND_FUNCTION(gmp_abs)
    1022              12 : {
    1023              12 :         gmp_unary_op(mpz_abs);
    1024              12 : }
    1025                 : /* }}} */
    1026                 : 
    1027                 : /* {{{ proto resource gmp_fact(int a)
    1028                 :    Calculates factorial function */
    1029                 : ZEND_FUNCTION(gmp_fact)
    1030              16 : {
    1031                 :         zval **a_arg;
    1032                 :         mpz_t *gmpnum_tmp;
    1033                 : 
    1034              16 :         if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &a_arg) == FAILURE){
    1035               2 :                 WRONG_PARAM_COUNT;
    1036                 :         }
    1037                 : 
    1038              14 :         if (Z_TYPE_PP(a_arg) == IS_RESOURCE) {
    1039               2 :                 FETCH_GMP_ZVAL(gmpnum_tmp, a_arg);
    1040               2 :                 if (mpz_sgn(*gmpnum_tmp) < 0) {
    1041               1 :                         php_error_docref(NULL TSRMLS_CC, E_WARNING,"Number has to be greater than or equal to 0");
    1042               1 :                         RETURN_FALSE;
    1043                 :                 }
    1044                 :         } else {
    1045              12 :                 convert_to_long_ex(a_arg);
    1046              12 :                 if (Z_LVAL_PP(a_arg) < 0) {
    1047               2 :                         php_error_docref(NULL TSRMLS_CC, E_WARNING,"Number has to be greater than or equal to 0");
    1048               2 :                         RETURN_FALSE;
    1049                 :                 }
    1050                 :         }
    1051                 :                 
    1052              11 :         gmp_zval_unary_ui_op(return_value, a_arg, mpz_fac_ui);
    1053                 : }
    1054                 : /* }}} */
    1055                 : 
    1056                 : /* {{{ proto resource gmp_pow(resource base, int exp)
    1057                 :    Raise base to power exp */
    1058                 : ZEND_FUNCTION(gmp_pow)
    1059              18 : {
    1060                 :         zval **base_arg, **exp_arg;
    1061                 :         mpz_t *gmpnum_result, *gmpnum_base;
    1062              18 :         int use_ui = 0;
    1063                 : 
    1064              18 :         if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &base_arg, &exp_arg) == FAILURE){
    1065               3 :                 WRONG_PARAM_COUNT;
    1066                 :         }
    1067                 : 
    1068              20 :         if (Z_TYPE_PP(base_arg) == IS_LONG && Z_LVAL_PP(base_arg) >= 0) {
    1069               5 :                 use_ui = 1;
    1070                 :         } else {
    1071              10 :                 FETCH_GMP_ZVAL(gmpnum_base, base_arg);
    1072                 :         }
    1073                 : 
    1074              13 :         convert_to_long_ex(exp_arg);
    1075                 : 
    1076              13 :         if (Z_LVAL_PP(exp_arg) < 0) {
    1077               2 :                 php_error_docref(NULL TSRMLS_CC, E_WARNING,"Negative exponent not supported");
    1078               2 :                 RETURN_FALSE;
    1079                 :         }
    1080                 :         
    1081              11 :         INIT_GMP_NUM(gmpnum_result);
    1082              11 :         if (use_ui) {
    1083               4 :                 mpz_ui_pow_ui(*gmpnum_result, Z_LVAL_PP(base_arg), Z_LVAL_PP(exp_arg));
    1084                 :         } else {
    1085               7 :                 mpz_pow_ui(*gmpnum_result, *gmpnum_base, Z_LVAL_PP(exp_arg));
    1086                 :         }
    1087                 : 
    1088              11 :         ZEND_REGISTER_RESOURCE(return_value, gmpnum_result, le_gmp);
    1089                 : }
    1090                 : /* }}} */
    1091                 : 
    1092                 : /* {{{ proto resource gmp_powm(resource base, resource exp, resource mod)
    1093                 :    Raise base to power exp and take result modulo mod */
    1094                 : ZEND_FUNCTION(gmp_powm)
    1095              18 : {
    1096                 :         zval **base_arg, **exp_arg, **mod_arg;
    1097                 :         mpz_t *gmpnum_base, *gmpnum_exp, *gmpnum_mod, *gmpnum_result;
    1098              18 :         int use_ui = 0;
    1099                 : 
    1100              18 :         if (ZEND_NUM_ARGS() != 3 || zend_get_parameters_ex(3, &base_arg, &exp_arg, &mod_arg) == FAILURE){
    1101               3 :                 WRONG_PARAM_COUNT;
    1102                 :         }
    1103                 : 
    1104              15 :         FETCH_GMP_ZVAL(gmpnum_base, base_arg);
    1105                 : 
    1106              20 :         if (Z_TYPE_PP(exp_arg) == IS_LONG && Z_LVAL_PP(exp_arg) >= 0) {
    1107               7 :                 use_ui = 1;
    1108                 :         } else {
    1109               6 :                 FETCH_GMP_ZVAL(gmpnum_exp, exp_arg);
    1110               5 :                 if (mpz_sgn(*gmpnum_exp) < 0) {
    1111               1 :                         php_error_docref(NULL TSRMLS_CC, E_WARNING,"Second parameter cannot be less than 0");
    1112               1 :                         RETURN_FALSE;
    1113                 :                 }
    1114                 :         }
    1115              11 :         FETCH_GMP_ZVAL(gmpnum_mod, mod_arg);
    1116                 : 
    1117              10 :         if (!mpz_cmp_ui(*gmpnum_mod, 0)) {
    1118               0 :                 RETURN_FALSE;
    1119                 :         }
    1120                 : 
    1121              10 :         INIT_GMP_NUM(gmpnum_result);
    1122              10 :         if (use_ui) {
    1123               7 :                 mpz_powm_ui(*gmpnum_result, *gmpnum_base, (unsigned long)Z_LVAL_PP(exp_arg), *gmpnum_mod);
    1124                 :         } else {
    1125               3 :                 mpz_powm(*gmpnum_result, *gmpnum_base, *gmpnum_exp, *gmpnum_mod);
    1126                 :         }
    1127                 : 
    1128              10 :         ZEND_REGISTER_RESOURCE(return_value, gmpnum_result, le_gmp);
    1129                 : 
    1130                 : }
    1131                 : /* }}} */
    1132                 : 
    1133                 : /* {{{ proto resource gmp_sqrt(resource a)
    1134                 :    Takes integer part of square root of a */
    1135                 : ZEND_FUNCTION(gmp_sqrt)
    1136              11 : {
    1137                 :         zval **a_arg;
    1138                 :         mpz_t *gmpnum_a, *gmpnum_result;
    1139                 : 
    1140              11 :         if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &a_arg) == FAILURE){
    1141               2 :                 WRONG_PARAM_COUNT;
    1142                 :         }
    1143                 :         
    1144               9 :         FETCH_GMP_ZVAL(gmpnum_a, a_arg);
    1145                 : 
    1146               8 :         if (mpz_sgn(*gmpnum_a) < 0) {
    1147               3 :                 php_error_docref(NULL TSRMLS_CC, E_WARNING,"Number has to be greater than or equal to 0");
    1148               3 :                 RETURN_FALSE;
    1149                 :         }       
    1150                 :         
    1151               5 :         INIT_GMP_NUM(gmpnum_result);
    1152               5 :         mpz_sqrt(*gmpnum_result, *gmpnum_a);
    1153                 : 
    1154               5 :         ZEND_REGISTER_RESOURCE(return_value, gmpnum_result, le_gmp);
    1155                 : }
    1156                 : /* }}} */
    1157                 : 
    1158                 : /* {{{ proto array gmp_sqrtrem(resource a)
    1159                 :    Square root with remainder */
    1160                 : ZEND_FUNCTION(gmp_sqrtrem)
    1161              13 : {
    1162                 :         zval **a_arg;
    1163                 :         mpz_t *gmpnum_a, *gmpnum_result1, *gmpnum_result2;
    1164                 :         zval r;
    1165                 : 
    1166              13 :         if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &a_arg) == FAILURE){
    1167               1 :                 WRONG_PARAM_COUNT;
    1168                 :         }
    1169                 : 
    1170              12 :         FETCH_GMP_ZVAL(gmpnum_a, a_arg);
    1171                 :         
    1172              11 :         if (mpz_sgn(*gmpnum_a) < 0) {
    1173               2 :                 php_error_docref(NULL TSRMLS_CC, E_WARNING,"Number has to be greater than or equal to 0");
    1174               2 :                 RETURN_FALSE;
    1175                 :         }
    1176                 :         
    1177               9 :         INIT_GMP_NUM(gmpnum_result1);
    1178               9 :         INIT_GMP_NUM(gmpnum_result2);
    1179                 : 
    1180               9 :         mpz_sqrtrem(*gmpnum_result1, *gmpnum_result2, *gmpnum_a);
    1181                 : 
    1182               9 :         array_init(return_value);
    1183               9 :         ZEND_REGISTER_RESOURCE(&r, gmpnum_result1, le_gmp);
    1184               9 :         add_index_resource(return_value, 0, Z_LVAL(r));
    1185               9 :         ZEND_REGISTER_RESOURCE(&r, gmpnum_result2, le_gmp);
    1186               9 :         add_index_resource(return_value, 1, Z_LVAL(r));
    1187                 : }
    1188                 : /* }}} */
    1189                 : 
    1190                 : /* {{{ proto bool gmp_perfect_square(resource a)
    1191                 :    Checks if a is an exact square */
    1192                 : ZEND_FUNCTION(gmp_perfect_square)
    1193              13 : {
    1194                 :         zval **a_arg;
    1195                 :         mpz_t *gmpnum_a;
    1196                 : 
    1197              13 :         if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &a_arg) == FAILURE){
    1198               1 :                 WRONG_PARAM_COUNT;
    1199                 :         }
    1200                 : 
    1201              12 :         FETCH_GMP_ZVAL(gmpnum_a, a_arg);
    1202                 : 
    1203              11 :         RETURN_BOOL((mpz_perfect_square_p(*gmpnum_a)!=0));
    1204                 : }
    1205                 : /* }}} */
    1206                 : 
    1207                 : /* {{{ proto int gmp_prob_prime(resource a[, int reps])
    1208                 :    Checks if a is "probably prime" */
    1209                 : ZEND_FUNCTION(gmp_prob_prime)
    1210              40 : {
    1211                 :         zval **gmpnumber_arg, **reps_arg;
    1212                 :         mpz_t *gmpnum_a;
    1213              40 :         int argc, reps = 10;
    1214                 : 
    1215              40 :         argc = ZEND_NUM_ARGS();
    1216              40 :         if (argc < 1 || argc > 2 || zend_get_parameters_ex(argc, &gmpnumber_arg, &reps_arg) == FAILURE){
    1217               1 :                 WRONG_PARAM_COUNT;
    1218                 :         }
    1219                 : 
    1220              39 :         FETCH_GMP_ZVAL(gmpnum_a, gmpnumber_arg);
    1221                 : 
    1222              38 :         switch (argc) {
    1223                 :                 case 2:
    1224              26 :                         convert_to_long_ex(reps_arg);
    1225              26 :                         reps = Z_LVAL_PP(reps_arg);
    1226              26 :                         break;
    1227                 :                 case 1:
    1228              12 :                         reps = 10;
    1229                 :                         break;
    1230                 :         }
    1231                 : 
    1232              38 :         RETURN_LONG(mpz_probab_prime_p(*gmpnum_a, reps));
    1233                 : }
    1234                 : /* }}} */
    1235                 : 
    1236                 : /* {{{ proto resource gmp_gcd(resource a, resource b)
    1237                 :    Computes greatest common denominator (gcd) of a and b */
    1238                 : ZEND_FUNCTION(gmp_gcd)
    1239              14 : {
    1240                 :         zval **a_arg, **b_arg;
    1241                 : 
    1242              14 :         if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &a_arg, &b_arg) == FAILURE){
    1243               3 :                 WRONG_PARAM_COUNT;
    1244                 :         }
    1245                 : 
    1246              11 :         gmp_zval_binary_ui_op_ex(return_value, a_arg, b_arg, mpz_gcd, (gmp_binary_ui_op_t)mpz_gcd_ui, 1, 0 TSRMLS_CC);
    1247                 : }
    1248                 : /* }}} */
    1249                 : 
    1250                 : /* {{{ proto array gmp_gcdext(resource a, resource b)
    1251                 :    Computes G, S, and T, such that AS + BT = G = `gcd' (A, B) */
    1252                 : ZEND_FUNCTION(gmp_gcdext)
    1253              15 : {
    1254                 :         zval **a_arg, **b_arg;
    1255                 :         mpz_t *gmpnum_a, *gmpnum_b, *gmpnum_t, *gmpnum_s, *gmpnum_g;
    1256                 :         zval r;
    1257                 : 
    1258              15 :         if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &a_arg, &b_arg) == FAILURE){
    1259               3 :                 WRONG_PARAM_COUNT;
    1260                 :         }
    1261                 : 
    1262              12 :         FETCH_GMP_ZVAL(gmpnum_a, a_arg);
    1263              11 :         FETCH_GMP_ZVAL(gmpnum_b, b_arg);
    1264                 : 
    1265              10 :         INIT_GMP_NUM(gmpnum_g);
    1266              10 :         INIT_GMP_NUM(gmpnum_s);
    1267              10 :         INIT_GMP_NUM(gmpnum_t);
    1268                 : 
    1269              10 :         mpz_gcdext(*gmpnum_g, *gmpnum_s, *gmpnum_t, *gmpnum_a, *gmpnum_b);
    1270                 : 
    1271              10 :         array_init(return_value);
    1272                 : 
    1273              10 :         ZEND_REGISTER_RESOURCE(&r, gmpnum_g, le_gmp);
    1274              10 :         add_assoc_resource(return_value, "g", Z_LVAL(r));
    1275              10 :         ZEND_REGISTER_RESOURCE(&r, gmpnum_s, le_gmp);
    1276              10 :         add_assoc_resource(return_value, "s", Z_LVAL(r));
    1277              10 :         ZEND_REGISTER_RESOURCE(&r, gmpnum_t, le_gmp);
    1278              10 :         add_assoc_resource(return_value, "t", Z_LVAL(r));
    1279                 : }
    1280                 : /* }}} */
    1281                 : 
    1282                 : /* {{{ proto resource gmp_invert(resource a, resource b)
    1283                 :    Computes the inverse of a modulo b */
    1284                 : ZEND_FUNCTION(gmp_invert)
    1285              14 : {
    1286                 :         zval **a_arg, **b_arg;
    1287                 :         mpz_t *gmpnum_a, *gmpnum_b, *gmpnum_result;
    1288                 : 
    1289              14 :         if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &a_arg, &b_arg) == FAILURE){
    1290               2 :                 WRONG_PARAM_COUNT;
    1291                 :         }
    1292                 : 
    1293              12 :         FETCH_GMP_ZVAL(gmpnum_a, a_arg);
    1294              10 :         FETCH_GMP_ZVAL(gmpnum_b, b_arg);
    1295                 : 
    1296               9 :         INIT_GMP_NUM(gmpnum_result);
    1297               9 :         if (mpz_invert(*gmpnum_result, *gmpnum_a, *gmpnum_b)) {
    1298               4 :                 ZEND_REGISTER_RESOURCE(return_value, gmpnum_result, le_gmp);
    1299                 :         } else {
    1300               5 :                 FREE_GMP_NUM(gmpnum_result);
    1301               5 :                 RETURN_FALSE;
    1302                 :         }
    1303                 : }
    1304                 : /* }}} */
    1305                 : 
    1306                 : /* {{{ proto int gmp_jacobi(resource a, resource b)
    1307                 :    Computes Jacobi symbol */
    1308                 : ZEND_FUNCTION(gmp_jacobi)
    1309              23 : {
    1310              23 :         gmp_binary_opl(mpz_jacobi);
    1311              23 : }
    1312                 : /* }}} */
    1313                 : 
    1314                 : /* {{{ proto int gmp_legendre(resource a, resource b)
    1315                 :    Computes Legendre symbol */
    1316                 : ZEND_FUNCTION(gmp_legendre)
    1317              23 : {
    1318              23 :         gmp_binary_opl(mpz_legendre);
    1319              23 : }
    1320                 : /* }}} */
    1321                 : 
    1322                 : /* {{{ proto int gmp_cmp(resource a, resource b)
    1323                 :    Compares two numbers */
    1324                 : ZEND_FUNCTION(gmp_cmp)
    1325              12 : {
    1326                 :         zval **a_arg, **b_arg;
    1327                 :         mpz_t *gmpnum_a, *gmpnum_b;
    1328              12 :         int use_si = 0, res;
    1329                 : 
    1330              12 :         if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &a_arg, &b_arg) == FAILURE){
    1331               3 :                 WRONG_PARAM_COUNT;
    1332                 :         }
    1333                 : 
    1334               9 :         FETCH_GMP_ZVAL(gmpnum_a, a_arg);
    1335                 : 
    1336               8 :         if (Z_TYPE_PP(b_arg) == IS_LONG) {
    1337               4 :                 use_si = 1;
    1338                 :         } else {
    1339               4 :                 FETCH_GMP_ZVAL(gmpnum_b, b_arg);
    1340                 :         }
    1341                 : 
    1342               8 :         if (use_si) {
    1343               4 :                 res = mpz_cmp_si(*gmpnum_a, Z_LVAL_PP(b_arg));
    1344                 :         } else {
    1345               4 :                 res = mpz_cmp(*gmpnum_a, *gmpnum_b);
    1346                 :         }
    1347                 :         
    1348               8 :         RETURN_LONG(res);
    1349                 : }
    1350                 : /* }}} */
    1351                 : 
    1352                 : /* {{{ proto int gmp_sign(resource a)
    1353                 :    Gets the sign of the number */
    1354                 : ZEND_FUNCTION(gmp_sign)
    1355              10 : {
    1356                 :         zval **a_arg;
    1357                 :         mpz_t *gmpnum_a;
    1358                 : 
    1359              10 :         if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &a_arg) == FAILURE){
    1360               2 :                 WRONG_PARAM_COUNT;
    1361                 :         }
    1362                 :         
    1363               8 :         FETCH_GMP_ZVAL(gmpnum_a, a_arg);
    1364                 : 
    1365               6 :         RETURN_LONG(mpz_sgn(*gmpnum_a));
    1366                 : }
    1367                 : /* }}} */
    1368                 : 
    1369                 : /* {{{ proto resource gmp_random([int limiter])
    1370                 :    Gets random number */
    1371                 : ZEND_FUNCTION(gmp_random)
    1372              10 : {
    1373                 :         zval **limiter_arg;
    1374                 :         int limiter, argc;
    1375                 :         mpz_t *gmpnum_result;
    1376                 : 
    1377              10 :         argc = ZEND_NUM_ARGS();
    1378                 : 
    1379              10 :         if (argc == 0) {
    1380               1 :                 limiter = 20;
    1381              17 :         } else if (argc == 1 && zend_get_parameters_ex(1, &limiter_arg) == SUCCESS) {
    1382               8 :                 convert_to_long_ex(limiter_arg);
    1383               8 :                 limiter = Z_LVAL_PP(limiter_arg);
    1384                 :         } else {
    1385               1 :                 WRONG_PARAM_COUNT;
    1386                 :         }
    1387                 : 
    1388               9 :         INIT_GMP_NUM(gmpnum_result);
    1389                 : 
    1390               9 :         if (!GMPG(rand_initialized)) {
    1391                 :                 /* Initialize */
    1392               1 :                 gmp_randinit_lc_2exp_size(GMPG(rand_state), 32L);
    1393                 : 
    1394                 :                 /* Seed */
    1395               1 :                 gmp_randseed_ui(GMPG(rand_state), GENERATE_SEED());
    1396                 : 
    1397               1 :                 GMPG(rand_initialized) = 1;
    1398                 :         }
    1399               9 :         mpz_urandomb(*gmpnum_result, GMPG(rand_state), GMP_ABS (limiter) * __GMP_BITS_PER_MP_LIMB);
    1400                 : 
    1401               9 :         ZEND_REGISTER_RESOURCE(return_value, gmpnum_result, le_gmp);
    1402                 : }
    1403                 : /* }}} */
    1404                 : 
    1405                 : /* {{{ proto resource gmp_and(resource a, resource b)
    1406                 :    Calculates logical AND of a and b */
    1407                 : ZEND_FUNCTION(gmp_and)
    1408              13 : {
    1409              13 :         gmp_binary_op(mpz_and);
    1410              13 : }
    1411                 : /* }}} */
    1412                 : 
    1413                 : /* {{{ proto resource gmp_or(resource a, resource b)
    1414                 :    Calculates logical OR of a and b */
    1415                 : ZEND_FUNCTION(gmp_or)
    1416              13 : {
    1417              13 :         gmp_binary_op(mpz_ior);
    1418              13 : }
    1419                 : /* }}} */
    1420                 : 
    1421                 : /* {{{ proto resource gmp_com(resource a)
    1422                 :    Calculates one's complement of a */
    1423                 : ZEND_FUNCTION(gmp_com)
    1424              11 : {
    1425              11 :         gmp_unary_op(mpz_com);
    1426              11 : }
    1427                 : /* }}} */
    1428                 : 
    1429                 : /* {{{ proto resource gmp_nextprime(resource a)
    1430                 :    Finds next prime of a */
    1431                 : ZEND_FUNCTION(gmp_nextprime)
    1432               8 : {
    1433               8 :    gmp_unary_op(mpz_nextprime);
    1434               8 : }
    1435                 : /* }}} */
    1436                 : 
    1437                 : /* {{{ proto resource gmp_xor(resource a, resource b)
    1438                 :    Calculates logical exclusive OR of a and b */
    1439                 : ZEND_FUNCTION(gmp_xor)
    1440              13 : {
    1441                 :         /* use formula: a^b = (a|b)&^(a&b) */
    1442                 :         zval **a_arg, **b_arg;
    1443                 :         mpz_t *gmpnum_a, *gmpnum_b, *gmpnum_result, *gmpnum_t;
    1444                 : 
    1445              13 :         if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &a_arg, &b_arg) == FAILURE){
    1446               2 :                 WRONG_PARAM_COUNT;
    1447                 :         }
    1448                 : 
    1449              11 :         FETCH_GMP_ZVAL(gmpnum_a, a_arg);
    1450               8 :         FETCH_GMP_ZVAL(gmpnum_b, b_arg);
    1451                 : 
    1452               7 :         INIT_GMP_NUM(gmpnum_result);
    1453               7 :         INIT_GMP_NUM(gmpnum_t);
    1454                 : 
    1455               7 :         mpz_and(*gmpnum_t, *gmpnum_a, *gmpnum_b);
    1456               7 :         mpz_com(*gmpnum_t, *gmpnum_t);
    1457                 : 
    1458               7 :         mpz_ior(*gmpnum_result, *gmpnum_a, *gmpnum_b);
    1459               7 :         mpz_and(*gmpnum_result, *gmpnum_result, *gmpnum_t);
    1460                 : 
    1461               7 :         FREE_GMP_NUM(gmpnum_t);
    1462                 : 
    1463               7 :         ZEND_REGISTER_RESOURCE(return_value, gmpnum_result, le_gmp);
    1464                 : }
    1465                 : /* }}} */
    1466                 : 
    1467                 : /* {{{ proto void gmp_setbit(resource &a, int index[, bool set_clear])
    1468                 :    Sets or clear bit in a */
    1469                 : ZEND_FUNCTION(gmp_setbit)
    1470              12 : {
    1471                 :         zval **a_arg, **ind_arg, **set_c_arg;
    1472              12 :         int argc, index, set = 1;
    1473                 :         mpz_t *gmpnum_a;
    1474                 : 
    1475              12 :         argc = ZEND_NUM_ARGS();
    1476              12 :         if (argc < 2 || argc > 3 || zend_get_parameters_ex(argc, &a_arg, &ind_arg, &set_c_arg) == FAILURE){
    1477               2 :                 WRONG_PARAM_COUNT;
    1478                 :         }
    1479                 : 
    1480              10 :         ZEND_FETCH_RESOURCE(gmpnum_a, mpz_t *, a_arg, -1, GMP_RESOURCE_NAME, le_gmp);
    1481                 : 
    1482               7 :         convert_to_long_ex(ind_arg);
    1483               7 :         index = Z_LVAL_PP(ind_arg);
    1484                 : 
    1485               7 :         switch (argc) {
    1486                 :                 case 3:
    1487               6 :                         convert_to_long_ex(set_c_arg);
    1488               6 :                         set = Z_LVAL_PP(set_c_arg);
    1489               6 :                         break;
    1490                 :                 case 2:
    1491               1 :                         set = 1;
    1492                 :                         break;
    1493                 :         }
    1494                 : 
    1495               7 :         if (index < 0) {
    1496               1 :                 php_error_docref(NULL TSRMLS_CC, E_WARNING, "Index must be greater than or equal to zero");
    1497               1 :                 return;
    1498                 :         }
    1499                 : 
    1500               6 :         if (set) {
    1501               4 :                 mpz_setbit(*gmpnum_a, index);
    1502                 :         } else {
    1503               2 :                 mpz_clrbit(*gmpnum_a, index);
    1504                 :         }
    1505                 : }
    1506                 : /* }}} */
    1507                 : 
    1508                 : /* {{{ proto void gmp_clrbit(resource &a, int index)
    1509                 :    Clears bit in a */
    1510                 : ZEND_FUNCTION(gmp_clrbit)
    1511              11 : {
    1512                 :         zval **a_arg, **ind_arg;
    1513                 :         int index;
    1514                 :         mpz_t *gmpnum_a;
    1515                 : 
    1516              11 :         if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &a_arg, &ind_arg) == FAILURE){
    1517               3 :                 WRONG_PARAM_COUNT;
    1518                 :         }
    1519                 : 
    1520               8 :         ZEND_FETCH_RESOURCE(gmpnum_a, mpz_t *, a_arg, -1, GMP_RESOURCE_NAME, le_gmp);
    1521                 : 
    1522               7 :         convert_to_long_ex(ind_arg);
    1523               7 :         index = Z_LVAL_PP(ind_arg);
    1524                 :         
    1525               7 :         if (index < 0) {
    1526               2 :                 php_error_docref(NULL TSRMLS_CC, E_WARNING, "Index must be greater than or equal to zero");
    1527               2 :                 return;
    1528                 :         }
    1529                 : 
    1530               5 :         mpz_clrbit(*gmpnum_a, index);
    1531                 : }
    1532                 : /* }}} */
    1533                 : 
    1534                 : /* {{{ proto int gmp_popcount(resource a)
    1535                 :    Calculates the population count of a */
    1536                 : ZEND_FUNCTION(gmp_popcount)
    1537               8 : {
    1538                 :         zval **a_arg;
    1539                 :         mpz_t *gmpnum_a;
    1540                 : 
    1541               8 :         if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &a_arg) == FAILURE){
    1542               1 :                 WRONG_PARAM_COUNT;
    1543                 :         }
    1544                 :         
    1545               7 :         FETCH_GMP_ZVAL(gmpnum_a, a_arg);
    1546                 : 
    1547               6 :         RETURN_LONG(mpz_popcount(*gmpnum_a));
    1548                 : }
    1549                 : /* }}} */
    1550                 : 
    1551                 : /* {{{ proto int gmp_hamdist(resource a, resource b)
    1552                 :    Calculates hamming distance between a and b */
    1553                 : ZEND_FUNCTION(gmp_hamdist)
    1554              12 : {
    1555                 :         zval **a_arg, **b_arg;
    1556                 :         mpz_t *gmpnum_a, *gmpnum_b;
    1557                 : 
    1558              12 :         if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &a_arg, &b_arg) == FAILURE){
    1559               2 :                 WRONG_PARAM_COUNT;
    1560                 :         }
    1561                 :         
    1562              10 :         FETCH_GMP_ZVAL(gmpnum_a, a_arg);
    1563               8 :         FETCH_GMP_ZVAL(gmpnum_b, b_arg);
    1564                 : 
    1565               7 :         RETURN_LONG(mpz_hamdist(*gmpnum_a, *gmpnum_b));
    1566                 : }
    1567                 : /* }}} */
    1568                 : 
    1569                 : /* {{{ proto int gmp_scan0(resource a, int start)
    1570                 :    Finds first zero bit */
    1571                 : ZEND_FUNCTION(gmp_scan0)
    1572               9 : {
    1573                 :         zval **a_arg, **start_arg;
    1574                 :         mpz_t *gmpnum_a;
    1575                 : 
    1576               9 :         if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &a_arg, &start_arg) == FAILURE){
    1577               2 :                 WRONG_PARAM_COUNT;
    1578                 :         }
    1579                 : 
    1580               7 :         FETCH_GMP_ZVAL(gmpnum_a, a_arg);
    1581               6 :         convert_to_long_ex(start_arg);
    1582                 : 
    1583               6 :         if (Z_LVAL_PP(start_arg) < 0) {
    1584               1 :                 php_error_docref(NULL TSRMLS_CC, E_WARNING, "Starting index must be greater than or equal to zero");
    1585               1 :                 RETURN_FALSE;
    1586                 :         }
    1587                 : 
    1588               5 :         RETURN_LONG(mpz_scan0(*gmpnum_a, Z_LVAL_PP(start_arg)));
    1589                 : }
    1590                 : /* }}} */
    1591                 : 
    1592                 : /* {{{ proto int gmp_scan1(resource a, int start)
    1593                 :    Finds first non-zero bit */
    1594                 : ZEND_FUNCTION(gmp_scan1)
    1595               9 : {
    1596                 :         zval **a_arg, **start_arg;
    1597                 :         mpz_t *gmpnum_a;
    1598                 : 
    1599               9 :         if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &a_arg, &start_arg) == FAILURE){
    1600               2 :                 WRONG_PARAM_COUNT;
    1601                 :         }
    1602                 : 
    1603               7 :         FETCH_GMP_ZVAL(gmpnum_a, a_arg);
    1604               6 :         convert_to_long_ex(start_arg);
    1605                 :         
    1606               6 :         if (Z_LVAL_PP(start_arg) < 0) {
    1607               1 :                 php_error_docref(NULL TSRMLS_CC, E_WARNING, "Starting index must be greater than or equal to zero");
    1608               1 :                 RETURN_FALSE;
    1609                 :         }
    1610                 : 
    1611               5 :         RETURN_LONG(mpz_scan1(*gmpnum_a, Z_LVAL_PP(start_arg)));
    1612                 : }
    1613                 : /* }}} */
    1614                 : 
    1615                 : /* {{{ _php_gmpnum_free
    1616                 :  */
    1617                 : static void _php_gmpnum_free(zend_rsrc_list_entry *rsrc TSRMLS_DC)
    1618            2679 : {
    1619            2679 :         mpz_t *gmpnum = (mpz_t *)rsrc->ptr;
    1620                 : 
    1621            2679 :         FREE_GMP_NUM(gmpnum);
    1622            2679 : }
    1623                 : /* }}} */
    1624                 : 
    1625                 : #endif  /* HAVE_GMP */
    1626                 : 
    1627                 : /*
    1628                 :  * Local variables:
    1629                 :  * tab-width: 4
    1630                 :  * c-basic-offset: 4
    1631                 :  * End:
    1632                 :  * vim600: noet sw=4 ts=4 fdm=marker
    1633                 :  * vim<600: noet sw=4 ts=4
    1634                 :  */

Generated by: LTP GCOV extension version 1.5

Generated at Thu, 19 Nov 2009 08:20:09 +0000 (5 days ago)

Copyright © 2005-2009 The PHP Group
All rights reserved.