/* calc_dose.cpp */ /* The dose at all voxels in the grid dose_mask is calculated using a convolution method that uses a polyenergetic kernel. Inhomogeneities are accounted for by kernel scaling, and an inverse square correction is applied after the convolution of the terma grid with the kernel, rather than being applied directly to the terma grid before the convolution. */ #include #include #include #include #include #include "defs.h" extern char errstr[200]; // error string that all routines have access to int calc_dose(FLOAT_GRID *density, FLOAT_GRID *terma,FLOAT_GRID *dose, KERNEL *kern, BEAM *bm, FLOAT_GRID *dose_mask) { int i, j, k, l; int p, q, r; int M, N, Q; // dimensions of CT array int inter_i, inter_j, inter_k; int r_index; int baseindex; float SAD; float del_x, del_y, del_z; float current_x, current_y, current_z; float inter_x, inter_y, inter_z; float r_eff, delr_eff, inter_r_eff; float kval; float cumval, last_cumval; float length; float *ip, *jp, *kp; float dx, dy, dz; float delr; // convolution step size float t, u, v, f; float *x, *y, *z; // vectors of CT coordinates float *phi, *theta, *sinphi, *cosphi, *sintheta, *costheta; float one; one = (float)1.0; // copy CT dimensions and voxel sizes for shorter references later M = density->x_count; N = density->y_count; Q = density->z_count; dx = density->inc.x; dy = density->inc.y; dz = density->inc.z; // copy vectors describing the beam's eye view coordinate system as well ip = fvector(0,2); // ip and jp span the beam's eye view jp = fvector(0,2); kp = fvector(0,2); // beam direction // create the unit vector describing the beam direction for (j=0;j<3;j++) ip[j] = bm->ip[j]; for (j=0;j<3;j++) jp[j] = bm->jp[j]; for (j=0;j<3;j++) kp[j] = bm->kp[j]; // vectors describing the location of each voxel x = fvector(0,M-1); y = fvector(0,N-1); z = fvector(0,Q-1); // lookup table for vectors in polar coordinates phi = fvector(0,NPHI-1); theta = fvector(0,NTHETA-1); sinphi = fvector(0,NPHI-1); cosphi = fvector(0,NPHI-1); sintheta = fvector(0,NTHETA-1); costheta = fvector(0,NTHETA-1); //kernel with fewer elements for faster calculation //see defs.h KERNEL smallkern; smallkern.radial_boundary = (float *)malloc(kern->nradii*sizeof(float)); smallkern.angular_boundary = (float *)malloc(kern->ntheta*sizeof(float)); //small kernel dimensions smallkern.nradii = N_KERNEL_RADII; //same as original kernel smallkern.ntheta = NTHETA; //must divide evenly into N_KERNEL_ANGLES SAD = bm->SAD; for (i=0;iradial_boundary[i]; //initialise for (i=0;i 0) KERNEL_VALUE(&smallkern,i,r,p) = KERNEL_VALUE(&smallkern,i,r-1,p) + KERNEL_VALUE(&smallkern,i,r,p); if (r > 0) KERNEL_TOTAL_VALUE(&smallkern,r,p) = KERNEL_TOTAL_VALUE(&smallkern,r-1,p) + KERNEL_TOTAL_VALUE(&smallkern,r,p); } // fill the coordinate vectors for (i=0;istart.x + (float)i*dx; for (j=0;jstart.y + (float)j*dy; for (k=0;kstart.z + (float)k*dz; // fill in the polar coordinates vectors for (q=0;q 0) // only calculate dose inside dose mask { // do the integral for the point in the ROI for (p=0;pstart.x)/dx); inter_j = (int) ((inter_y - density->start.y)/dy); inter_k = (int) ((inter_z - density->start.z)/dz); // stop the integral if interaction point is outside the dose calculation limits if ( (inter_i < 0) || (inter_i + 1 >= M) || (inter_j < 0) || (inter_j + 1 >= N) || (inter_k < 0) || (inter_k + 1 >= Q) || (GRID_VALUE(dose_mask,inter_i,inter_j,inter_k) <= 0.0)) break; // Position of the end of the increment. Interaction point is at the // midpoint. current_x += del_x; current_y += del_y; current_z += del_z; //effective distance increment delr_eff = delr*GRID_VALUE(density,inter_i,inter_j,inter_k); //effective radius of interaction point inter_r_eff = r_eff + (float)0.5*delr_eff; r_eff += delr_eff; // trilinear interpolation method of the terma contribution, f // relative differences between the interaction point and the lower voxel bound t = (inter_x - x[inter_i])/dx; u = (inter_y - y[inter_j])/dy; v = (inter_z - z[inter_k])/dz; f = GRID_VALUE(terma,inter_i,inter_j,inter_k)*(one-t)*(one-u)*(one-v) + GRID_VALUE(terma,inter_i,inter_j,inter_k+1)*(one-t)*(one-u)*v + GRID_VALUE(terma,inter_i,inter_j+1,inter_k+1)*(one-t)*u*v + GRID_VALUE(terma,inter_i+1,inter_j+1,inter_k+1)*t*u*v + GRID_VALUE(terma,inter_i,inter_j+1,inter_k)*(one-t)*u*(one-v) + GRID_VALUE(terma,inter_i+1,inter_j+1,inter_k)*t*u*(one-v) + GRID_VALUE(terma,inter_i+1,inter_j,inter_k+1)*t*(one-u)*v + GRID_VALUE(terma,inter_i+1,inter_j,inter_k)*t*(one-u)*(one-v); // perform kernel lookup for r_eff, r_index is the kernel index of the first // bin boundary below the effective radius of the voxel r_index = binSearch(smallkern.radial_boundary,inter_r_eff,smallkern.nradii); // interpolate the obtain the effective cumulative kernel value if (r_index == -1) // radius is between zero and the first bin boundary { // fractional difference between inter_r_eff and the first bin boundary t = inter_r_eff/smallkern.radial_boundary[0]; cumval = (1-t)*KERNEL_TOTAL_VALUE(&smallkern,0,p); } else if (r_index >= smallkern.nradii-1) // overshot the kernel bin boundaries { cumval = KERNEL_TOTAL_VALUE(&smallkern,smallkern.nradii-1,p); } else // inter_r_eff is between the first upper bin boundary and the last { t = (inter_r_eff - smallkern.radial_boundary[r_index]) /(smallkern.radial_boundary[r_index + 1] - smallkern.radial_boundary[r_index]); cumval = (1-t)*KERNEL_TOTAL_VALUE(&smallkern,r_index,p) + t*KERNEL_TOTAL_VALUE(&smallkern,r_index+1,p); } kval = cumval - last_cumval; last_cumval = cumval; // Kernel value to use is current increment in cumulative value // Note that this is the fractional dose deposited at i,j,k due to // terma in an effective increment (increment*density) along the kernel ray at the // interaction point. The value comes from the fractional dose deposited in an // effective increment along the ray at i,j,k due to terma at the current // interaction point. //Increment dose value. //Note that to include the effect of density at i,j,k on ENERGY deposited at i,j,k, //there should be a density term, but this cancels on conversion of energy to dose as indicated below // if (GRID_VALUE(terma,inter_i,inter_j,inter_k) > 0.001) GRID_VALUE(dose,i,j,k) += f*kval; r++; } while (r<10000); } //p,q GRID_VALUE(dose,i,j,k)/= NPHI; } //Inverse square correction to dose //This works better than applying the inverse square correction to terma //See Papanikolaou and Mackie 1993 for (k=0;k 0.0) { // squared difference between the source and the voxel length = (float)pow(x[i] - bm->y_vec[0],2.0) + (float)pow(y[j] - bm->y_vec[1],2.0) + (float)pow(z[k] - bm->y_vec[2],2.0); if (length > 0.0) GRID_VALUE(dose,i,j,k) *= SAD*SAD/length; } // free vectors free_fvector(ip,0,2); free_fvector(jp,0,2); free_fvector(kp,0,2); free_fvector(x,0,M-1); free_fvector(y,0,N-1); free_fvector(z,0,Q-1); free(smallkern.angular_boundary); free(smallkern.radial_boundary); // free(poly_kernel.total_matrix); for (j=0;j