/* Author: G. Jungman * RCS: $Id: rk2.c,v 1.6 1999/08/26 08:54:37 jungman Exp $ */ #include #include #include #include #include "odeiv_util.h" #include "gsl_odeiv.h" typedef struct gsl_odeiv_step_rk2_struct gsl_odeiv_step_rk2; struct gsl_odeiv_step_rk2_struct { gsl_odeiv_step parent; /* inherits from gsl_odeiv_step */ double * work; /* generic work space */ }; static int rk2_step(void * self, double t, double h, double y[], double yerr[], const double dydt_in[], double dydt_out[], const gsl_odeiv_system * dydt); static void rk2_free(void * self); gsl_odeiv_step * gsl_odeiv_step_rk2_new(void) { gsl_odeiv_step_rk2 * s = (gsl_odeiv_step_rk2 *) malloc(sizeof(gsl_odeiv_step_rk2)); if(s != 0) { gsl_odeiv_step_construct(&(s->parent), "rk2", rk2_step, 0, rk2_free, 0, 0, 2); s->work = 0; } return (gsl_odeiv_step *) s; } static int rk2_step( void * self, double t, double h, double y[], double yerr[], const double dydt_in[], double dydt_out[], const gsl_odeiv_system * sys) { int i; int status = 0; size_t dim; double * k1; double * k2; double * k3; double * ytmp; gsl_odeiv_step_rk2 * my = (gsl_odeiv_step_rk2 *) self; if(sys->dimension == 0) { return GSL_EINVAL; } if(sys->dimension != my->parent.dimension) { if(my->work != 0) free(my->work); my->parent.dimension = sys->dimension; my->work = (double *) malloc(4 * sys->dimension * sizeof(double)); if(my->work == 0) { my->parent.dimension = 0; return GSL_ENOMEM; } } dim = my->parent.dimension; /* divide up the work space */ k1 = my->work; k2 = my->work + dim; k3 = my->work + 2*dim; ytmp = my->work + 3*dim; /* k1 step */ if(dydt_in != 0) { memcpy(k1, dydt_in, dim * sizeof(double)); } else { status += ( GSL_ODEIV_FN_EVAL(sys, t, y, k1) != 0 ); } for(i=0;iwork != 0) free(my->work); free(self); } }