Retired Document
Important: This sample code may not represent best practices for current development. The project may use deprecated symbols and illustrate technologies and techniques that are no longer recommended.
gmp/piGmp/piGmp.c
// |
// |
// File: <piGmp.c> |
// |
// Abstract: Calculate pi via the Gauss-Legendre algorithm, using the GMP library |
// for MP floating point arithmetic. |
// See <http://en.wikipedia.org/wiki/Gauss-Legendre_algorithm> for a good |
// discussion of this algorithm. |
// |
// Version: <1.0 > |
// |
// Disclaimer: IMPORTANT: This Apple software is supplied to you by |
// Apple Inc. ( "Apple" ) in consideration of your agreement to the |
// following terms, and your use, installation, modification or |
// redistribution of this Apple software constitutes acceptance of these |
// terms. If you do not agree with these terms, please do not use, |
// install, modify or redistribute this Apple software. |
// |
// In consideration of your agreement to abide by the following terms, and |
// subject to these terms, Apple grants you a personal, non - exclusive |
// license, under Apple's copyrights in this original Apple software ( the |
// "Apple Software" ), to use, reproduce, modify and redistribute the Apple |
// Software, with or without modifications, in source and / or binary forms; |
// provided that if you redistribute the Apple Software in its entirety and |
// without modifications, you must retain this notice and the following |
// text and disclaimers in all such redistributions of the Apple Software. |
// Neither the name, trademarks, service marks or logos of Apple Inc. |
// may be used to endorse or promote products derived from the Apple |
// Software without specific prior written permission from Apple. Except |
// as expressly stated in this notice, no other rights or licenses, express |
// or implied, are granted by Apple herein, including but not limited to |
// any patent rights that may be infringed by your derivative works or by |
// other works in which the Apple Software may be incorporated. |
// |
// The Apple Software is provided by Apple on an "AS IS" basis. APPLE |
// MAKES NO WARRANTIES, EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION |
// THE IMPLIED WARRANTIES OF NON - INFRINGEMENT, MERCHANTABILITY AND FITNESS |
// FOR A PARTICULAR PURPOSE, REGARDING THE APPLE SOFTWARE OR ITS USE AND |
// OPERATION ALONE OR IN COMBINATION WITH YOUR PRODUCTS. |
// |
// IN NO EVENT SHALL APPLE BE LIABLE FOR ANY SPECIAL, INDIRECT, INCIDENTAL |
// OR CONSEQUENTIAL DAMAGES ( INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS |
// INTERRUPTION ) ARISING IN ANY WAY OUT OF THE USE, REPRODUCTION, |
// MODIFICATION AND / OR DISTRIBUTION OF THE APPLE SOFTWARE, HOWEVER CAUSED |
// AND WHETHER UNDER THEORY OF CONTRACT, TORT ( INCLUDING NEGLIGENCE ), |
// STRICT LIABILITY OR OTHERWISE, EVEN IF APPLE HAS BEEN ADVISED OF THE |
// POSSIBILITY OF SUCH DAMAGE. |
// |
// Copyright ( C ) 2007 Apple Inc. All Rights Reserved. |
// |
// **************************************************** |
#pragma mark - |
#pragma mark * includes & imports * |
// ---------------------------------------------------- |
#include <stdlib.h> |
#include <strings.h> |
#include <stdio.h> |
#include <unistd.h> |
#include <gmp.h> |
#include <unistd.h> |
#include <mach/mach_time.h> |
// **************************************************** |
#pragma mark - |
#pragma mark * local ( static ) function prototypes * |
// ---------------------------------------------------- |
static void usage(char **argv); |
static double currentTime(void); |
static void calculatePi( unsigned bits, mpf_t pi); |
// **************************************************** |
#pragma mark - |
#pragma mark * exported function implementations * |
// ---------------------------------------------------- |
int main(int argc, char **argv) |
{ |
unsigned bits; |
double start, end; |
mpf_t pi; |
if(argc < 2) { |
usage(argv); |
} |
bits = atoi(argv[1]); |
if(bits == 0) { |
usage(argv); |
} |
printf("Calculating pi..."); |
fflush(stdout); |
start = currentTime(); |
mpf_set_default_prec(bits); |
mpf_init(pi); |
calculatePi(bits, pi); |
end = currentTime(); |
printf("\npi = "); |
mpf_out_str(stdout, 10, 0, pi); |
printf("\n"); |
printf("total elapsed time : %.2f seconds\n", end - start); |
return 0; |
} |
// **************************************************** |
#pragma mark - |
#pragma mark * local ( static ) function implementations * |
// ---------------------------------------------------- |
static void usage(char **argv) |
{ |
printf("usage: %s bits\n", argv[0]); |
exit(1); |
} |
static double currentTime(void) |
{ |
static double scale = 0.0; |
if (0.0 == scale) { |
mach_timebase_info_data_t info; |
mach_timebase_info(&info); |
scale = info.numer / info.denom * 1e-9; |
} |
return mach_absolute_time() * scale; |
} |
// |
// The core Gauss-Legendre routine. |
// On input, 'bits' is the desired precision in bits. |
// On output, 'pi' contains the calculated value. |
// |
static void calculatePi( unsigned bits, mpf_t pi) |
{ |
mpf_t lastPi; |
mpf_t scratch; |
// variables per the formal Gauss-Legendre formulae |
mpf_t a; |
mpf_t b; |
mpf_t t; |
mpf_t x; |
mpf_t y; |
unsigned p = 1; |
mpf_init_set_ui(lastPi, 0); |
mpf_init(x); |
mpf_init(y); |
mpf_init(scratch); |
// initial conditions |
mpf_init_set_ui(a, 1); // a := 1 |
mpf_init(b); // b := 1 / sqrt(2) |
mpf_sqrt_ui(b, 2); |
mpf_ui_div(b, 1, b); |
mpf_init_set_ui(t, 4); // t := 1/4 |
mpf_ui_div(t, 1, t); |
for(;;) { |
// x := (a+b)/2 |
mpf_add(x, a, b); |
mpf_div_ui(x, x, 2); |
// y := sqrt(a*b) |
mpf_mul(y, a, b); |
mpf_sqrt(y, y); |
// t := t - p * (a-x)**2 |
mpf_sub(scratch, a, x); |
mpf_pow_ui(scratch, scratch, 2); |
mpf_mul_ui(scratch, scratch, p); |
mpf_sub(t, t, scratch); |
// a := x |
// b := y |
// p := 2p |
mpf_set(a, x); |
mpf_set(b, y); |
p <<= 1; |
// pi := ((a + b)**2) / 4t |
mpf_add(pi, a, b); |
mpf_pow_ui(pi, pi, 2); |
mpf_mul_ui(scratch, t, 4); |
mpf_div(pi, pi, scratch); |
// if pi == lastPi, within the requested precision, we're done |
if(mpf_eq(pi, lastPi, bits)) { |
break; |
} |
mpf_set(lastPi, pi); |
} |
// free memory associated with mpf_t's we allocated |
mpf_clear(a); |
mpf_clear(b); |
mpf_clear(t); |
mpf_clear(x); |
mpf_clear(y); |
mpf_clear(lastPi); |
mpf_clear(scratch); |
} |
Copyright © 2007 Apple Inc. All Rights Reserved. Terms of Use | Privacy Policy | Updated: 2007-03-26