* copy vendor drop to trunk
[lab.git] / Dev / utvpn / utvpn-unix-v101-7101-public / src / Cedar / Bridge.c
1 // SoftEther UT-VPN SourceCode\r
2 // \r
3 // Copyright (C) 2004-2010 SoftEther Corporation.\r
4 // Copyright (C) 2004-2010 University of Tsukuba, Japan.\r
5 // Copyright (C) 2003-2010 Daiyuu Nobori.\r
6 // All Rights Reserved.\r
7 // \r
8 // http://utvpn.tsukuba.ac.jp/\r
9 // \r
10 // This program is free software; you can redistribute it and/or\r
11 // modify it under the terms of the GNU General Public License\r
12 // version 2 as published by the Free Software Foundation.\r
13 // \r
14 // This program is distributed in the hope that it will be useful,\r
15 // but WITHOUT ANY WARRANTY; without even the implied warranty of\r
16 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\r
17 // GNU General Public License for more details.\r
18 // \r
19 // You should have received a copy of the GNU General Public License version 2\r
20 // along with this program; if not, write to the Free Software\r
21 // Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.\r
22 // \r
23 // このファイルは GPL バージョン 2 ライセンスで公開されています。\r
24 // 誰でもこのファイルの内容を複製、改変したり、改変したバージョンを再配布\r
25 // することができます。ただし、原著作物を改変した場合は、原著作物の著作権表示\r
26 // を除去することはできません。改変した著作物を配布する場合は、改変実施者の\r
27 // 著作権表示を原著作物の著作権表示に付随して記載するようにしてください。\r
28 // \r
29 // この SoftEther UT-VPN オープンソース・プロジェクトは、日本国の\r
30 // ソフトイーサ株式会社 (SoftEther Corporation, http://www.softether.co.jp/ )\r
31 // および筑波大学 (University of Tsukuba, http://www.tsukuba.ac.jp/ ) によって\r
32 // ホストされています。\r
33 // 本プログラムの配布者は、本プログラムを、業としての利用以外のため、\r
34 // および、試験または研究のために利用が行われることを想定して配布\r
35 // しています。\r
36 // SoftEther UT-VPN プロジェクトの Web サイトは http://utvpn.tsukuba.ac.jp/ に\r
37 // あります。\r
38 // 本ソフトウェアの不具合の修正、機能改良、セキュリティホールの修復などのコード\r
39 // の改変を行った場合で、その成果物を SoftEther UT-VPN プロジェクトに提出して\r
40 // いただける場合は、 http://utvpn.tsukuba.ac.jp/ までソースコードを送付して\r
41 // ください。SoftEther UT-VPN プロジェクトの本体リリースまたはブランチリリース\r
42 // に組み込みさせていただきます。\r
43 // \r
44 // GPL に基づいて原著作物が提供される本ソフトウェアの改良版を配布、販売する\r
45 // 場合は、そのソースコードを GPL に基づいて誰にでも開示する義務が生じます。\r
46 // \r
47 // 本ソフトウェアに関連する著作権、特許権、商標権はソフトイーサ株式会社\r
48 // (SoftEther Corporation) およびその他の著作権保持者が保有しています。\r
49 // ソフトイーサ株式会社等はこれらの権利を放棄していません。本ソフトウェアの\r
50 // 二次著作物を配布、販売する場合は、これらの権利を侵害しないようにご注意\r
51 // ください。\r
52 // \r
53 // お願い: どのような通信ソフトウェアにも通常は必ず未発見の\r
54 // セキュリティホールが潜んでいます。本ソースコードをご覧いただいた結果、\r
55 // UT-VPN にセキュリティホールを発見された場合は、当該セキュリティホールの\r
56 // 情報を不特定多数に開示される前に、必ず、ソフトイーサ株式会社\r
57 // および脆弱性情報の届出を受け付ける公的機関まで通報いただき、\r
58 // 公益保護にご協力いただきますようお願い申し上げます。\r
59 // \r
60 // ソフトイーサ株式会社は、当該セキュリティホールについて迅速に対処を\r
61 // 行い、UT-VPN および UT-VPN に関連するソフトウェアのユーザー・顧客\r
62 // を保護するための努力を行います。\r
63 // \r
64 // ソフトイーサへの届出先: http://www.softether.co.jp/jp/contact/\r
65 // 日本国内の脆弱性情報届出受付公的機関:\r
66 //         独立行政法人 情報処理推進機構\r
67 //         http://www.ipa.go.jp/security/vuln/report/\r
68 // \r
69 // 上記各事項について不明な点は、ソフトイーサ株式会社までご連絡ください。\r
70 // 連絡先: http://www.softether.co.jp/jp/contact/\r
71 \r
72 // -----------------------------------------------\r
73 // [ChangeLog]\r
74 // 2010.05.20\r
75 //  新規リリース by SoftEther\r
76 // -----------------------------------------------\r
77 \r
78 // Bridge.c\r
79 // Ethernet ブリッジプログラム\r
80 \r
81 #define BRIDGE_C\r
82 \r
83 #ifdef  WIN32\r
84 #define OS_WIN32\r
85 #endif\r
86 \r
87 #ifdef  OS_WIN32\r
88 \r
89 // Win32 用\r
90 #include "BridgeWin32.c"\r
91 \r
92 #else\r
93 \r
94 // Unix 用\r
95 #include "BridgeUnix.c"\r
96 \r
97 #endif  // OS_WIN32\r
98 \r
99 // 現在の Ethernet デバイス一覧をハッシュする\r
100 UINT GetEthDeviceHash()\r
101 {\r
102 #ifdef  OS_UNIX\r
103         // UNIX 版\r
104         UINT num;\r
105         UINT i;\r
106         char tmp[4096];\r
107         UCHAR hash[SHA1_SIZE];\r
108         TOKEN_LIST *t = GetEthList();\r
109 \r
110         num = t->NumTokens;\r
111         tmp[0] = 0;\r
112         for (i = 0;i < t->NumTokens;i++)\r
113         {\r
114                 StrCat(tmp, sizeof(tmp), t->Token[i]);\r
115         }\r
116         FreeToken(t);\r
117 \r
118         Hash(hash, tmp, StrLen(tmp), true);\r
119 \r
120         Copy(&num, hash, sizeof(UINT));\r
121 \r
122         return num;\r
123 #else   // OS_UNIX\r
124         // Win32 版\r
125         UINT ret = 0;\r
126         MS_ADAPTER_LIST *a = MsCreateAdapterListEx(true);\r
127         UINT num;\r
128         UINT i;\r
129         char tmp[4096];\r
130         UCHAR hash[SHA1_SIZE];\r
131 \r
132         tmp[0] = 0;\r
133         if (a != NULL)\r
134         {\r
135                 for (i = 0;i < a->Num;i++)\r
136                 {\r
137                         StrCat(tmp, sizeof(tmp), a->Adapters[i]->Title);\r
138                 }\r
139         }\r
140         MsFreeAdapterList(a);\r
141 \r
142         Hash(hash, tmp, StrLen(tmp), true);\r
143 \r
144         Copy(&num, hash, sizeof(UINT));\r
145 \r
146         return num;\r
147 #endif  // OS_UNIU\r
148 }\r
149 \r
150 // WinPcap が必要かどうか取得する\r
151 bool IsNeedWinPcap()\r
152 {\r
153         if (IsBridgeSupported() == false)\r
154         {\r
155                 // Windows 以外\r
156                 return false;\r
157         }\r
158         else\r
159         {\r
160                 // Windows\r
161                 if (IsEthSupported())\r
162                 {\r
163                         // すでに Ethernet デバイスへのアクセスに成功している\r
164                         return false;\r
165                 }\r
166                 else\r
167                 {\r
168                         // Ethernet デバイスへのアクセスに失敗している\r
169                         return true;\r
170                 }\r
171         }\r
172 }\r
173 \r
174 // 現在の OS でブリッジがサポートされているかどうか取得する\r
175 bool IsBridgeSupported()\r
176 {\r
177         UINT type = GetOsInfo()->OsType;\r
178 \r
179         if (OS_IS_WINDOWS(type))\r
180         {\r
181                 return true;\r
182         }\r
183         else\r
184         {\r
185                 return IsEthSupported();\r
186         }\r
187 }\r
188 \r
189 // ローカルブリッジの削除\r
190 bool DeleteLocalBridge(CEDAR *c, char *hubname, char *devicename)\r
191 {\r
192         bool ret = false;\r
193         // 引数チェック\r
194         if (c == NULL || hubname == NULL || devicename == NULL)\r
195         {\r
196                 return false;\r
197         }\r
198 \r
199         LockList(c->HubList);\r
200         {\r
201                 LockList(c->LocalBridgeList);\r
202                 {\r
203                         UINT i;\r
204 \r
205                         for (i = 0;i < LIST_NUM(c->LocalBridgeList);i++)\r
206                         {\r
207                                 LOCALBRIDGE *br = LIST_DATA(c->LocalBridgeList, i);\r
208 \r
209                                 if (StrCmpi(br->HubName, hubname) == 0)\r
210                                 {\r
211                                         if (StrCmpi(br->DeviceName, devicename) == 0)\r
212                                         {\r
213                                                 if (br->Bridge != NULL)\r
214                                                 {\r
215                                                         BrFreeBridge(br->Bridge);\r
216                                                         br->Bridge = NULL;\r
217                                                 }\r
218 \r
219                                                 Delete(c->LocalBridgeList, br);\r
220                                                 Free(br);\r
221 \r
222                                                 ret = true;\r
223                                                 break;\r
224                                         }\r
225                                 }\r
226                         }\r
227                 }\r
228                 UnlockList(c->LocalBridgeList);\r
229         }\r
230         UnlockList(c->HubList);\r
231 \r
232         return ret;\r
233 }\r
234 \r
235 // ローカルブリッジの追加\r
236 void AddLocalBridge(CEDAR *c, char *hubname, char *devicename, bool local, bool monitor, bool tapmode, char *tapaddr, bool fullbcast)\r
237 {\r
238         UINT i;\r
239         HUB *h = NULL;\r
240         LOCALBRIDGE *br = NULL;\r
241         // 引数チェック\r
242         if (c == NULL || hubname == NULL || devicename == NULL)\r
243         {\r
244                 return;\r
245         }\r
246 \r
247         if (OS_IS_UNIX(GetOsInfo()->OsType) == false)\r
248         {\r
249                 tapmode = false;\r
250         }\r
251 \r
252         LockList(c->HubList);\r
253         {\r
254                 LockList(c->LocalBridgeList);\r
255                 {\r
256                         bool exists = false;\r
257 \r
258                         // 全く同一のブリッジ設定が無いかどうか調べる\r
259                         for (i = 0;i < LIST_NUM(c->LocalBridgeList);i++)\r
260                         {\r
261                                 LOCALBRIDGE *br = LIST_DATA(c->LocalBridgeList, i);\r
262                                 if (StrCmpi(br->DeviceName, devicename) == 0)\r
263                                 {\r
264                                         if (StrCmpi(br->HubName, hubname) == 0)\r
265                                         {\r
266                                                 if (br->TapMode == tapmode)\r
267                                                 {\r
268                                                         exists = true;\r
269                                                 }\r
270                                         }\r
271                                 }\r
272                         }\r
273 \r
274                         if (exists == false)\r
275                         {\r
276                                 // 設定を追加する\r
277                                 br = ZeroMalloc(sizeof(LOCALBRIDGE));\r
278                                 StrCpy(br->HubName, sizeof(br->HubName), hubname);\r
279                                 StrCpy(br->DeviceName, sizeof(br->DeviceName), devicename);\r
280                                 br->Bridge = NULL;\r
281                                 br->Local = local;\r
282                                 br->TapMode = tapmode;\r
283                                 br->FullBroadcast = fullbcast;\r
284                                 br->Monitor = monitor;\r
285                                 if (br->TapMode)\r
286                                 {\r
287                                         if (tapaddr != NULL && IsZero(tapaddr, 6) == false)\r
288                                         {\r
289                                                 Copy(br->TapMacAddress, tapaddr, 6);\r
290                                         }\r
291                                         else\r
292                                         {\r
293                                                 GenMacAddress(br->TapMacAddress);\r
294                                         }\r
295                                 }\r
296 \r
297                                 Add(c->LocalBridgeList, br);\r
298 \r
299                                 // HUB を検索する\r
300                                 for (i = 0;i < LIST_NUM(c->HubList);i++)\r
301                                 {\r
302                                         HUB *hub = LIST_DATA(c->HubList, i);\r
303                                         if (StrCmpi(hub->Name, br->HubName) == 0)\r
304                                         {\r
305                                                 h = hub;\r
306                                                 AddRef(h->ref);\r
307                                                 break;\r
308                                         }\r
309                                 }\r
310                         }\r
311                 }\r
312                 UnlockList(c->LocalBridgeList);\r
313         }\r
314         UnlockList(c->HubList);\r
315 \r
316         // 早速ブリッジを開始する\r
317         if (h != NULL && br != NULL && h->Type != HUB_TYPE_FARM_DYNAMIC)\r
318         {\r
319                 Lock(h->lock_online);\r
320                 {\r
321                         if (h->Offline == false)\r
322                         {\r
323                                 LockList(c->LocalBridgeList);\r
324                                 {\r
325                                         if (IsInList(c->LocalBridgeList, br))\r
326                                         {\r
327                                                 if (br->Bridge == NULL)\r
328                                                 {\r
329                                                         br->Bridge = BrNewBridge(h, br->DeviceName, NULL, br->Local, br->Monitor, br->TapMode, br->TapMacAddress, br->FullBroadcast);\r
330                                                 }\r
331                                         }\r
332                                 }\r
333                                 UnlockList(c->LocalBridgeList);\r
334                         }\r
335                 }\r
336                 Unlock(h->lock_online);\r
337         }\r
338 \r
339         ReleaseHub(h);\r
340 }\r
341 \r
342 // ローカルブリッジリストの初期化\r
343 void InitLocalBridgeList(CEDAR *c)\r
344 {\r
345         // 引数チェック\r
346         if (c == NULL)\r
347         {\r
348                 return;\r
349         }\r
350 \r
351         c->LocalBridgeList = NewList(NULL);\r
352 }\r
353 \r
354 // ローカルブリッジリストの解放\r
355 void FreeLocalBridgeList(CEDAR *c)\r
356 {\r
357         UINT i;\r
358         // 引数チェック\r
359         if (c == NULL)\r
360         {\r
361                 return;\r
362         }\r
363 \r
364         for (i = 0;i < LIST_NUM(c->LocalBridgeList);i++)\r
365         {\r
366                 LOCALBRIDGE *br = LIST_DATA(c->LocalBridgeList, i);\r
367                 Free(br);\r
368         }\r
369 \r
370         ReleaseList(c->LocalBridgeList);\r
371         c->LocalBridgeList = NULL;\r
372 }\r
373 \r
374 // ブリッジ用スレッド\r
375 void BrBridgeThread(THREAD *thread, void *param)\r
376 {\r
377         BRIDGE *b;\r
378         CONNECTION *c;\r
379         SESSION *s;\r
380         HUB *h;\r
381         char name[MAX_SIZE];\r
382         // 引数チェック\r
383         if (thread == NULL || param == NULL)\r
384         {\r
385                 return;\r
386         }\r
387 \r
388         b = (BRIDGE *)param;\r
389 \r
390         // コネクションの作成\r
391         c = NewServerConnection(b->Cedar, NULL, thread);\r
392         c->Protocol = CONNECTION_HUB_BRIDGE;\r
393 \r
394         // セッションの作成\r
395         s = NewServerSession(b->Cedar, c, b->Hub, BRIDGE_USER_NAME, b->Policy);\r
396         HLog(b->Hub, "LH_START_BRIDGE", b->Name, s->Name);\r
397         StrCpy(name, sizeof(name), b->Name);\r
398         h = b->Hub;\r
399         AddRef(h->ref);\r
400         s->BridgeMode = true;\r
401         s->Bridge = b;\r
402         c->Session = s;\r
403         ReleaseConnection(c);\r
404 \r
405         // ユーザー名\r
406         s->Username = CopyStr(BRIDGE_USER_NAME_PRINT);\r
407 \r
408         b->Session = s;\r
409         AddRef(s->ref);\r
410 \r
411         // 初期化完了を通知\r
412         NoticeThreadInit(thread);\r
413 \r
414         // セッションのメイン関数\r
415         Debug("Bridge %s Start.\n", b->Name);\r
416         SessionMain(s);\r
417         Debug("Bridge %s Stop.\n", b->Name);\r
418 \r
419         HLog(h, "LH_STOP_BRIDGE", name);\r
420 \r
421         ReleaseHub(h);\r
422 \r
423         ReleaseSession(s);\r
424 }\r
425 \r
426 // ブリッジの解放\r
427 void BrFreeBridge(BRIDGE *b)\r
428 {\r
429         // 引数チェック\r
430         if (b == NULL)\r
431         {\r
432                 return;\r
433         }\r
434 \r
435         // セッション停止\r
436         StopSession(b->Session);\r
437         ReleaseSession(b->Session);\r
438         Free(b->Name);\r
439 \r
440         Free(b);\r
441 }\r
442 \r
443 // 新しいブリッジの作成\r
444 BRIDGE *BrNewBridge(HUB *h, char *name, POLICY *p, bool local, bool monitor, bool tapmode, char *tapaddr, bool fullbcast)\r
445 {\r
446         BRIDGE *b;\r
447         POLICY *policy;\r
448         THREAD *t;\r
449         // 引数チェック\r
450         if (h == NULL || name == NULL)\r
451         {\r
452                 return NULL;\r
453         }\r
454 \r
455         if (p == NULL)\r
456         {\r
457                 policy = ClonePolicy(GetDefaultPolicy());\r
458         }\r
459         else\r
460         {\r
461                 policy = ClonePolicy(p);\r
462         }\r
463 \r
464         policy->CheckMac = true;\r
465 \r
466 #ifdef  UNIX_LINUX\r
467         policy->CheckMac = false;\r
468 #endif  // UNIX_LINUX\r
469 \r
470         policy->NoBroadcastLimiter = true;\r
471 \r
472         b = ZeroMalloc(sizeof(BRIDGE));\r
473         b->Cedar = h->Cedar;\r
474         b->Hub = h;\r
475         b->Name = CopyStr(name);\r
476         b->Policy = policy;\r
477         b->Local = local;\r
478         b->Monitor = monitor;\r
479         b->TapMode = tapmode;\r
480         b->FullBroadcast = fullbcast;\r
481 \r
482         if (b->TapMode)\r
483         {\r
484                 if (tapaddr != NULL && IsZero(tapaddr, 6) == false)\r
485                 {\r
486                         Copy(b->TapMacAddress, tapaddr, 6);\r
487                 }\r
488                 else\r
489                 {\r
490                         GenMacAddress(b->TapMacAddress);\r
491                 }\r
492         }\r
493 \r
494         if (monitor)\r
495         {\r
496                 // モニタリングモード\r
497                 policy->MonitorPort = true;\r
498         }\r
499 \r
500         if (b->FullBroadcast)\r
501         {\r
502                 // ブロードキャストを制限しない\r
503                 policy->NoBroadcastLimiter = true;\r
504         }\r
505 \r
506         // スレッド作成\r
507         t = NewThread(BrBridgeThread, b);\r
508         WaitThreadInit(t);\r
509         ReleaseThread(t);\r
510 \r
511         return b;\r
512 }\r
513 \r