VirtualBox

source: vbox/trunk/src/VBox/NetworkServices/NetLib/VBoxNetPortForwardString.cpp@ 46165

Last change on this file since 46165 was 45559, checked in by vboxsync, 12 years ago

NAT/Lwip-service: Main client now, dynamic port-forward + managment.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 10.0 KB
Line 
1/* $Id: VBoxNetPortForwardString.cpp 45559 2013-04-16 06:11:59Z vboxsync $ */
2/** @file
3 * VBoxNetPortForwardString - Routines for managing port-forward strings.
4 */
5
6/*
7 * Copyright (C) 2006-2013 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18/*******************************************************************************
19* Header Files *
20*******************************************************************************/
21
22#include <netinet/in.h>
23
24#include <iprt/cdefs.h>
25#define IPv6
26#include <iprt/cidr.h>
27#include <iprt/param.h>
28#include <iprt/path.h>
29#include <iprt/stream.h>
30#include <iprt/string.h>
31#include <iprt/net.h>
32#include <iprt/getopt.h>
33#include <iprt/ctype.h>
34
35
36#include <VBox/log.h>
37
38#include "VBoxPortForwardString.h"
39
40
41#define PF_FIELD_SEPARATOR ':'
42#define PF_ADDRESS_FIELD_STARTS '['
43#define PF_ADDRESS_FIELD_ENDS ']'
44
45#define PF_STR_FIELD_SEPARATOR ":"
46#define PF_STR_ADDRESS_FIELD_STARTS "["
47#define PF_STR_ADDRESS_FIELD_ENDS "]"
48
49static int netPfStrAddressParse(char *pszRaw, int cbRaw,
50 char *pszAddress, int cbAddress,
51 bool fEmptyAcceptable)
52{
53 int idxRaw = 0;
54 int cbField = 0;
55
56 AssertPtrReturn(pszRaw, -1);
57 AssertPtrReturn(pszAddress, -1);
58 AssertReturn(pszRaw[0] == PF_ADDRESS_FIELD_STARTS, -1);
59
60 if (pszRaw[0] == PF_ADDRESS_FIELD_STARTS)
61 {
62 /* shift pszRaw to next symbol */
63 pszRaw++;
64 cbRaw--;
65
66
67 /* we shouldn't face with ending here */
68 AssertReturn(cbRaw > 0, VERR_INVALID_PARAMETER);
69
70 char *pszEndOfAddress = RTStrStr(pszRaw, PF_STR_ADDRESS_FIELD_ENDS);
71
72 /* no pair closing sign */
73 AssertPtrReturn(pszEndOfAddress, VERR_INVALID_PARAMETER);
74
75 cbField = pszEndOfAddress - pszRaw;
76
77 /* field should be less then the rest of the string */
78 AssertReturn(cbField < cbRaw, VERR_INVALID_PARAMETER);
79
80 if (cbField != 0)
81 RTStrCopy(pszAddress, RT_MIN(cbField + 1, cbAddress), pszRaw);
82 else if (!fEmptyAcceptable)
83 return -1;
84 }
85
86 AssertReturn(pszRaw[cbField] == PF_ADDRESS_FIELD_ENDS, -1);
87
88 return cbField + 2; /* length of the field and closing braces */
89}
90
91
92static int netPfStrPortParse(char *pszRaw, int cbRaw, uint16_t *pu16Port)
93{
94 char *pszEndOfPort = NULL;
95 uint16_t u16Port = 0;
96 int idxRaw = 1; /* we increment pszRaw after checks. */
97 int cbRest = 0;
98 size_t cbPort = 0;
99
100 AssertPtrReturn(pszRaw, -1);
101 AssertPtrReturn(pu16Port, -1);
102 AssertReturn(pszRaw[0] == PF_FIELD_SEPARATOR, -1);
103
104 pszRaw++; /* skip line separator */
105 cbRaw --;
106
107 pszEndOfPort = RTStrStr(pszRaw, ":");
108 if (!pszEndOfPort)
109 {
110 cbRest = strlen(pszRaw);
111
112 Assert(cbRaw == cbRest);
113
114 /* XXX: Assumption that if string is too big, it will be reported by
115 * RTStrToUint16.
116 */
117 if (cbRest > 0)
118 {
119 pszEndOfPort = pszRaw + cbRest;
120 cbPort = cbRest;
121 }
122 else
123 return -1;
124 }
125 else
126 cbPort = pszEndOfPort - pszRaw;
127
128
129 idxRaw += cbPort;
130
131 Assert(cbRest || pszRaw[idxRaw - 1] == PF_FIELD_SEPARATOR); /* we are 1 char ahead */
132
133 char aszPort[10];
134 RT_ZERO(aszPort);
135
136 Assert(idxRaw > 0);
137 RTStrCopy(aszPort, RT_MIN(RT_ELEMENTS(aszPort), (size_t)(cbPort) + 1), pszRaw);
138
139 if (!(u16Port = RTStrToUInt16(aszPort)))
140 return -1;
141
142 *pu16Port = u16Port;
143
144 return idxRaw;
145}
146
147
148static int netPfStrAddressPortPairParse(char *pszRaw, int cbRaw,
149 char *pszAddress, int cbAddress,
150 bool fEmptyAddressAcceptable,
151 uint16_t *pu16Port)
152{
153 int idxRaw = 0;
154 int idxRawTotal = 0;
155
156 AssertPtrReturn(pszRaw, -1);
157 AssertPtrReturn(pszAddress, -1);
158 AssertPtrReturn(pu16Port, -2);
159
160 /* XXX: Here we should check 0 - ':' and 1 - '[' */
161 Assert( pszRaw[0] == PF_FIELD_SEPARATOR
162 && pszRaw[1] == PF_ADDRESS_FIELD_STARTS);
163
164 pszRaw++; /* field separator skip */
165 cbRaw--;
166 AssertReturn(cbRaw > 0, VERR_INVALID_PARAMETER);
167
168 idxRaw = 0;
169
170 if (pszRaw[0] == PF_ADDRESS_FIELD_STARTS)
171 {
172 idxRaw += netPfStrAddressParse(pszRaw,
173 cbRaw - idxRaw,
174 pszAddress,
175 cbAddress,
176 fEmptyAddressAcceptable);
177 if (idxRaw == -1)
178 return -1;
179
180 Assert(pszRaw[idxRaw] == PF_FIELD_SEPARATOR);
181 }
182 else return -1;
183
184 pszRaw += idxRaw;
185 idxRawTotal += idxRaw;
186 cbRaw -= idxRaw;
187
188 AssertReturn(cbRaw > 0, VERR_INVALID_PARAMETER);
189
190 idxRaw = 0;
191
192 Assert(pszRaw[0] == PF_FIELD_SEPARATOR);
193
194 if (pszRaw[0] == PF_FIELD_SEPARATOR)
195 {
196 idxRaw = netPfStrPortParse(pszRaw, strlen(pszRaw), pu16Port);
197
198 Assert(strlen(&pszRaw[idxRaw]) == 0 || pszRaw[idxRaw] == PF_FIELD_SEPARATOR);
199
200 if (idxRaw == -1)
201 return -2;
202
203 idxRawTotal += idxRaw;
204
205 return idxRawTotal + 1;
206 }
207 else return -1; /* trailing garbage in the address */
208}
209
210/* XXX: Having fIPv6 we might emprove adress verification comparing address length
211 * with INET[6]_ADDRLEN
212 */
213int netPfStrToPf(const char *pcszStrPortForward, int fIPv6, PPORTFORWARDRULE pPfr)
214{
215 char *pszName;
216 int proto;
217 char *pszHostAddr;
218 char *pszGuestAddr;
219 uint16_t u16HostPort;
220 uint16_t u16GuestPort;
221 bool fTcpProto = false;
222
223 char *pszRawBegin = NULL;
224 char *pszRaw = NULL;
225 int idxRaw = 0;
226 int cbToken = 0;
227 int cbRaw = 0;
228 int rc = VINF_SUCCESS;
229
230 AssertPtrReturn(pcszStrPortForward, VERR_INVALID_PARAMETER);
231 AssertPtrReturn(pPfr, VERR_INVALID_PARAMETER);
232
233 memset(pPfr, 0, sizeof(PORTFORWARDRULE));
234
235 pszHostAddr = &pPfr->aszPfrHostAddr[0];
236 pszGuestAddr = &pPfr->aszPfrGuestAddr[0];
237 pszName = &pPfr->aszPfrName[0];
238
239 cbRaw = strlen(pcszStrPortForward);
240
241 /* Minimal rule ":tcp:[]:0:[]:0" has got lenght 14 */
242 AssertReturn(cbRaw > 14, VERR_INVALID_PARAMETER);
243
244 pszRaw = RTStrDup(pcszStrPortForward);
245
246 AssertPtrReturn(pszRaw, VERR_NO_MEMORY);
247
248 pszRawBegin = pszRaw;
249
250 /* name */
251 if (pszRaw[idxRaw] == PF_FIELD_SEPARATOR)
252 idxRaw = 1; /* begin of the next segment */
253 else
254 {
255 char *pszEndOfName = RTStrStr(pszRaw + 1, PF_STR_FIELD_SEPARATOR);
256 if (!pszEndOfName)
257 goto invalid_parameter;
258
259 cbToken = (pszEndOfName) - pszRaw; /* don't take : into account */
260 /* XXX it's unacceptable to have only name entry in PF */
261 AssertReturn(cbToken < cbRaw, VERR_INVALID_PARAMETER);
262
263 if ( cbToken < 0
264 || (size_t)cbToken >= PF_NAMELEN)
265 goto invalid_parameter;
266
267 RTStrCopy(pszName,
268 RT_MIN((size_t)cbToken + 1, PF_NAMELEN),
269 pszRaw);
270 pszRaw += cbToken; /* move to separator */
271 }
272
273 AssertReturn(pszRaw[0] == PF_FIELD_SEPARATOR, VERR_INVALID_PARAMETER);
274 /* protocol */
275
276 pszRaw++; /* skip separator */
277 idxRaw = 0;
278
279 cbRaw--;
280
281 if ( (( fTcpProto = (RTStrNICmp(pszRaw, "tcp", 3) == 0)
282 || (RTStrNICmp(pszRaw, "udp", 3) == 0))
283 && (pszRaw[3] == PF_FIELD_SEPARATOR)))
284 {
285 proto = (fTcpProto ? IPPROTO_TCP : IPPROTO_UDP);
286 idxRaw = 3;
287 }
288 else
289 goto invalid_parameter;
290
291 pszRaw += idxRaw;
292 cbRaw -= idxRaw;
293 idxRaw = 0;
294
295 idxRaw = netPfStrAddressPortPairParse(pszRaw, cbRaw,
296 pszHostAddr, INET6_ADDRSTRLEN,
297 true, &u16HostPort);
298 if (idxRaw < 0)
299 return VERR_INVALID_PARAMETER;
300
301 pszRaw += idxRaw;
302 cbRaw -= idxRaw;
303
304 Assert(pszRaw[0] == PF_FIELD_SEPARATOR);
305
306 idxRaw = 0;
307
308 idxRaw = netPfStrAddressPortPairParse(pszRaw, cbRaw,
309 pszGuestAddr,
310 INET6_ADDRSTRLEN,
311 false,
312 &u16GuestPort);
313
314 if (idxRaw < 0)
315 goto invalid_parameter;
316
317 /* XXX: fill the rule */
318 pPfr->fPfrIPv6 = fIPv6;
319 pPfr->iPfrProto = proto;
320
321
322 if (strlen(pszHostAddr))
323 {
324 if (!fIPv6)
325 rc = RTNetStrToIPv4Addr(pszHostAddr, &pPfr->uPfrHostAddr.IPv4);
326#if 0 /* No IPv6 yet */
327 else
328 rc = RTNetStrToIPv6Addr(pszHostAddr, &pPfr->uPfrHostAddr.IPv6);
329#endif
330
331 if (RT_FAILURE(rc))
332 goto invalid_parameter;
333 }
334
335 pPfr->u16PfrHostPort = u16HostPort;
336
337 if (strlen(pszGuestAddr))
338 {
339 if (!fIPv6)
340 rc = RTNetStrToIPv4Addr(pszGuestAddr, &pPfr->uPfrGuestAddr.IPv4);
341#if 0
342 else
343 rc = RTNetStrToIPv6Addr(pszGuestAddr, &pPfr->uPfrGuestAddr.IPv6);
344#endif
345 if (RT_FAILURE(rc))
346 goto invalid_parameter;
347 }
348 else
349 goto invalid_parameter; /* guest address should be defined */
350
351 pPfr->u16PfrGuestPort = u16GuestPort;
352
353 Log(("name: %s\n"
354 "proto: %d\n"
355 "host address: %s\n"
356 "host port: %d\n"
357 "guest address: %s\n"
358 "guest port:%d\n",
359 pszName, proto,
360 pszHostAddr, u16HostPort,
361 pszGuestAddr, u16GuestPort));
362
363 RTStrFree(pszRawBegin);
364 return VINF_SUCCESS;
365
366invalid_parameter:
367 RTStrFree(pszRawBegin);
368 if (pPfr)
369 memset(pPfr, 0, sizeof(PORTFORWARDRULE));
370 return VERR_INVALID_PARAMETER;
371}
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette