Annotation of 43BSD/contrib/icon/operators/power.c, revision 1.1

1.1     ! root        1: #include "../h/rt.h"
        !             2: 
        !             3: /*
        !             4:  * x ^ y - raise x to the y power.
        !             5:  */
        !             6: 
        !             7: power(nargs, arg2, arg1, arg0)
        !             8: int nargs;
        !             9: struct descrip arg2, arg1, arg0;
        !            10:    {
        !            11:    register int t1, t2;
        !            12:    union numeric n1, n2;
        !            13:    extern double pow();
        !            14:    extern long ipow();
        !            15: 
        !            16:    SetBound;
        !            17:    /*
        !            18:     * x and y must be numeric.  Save the cvnum return values for later use.
        !            19:     */
        !            20:    if ((t1 = cvnum(&arg1, &n1)) == NULL)
        !            21:       runerr(102, &arg1);
        !            22:    if ((t2 = cvnum(&arg2, &n2)) == NULL)
        !            23:       runerr(102, &arg2);
        !            24: 
        !            25:    if (t1 == T_LONGINT && t2 == T_LONGINT)
        !            26:       /*
        !            27:        * Both x and y are integers.  Perform integer exponentiation
        !            28:        *  and place the result in arg0 as the return value.
        !            29:        */
        !            30:       mkint(ipow(n1.integer, n2.integer), &arg0);
        !            31:    else {
        !            32:       /*
        !            33:        * Either x or y is real, convert the other to a real, perform
        !            34:        *  real exponentiation and place the result in arg0 as the
        !            35:        *  return value.
        !            36:        */
        !            37:       if (t1 == T_LONGINT)
        !            38:          n1.real = n1.integer;
        !            39:       if (t2 == T_LONGINT)
        !            40:          n2.real = n2.integer;
        !            41:       if (n1.real == 0.0 && n2.real <= 0.0)
        !            42:          /*
        !            43:           * Tried to raise zero to a negative power.
        !            44:           */
        !            45:          runerr(204, NULL);
        !            46:       if (n1.real < 0.0 && t2 == T_REAL)
        !            47:          /*
        !            48:           * Tried to raise a negative number to a real power.
        !            49:           */
        !            50:          runerr(206, NULL);
        !            51:       mkreal(pow(n1.real,n2.real), &arg0);
        !            52:       }
        !            53:    ClearBound;
        !            54:    }
        !            55: 
        !            56: Opblock(power,2,"^")
        !            57: 
        !            58: long ipow(n1, n2)
        !            59: long n1, n2;
        !            60:    {
        !            61:    long result;
        !            62: 
        !            63:    if (n1 == 0 && n2 <= 0)
        !            64:       runerr(204, NULL);
        !            65:    if (n2 < 0)
        !            66:       return (0.0);
        !            67:    result = 1L;
        !            68:    while (n2 > 0) {
        !            69:       if (n2 & 01L)
        !            70:          result *= n1;
        !            71:       n1 *= n1;
        !            72:       n2 >>= 1;
        !            73:       }
        !            74:    return (result);
        !            75:    }
        !            76: 

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.