comparison rDiff/mex/interval_overlap.cpp @ 0:0f80a5141704

version 0.3 uploaded
author vipints
date Thu, 14 Feb 2013 23:38:36 -0500
parents
children
comparison
equal deleted inserted replaced
-1:000000000000 0:0f80a5141704
1 /*
2 * This program is free software; you can redistribute it and/or modify
3 * it under the terms of the GNU General Public License as published by
4 * the Free Software Foundation; either version 3 of the License, or
5 * (at your option) any later version.
6 *
7 * Written (W) 2010-2011 Jonas Behr
8 * Copyright (C) 2010-2011 Max Planck Society
9 */
10
11
12 #include <stdio.h>
13 #include <stdarg.h>
14 #include <errno.h>
15 #include <sys/types.h>
16 #include <sys/stat.h>
17 #include <fcntl.h>
18 #include <ctype.h>
19 #include <sys/stat.h>
20 #include <stdlib.h>
21 #include <unistd.h>
22 #include <string.h>
23 #include <sys/mman.h>
24 #include <time.h>
25 #include <math.h>
26 #include <limits>
27 #include <mex.h>
28 #include <assert.h>
29 #include <vector>
30 using std::vector;
31 #include <algorithm>
32 using std::sort;
33 using std::min;
34 using std::max;
35
36 typedef struct {
37 int start;
38 int stop;
39 int idx;
40 int set_id;
41 } interval_t;
42
43 bool compare (interval_t i, interval_t j)
44 {
45 return (i.start<j.start);
46 }
47
48 bool overlaps(interval_t a, interval_t b)
49 {
50 int v = min(a.stop,b.stop) - max(a.start,b.start) + 1;
51 return (v >= 1);
52 }
53 bool leftOf(interval_t a, interval_t b)
54 {
55 return (a.stop < b.start);
56 }
57
58 void scan(interval_t f, vector<interval_t>* Wf, interval_t g, vector<interval_t>* Wg, vector<int>* overlap)
59 {
60 vector<interval_t>::iterator i;
61 i=Wg->begin();
62 while (i<Wg->end())
63 {
64 interval_t g2 = *i;
65 if (leftOf(g2,f))
66 {
67 Wg->erase(i);// inefficient if Wg is large
68 // this moves all elements, therefore i is not incremented
69 }
70 else if (overlaps(g2,f))
71 {
72 if (g2.set_id==1)
73 {
74 //printf("overlap: [%i | %i, %i] [%i | %i, %i]\n", g2.idx, g2.start, g2.stop, f.idx, f.start, f.stop);
75 overlap->push_back(g2.idx);
76 overlap->push_back(f.idx);
77 }
78 else if (f.set_id==1)
79 {
80 //printf("overlap: [%i | %i, %i] [%i | %i, %i]\n", f.idx, f.start, f.stop, g2.idx, g2.start, g2.stop);
81 overlap->push_back(f.idx);
82 overlap->push_back(g2.idx);
83 }
84 i++;
85 }
86 else
87 {
88 printf("never happens??\n");
89 i++;
90 }
91 }
92 if (!leftOf(f, g))
93 {
94 Wf->push_back(f);
95 //printf("push: [%i, %i] size:%i\n", f.start, f.stop, Wf->size());
96 }
97 }
98
99 /*
100 * prhs[0] first interval set starts
101 * prhs[1] first interval set stops
102 * prhs[2] second interval set starts
103 * prhs[3] second interval set stops
104 *
105 * return:
106 * plhs[0] one based index in first interval set overlapping with a interval in the second set
107 * plhs[1] corresponding index in the second set
108 *
109 */
110 void mexFunction(int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[])
111 {
112 if (nrhs!=4)
113 mexErrMsgTxt("Expected 4 arguments: starts1, stops1, starts2, stops2 \n");
114 if (nlhs!=2)
115 mexErrMsgTxt("Expected 2 output arguments \n");
116
117 int num_intervals1 = mxGetNumberOfElements(prhs[0]);
118 assert(num_intervals1 == mxGetNumberOfElements(prhs[1]));
119 int num_intervals2 = mxGetNumberOfElements(prhs[2]);
120 assert(num_intervals2 == mxGetNumberOfElements(prhs[3]));
121
122 //printf("num_intervals1: %i\n", num_intervals1);
123 //printf("num_intervals2: %i\n", num_intervals2);
124
125 double* starts1 = mxGetPr(prhs[0]);
126 double* stops1 = mxGetPr(prhs[1]);
127 double* starts2 = mxGetPr(prhs[2]);
128 double* stops2 = mxGetPr(prhs[3]);
129
130 vector<interval_t> intervals1;
131 for (int i=0; i<num_intervals1; i++)
132 {
133 interval_t interval;
134 interval.start = starts1[i];
135 interval.stop = stops1[i];
136 interval.set_id = 1;
137 interval.idx = i+1;
138 intervals1.push_back(interval);
139 //printf("int1: [%i, %i] \n",intervals1[i].start, intervals1[i].stop);
140 }
141 interval_t i;
142 i.start = std::numeric_limits<int>::max();
143 i.stop = std::numeric_limits<int>::max();
144 i.set_id = std::numeric_limits<int>::max();
145 i.idx = std::numeric_limits<int>::max();
146 intervals1.push_back(i);
147
148 //printf("num_intervals1: %i\n", intervals1.size());
149 vector<interval_t> intervals2;
150 for (int i=0; i<num_intervals2; i++)
151 {
152 interval_t interval;
153 interval.start = starts2[i];
154 interval.stop = stops2[i];
155 interval.set_id = 2;
156 interval.idx = i+1;
157 intervals2.push_back(interval);
158 //printf("int2: [%i, %i] \n",intervals2[i].start, intervals2[i].stop);
159 }
160 intervals2.push_back(i);
161 //printf("num_intervals2: %i\n", intervals2.size());
162
163 sort(intervals1.begin(), intervals1.end(), compare);
164 sort(intervals2.begin(), intervals2.end(), compare);
165
166
167 vector<int> overlap;
168 vector<interval_t> Wx;
169 vector<interval_t> Wy;
170 vector<interval_t>::iterator x = intervals1.begin();
171 vector<interval_t>::iterator y = intervals2.begin();
172 while (x<intervals1.end() && y<intervals2.end())
173 {
174 //vector<interval_t>::iterator x;
175 //vector<interval_t>::iterator y;
176 //if (it1>intervals1.end())
177 // x = inf_interval();
178 //else
179 // x = it1;
180 //if (it2>intervals2.end())
181 // y = inf_interval();
182 //else
183 // y=it2;
184
185 if (x->start <= y->start)
186 {
187 scan(*x, &Wx, *y, &Wy, &overlap);
188 x++;
189 }
190 else
191 {
192 if (y<=intervals2.end())
193 {
194 scan(*y, &Wy, *x, &Wx, &overlap);
195 y++;
196 }
197 }
198 }
199
200 plhs[0] = mxCreateDoubleMatrix(1, overlap.size()/2, mxREAL);
201 double* idx1 = mxGetPr(plhs[0]);
202
203 plhs[1] = mxCreateDoubleMatrix(1, overlap.size()/2, mxREAL);
204 double* idx2 = mxGetPr(plhs[1]);
205
206 for (int i=0; i<overlap.size(); i+=2)
207 {
208 //printf("ov: %i [%i, %i] \n", i, overlap[i], overlap[i+1]);
209 idx1[i/2] = (double) overlap[i];
210 idx2[i/2] = (double) overlap[i+1];
211 }
212 }
213
214
215
216
217