-
Notifications
You must be signed in to change notification settings - Fork 21
/
Copy pathpyears2.c
156 lines (148 loc) · 4.79 KB
/
pyears2.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
/*
** Person-years calculations.
** same as pyears1, but no expected rates
**
** Input:
** n number of subjects
** ny number of columns of y.
** doevent does y have an 'events' column? 1=yes, 0=no
** if ny=2 and doevent=1, then "start" is missing.
** y[3,n] contains start, stop, and event for each subject
** wt contains the case weights
**
** output table's description
** odim number of dimensions
** ofac[odim] 1=is a factor, 0=continuous (time based)
** odims[odim] the number of rows, columns, etc
** ocut[] for each non-factor dimension, the odim[i]+1 cutpoints
** that define the intervals; concatonated.
** odata[odim, n] the subject data-- where each indexes into the
** expected table, at time 0.
**
** Output:
** pyears output table of person years
** pn number of observations that contribute to each cell
** pcount number of events
** offtable total person years that did not fall into the output table
**
** Scratch allocated on the fly
** scratch[edim]
*/
#include "survS.h"
#include "survproto.h"
/* names that begin with "s" will be re-declared in the main body */
void pyears2(int *sn, int *sny, int *sdoevent,
double *sy, double *wt, int *sodim, int *ofac,
int *odims, double *socut, double *sodata,
double *pyears, double *pn, double *pcount,
double *offtable)
{
int i,j;
int n,
ny,
doevent,
odim;
double *start,
*stop,
*event,
**ocut,
**odata;
double *data;
double timeleft,
thiscell;
int index;
int dostart;
int d1; /* a dummy */
double d2; /* a dummy for pystep */
double eps; /* round off protection */
n = *sn;
ny= *sny;
doevent = *sdoevent;
odim = *sodim;
start = sy;
if (ny==3 || (ny==2 && doevent==0)) {
/* each subject has a "start" time */
stop = sy +n;
dostart =1;
}
else {
/* followup starts at time 0 */
stop = sy;
dostart =0;
}
event = stop +n;
odata = dmatrix(sodata, n, odim);
data = (double *) ALLOC(odim, sizeof(double));
/*
** will be a ragged array
*/
ocut = (double **)ALLOC(odim, sizeof(double *));
for (i=0; i<odim; i++) {
ocut[i] = socut;
if (ofac[i]==0) socut += odims[i] +1;
}
/*
** Set the round off error to min(time[time>0]) * 1e-8
** The events are counted in the last cell to which person years are
** added in the while() loop below. We don't want to "spill over" into
** a next (incorrect) cell due to accumulated round off, in the case
** that a subjects fu time exactly matches one of the cell boundaries.
*/
eps =0; /* guard against the rare case that all(time==0) */
for (i=0; i<n; i++) {
if (dostart==1) timeleft = stop[i] - start[i];
else timeleft= stop[i];
if (timeleft >0) {
eps = timeleft; /* starting guess for min = first non-zero value*/
break;
}
}
for (; i<n; i++) {
if (dostart==1) timeleft = stop[i] - start[i];
else timeleft= stop[i];
if ((timeleft >0) && (timeleft < eps)) eps = timeleft;
}
eps *= 1e-8;
*offtable =0;
for (i=0; i<n; i++) {
R_CheckUserInterrupt(); /* in case of long calculations */
/*
** initialize
** "data" will be the vector of starting values for each subject
** for a factor variable this is just the cell number (1,2,3,...)
** for a continuous one it is the value, which will be matched to
** the ocuts list, by pystep, to figure out a cell number
*/
for (j=0; j<odim; j++) {
if (ofac[j] ==1 || dostart==0) data[j] = odata[j][i];
else data[j] = odata[j][i] + start[i];
}
if (dostart==1) timeleft = stop[i] - start[i];
else timeleft = stop[i];
/*
** add up p-yrs
** d1 and d2 are only relevant to rate tables, so ignored here
** If there are only factor variables, the loop below will finish in
** one iteration. Otherwise, the value "thiscell" is the amount of
** person-years in the current cell, up to the next cell boundary
** (or the amount of time left, whichever is lesser).
*/
if (timeleft <=eps && doevent) {
/* we have to call pystep at least once to set the index */
pystep(odim, &index, &d1, &d2, data, ofac, odims, ocut, 1, 0);
}
while (timeleft > eps) {
thiscell = pystep(odim, &index, &d1, &d2, data, ofac, odims, ocut,
timeleft, 0);
if (index >=0) {
pyears[index] += thiscell * wt[i];
pn[index] += 1;
}
else *offtable += thiscell * wt[i];
for (j=0; j<odim; j++)
if (ofac[j] ==0) data[j] += thiscell;
timeleft -= thiscell;
}
if (index >=0 && doevent) pcount[index] += event[i] * wt[i];
}
}