fix ucspi-tcp compile
[packages.git] / net / djbdns / patches / 310-bugfix2-merge-outgoing-requests.patch
1 diff --git a/Makefile b/Makefile
2 index 1429643..bc047c0 100644
3 --- a/Makefile
4 +++ b/Makefile
5 @@ -318,11 +318,11 @@ stralloc.h iopause.h taia.h tai.h uint64.h taia.h
6         ./compile dns_txt.c
7  
8  dnscache: \
9 -load dnscache.o droproot.o okclient.o log.o cache.o query.o \
10 +load dnscache.o droproot.o okclient.o log.o cache.o query.o qmerge.o \
11  response.o dd.o roots.o iopause.o prot.o dns.a env.a alloc.a buffer.a \
12  libtai.a unix.a byte.a socket.lib
13         ./load dnscache droproot.o okclient.o log.o cache.o \
14 -       query.o response.o dd.o roots.o iopause.o prot.o dns.a \
15 +       query.o qmerge.o response.o dd.o roots.o iopause.o prot.o dns.a \
16         env.a alloc.a buffer.a libtai.a unix.a byte.a  `cat \
17         socket.lib`
18  
19 @@ -343,7 +343,7 @@ compile dnscache.c env.h exit.h scan.h strerr.h error.h ip4.h \
20  uint16.h uint64.h socket.h uint16.h dns.h stralloc.h gen_alloc.h \
21  iopause.h taia.h tai.h uint64.h taia.h taia.h byte.h roots.h fmt.h \
22  iopause.h query.h dns.h uint32.h alloc.h response.h uint32.h cache.h \
23 -uint32.h uint64.h ndelay.h log.h uint64.h okclient.h droproot.h
24 +uint32.h uint64.h ndelay.h log.h uint64.h okclient.h droproot.h maxclient.h
25         ./compile dnscache.c
26  
27  dnsfilter: \
28 @@ -687,11 +687,16 @@ qlog.o: \
29  compile qlog.c buffer.h qlog.h uint16.h
30         ./compile qlog.c
31  
32 +qmerge.o: \
33 +compile qmerge.c qmerge.h dns.h stralloc.h gen_alloc.h iopause.h \
34 +taia.h tai.h uint64.h log.h maxclient.h
35 +       ./compile qmerge.c
36 +
37  query.o: \
38  compile query.c error.h roots.h log.h uint64.h case.h cache.h \
39  uint32.h uint64.h byte.h dns.h stralloc.h gen_alloc.h iopause.h \
40  taia.h tai.h uint64.h taia.h uint64.h uint32.h uint16.h dd.h alloc.h \
41 -response.h uint32.h query.h dns.h uint32.h
42 +response.h uint32.h query.h dns.h uint32.h qmerge.h
43         ./compile query.c
44  
45  random-ip: \
46 diff --git a/dnscache.c b/dnscache.c
47 index 8c899a3..5ccb16a 100644
48 --- a/dnscache.c
49 +++ b/dnscache.c
50 @@ -22,6 +22,7 @@
51  #include "log.h"
52  #include "okclient.h"
53  #include "droproot.h"
54 +#include "maxclient.h"
55  
56  static int packetquery(char *buf,unsigned int len,char **q,char qtype[2],char qclass[2],char id[2])
57  {
58 @@ -54,7 +55,6 @@ uint64 numqueries = 0;
59  
60  static int udp53;
61  
62 -#define MAXUDP 200
63  static struct udpclient {
64    struct query q;
65    struct taia start;
66 @@ -131,7 +131,6 @@ void u_new(void)
67  
68  static int tcp53;
69  
70 -#define MAXTCP 20
71  struct tcpclient {
72    struct query q;
73    struct taia start;
74 diff --git a/log.c b/log.c
75 index c43e8b0..b8cd7ce 100644
76 --- a/log.c
77 +++ b/log.c
78 @@ -150,6 +150,13 @@ void log_tx(const char *q,const char qtype[2],const char *control,const char ser
79    line();
80  }
81  
82 +void log_tx_piggyback(const char *q, const char qtype[2], const char *control)
83 +{
84 +  string("txpb ");
85 +  logtype(qtype); space(); name(q); space(); name(control);
86 +  line();
87 +}
88 +
89  void log_cachedanswer(const char *q,const char type[2])
90  {
91    string("cached "); logtype(type); space();
92 diff --git a/log.h b/log.h
93 index fe62fa3..d9a829b 100644
94 --- a/log.h
95 +++ b/log.h
96 @@ -18,6 +18,7 @@ extern void log_cachednxdomain(const char *);
97  extern void log_cachedns(const char *,const char *);
98  
99  extern void log_tx(const char *,const char *,const char *,const char *,unsigned int);
100 +extern void log_tx_piggyback(const char *,const char *,const char *);
101  
102  extern void log_nxdomain(const char *,const char *,unsigned int);
103  extern void log_nodata(const char *,const char *,const char *,unsigned int);
104 diff --git a/maxclient.h b/maxclient.h
105 new file mode 100644
106 index 0000000..e52fcd1
107 --- /dev/null
108 +++ b/maxclient.h
109 @@ -0,0 +1,7 @@
110 +#ifndef MAXCLIENT_H
111 +#define MAXCLIENT_H
112 +
113 +#define MAXUDP 200
114 +#define MAXTCP 20
115 +
116 +#endif /* MAXCLIENT_H */
117 diff --git a/qmerge.c b/qmerge.c
118 new file mode 100644
119 index 0000000..7c92299
120 --- /dev/null
121 +++ b/qmerge.c
122 @@ -0,0 +1,115 @@
123 +#include "qmerge.h"
124 +#include "byte.h"
125 +#include "log.h"
126 +#include "maxclient.h"
127 +
128 +#define QMERGE_MAX (MAXUDP+MAXTCP)
129 +struct qmerge inprogress[QMERGE_MAX];
130 +
131 +static
132 +int qmerge_key_init(struct qmerge_key *qmk, const char *q, const char qtype[2],
133 +    const char *control)
134 +{
135 +  if (!dns_domain_copy(&qmk->q, q)) return 0;
136 +  byte_copy(qmk->qtype, 2, qtype);
137 +  if (!dns_domain_copy(&qmk->control, control)) return 0;
138 +  return 1;
139 +}
140 +
141 +static
142 +int qmerge_key_equal(struct qmerge_key *a, struct qmerge_key *b)
143 +{
144 +  return
145 +    byte_equal(a->qtype, 2, b->qtype) &&
146 +    dns_domain_equal(a->q, b->q) &&
147 +    dns_domain_equal(a->control, b->control);
148 +}
149 +
150 +static
151 +void qmerge_key_free(struct qmerge_key *qmk)
152 +{
153 +  dns_domain_free(&qmk->q);
154 +  dns_domain_free(&qmk->control);
155 +}
156 +
157 +void qmerge_free(struct qmerge **x)
158 +{
159 +  struct qmerge *qm;
160 +
161 +  qm = *x;
162 +  *x = 0;
163 +  if (!qm || !qm->active) return;
164 +
165 +  qm->active--;
166 +  if (!qm->active) {
167 +    qmerge_key_free(&qm->key);
168 +    dns_transmit_free(&qm->dt);
169 +  }
170 +}
171 +
172 +int qmerge_start(struct qmerge **qm, const char servers[64], int flagrecursive,
173 +    const char *q, const char qtype[2], const char localip[4],
174 +    const char *control)
175 +{
176 +  struct qmerge_key k;
177 +  int i;
178 +  int r;
179 +
180 +  qmerge_free(qm);
181 +
182 +  byte_zero(&k, sizeof k);
183 +  if (!qmerge_key_init(&k, q, qtype, control)) return -1;
184 +  for (i = 0; i < QMERGE_MAX; i++) {
185 +    if (!inprogress[i].active) continue;
186 +    if (!qmerge_key_equal(&k, &inprogress[i].key)) continue;
187 +    log_tx_piggyback(q, qtype, control);
188 +    inprogress[i].active++;
189 +    *qm = &inprogress[i];
190 +    qmerge_key_free(&k);
191 +    return 0;
192 +  }
193 +
194 +  for (i = 0; i < QMERGE_MAX; i++)
195 +    if (!inprogress[i].active)
196 +      break;
197 +  if (i == QMERGE_MAX) return -1;
198 +
199 +  log_tx(q, qtype, control, servers, 0);
200 +  r = dns_transmit_start(&inprogress[i].dt, servers, flagrecursive, q, qtype, localip);
201 +  if (r == -1) { qmerge_key_free(&k); return -1; }
202 +  inprogress[i].active++;
203 +  inprogress[i].state = 0;
204 +  qmerge_key_free(&inprogress[i].key);
205 +  byte_copy(&inprogress[i].key, sizeof k, &k);
206 +  *qm = &inprogress[i];
207 +  return 0;
208 +}
209 +
210 +void qmerge_io(struct qmerge *qm, iopause_fd *io, struct taia *deadline)
211 +{
212 +  if (qm->state == 0) {
213 +    dns_transmit_io(&qm->dt, io, deadline);
214 +    qm->state = 1;
215 +  }
216 +  else {
217 +    io->fd = -1;
218 +    io->events = 0;
219 +  }
220 +}
221 +
222 +int qmerge_get(struct qmerge **x, const iopause_fd *io, const struct taia *when)
223 +{
224 +  int r;
225 +  struct qmerge *qm;
226 +
227 +  qm = *x;
228 +  if (qm->state == -1) return -1; /* previous error */
229 +  if (qm->state == 0) return 0; /* no packet */
230 +  if (qm->state == 2) return 1; /* already got packet */
231 +
232 +  r = dns_transmit_get(&qm->dt, io, when);
233 +  if (r == -1) { qm->state = -1; return -1; } /* error */
234 +  if (r == 0) { qm->state = 0; return 0; } /* must wait for i/o */
235 +  if (r == 1) { qm->state = 2; return 1; } /* got packet */
236 +  return -1; /* bug */
237 +}
238 diff --git a/qmerge.h b/qmerge.h
239 new file mode 100644
240 index 0000000..9a58157
241 --- /dev/null
242 +++ b/qmerge.h
243 @@ -0,0 +1,24 @@
244 +#ifndef QMERGE_H
245 +#define QMERGE_H
246 +
247 +#include "dns.h"
248 +
249 +struct qmerge_key {
250 +  char *q;
251 +  char qtype[2];
252 +  char *control;
253 +};
254 +
255 +struct qmerge {
256 +  int active;
257 +  struct qmerge_key key;
258 +  struct dns_transmit dt;
259 +  int state; /* -1 = error, 0 = need io, 1 = need get, 2 = got packet */
260 +};
261 +
262 +extern int qmerge_start(struct qmerge **,const char *,int,const char *,const char *,const char *,const char *);
263 +extern void qmerge_io(struct qmerge *,iopause_fd *,struct taia *);
264 +extern int qmerge_get(struct qmerge **,const iopause_fd *,const struct taia *);
265 +extern void qmerge_free(struct qmerge **);
266 +
267 +#endif /* QMERGE_H */
268 diff --git a/query.c b/query.c
269 index 46cdc00..f091fdd 100644
270 --- a/query.c
271 +++ b/query.c
272 @@ -81,7 +81,7 @@ static void cleanup(struct query *z)
273    int j;
274    int k;
275  
276 -  dns_transmit_free(&z->dt);
277 +  qmerge_free(&z->qm);
278    for (j = 0;j < QUERY_MAXALIAS;++j)
279      dns_domain_free(&z->alias[j]);
280    for (j = 0;j < QUERY_MAXLEVEL;++j) {
281 @@ -429,14 +429,8 @@ static int doit(struct query *z,int state)
282    if (j == 64) goto SERVFAIL;
283  
284    dns_sortip(z->servers[z->level],64);
285 -  if (z->level) {
286 -    log_tx(z->name[z->level],DNS_T_A,z->control[z->level],z->servers[z->level],z->level);
287 -    if (dns_transmit_start(&z->dt,z->servers[z->level],flagforwardonly,z->name[z->level],DNS_T_A,z->localip) == -1) goto DIE;
288 -  }
289 -  else {
290 -    log_tx(z->name[0],z->type,z->control[0],z->servers[0],0);
291 -    if (dns_transmit_start(&z->dt,z->servers[0],flagforwardonly,z->name[0],z->type,z->localip) == -1) goto DIE;
292 -  }
293 +  dtype = z->level ? DNS_T_A : z->type;
294 +  if (qmerge_start(&z->qm,z->servers[z->level],flagforwardonly,z->name[z->level],dtype,z->localip,z->control[z->level]) == -1) goto DIE;
295    return 0;
296  
297  
298 @@ -450,10 +444,10 @@ static int doit(struct query *z,int state)
299  
300    HAVEPACKET:
301    if (++z->loop == 100) goto DIE;
302 -  buf = z->dt.packet;
303 -  len = z->dt.packetlen;
304 +  buf = z->qm->dt.packet;
305 +  len = z->qm->dt.packetlen;
306  
307 -  whichserver = z->dt.servers + 4 * z->dt.curserver;
308 +  whichserver = z->qm->dt.servers + 4 * z->qm->dt.curserver;
309    control = z->control[z->level];
310    d = z->name[z->level];
311    dtype = z->level ? DNS_T_A : z->type;
312 @@ -836,7 +830,7 @@ int query_start(struct query *z,char *dn,char type[2],char class[2],char localip
313  
314  int query_get(struct query *z,iopause_fd *x,struct taia *stamp)
315  {
316 -  switch(dns_transmit_get(&z->dt,x,stamp)) {
317 +  switch(qmerge_get(&z->qm,x,stamp)) {
318      case 1:
319        return doit(z,1);
320      case -1:
321 @@ -847,5 +841,5 @@ int query_get(struct query *z,iopause_fd *x,struct taia *stamp)
322  
323  void query_io(struct query *z,iopause_fd *x,struct taia *deadline)
324  {
325 -  dns_transmit_io(&z->dt,x,deadline);
326 +  qmerge_io(z->qm,x,deadline);
327  }
328 diff --git a/query.h b/query.h
329 index eff68b2..06feab4 100644
330 --- a/query.h
331 +++ b/query.h
332 @@ -1,7 +1,7 @@
333  #ifndef QUERY_H
334  #define QUERY_H
335  
336 -#include "dns.h"
337 +#include "qmerge.h"
338  #include "uint32.h"
339  
340  #define QUERY_MAXLEVEL 5
341 @@ -20,7 +20,7 @@ struct query {
342    char localip[4];
343    char type[2];
344    char class[2];
345 -  struct dns_transmit dt;
346 +  struct qmerge *qm;
347  } ;
348  
349  extern int query_start(struct query *,char *,char *,char *,char *);