mirror of
				https://gitea.osmocom.org/cellular-infrastructure/osmo-mgw.git
				synced 2025-10-30 19:43:52 +00:00 
			
		
		
		
	Compare commits
	
		
			307 Commits
		
	
	
		
			osmith/fix
			...
			on-waves/0
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 2b4e366083 | ||
|  | bf1eb64b02 | ||
|  | f0fbae94ea | ||
|  | 8fe4df503c | ||
|  | 8da7103070 | ||
|  | f73f6fad8c | ||
|  | 25cb84be12 | ||
|  | d9ae25c1bf | ||
|  | 5c011366c9 | ||
|  | 79e2d4230d | ||
|  | 8ecd029b12 | ||
|  | 3c0508e94a | ||
|  | f535aad612 | ||
|  | d0ac8866f1 | ||
|  | 73f9a65f12 | ||
|  | b2c55c49a8 | ||
|  | 8dc241959c | ||
|  | f99709430a | ||
|  | b9bc45b1b0 | ||
|  | 65d10c1320 | ||
|  | 414ba77f75 | ||
|  | 59f2470650 | ||
|  | 339dfdb624 | ||
|  | 9e2e2e04d1 | ||
|  | 2ab6db0153 | ||
|  | 6cb97bdebe | ||
|  | 8c3694a282 | ||
|  | 191d23a889 | ||
|  | a5963097ac | ||
|  | 221fb37518 | ||
|  | 4ec8a390cc | ||
|  | cf3f1c8b3d | ||
|  | 984f3b8047 | ||
|  | ec1f15d513 | ||
|  | b76cd5ed7e | ||
|  | 1592550d98 | ||
|  | 5cdf42b1a4 | ||
|  | 3a6b1a41fb | ||
|  | 1b5b3bbfdb | ||
|  | 3a67035411 | ||
|  | cb1937a4c5 | ||
|  | 3cfd5d6a02 | ||
|  | 6cc4dbfd46 | ||
|  | 9960d59fff | ||
|  | 161bd6d253 | ||
|  | add3472e9f | ||
|  | 33b0bee457 | ||
|  | 6949db1bd8 | ||
|  | 8ae0080e21 | ||
|  | 546c296c4a | ||
|  | 86f42eb6a5 | ||
|  | 494c086dca | ||
|  | 6b18c8f3b6 | ||
|  | 87f6d26c2e | ||
|  | fab2ff34c4 | ||
|  | 06d353e02e | ||
|  | dfe47549c6 | ||
|  | c70e8c2103 | ||
|  | b462a03c35 | ||
|  | 6e0ec5b6fa | ||
|  | 6768387f16 | ||
|  | 5ef1234dd3 | ||
|  | 581e58d166 | ||
|  | e308bb466a | ||
|  | e4be5394ef | ||
|  | 81e1edd3e6 | ||
|  | cfd1c28604 | ||
|  | 3ba8963a1d | ||
|  | 238d156481 | ||
|  | 516c4f073a | ||
|  | fa22aa6bbd | ||
|  | 4072ceed32 | ||
|  | cf6bf63a0d | ||
|  | 88b614110f | ||
|  | d9b825a5f5 | ||
|  | b91e5f1da4 | ||
|  | 07bb509434 | ||
|  | 08db6ca509 | ||
|  | 6446ded81c | ||
|  | 7b8f6064d6 | ||
|  | c6a1fe773d | ||
|  | 729d468fdf | ||
|  | b37ce4c5a4 | ||
|  | 5cd62c0ba5 | ||
|  | 1e1acafafd | ||
|  | fb83b7a86d | ||
|  | ef0b641f63 | ||
|  | 27e0bfd3c7 | ||
|  | bbfff6ec39 | ||
|  | dc0914df09 | ||
|  | 0db691dcf6 | ||
|  | bb45b73b20 | ||
|  | 5f5c1b6bcb | ||
|  | e51cf4f946 | ||
|  | 749ba7f5ad | ||
|  | 860c8955c3 | ||
|  | c33701c4e5 | ||
|  | 44d92b4728 | ||
|  | 8aaec620da | ||
|  | a5a4014d67 | ||
|  | 9d519189ae | ||
|  | f0fc618782 | ||
|  | c57575bea8 | ||
|  | 8cdfe9fc37 | ||
|  | 0959f8cbe6 | ||
|  | f21028985e | ||
|  | 483b768ab2 | ||
|  | 82cb311c4f | ||
|  | 2980442e33 | ||
|  | fa7afb31e9 | ||
|  | 7513b3a1c2 | ||
|  | 135d99b36e | ||
|  | 5aaf7c164c | ||
|  | 790db1e01b | ||
|  | 81a8975662 | ||
|  | fd876b7488 | ||
|  | 2ffe7aa340 | ||
|  | 538ea6d5c6 | ||
|  | e14ec0dab4 | ||
|  | 8252b9b947 | ||
|  | 9fb88021dd | ||
|  | b031d6ecae | ||
|  | fcfdde5390 | ||
|  | 571ba8e4da | ||
|  | bed6234e26 | ||
|  | 9d24578812 | ||
|  | a087c4e75d | ||
|  | 6b64b26d8b | ||
|  | 22252a98e3 | ||
|  | 957bc93244 | ||
|  | 18bbe2e8a0 | ||
|  | 1b17913cbc | ||
|  | ce2a36840d | ||
|  | 0e09feccb0 | ||
|  | 40a1de699a | ||
|  | d906a366c8 | ||
|  | d44d4c8c8b | ||
|  | af0e1d7a85 | ||
|  | d21b4d7f98 | ||
|  | 3bdaa69fb2 | ||
|  | 5c0132882a | ||
|  | ed443e949e | ||
|  | 1df69f3c64 | ||
|  | d7cafafeee | ||
|  | e09348d366 | ||
|  | 5f1b7c14f5 | ||
|  | 5b3e9198f0 | ||
|  | f0b21dfd25 | ||
|  | e165d1aaa4 | ||
|  | 649496eb57 | ||
|  | 135a45c833 | ||
|  | 1a3d9dbabf | ||
|  | a91d15df7e | ||
|  | 3368e2a3d1 | ||
|  | 929d788e21 | ||
|  | 6958065f85 | ||
|  | 097c82b2bc | ||
|  | abaeb3f55f | ||
|  | f42e29c79c | ||
|  | 3177580cc1 | ||
|  | cbe77e1657 | ||
|  | 3cedc4738f | ||
|  | 0834fd9b85 | ||
|  | 7b65c986eb | ||
|  | 17d751531e | ||
|  | 3c1221e2b2 | ||
|  | 92e9caed63 | ||
|  | facb5cdfc2 | ||
|  | aebea482f5 | ||
|  | 12f20d369c | ||
|  | 2008d3f54c | ||
|  | a26ebe40f5 | ||
|  | a52f1cacb3 | ||
|  | f5e71415a2 | ||
|  | 82a8d6e393 | ||
|  | 1226c93937 | ||
|  | b9c520f9b3 | ||
|  | 8a7ca57d3e | ||
|  | 29f9f9fc79 | ||
|  | d512e454b3 | ||
|  | 22481bf76d | ||
|  | b973955295 | ||
|  | 9d51a36528 | ||
|  | ba3bbe55c1 | ||
|  | 0619478073 | ||
|  | f8f184edab | ||
|  | d838951302 | ||
|  | f8e1b45a78 | ||
|  | dd2c9fdbcf | ||
|  | 9991421cfb | ||
|  | e30f0e1c75 | ||
|  | 18598ff66d | ||
|  | 8882c9e3a8 | ||
|  | fdc64f6806 | ||
|  | 16b331d14f | ||
|  | ab46372e2a | ||
|  | be807e4250 | ||
|  | 71ddbf5c4f | ||
|  | 63bb29fac0 | ||
|  | 04b4f915a7 | ||
|  | 4d95ab2231 | ||
|  | 17944f7285 | ||
|  | d2964b6cd1 | ||
|  | 1ce5d7c8b7 | ||
|  | 846457b10a | ||
|  | e7b9771c4d | ||
|  | d709900efa | ||
|  | 55b4f5cc2e | ||
|  | 1ac5ac75a9 | ||
|  | 5cf38ed1ab | ||
|  | 35d1531089 | ||
|  | 47e3777caa | ||
|  | 710f3c615c | ||
|  | 3111560e8a | ||
|  | 7396afbba4 | ||
|  | 52a72e217e | ||
|  | ff0a562f9a | ||
|  | 556008d724 | ||
|  | 0094f84f30 | ||
|  | 86069143ff | ||
|  | 44f0be88a3 | ||
|  | 5d88b372d7 | ||
|  | 71c7bf5907 | ||
|  | 869033148c | ||
|  | bc0f7c0988 | ||
|  | 7d06063cfb | ||
|  | 4e42b637fd | ||
|  | f44de9942b | ||
|  | 3a110ae60b | ||
|  | bb84adc465 | ||
|  | 8d123ea3c0 | ||
|  | 88ca894df7 | ||
|  | 42b0d6b494 | ||
|  | 82d8b0457b | ||
|  | 433d6ee1a2 | ||
|  | 203a6eddf8 | ||
|  | 56ef6249e3 | ||
|  | b2a96b1be7 | ||
|  | d4c29c1574 | ||
|  | 3d947e6d67 | ||
|  | b62c9a19cf | ||
|  | ff5957568f | ||
|  | 7d2e1ca4be | ||
|  | 7ce2e0c8b0 | ||
|  | 78d442420b | ||
|  | 8cd2709ebf | ||
|  | 41a1780102 | ||
|  | 2f84715984 | ||
|  | 7253154fc5 | ||
|  | 6c1c76683f | ||
|  | a92fe9a4ca | ||
|  | e83a3f584e | ||
|  | 118ddebc36 | ||
|  | bb53004d47 | ||
|  | 6af20842cb | ||
|  | cc41cb07e7 | ||
|  | d6fb23523a | ||
|  | 2aa0b45cc0 | ||
|  | 619df61ad2 | ||
|  | 893ea65f38 | ||
|  | 64b811f113 | ||
|  | 91fc9bf862 | ||
|  | 111a58dd37 | ||
|  | d1a2563a74 | ||
|  | 7d3ef919ce | ||
|  | cba98d87d6 | ||
|  | 5c18ad0829 | ||
|  | 0d9ed87d5c | ||
|  | ec7be0c969 | ||
|  | 9be3347601 | ||
|  | 3eef7b7d81 | ||
|  | 9de4a6daa9 | ||
|  | 851ace9f33 | ||
|  | d1dd069b48 | ||
|  | 401db32ca2 | ||
|  | 17e03d21d2 | ||
|  | 26a9bff201 | ||
|  | 80fb260a60 | ||
|  | 55a0716da7 | ||
|  | c88fb75616 | ||
|  | d55a4dc326 | ||
|  | a4e6f2e6e1 | ||
|  | 7f71d99cc3 | ||
|  | b92167cf80 | ||
|  | 4b6a6dbe7e | ||
|  | 763e8c7766 | ||
|  | 823ff16088 | ||
|  | 6f93c6a1e0 | ||
|  | f97e48b0de | ||
|  | 761600b0fd | ||
|  | 8549462bc6 | ||
|  | 436e5c6308 | ||
|  | f8b9d844c1 | ||
|  | 58ec07d580 | ||
|  | 71465c21f4 | ||
|  | 16d0a833f8 | ||
|  | ea72b62cac | ||
|  | 49a84ec6e9 | ||
|  | 42c636b6c8 | ||
|  | a0a55f555e | ||
|  | 23ed00e410 | ||
|  | 3fe910b9f1 | ||
|  | 097bdeb77d | ||
|  | 1b85de02e0 | ||
|  | 2281d1835f | ||
|  | fb4433a129 | ||
|  | d954dcf9e1 | 
| @@ -1,9 +1,7 @@ | |||||||
| dnl Process this file with autoconf to produce a configure script | dnl Process this file with autoconf to produce a configure script | ||||||
| AC_INIT([openbsc], | AC_INIT | ||||||
| 	m4_esyscmd([./git-version-gen .tarball-version]), |  | ||||||
| 	[openbsc-devel@lists.openbsc.org]) |  | ||||||
|  |  | ||||||
| AM_INIT_AUTOMAKE([dist-bzip2]) | AM_INIT_AUTOMAKE(openbsc, 0.3.94onwaves) | ||||||
|  |  | ||||||
| dnl kernel style compile messages | dnl kernel style compile messages | ||||||
| m4_ifdef([AM_SILENT_RULES], [AM_SILENT_RULES([yes])]) | m4_ifdef([AM_SILENT_RULES], [AM_SILENT_RULES([yes])]) | ||||||
| @@ -54,4 +52,5 @@ AC_OUTPUT( | |||||||
|     tests/db/Makefile |     tests/db/Makefile | ||||||
|     tests/channel/Makefile |     tests/channel/Makefile | ||||||
|     tests/sccp/Makefile |     tests/sccp/Makefile | ||||||
|  |     tests/bsc-nat/Makefile | ||||||
|     Makefile) |     Makefile) | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								openbsc/contrib/README
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								openbsc/contrib/README
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | |||||||
|  | This contains a set of scripts used for the development of the | ||||||
|  | MSC functionality. | ||||||
| @@ -10,7 +10,7 @@ rsip_resp = """200 321321332\r\n""" | |||||||
| audit_packet = """AUEP %d 13@mgw MGCP 1.0\r\n""" | audit_packet = """AUEP %d 13@mgw MGCP 1.0\r\n""" | ||||||
| crcx_packet = """CRCX %d 14@mgw MGCP 1.0\r\nC: 4a84ad5d25f\r\nL: p:20, a:GSM-EFR, nt:IN\r\nM: recvonly\r\n""" | crcx_packet = """CRCX %d 14@mgw MGCP 1.0\r\nC: 4a84ad5d25f\r\nL: p:20, a:GSM-EFR, nt:IN\r\nM: recvonly\r\n""" | ||||||
| dlcx_packet = """DLCX %d 14@mgw MGCP 1.0\r\nC: 4a84ad5d25f\r\nI: %d\r\n""" | dlcx_packet = """DLCX %d 14@mgw MGCP 1.0\r\nC: 4a84ad5d25f\r\nI: %d\r\n""" | ||||||
| mdcx_packet = """MDCX %d 14@mgw MGCP 1.0\r\nC: 4a84ad5d25f\r\nI: %d\r\nL: p:20, a:GSM-EFR, nt:IN\r\nM: recvonly\r\n\r\nv=0\r\no=- 258696477 0 IN IP4 172.16.1.107\r\ns=-\r\nc=IN IP4 172.16.1.107\r\nt=0 0\r\nm=audio 4400 RTP/AVP 127\r\na=rtpmap:127 GSM-EFR/8000/1\r\na=ptime:20\r\na=recvonly\r\nm=image 4402 udptl t38\r\na=T38FaxVersion:0\r\na=T38MaxBitRate:14400\r\n""" | mdcx_packet = """MDCX %d 14@mgw MGCP 1.0\r\nC: 4a84ad5d25f\r\nI: %d\r\nL: p:20, a:GSM-EFR, nt:IN\r\nM: recvonly\r\n\r\nv=0\r\no=- 258696477 0 IN IP4 172.16.1.107\r\ns=-\r\nc=IN IP4 172.16.1.107\r\nt=0 0\r\nm=audio 6666 RTP/AVP 127\r\na=rtpmap:127 GSM-EFR/8000/1\r\na=ptime:20\r\na=recvonly\r\nm=image 4402 udptl t38\r\na=T38FaxVersion:0\r\na=T38MaxBitRate:14400\r\n""" | ||||||
|  |  | ||||||
| def hexdump(src, length=8): | def hexdump(src, length=8): | ||||||
|     """Recipe is from http://code.activestate.com/recipes/142812/""" |     """Recipe is from http://code.activestate.com/recipes/142812/""" | ||||||
| @@ -25,15 +25,24 @@ def hexdump(src, length=8): | |||||||
|  |  | ||||||
| server_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) | server_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) | ||||||
| server_socket.bind(("127.0.0.1", MGCP_CALLAGENT_PORT)) | server_socket.bind(("127.0.0.1", MGCP_CALLAGENT_PORT)) | ||||||
| server_socket.setblocking(0) | server_socket.setblocking(1) | ||||||
|  |  | ||||||
|  | last_ci = 1 | ||||||
| def send_receive(packet): | def send_and_receive(packet): | ||||||
|  |     global last_ci | ||||||
|     server_socket.sendto(packet, ("127.0.0.1", MGCP_GATEWAY_PORT)) |     server_socket.sendto(packet, ("127.0.0.1", MGCP_GATEWAY_PORT)) | ||||||
|     try: |     try: | ||||||
|         data, addr = server_socket.recvfrom(4096) |         data, addr = server_socket.recvfrom(4096) | ||||||
|  |  | ||||||
|  |         # attempt to store the CI of the response | ||||||
|  |         list = data.split("\n") | ||||||
|  |         for item in list: | ||||||
|  |            if item.startswith("I: "): | ||||||
|  |                last_ci = int(item[3:]) | ||||||
|  |  | ||||||
|         print hexdump(data), addr |         print hexdump(data), addr | ||||||
|     except socket.error: |     except socket.error, e: | ||||||
|  |         print e | ||||||
|         pass |         pass | ||||||
|  |  | ||||||
| def generate_tid(): | def generate_tid(): | ||||||
| @@ -42,13 +51,10 @@ def generate_tid(): | |||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| i = 1 |  | ||||||
| while True: | while True: | ||||||
|     send_receive(rsip_resp) |     send_and_receive(audit_packet % generate_tid()) | ||||||
|     send_receive(audit_packet) |     send_and_receive(crcx_packet % generate_tid() ) | ||||||
|     send_receive(crcx_packet % generate_tid() ) |     send_and_receive(mdcx_packet % (generate_tid(), last_ci)) | ||||||
|     send_receive(mdcx_packet % (generate_tid(), i)) |     send_and_receive(dlcx_packet % (generate_tid(), last_ci)) | ||||||
|     send_receive(dlcx_packet % (generate_tid(), i)) |  | ||||||
|     i = i + 1 |  | ||||||
|  |  | ||||||
|     time.sleep(3) |     time.sleep(3) | ||||||
|   | |||||||
							
								
								
									
										30
									
								
								openbsc/contrib/send_handshake.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										30
									
								
								openbsc/contrib/send_handshake.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,30 @@ | |||||||
|  | #!/usr/bin/env python | ||||||
|  |  | ||||||
|  | import sys | ||||||
|  |  | ||||||
|  | # packages | ||||||
|  | ACK ="\x00\x01\xfe\x06" | ||||||
|  | RESET_ACK = "\x00\x13\xfd\x09\x00\x03\x07\x0b\x04\x43\x01\x00\xfe\x04\x43\x5c\x00\xfe\x03\x00\x01\x31" | ||||||
|  | PAGE = "\x00\x20\xfd\x09\x00\x03\x07\x0b\x04\x43\x01\x00\xfe\x04\x43\x5c\x00\xfe\x10\x00\x0e\x52\x08\x08\x29\x42\x08\x05\x03\x12\x23\x42\x1a\x01\x06" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # simple handshake... | ||||||
|  | sys.stdout.write(ACK) | ||||||
|  | sys.stdout.flush() | ||||||
|  | sys.stdin.read(4) | ||||||
|  |  | ||||||
|  | # wait for some data and send reset ack | ||||||
|  | sys.stdin.read(21) | ||||||
|  | sys.stdout.write(RESET_ACK) | ||||||
|  | sys.stdout.flush() | ||||||
|  |  | ||||||
|  | sys.stdout.write(RESET_ACK) | ||||||
|  | sys.stdout.flush() | ||||||
|  |  | ||||||
|  | # page a subscriber | ||||||
|  | sys.stdout.write(PAGE) | ||||||
|  | sys.stdout.flush() | ||||||
|  |  | ||||||
|  | while True: | ||||||
|  |     sys.stdin.read(1) | ||||||
|  |  | ||||||
| @@ -6,7 +6,7 @@ noinst_HEADERS = abis_nm.h abis_rsl.h db.h gsm_04_08.h gsm_data.h \ | |||||||
| 		 bsc_rll.h mncc.h transaction.h ussd.h gsm_04_80.h \ | 		 bsc_rll.h mncc.h transaction.h ussd.h gsm_04_80.h \ | ||||||
| 		 silent_call.h mgcp.h meas_rep.h rest_octets.h \ | 		 silent_call.h mgcp.h meas_rep.h rest_octets.h \ | ||||||
| 		 system_information.h handover.h mgcp_internal.h \ | 		 system_information.h handover.h mgcp_internal.h \ | ||||||
| 		 vty.h | 		 vty.h bssap.h bsc_msc.h bsc_nat.h | ||||||
|  |  | ||||||
| openbsc_HEADERS = gsm_04_08.h meas_rep.h bsc_api.h | openbsc_HEADERS = gsm_04_08.h meas_rep.h bsc_api.h | ||||||
| openbscdir = $(includedir)/openbsc | openbscdir = $(includedir)/openbsc | ||||||
|   | |||||||
| @@ -68,12 +68,14 @@ unsigned int get_paging_group(u_int64_t imsi, unsigned int bs_cc_chans, | |||||||
| unsigned int n_pag_blocks(int bs_ccch_sdcch_comb, unsigned int bs_ag_blks_res); | unsigned int n_pag_blocks(int bs_ccch_sdcch_comb, unsigned int bs_ag_blks_res); | ||||||
| u_int64_t str_to_imsi(const char *imsi_str); | u_int64_t str_to_imsi(const char *imsi_str); | ||||||
| u_int8_t lchan2chan_nr(const struct gsm_lchan *lchan); | u_int8_t lchan2chan_nr(const struct gsm_lchan *lchan); | ||||||
| int rsl_release_request(struct gsm_lchan *lchan, u_int8_t link_id); | int rsl_release_request(struct gsm_lchan *lchan, u_int8_t link_id, u_int8_t release_reason); | ||||||
|  |  | ||||||
|  | int rsl_lchan_set_state(struct gsm_lchan *lchan, int); | ||||||
|  |  | ||||||
| /* to be provided by external code */ | /* to be provided by external code */ | ||||||
| int abis_rsl_sendmsg(struct msgb *msg); | int abis_rsl_sendmsg(struct msgb *msg); | ||||||
| int rsl_deact_sacch(struct gsm_lchan *lchan); | int rsl_deact_sacch(struct gsm_lchan *lchan); | ||||||
| int rsl_chan_release(struct gsm_lchan *lchan); | int rsl_lchan_rll_release(struct gsm_lchan *lchan, u_int8_t link_id); | ||||||
|  |  | ||||||
| /* BCCH related code */ | /* BCCH related code */ | ||||||
| int rsl_ccch_conf_to_bs_cc_chans(int ccch_conf); | int rsl_ccch_conf_to_bs_cc_chans(int ccch_conf); | ||||||
|   | |||||||
							
								
								
									
										46
									
								
								openbsc/include/openbsc/bsc_msc.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								openbsc/include/openbsc/bsc_msc.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | |||||||
|  | /* Routines to talk to the MSC using the IPA Protocol */ | ||||||
|  | /* | ||||||
|  |  * (C) 2010 by Holger Hans Peter Freyther <zecke@selfish.org> | ||||||
|  |  * (C) 2010 by On-Waves | ||||||
|  |  * All Rights Reserved | ||||||
|  |  * | ||||||
|  |  * This program is free software; you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation; either version 2 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License along | ||||||
|  |  * with this program; if not, write to the Free Software Foundation, Inc., | ||||||
|  |  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #ifndef BSC_MSC_H | ||||||
|  | #define BSC_MSC_H | ||||||
|  |  | ||||||
|  | #include <osmocore/write_queue.h> | ||||||
|  | #include <osmocore/timer.h> | ||||||
|  |  | ||||||
|  | struct bsc_msc_connection { | ||||||
|  | 	struct write_queue write_queue; | ||||||
|  | 	int is_connected; | ||||||
|  | 	const char *ip; | ||||||
|  | 	int port; | ||||||
|  |  | ||||||
|  | 	void (*connection_loss) (struct bsc_msc_connection *); | ||||||
|  | 	void (*connected) (struct bsc_msc_connection *); | ||||||
|  | 	struct timer_list reconnect_timer; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | struct bsc_msc_connection *bsc_msc_create(const char *ip, int port); | ||||||
|  | int bsc_msc_connect(struct bsc_msc_connection *); | ||||||
|  | void bsc_msc_schedule_connect(struct bsc_msc_connection *); | ||||||
|  |  | ||||||
|  | void bsc_msc_lost(struct bsc_msc_connection *); | ||||||
|  |  | ||||||
|  | #endif | ||||||
							
								
								
									
										217
									
								
								openbsc/include/openbsc/bsc_nat.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										217
									
								
								openbsc/include/openbsc/bsc_nat.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,217 @@ | |||||||
|  | /* | ||||||
|  |  * (C) 2010 by Holger Hans Peter Freyther <zecke@selfish.org> | ||||||
|  |  * (C) 2010 by On-Waves | ||||||
|  |  * All Rights Reserved | ||||||
|  |  * | ||||||
|  |  * This program is free software; you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation; either version 2 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License along | ||||||
|  |  * with this program; if not, write to the Free Software Foundation, Inc., | ||||||
|  |  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #ifndef BSC_NAT_H | ||||||
|  | #define BSC_NAT_H | ||||||
|  |  | ||||||
|  | #include "mgcp.h" | ||||||
|  |  | ||||||
|  | #include <sys/types.h> | ||||||
|  | #include <sccp/sccp_types.h> | ||||||
|  |  | ||||||
|  | #include <osmocore/select.h> | ||||||
|  | #include <osmocore/msgb.h> | ||||||
|  | #include <osmocore/timer.h> | ||||||
|  | #include <osmocore/write_queue.h> | ||||||
|  |  | ||||||
|  | #define DIR_BSC 1 | ||||||
|  | #define DIR_MSC 2 | ||||||
|  |  | ||||||
|  | #define NAT_IPAC_PROTO_MGCP	0xfc | ||||||
|  |  | ||||||
|  | struct bsc_nat; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * For the NAT we will need to analyze and later patch | ||||||
|  |  * the received message. This would require us to parse | ||||||
|  |  * the IPA and SCCP header twice. Instead of doing this | ||||||
|  |  * we will have one analyze structure and have the patching | ||||||
|  |  * and filter operate on the same structure. | ||||||
|  |  */ | ||||||
|  | struct bsc_nat_parsed { | ||||||
|  | 	/* ip access prototype */ | ||||||
|  | 	int ipa_proto; | ||||||
|  |  | ||||||
|  | 	/* source local reference */ | ||||||
|  | 	struct sccp_source_reference *src_local_ref; | ||||||
|  |  | ||||||
|  | 	/* destination local reference */ | ||||||
|  | 	struct sccp_source_reference *dest_local_ref; | ||||||
|  |  | ||||||
|  | 	/* called ssn number */ | ||||||
|  | 	int called_ssn; | ||||||
|  |  | ||||||
|  | 	/* calling ssn number */ | ||||||
|  | 	int calling_ssn; | ||||||
|  |  | ||||||
|  | 	/* sccp message type */ | ||||||
|  | 	int sccp_type; | ||||||
|  |  | ||||||
|  | 	/* bssap type, e.g. 0 for BSS Management */ | ||||||
|  | 	int bssap; | ||||||
|  |  | ||||||
|  | 	/* the gsm0808 message type */ | ||||||
|  | 	int gsm_type; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Per BSC data structure | ||||||
|  |  */ | ||||||
|  | struct bsc_connection { | ||||||
|  | 	struct llist_head list_entry; | ||||||
|  |  | ||||||
|  | 	/* do we know anything about this BSC? */ | ||||||
|  | 	int authenticated; | ||||||
|  |  | ||||||
|  | 	/* the fd we use to communicate */ | ||||||
|  | 	struct write_queue write_queue; | ||||||
|  |  | ||||||
|  | 	/* the BSS associated */ | ||||||
|  | 	struct bsc_config *cfg; | ||||||
|  |  | ||||||
|  | 	/* a timeout node */ | ||||||
|  | 	struct timer_list id_timeout; | ||||||
|  |  | ||||||
|  | 	/* a back pointer */ | ||||||
|  | 	struct bsc_nat *nat; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Per SCCP source local reference patch table. It needs to | ||||||
|  |  * be updated on new SCCP connections, connection confirm and reject, | ||||||
|  |  * and on the loss of the BSC connection. | ||||||
|  |  */ | ||||||
|  | struct sccp_connections { | ||||||
|  | 	struct llist_head list_entry; | ||||||
|  |  | ||||||
|  | 	struct bsc_connection *bsc; | ||||||
|  |  | ||||||
|  | 	struct sccp_source_reference real_ref; | ||||||
|  | 	struct sccp_source_reference patched_ref; | ||||||
|  | 	struct sccp_source_reference remote_ref; | ||||||
|  |  | ||||||
|  | 	/* GSM audio handling. That is 32 * multiplex + ts */ | ||||||
|  | 	int msc_timeslot; | ||||||
|  | 	int bsc_timeslot; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * One BSC entry in the config | ||||||
|  |  */ | ||||||
|  | struct bsc_config { | ||||||
|  | 	struct llist_head entry; | ||||||
|  |  | ||||||
|  | 	char *token; | ||||||
|  | 	unsigned int lac; | ||||||
|  | 	int nr; | ||||||
|  |  | ||||||
|  | 	struct bsc_nat *nat; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * BSCs point of view of endpoints | ||||||
|  |  */ | ||||||
|  | struct bsc_endpoint { | ||||||
|  | 	/* the pending transaction id */ | ||||||
|  | 	char *transaction_id; | ||||||
|  | 	/* the bsc we are talking to */ | ||||||
|  | 	struct bsc_connection *bsc; | ||||||
|  | 	/* pending delete */ | ||||||
|  | 	int pending_delete; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * the structure of the "nat" network | ||||||
|  |  */ | ||||||
|  | struct bsc_nat { | ||||||
|  | 	/* active SCCP connections that need patching */ | ||||||
|  | 	struct llist_head sccp_connections; | ||||||
|  |  | ||||||
|  | 	/* active BSC connections that need patching */ | ||||||
|  | 	struct llist_head bsc_connections; | ||||||
|  |  | ||||||
|  | 	/* known BSC's */ | ||||||
|  | 	struct llist_head bsc_configs; | ||||||
|  | 	int num_bsc; | ||||||
|  |  | ||||||
|  | 	/* MGCP config */ | ||||||
|  | 	struct mgcp_config *mgcp_cfg; | ||||||
|  | 	struct write_queue mgcp_queue; | ||||||
|  | 	u_int8_t mgcp_msg[4096]; | ||||||
|  | 	int mgcp_length; | ||||||
|  |  | ||||||
|  | 	/* msc things */ | ||||||
|  | 	int first_contact; | ||||||
|  |  | ||||||
|  | 	struct bsc_endpoint *bsc_endpoints; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /* create and init the structures */ | ||||||
|  | struct bsc_config *bsc_config_alloc(struct bsc_nat *nat, const char *token, unsigned int lac); | ||||||
|  | struct bsc_config *bsc_config_num(struct bsc_nat *nat, int num); | ||||||
|  | struct bsc_nat *bsc_nat_alloc(void); | ||||||
|  | struct bsc_connection *bsc_connection_alloc(struct bsc_nat *nat); | ||||||
|  |  | ||||||
|  | void sccp_connection_destroy(struct sccp_connections *); | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * parse the given message into the above structure | ||||||
|  |  */ | ||||||
|  | struct bsc_nat_parsed *bsc_nat_parse(struct msgb *msg); | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * filter based on IP Access header in both directions | ||||||
|  |  */ | ||||||
|  | int bsc_nat_filter_ipa(int direction, struct msgb *msg, struct bsc_nat_parsed *parsed); | ||||||
|  | int bsc_nat_vty_init(struct bsc_nat *nat); | ||||||
|  | struct bsc_connection *bsc_nat_find_bsc(struct bsc_nat *nat, struct msgb *msg); | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * SCCP patching and handling | ||||||
|  |  */ | ||||||
|  | int create_sccp_src_ref(struct bsc_connection *bsc, struct msgb *msg, struct bsc_nat_parsed *parsed); | ||||||
|  | int update_sccp_src_ref(struct sccp_connections *sccp, struct bsc_nat_parsed *parsed); | ||||||
|  | void remove_sccp_src_ref(struct bsc_connection *bsc, struct msgb *msg, struct bsc_nat_parsed *parsed); | ||||||
|  | struct sccp_connections *patch_sccp_src_ref_to_bsc(struct msgb *, struct bsc_nat_parsed *, struct bsc_nat *); | ||||||
|  | struct sccp_connections *patch_sccp_src_ref_to_msc(struct msgb *, struct bsc_nat_parsed *, struct bsc_nat *); | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * MGCP/Audio handling | ||||||
|  |  */ | ||||||
|  | int bsc_write_mgcp(struct bsc_connection *bsc, const u_int8_t *data, unsigned int length); | ||||||
|  | int bsc_mgcp_assign(struct sccp_connections *, struct msgb *msg); | ||||||
|  | void bsc_mgcp_clear(struct sccp_connections *); | ||||||
|  | void bsc_mgcp_free_endpoint(struct bsc_nat *nat, int); | ||||||
|  | void bsc_mgcp_free_endpoints(struct bsc_nat *nat); | ||||||
|  | int bsc_mgcp_init(struct bsc_nat *nat); | ||||||
|  |  | ||||||
|  | struct bsc_connection *bsc_mgcp_find_con(struct bsc_nat *, int endpoint_number); | ||||||
|  | struct msgb *bsc_mgcp_rewrite(char *input, int length, const char *ip, int port); | ||||||
|  | void bsc_mgcp_forward(struct bsc_connection *bsc, struct msgb *msg); | ||||||
|  |  | ||||||
|  | void bsc_mgcp_clear_endpoints_for(struct bsc_connection *bsc); | ||||||
|  | int bsc_mgcp_parse_response(const char *str, int *code, char transaction[60]); | ||||||
|  | int bsc_mgcp_extract_ci(const char *resp); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | int bsc_write(struct bsc_connection *bsc, struct msgb *msg, int id); | ||||||
|  |  | ||||||
|  | #endif | ||||||
							
								
								
									
										335
									
								
								openbsc/include/openbsc/bssap.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										335
									
								
								openbsc/include/openbsc/bssap.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,335 @@ | |||||||
|  | /* From GSM08.08 */ | ||||||
|  |  | ||||||
|  | #ifndef BSSAP_H | ||||||
|  | #define BSSAP_H | ||||||
|  |  | ||||||
|  | #include <stdlib.h> | ||||||
|  |  | ||||||
|  | #include <osmocore/msgb.h> | ||||||
|  | #include <openbsc/gsm_data.h> | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * this is from GSM 03.03 CGI but is copied in GSM 08.08 | ||||||
|  |  * in § 3.2.2.27 for Cell Identifier List | ||||||
|  |  */ | ||||||
|  | enum CELL_IDENT { | ||||||
|  | 	CELL_IDENT_WHOLE_GLOBAL		= 0, | ||||||
|  | 	CELL_IDENT_LAC_AND_CI		= 1, | ||||||
|  | 	CELL_IDENT_CI			= 2, | ||||||
|  | 	CELL_IDENT_NO_CELL		= 3, | ||||||
|  | 	CELL_IDENT_LAI_AND_LAC		= 4, | ||||||
|  | 	CELL_IDENT_LAC			= 5, | ||||||
|  | 	CELL_IDENT_BSS			= 6, | ||||||
|  | 	CELL_IDENT_UTRAN_PLMN_LAC_RNC	= 8, | ||||||
|  | 	CELL_IDENT_UTRAN_RNC		= 9, | ||||||
|  | 	CELL_IDENT_UTRAN_LAC_RNC	= 10, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | /* GSM 08.06 § 6.3 */ | ||||||
|  | enum BSSAP_MSG_TYPE { | ||||||
|  | 	BSSAP_MSG_BSS_MANAGEMENT    = 0x0, | ||||||
|  | 	BSSAP_MSG_DTAP		    = 0x1, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | struct bssmap_header { | ||||||
|  | 	u_int8_t type; | ||||||
|  | 	u_int8_t length; | ||||||
|  | } __attribute__((packed)); | ||||||
|  |  | ||||||
|  | struct dtap_header { | ||||||
|  | 	u_int8_t type; | ||||||
|  | 	u_int8_t link_id; | ||||||
|  | 	u_int8_t length; | ||||||
|  | } __attribute__((packed)); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | enum BSS_MAP_MSG_TYPE { | ||||||
|  | 	BSS_MAP_MSG_RESERVED_0		= 0, | ||||||
|  |  | ||||||
|  | 	/* ASSIGNMENT MESSAGES */ | ||||||
|  | 	BSS_MAP_MSG_ASSIGMENT_RQST	= 1, | ||||||
|  | 	BSS_MAP_MSG_ASSIGMENT_COMPLETE	= 2, | ||||||
|  | 	BSS_MAP_MSG_ASSIGMENT_FAILURE	= 3, | ||||||
|  |  | ||||||
|  | 	/*  HANDOVER MESSAGES */ | ||||||
|  | 	BSS_MAP_MSG_HANDOVER_RQST		= 16, | ||||||
|  | 	BSS_MAP_MSG_HANDOVER_REQUIRED		= 17, | ||||||
|  | 	BSS_MAP_MSG_HANDOVER_RQST_ACKNOWLEDGE= 18, | ||||||
|  | 	BSS_MAP_MSG_HANDOVER_CMD		= 19, | ||||||
|  | 	BSS_MAP_MSG_HANDOVER_COMPLETE		= 20, | ||||||
|  | 	BSS_MAP_MSG_HANDOVER_SUCCEEDED		= 21, | ||||||
|  | 	BSS_MAP_MSG_HANDOVER_FAILURE		= 22, | ||||||
|  | 	BSS_MAP_MSG_HANDOVER_PERFORMED		= 23, | ||||||
|  | 	BSS_MAP_MSG_HANDOVER_CANDIDATE_ENQUIRE	= 24, | ||||||
|  | 	BSS_MAP_MSG_HANDOVER_CANDIDATE_RESPONSE	= 25, | ||||||
|  | 	BSS_MAP_MSG_HANDOVER_REQUIRED_REJECT	= 26, | ||||||
|  | 	BSS_MAP_MSG_HANDOVER_DETECT		= 27, | ||||||
|  |  | ||||||
|  | 	/* RELEASE MESSAGES */ | ||||||
|  | 	BSS_MAP_MSG_CLEAR_CMD		= 32, | ||||||
|  | 	BSS_MAP_MSG_CLEAR_COMPLETE		= 33, | ||||||
|  | 	BSS_MAP_MSG_CLEAR_RQST		= 34, | ||||||
|  | 	BSS_MAP_MSG_RESERVED_1			= 35, | ||||||
|  | 	BSS_MAP_MSG_RESERVED_2			= 36, | ||||||
|  | 	BSS_MAP_MSG_SAPI_N_REJECT		= 37, | ||||||
|  | 	BSS_MAP_MSG_CONFUSION			= 38, | ||||||
|  |  | ||||||
|  | 	/* OTHER CONNECTION RELATED MESSAGES */ | ||||||
|  | 	BSS_MAP_MSG_SUSPEND			= 40, | ||||||
|  | 	BSS_MAP_MSG_RESUME			= 41, | ||||||
|  | 	BSS_MAP_MSG_CONNECTION_ORIENTED_INFORMATION = 42, | ||||||
|  | 	BSS_MAP_MSG_PERFORM_LOCATION_RQST	= 43, | ||||||
|  | 	BSS_MAP_MSG_LSA_INFORMATION		= 44, | ||||||
|  | 	BSS_MAP_MSG_PERFORM_LOCATION_RESPONSE	= 45, | ||||||
|  | 	BSS_MAP_MSG_PERFORM_LOCATION_ABORT	= 46, | ||||||
|  | 	BSS_MAP_MSG_COMMON_ID			= 47, | ||||||
|  |  | ||||||
|  | 	/* GENERAL MESSAGES */ | ||||||
|  | 	BSS_MAP_MSG_RESET			= 48, | ||||||
|  | 	BSS_MAP_MSG_RESET_ACKNOWLEDGE		= 49, | ||||||
|  | 	BSS_MAP_MSG_OVERLOAD			= 50, | ||||||
|  | 	BSS_MAP_MSG_RESERVED_3			= 51, | ||||||
|  | 	BSS_MAP_MSG_RESET_CIRCUIT		= 52, | ||||||
|  | 	BSS_MAP_MSG_RESET_CIRCUIT_ACKNOWLEDGE	= 53, | ||||||
|  | 	BSS_MAP_MSG_MSC_INVOKE_TRACE		= 54, | ||||||
|  | 	BSS_MAP_MSG_BSS_INVOKE_TRACE		= 55, | ||||||
|  | 	BSS_MAP_MSG_CONNECTIONLESS_INFORMATION	= 58, | ||||||
|  |  | ||||||
|  | 	/* TERRESTRIAL RESOURCE MESSAGES */ | ||||||
|  | 	BSS_MAP_MSG_BLOCK			= 64, | ||||||
|  | 	BSS_MAP_MSG_BLOCKING_ACKNOWLEDGE	= 65, | ||||||
|  | 	BSS_MAP_MSG_UNBLOCK			= 66, | ||||||
|  | 	BSS_MAP_MSG_UNBLOCKING_ACKNOWLEDGE	= 67, | ||||||
|  | 	BSS_MAP_MSG_CIRCUIT_GROUP_BLOCK		= 68, | ||||||
|  | 	BSS_MAP_MSG_CIRCUIT_GROUP_BLOCKING_ACKNOWLEDGE	= 69, | ||||||
|  | 	BSS_MAP_MSG_CIRCUIT_GROUP_UNBLOCK	= 70, | ||||||
|  | 	BSS_MAP_MSG_CIRCUIT_GROUP_UNBLOCKING_ACKNOWLEDGE = 71, | ||||||
|  | 	BSS_MAP_MSG_UNEQUIPPED_CIRCUIT		= 72, | ||||||
|  | 	BSS_MAP_MSG_CHANGE_CIRCUIT		= 78, | ||||||
|  | 	BSS_MAP_MSG_CHANGE_CIRCUIT_ACKNOWLEDGE	= 79, | ||||||
|  |  | ||||||
|  | 	/* RADIO RESOURCE MESSAGES */ | ||||||
|  | 	BSS_MAP_MSG_RESOURCE_RQST		= 80, | ||||||
|  | 	BSS_MAP_MSG_RESOURCE_INDICATION		= 81, | ||||||
|  | 	BSS_MAP_MSG_PAGING			= 82, | ||||||
|  | 	BSS_MAP_MSG_CIPHER_MODE_CMD		= 83, | ||||||
|  | 	BSS_MAP_MSG_CLASSMARK_UPDATE		= 84, | ||||||
|  | 	BSS_MAP_MSG_CIPHER_MODE_COMPLETE	= 85, | ||||||
|  | 	BSS_MAP_MSG_QUEUING_INDICATION		= 86, | ||||||
|  | 	BSS_MAP_MSG_COMPLETE_LAYER_3		= 87, | ||||||
|  | 	BSS_MAP_MSG_CLASSMARK_RQST		= 88, | ||||||
|  | 	BSS_MAP_MSG_CIPHER_MODE_REJECT		= 89, | ||||||
|  | 	BSS_MAP_MSG_LOAD_INDICATION		= 90, | ||||||
|  |  | ||||||
|  | 	/* VGCS/VBS */ | ||||||
|  | 	BSS_MAP_MSG_VGCS_VBS_SETUP		= 4, | ||||||
|  | 	BSS_MAP_MSG_VGCS_VBS_SETUP_ACK		= 5, | ||||||
|  | 	BSS_MAP_MSG_VGCS_VBS_SETUP_REFUSE	= 6, | ||||||
|  | 	BSS_MAP_MSG_VGCS_VBS_ASSIGNMENT_RQST	= 7, | ||||||
|  | 	BSS_MAP_MSG_VGCS_VBS_ASSIGNMENT_RESULT	= 28, | ||||||
|  | 	BSS_MAP_MSG_VGCS_VBS_ASSIGNMENT_FAILURE	= 29, | ||||||
|  | 	BSS_MAP_MSG_VGCS_VBS_QUEUING_INDICATION	= 30, | ||||||
|  | 	BSS_MAP_MSG_UPLINK_RQST		= 31, | ||||||
|  | 	BSS_MAP_MSG_UPLINK_RQST_ACKNOWLEDGE	= 39, | ||||||
|  | 	BSS_MAP_MSG_UPLINK_RQST_CONFIRMATION	= 73, | ||||||
|  | 	BSS_MAP_MSG_UPLINK_RELEASE_INDICATION	= 74, | ||||||
|  | 	BSS_MAP_MSG_UPLINK_REJECT_CMD	= 75, | ||||||
|  | 	BSS_MAP_MSG_UPLINK_RELEASE_CMD	= 76, | ||||||
|  | 	BSS_MAP_MSG_UPLINK_SEIZED_CMD	= 77, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | enum GSM0808_IE_CODING { | ||||||
|  | 	GSM0808_IE_CIRCUIT_IDENTITY_CODE	= 1, | ||||||
|  | 	GSM0808_IE_RESERVED_0			= 2, | ||||||
|  | 	GSM0808_IE_RESOURCE_AVAILABLE		= 3, | ||||||
|  | 	GSM0808_IE_CAUSE			= 4, | ||||||
|  | 	GSM0808_IE_CELL_IDENTIFIER		= 5, | ||||||
|  | 	GSM0808_IE_PRIORITY			= 6, | ||||||
|  | 	GSM0808_IE_LAYER_3_HEADER_INFORMATION	= 7, | ||||||
|  | 	GSM0808_IE_IMSI				= 8, | ||||||
|  | 	GSM0808_IE_TMSI				= 9, | ||||||
|  | 	GSM0808_IE_ENCRYPTION_INFORMATION	= 10, | ||||||
|  | 	GSM0808_IE_CHANNEL_TYPE			= 11, | ||||||
|  | 	GSM0808_IE_PERIODICITY			= 12, | ||||||
|  | 	GSM0808_IE_EXTENDED_RESOURCE_INDICATOR	= 13, | ||||||
|  | 	GSM0808_IE_NUMBER_OF_MSS		= 14, | ||||||
|  | 	GSM0808_IE_RESERVED_1			= 15, | ||||||
|  | 	GSM0808_IE_RESERVED_2			= 16, | ||||||
|  | 	GSM0808_IE_RESERVED_3			= 17, | ||||||
|  | 	GSM0808_IE_CLASSMARK_INFORMATION_T2	= 18, | ||||||
|  | 	GSM0808_IE_CLASSMARK_INFORMATION_T3	= 19, | ||||||
|  | 	GSM0808_IE_INTERFERENCE_BAND_TO_USE	= 20, | ||||||
|  | 	GSM0808_IE_RR_CAUSE			= 21, | ||||||
|  | 	GSM0808_IE_RESERVED_4			= 22, | ||||||
|  | 	GSM0808_IE_LAYER_3_INFORMATION		= 23, | ||||||
|  | 	GSM0808_IE_DLCI				= 24, | ||||||
|  | 	GSM0808_IE_DOWNLINK_DTX_FLAG		= 25, | ||||||
|  | 	GSM0808_IE_CELL_IDENTIFIER_LIST		= 26, | ||||||
|  | 	GSM0808_IE_RESPONSE_RQST		= 27, | ||||||
|  | 	GSM0808_IE_RESOURCE_INDICATION_METHOD	= 28, | ||||||
|  | 	GSM0808_IE_CLASSMARK_INFORMATION_TYPE_1	= 29, | ||||||
|  | 	GSM0808_IE_CIRCUIT_IDENTITY_CODE_LIST	= 30, | ||||||
|  | 	GSM0808_IE_DIAGNOSTIC			= 31, | ||||||
|  | 	GSM0808_IE_LAYER_3_MESSAGE_CONTENTS	= 32, | ||||||
|  | 	GSM0808_IE_CHOSEN_CHANNEL		= 33, | ||||||
|  | 	GSM0808_IE_TOTAL_RESOURCE_ACCESSIBLE	= 34, | ||||||
|  | 	GSM0808_IE_CIPHER_RESPONSE_MODE		= 35, | ||||||
|  | 	GSM0808_IE_CHANNEL_NEEDED		= 36, | ||||||
|  | 	GSM0808_IE_TRACE_TYPE			= 37, | ||||||
|  | 	GSM0808_IE_TRIGGERID			= 38, | ||||||
|  | 	GSM0808_IE_TRACE_REFERENCE		= 39, | ||||||
|  | 	GSM0808_IE_TRANSACTIONID		= 40, | ||||||
|  | 	GSM0808_IE_MOBILE_IDENTITY		= 41, | ||||||
|  | 	GSM0808_IE_OMCID			= 42, | ||||||
|  | 	GSM0808_IE_FORWARD_INDICATOR		= 43, | ||||||
|  | 	GSM0808_IE_CHOSEN_ENCR_ALG		= 44, | ||||||
|  | 	GSM0808_IE_CIRCUIT_POOL			= 45, | ||||||
|  | 	GSM0808_IE_CIRCUIT_POOL_LIST		= 46, | ||||||
|  | 	GSM0808_IE_TIME_INDICATION		= 47, | ||||||
|  | 	GSM0808_IE_RESOURCE_SITUATION		= 48, | ||||||
|  | 	GSM0808_IE_CURRENT_CHANNEL_TYPE_1	= 49, | ||||||
|  | 	GSM0808_IE_QUEUEING_INDICATOR		= 50, | ||||||
|  | 	GSM0808_IE_SPEECH_VERSION		= 64, | ||||||
|  | 	GSM0808_IE_ASSIGNMENT_REQUIREMENT	= 51, | ||||||
|  | 	GSM0808_IE_TALKER_FLAG			= 53, | ||||||
|  | 	GSM0808_IE_CONNECTION_RELEASE_RQSTED	= 54, | ||||||
|  | 	GSM0808_IE_GROUP_CALL_REFERENCE		= 55, | ||||||
|  | 	GSM0808_IE_EMLPP_PRIORITY		= 56, | ||||||
|  | 	GSM0808_IE_CONFIG_EVO_INDI		= 57, | ||||||
|  | 	GSM0808_IE_OLD_BSS_TO_NEW_BSS_INFORMATION	= 58, | ||||||
|  | 	GSM0808_IE_LSA_IDENTIFIER		= 59, | ||||||
|  | 	GSM0808_IE_LSA_IDENTIFIER_LIST		= 60, | ||||||
|  | 	GSM0808_IE_LSA_INFORMATION		= 61, | ||||||
|  | 	GSM0808_IE_LCS_QOS			= 62, | ||||||
|  | 	GSM0808_IE_LSA_ACCESS_CTRL_SUPPR	= 63, | ||||||
|  | 	GSM0808_IE_LCS_PRIORITY			= 67, | ||||||
|  | 	GSM0808_IE_LOCATION_TYPE		= 68, | ||||||
|  | 	GSM0808_IE_LOCATION_ESTIMATE		= 69, | ||||||
|  | 	GSM0808_IE_POSITIONING_DATA		= 70, | ||||||
|  | 	GSM0808_IE_LCS_CAUSE			= 71, | ||||||
|  | 	GSM0808_IE_LCS_CLIENT_TYPE		= 72, | ||||||
|  | 	GSM0808_IE_APDU				= 73, | ||||||
|  | 	GSM0808_IE_NETWORK_ELEMENT_IDENTITY	= 74, | ||||||
|  | 	GSM0808_IE_GPS_ASSISTANCE_DATA		= 75, | ||||||
|  | 	GSM0808_IE_DECIPHERING_KEYS		= 76, | ||||||
|  | 	GSM0808_IE_RETURN_ERROR_RQST		= 77, | ||||||
|  | 	GSM0808_IE_RETURN_ERROR_CAUSE		= 78, | ||||||
|  | 	GSM0808_IE_SEGMENTATION			= 79, | ||||||
|  | 	GSM0808_IE_SERVICE_HANDOVER		= 80, | ||||||
|  | 	GSM0808_IE_SOURCE_RNC_TO_TARGET_RNC_TRANSPARENT_UMTS	= 81, | ||||||
|  | 	GSM0808_IE_SOURCE_RNC_TO_TARGET_RNC_TRANSPARENT_CDMA2000= 82, | ||||||
|  | 	GSM0808_IE_RESERVED_5			= 65, | ||||||
|  | 	GSM0808_IE_RESERVED_6			= 66, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | enum gsm0808_cause { | ||||||
|  | 	GSM0808_CAUSE_RADIO_INTERFACE_MESSAGE_FAILURE			= 0, | ||||||
|  | 	GSM0808_CAUSE_RADIO_INTERFACE_FAILURE				= 1, | ||||||
|  | 	GSM0808_CAUSE_UPLINK_QUALITY					= 2, | ||||||
|  | 	GSM0808_CAUSE_UPLINK_STRENGTH					= 3, | ||||||
|  | 	GSM0808_CAUSE_DOWNLINK_QUALITY					= 4, | ||||||
|  | 	GSM0808_CAUSE_DOWNLINK_STRENGTH					= 5, | ||||||
|  | 	GSM0808_CAUSE_DISTANCE						= 6, | ||||||
|  | 	GSM0808_CAUSE_O_AND_M_INTERVENTION				= 7, | ||||||
|  | 	GSM0808_CAUSE_RESPONSE_TO_MSC_INVOCATION			= 8, | ||||||
|  | 	GSM0808_CAUSE_CALL_CONTROL					= 9, | ||||||
|  | 	GSM0808_CAUSE_RADIO_INTERFACE_FAILURE_REVERSION			= 10, | ||||||
|  | 	GSM0808_CAUSE_HANDOVER_SUCCESSFUL				= 11, | ||||||
|  | 	GSM0808_CAUSE_BETTER_CELL					= 12, | ||||||
|  | 	GSM0808_CAUSE_DIRECTED_RETRY					= 13, | ||||||
|  | 	GSM0808_CAUSE_JOINED_GROUP_CALL_CHANNEL				= 14, | ||||||
|  | 	GSM0808_CAUSE_TRAFFIC						= 15, | ||||||
|  | 	GSM0808_CAUSE_EQUIPMENT_FAILURE					= 32, | ||||||
|  | 	GSM0808_CAUSE_NO_RADIO_RESOURCE_AVAILABLE			= 33, | ||||||
|  | 	GSM0808_CAUSE_RQSTED_TERRESTRIAL_RESOURCE_UNAVAILABLE	= 34, | ||||||
|  | 	GSM0808_CAUSE_CCCH_OVERLOAD					= 35, | ||||||
|  | 	GSM0808_CAUSE_PROCESSOR_OVERLOAD				= 36, | ||||||
|  | 	GSM0808_CAUSE_BSS_NOT_EQUIPPED					= 37, | ||||||
|  | 	GSM0808_CAUSE_MS_NOT_EQUIPPED					= 38, | ||||||
|  | 	GSM0808_CAUSE_INVALID_CELL					= 39, | ||||||
|  | 	GSM0808_CAUSE_TRAFFIC_LOAD					= 40, | ||||||
|  | 	GSM0808_CAUSE_PREEMPTION					= 41, | ||||||
|  | 	GSM0808_CAUSE_RQSTED_TRANSCODING_RATE_ADAPTION_UNAVAILABLE	= 48, | ||||||
|  | 	GSM0808_CAUSE_CIRCUIT_POOL_MISMATCH				= 49, | ||||||
|  | 	GSM0808_CAUSE_SWITCH_CIRCUIT_POOL				= 50, | ||||||
|  | 	GSM0808_CAUSE_RQSTED_SPEECH_VERSION_UNAVAILABLE		= 51, | ||||||
|  | 	GSM0808_CAUSE_LSA_NOT_ALLOWED					= 52, | ||||||
|  | 	GSM0808_CAUSE_CIPHERING_ALGORITHM_NOT_SUPPORTED			= 64, | ||||||
|  | 	GSM0808_CAUSE_TERRESTRIAL_CIRCUIT_ALREADY_ALLOCATED		= 80, | ||||||
|  | 	GSM0808_CAUSE_INVALID_MESSAGE_CONTENTS				= 81, | ||||||
|  | 	GSM0808_CAUSE_INFORMATION_ELEMENT_OR_FIELD_MISSING		= 82, | ||||||
|  | 	GSM0808_CAUSE_INCORRECT_VALUE					= 83, | ||||||
|  | 	GSM0808_CAUSE_UNKNOWN_MESSAGE_TYPE				= 84, | ||||||
|  | 	GSM0808_CAUSE_UNKNOWN_INFORMATION_ELEMENT			= 85, | ||||||
|  | 	GSM0808_CAUSE_PROTOCOL_ERROR_BETWEEN_BSS_AND_MSC		= 96, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /* GSM 08.08 3.2.2.11 Channel Type */ | ||||||
|  | enum gsm0808_chan_indicator { | ||||||
|  | 	GSM0808_CHAN_SPEECH = 1, | ||||||
|  | 	GSM0808_CHAN_DATA   = 2, | ||||||
|  | 	GSM0808_CHAN_SIGN   = 3, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | enum gsm0808_chan_rate_type_data { | ||||||
|  | 	GSM0808_DATA_FULL_BM	= 0x8, | ||||||
|  | 	GSM0808_DATA_HALF_LM	= 0x9, | ||||||
|  | 	GSM0808_DATA_FULL_RPREF	= 0xa, | ||||||
|  | 	GSM0808_DATA_HALF_PREF	= 0xb, | ||||||
|  | 	GSM0808_DATA_FULL_PREF_NO_CHANGE	= 0x1a, | ||||||
|  | 	GSM0808_DATA_HALF_PREF_NO_CHANGE	= 0x1b, | ||||||
|  | 	GSM0808_DATA_MULTI_MASK	= 0x20, | ||||||
|  | 	GSM0808_DATA_MULTI_MASK_NO_CHANGE	= 0x30, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | enum gsm0808_chan_rate_type_speech { | ||||||
|  | 	GSM0808_SPEECH_FULL_BM	= 0x8, | ||||||
|  | 	GSM0808_SPEECH_HALF_LM	= 0x9, | ||||||
|  | 	GSM0808_SPEECH_FULL_PREF= 0xa, | ||||||
|  | 	GSM0808_SPEECH_HALF_PREF= 0xb, | ||||||
|  | 	GSM0808_SPEECH_FULL_PREF_NO_CHANGE	= 0x1a, | ||||||
|  | 	GSM0808_SPEECH_HALF_PREF_NO_CHANGE	= 0x1b, | ||||||
|  | 	GSM0808_SPEECH_PERM	= 0xf, | ||||||
|  | 	GSM0808_SPEECH_PERM_NO_CHANGE = 0x1f, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | enum gsm0808_permitted_speech { | ||||||
|  | 	GSM0808_PERM_FR1	= 0x01, | ||||||
|  | 	GSM0808_PERM_FR2	= 0x11, | ||||||
|  | 	GSM0808_PERM_FR3	= 0x21, | ||||||
|  | 	GSM0808_PERM_HR1	= GSM0808_PERM_FR1 | 0x4, | ||||||
|  | 	GSM0808_PERM_HR2	= GSM0808_PERM_FR2 | 0x4, | ||||||
|  | 	GSM0808_PERM_HR3	= GSM0808_PERM_FR3 | 0x4, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | int bssmap_rcvmsg_dt1(struct sccp_connection *conn, struct msgb *msg, unsigned int length); | ||||||
|  | int bssmap_rcvmsg_udt(struct gsm_network *net, struct msgb *msg, unsigned int length); | ||||||
|  |  | ||||||
|  | struct msgb *bssmap_create_layer3(struct msgb *msg); | ||||||
|  | struct msgb *bssmap_create_reset(void); | ||||||
|  | struct msgb *bssmap_create_clear_complete(void); | ||||||
|  | struct msgb *bssmap_create_cipher_complete(struct msgb *layer3); | ||||||
|  | struct msgb *bssmap_create_cipher_reject(u_int8_t cause); | ||||||
|  | struct msgb *bssmap_create_sapi_reject(u_int8_t link_id); | ||||||
|  | struct msgb *bssmap_create_assignment_completed(struct gsm_lchan *lchan, u_int8_t rr_cause); | ||||||
|  | struct msgb *bssmap_create_assignment_failure(u_int8_t cause, u_int8_t *rr_cause); | ||||||
|  | struct msgb *bssmap_create_classmark_update(const u_int8_t *classmark, u_int8_t length); | ||||||
|  |  | ||||||
|  | void gsm0808_send_assignment_failure(struct gsm_lchan *l, u_int8_t cause, u_int8_t *rr_value); | ||||||
|  | void gsm0808_send_assignment_compl(struct gsm_lchan *l, u_int8_t rr_value); | ||||||
|  |  | ||||||
|  | int dtap_rcvmsg(struct gsm_lchan *lchan, struct msgb *msg, unsigned int length); | ||||||
|  | struct msgb *dtap_create_msg(struct msgb *msg_l3, u_int8_t link_id); | ||||||
|  |  | ||||||
|  | void bsc_queue_connection_write(struct sccp_connection *conn, struct msgb *msg); | ||||||
|  | void bsc_free_queued(struct sccp_connection *conn); | ||||||
|  | void bsc_send_queued(struct sccp_connection *conn); | ||||||
|  |  | ||||||
|  | void bts_send_queued(struct bss_sccp_connection_data*); | ||||||
|  | void bts_free_queued(struct bss_sccp_connection_data*); | ||||||
|  | void bts_unblock_queue(struct bss_sccp_connection_data*); | ||||||
|  |  | ||||||
|  | const struct tlv_definition *gsm0808_att_tlvdef(); | ||||||
|  |  | ||||||
|  | #endif | ||||||
| @@ -23,6 +23,28 @@ | |||||||
|  |  | ||||||
| #include "gsm_subscriber.h" | #include "gsm_subscriber.h" | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Refcounting for the lchan. If the refcount drops to zero | ||||||
|  |  * the channel will send a RSL release request. | ||||||
|  |  */ | ||||||
|  | #define use_subscr_con(con) \ | ||||||
|  | 	do {	(con)->use_count++; \ | ||||||
|  | 		DEBUGP(DREF, "lchan (bts=%d,trx=%d,ts=%d,ch=%d) increases usage to: %d\n", \ | ||||||
|  | 			(con)->lchan->ts->trx->bts->nr, (con)->lchan->ts->trx->nr, (con)->lchan->ts->nr, \ | ||||||
|  | 			(con)->lchan->nr, (con)->use_count); \ | ||||||
|  | 	} while(0); | ||||||
|  |  | ||||||
|  | #define put_subscr_con(con, reason) \ | ||||||
|  | 	do { (con)->use_count--; \ | ||||||
|  | 		DEBUGP(DREF, "lchan (bts=%d,trx=%d,ts=%d,ch=%d) decreases usage to: %d\n", \ | ||||||
|  | 			(con)->lchan->ts->trx->bts->nr, (con)->lchan->ts->trx->nr, (con)->lchan->ts->nr, \ | ||||||
|  | 			(con)->lchan->nr, (con)->use_count); \ | ||||||
|  | 	    if ((con)->use_count <= 0) \ | ||||||
|  | 		_lchan_release((con)->lchan, reason); \ | ||||||
|  | 	} while(0); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| /* Special allocator for C0 of BTS */ | /* Special allocator for C0 of BTS */ | ||||||
| struct gsm_bts_trx_ts *ts_c0_alloc(struct gsm_bts *bts, | struct gsm_bts_trx_ts *ts_c0_alloc(struct gsm_bts *bts, | ||||||
| 				   enum gsm_phys_chan_config pchan); | 				   enum gsm_phys_chan_config pchan); | ||||||
| @@ -46,8 +68,8 @@ struct gsm_lchan *lchan_alloc(struct gsm_bts *bts, enum gsm_chan_t type); | |||||||
| /* Free a logical channel (SDCCH, TCH, ...) */ | /* Free a logical channel (SDCCH, TCH, ...) */ | ||||||
| void lchan_free(struct gsm_lchan *lchan); | void lchan_free(struct gsm_lchan *lchan); | ||||||
|  |  | ||||||
| /* Consider releasing the channel */ | /* internal.. do not use */ | ||||||
| int lchan_auto_release(struct gsm_lchan *lchan); | int _lchan_release(struct gsm_lchan *lchan, u_int8_t release_reason); | ||||||
|  |  | ||||||
| struct load_counter { | struct load_counter { | ||||||
| 	unsigned int total; | 	unsigned int total; | ||||||
|   | |||||||
| @@ -29,6 +29,7 @@ enum { | |||||||
| 	DHO, | 	DHO, | ||||||
| 	DDB, | 	DDB, | ||||||
| 	DREF, | 	DREF, | ||||||
|  | 	DNAT, | ||||||
| 	Debug_LastEntry, | 	Debug_LastEntry, | ||||||
| }; | }; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -16,8 +16,9 @@ struct gsm_trans; | |||||||
| void gsm0408_allow_everyone(int allow); | void gsm0408_allow_everyone(int allow); | ||||||
|  |  | ||||||
| int gsm0408_rcvmsg(struct msgb *msg, u_int8_t link_id); | int gsm0408_rcvmsg(struct msgb *msg, u_int8_t link_id); | ||||||
| enum gsm_chan_t get_ctype_by_chreq(struct gsm_bts *bts, u_int8_t ra, int neci); | enum gsm_chan_t get_ctype_by_chreq(struct gsm_network *bts, u_int8_t ra); | ||||||
| enum gsm_chreq_reason_t get_reason_by_chreq(struct gsm_bts *bts, u_int8_t ra, int neci); | enum gsm_chreq_reason_t get_reason_by_chreq(u_int8_t ra, int neci); | ||||||
|  | void gsm_net_update_ctype(struct gsm_network *net); | ||||||
|  |  | ||||||
| int gsm48_tx_mm_info(struct gsm_lchan *lchan); | int gsm48_tx_mm_info(struct gsm_lchan *lchan); | ||||||
| int gsm48_tx_mm_auth_req(struct gsm_lchan *lchan, u_int8_t *rand, int key_seq); | int gsm48_tx_mm_auth_req(struct gsm_lchan *lchan, u_int8_t *rand, int key_seq); | ||||||
|   | |||||||
| @@ -79,31 +79,13 @@ typedef int gsm_cbfn(unsigned int hooknum, | |||||||
| 		     struct msgb *msg, | 		     struct msgb *msg, | ||||||
| 		     void *data, void *param); | 		     void *data, void *param); | ||||||
|  |  | ||||||
| /* |  | ||||||
|  * Use the channel. As side effect the lchannel recycle timer |  | ||||||
|  * will be started. |  | ||||||
|  */ |  | ||||||
| #define LCHAN_RELEASE_TIMEOUT 20, 0 |  | ||||||
| #define use_subscr_con(con) \ |  | ||||||
| 	do {	(con)->use_count++; \ |  | ||||||
| 		DEBUGP(DREF, "lchan (bts=%d,trx=%d,ts=%d,ch=%d) increases usage to: %d\n", \ |  | ||||||
| 			(con)->lchan->ts->trx->bts->nr, (con)->lchan->ts->trx->nr, (con)->lchan->ts->nr, \ |  | ||||||
| 			(con)->lchan->nr, (con)->use_count); \ |  | ||||||
| 		bsc_schedule_timer(&(con)->release_timer, LCHAN_RELEASE_TIMEOUT); } while(0); |  | ||||||
|  |  | ||||||
| #define put_subscr_con(con) \ |  | ||||||
| 	do { (con)->use_count--; \ |  | ||||||
| 		DEBUGP(DREF, "lchan (bts=%d,trx=%d,ts=%d,ch=%d) decreases usage to: %d\n", \ |  | ||||||
| 			(con)->lchan->ts->trx->bts->nr, (con)->lchan->ts->trx->nr, (con)->lchan->ts->nr, \ |  | ||||||
| 			(con)->lchan->nr, (con)->use_count); \ |  | ||||||
| 	} while(0); |  | ||||||
|  |  | ||||||
|  |  | ||||||
| /* communications link with a BTS */ | /* communications link with a BTS */ | ||||||
| struct gsm_bts_link { | struct gsm_bts_link { | ||||||
| 	struct gsm_bts *bts; | 	struct gsm_bts *bts; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | struct sccp_connection; | ||||||
|  |  | ||||||
| /* Real authentication information containing Ki */ | /* Real authentication information containing Ki */ | ||||||
| enum gsm_auth_algo { | enum gsm_auth_algo { | ||||||
| 	AUTH_ALGO_NONE, | 	AUTH_ALGO_NONE, | ||||||
| @@ -131,6 +113,43 @@ struct gsm_subscriber; | |||||||
| struct gsm_mncc; | struct gsm_mncc; | ||||||
| struct rtp_socket; | struct rtp_socket; | ||||||
|  |  | ||||||
|  | /* BSC/MSC data holding them together */ | ||||||
|  | struct bss_sccp_connection_data { | ||||||
|  | 	struct gsm_lchan *lchan; | ||||||
|  | 	struct gsm_lchan *secondary_lchan; | ||||||
|  | 	struct sccp_connection *sccp; | ||||||
|  | 	int ciphering_handled : 1; | ||||||
|  |  | ||||||
|  |         /* Timers... */ | ||||||
|  |  | ||||||
|  |         /* for assginment command */ | ||||||
|  |         struct timer_list T10; | ||||||
|  |  | ||||||
|  | 	/* for SCCP ... */ | ||||||
|  | 	struct timer_list sccp_it; | ||||||
|  |  | ||||||
|  | 	/* audio handling */ | ||||||
|  | 	int rtp_port; | ||||||
|  |  | ||||||
|  | 	/* Queue SCCP and GSM0408 messages */ | ||||||
|  | 	int block_gsm; | ||||||
|  | 	struct llist_head gsm_queue; | ||||||
|  | 	unsigned int gsm_queue_size; | ||||||
|  |  | ||||||
|  | 	struct llist_head sccp_queue; | ||||||
|  | 	unsigned int sccp_queue_size; | ||||||
|  |  | ||||||
|  | 	/* Active connections */ | ||||||
|  | 	struct llist_head active_connections; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | #define GSM0808_T10_VALUE	6, 0 | ||||||
|  | #define sccp_get_lchan(data_ctx) ((struct bss_sccp_connection_data *)data_ctx)->lchan | ||||||
|  | #define lchan_get_sccp(lchan) lchan->msc_data->sccp | ||||||
|  | struct bss_sccp_connection_data *bss_sccp_create_data(); | ||||||
|  | void bss_sccp_free_data(struct bss_sccp_connection_data *); | ||||||
|  |  | ||||||
|  |  | ||||||
| /* Network Management State */ | /* Network Management State */ | ||||||
| struct gsm_nm_state { | struct gsm_nm_state { | ||||||
| 	u_int8_t operational; | 	u_int8_t operational; | ||||||
| @@ -179,6 +198,7 @@ enum gsm_lchan_state { | |||||||
| 	LCHAN_S_ACT_REQ,	/* channel activatin requested */ | 	LCHAN_S_ACT_REQ,	/* channel activatin requested */ | ||||||
| 	LCHAN_S_ACTIVE,		/* channel is active and operational */ | 	LCHAN_S_ACTIVE,		/* channel is active and operational */ | ||||||
| 	LCHAN_S_REL_REQ,	/* channel release has been requested */ | 	LCHAN_S_REL_REQ,	/* channel release has been requested */ | ||||||
|  | 	LCHAN_S_REL_ERR,	/* channel is in an error state */ | ||||||
| 	LCHAN_S_INACTIVE,	/* channel is set inactive */ | 	LCHAN_S_INACTIVE,	/* channel is set inactive */ | ||||||
| }; | }; | ||||||
|  |  | ||||||
| @@ -187,9 +207,6 @@ struct gsm_subscriber_connection { | |||||||
| 	/* To whom we are allocated at the moment */ | 	/* To whom we are allocated at the moment */ | ||||||
| 	struct gsm_subscriber *subscr; | 	struct gsm_subscriber *subscr; | ||||||
|  |  | ||||||
| 	/* Timer started to release the channel */ |  | ||||||
| 	struct timer_list release_timer; |  | ||||||
|  |  | ||||||
| 	/* | 	/* | ||||||
| 	 * Operations that have a state and might be pending | 	 * Operations that have a state and might be pending | ||||||
| 	 */ | 	 */ | ||||||
| @@ -230,6 +247,8 @@ struct gsm_lchan { | |||||||
| 	} encr; | 	} encr; | ||||||
|  |  | ||||||
| 	struct timer_list T3101; | 	struct timer_list T3101; | ||||||
|  | 	struct timer_list T3111; | ||||||
|  | 	struct timer_list error_timer; | ||||||
|  |  | ||||||
| 	/* AMR bits */ | 	/* AMR bits */ | ||||||
| 	struct gsm48_multi_rate_conf mr_conf; | 	struct gsm48_multi_rate_conf mr_conf; | ||||||
| @@ -237,6 +256,12 @@ struct gsm_lchan { | |||||||
| 	/* Established data link layer services */ | 	/* Established data link layer services */ | ||||||
| 	u_int8_t sapis[8]; | 	u_int8_t sapis[8]; | ||||||
|  |  | ||||||
|  | 	/* | ||||||
|  | 	 * MSC handling... | ||||||
|  | 	 */ | ||||||
|  | 	struct bss_sccp_connection_data *msc_data; | ||||||
|  |  | ||||||
|  |  | ||||||
| 	/* cache of last measurement reports on this lchan */ | 	/* cache of last measurement reports on this lchan */ | ||||||
| 	struct gsm_meas_rep meas_rep[6]; | 	struct gsm_meas_rep meas_rep[6]; | ||||||
| 	int meas_rep_idx; | 	int meas_rep_idx; | ||||||
| @@ -256,6 +281,9 @@ struct gsm_lchan { | |||||||
| 	} abis_ip; | 	} abis_ip; | ||||||
|  |  | ||||||
| 	struct gsm_subscriber_connection conn; | 	struct gsm_subscriber_connection conn; | ||||||
|  |  | ||||||
|  | 	/* release reason */ | ||||||
|  | 	u_int8_t release_reason; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| struct gsm_e1_subslot { | struct gsm_e1_subslot { | ||||||
| @@ -546,6 +574,14 @@ enum gsm_auth_policy { | |||||||
| #define GSM_T3101_DEFAULT 10 | #define GSM_T3101_DEFAULT 10 | ||||||
| #define GSM_T3113_DEFAULT 60 | #define GSM_T3113_DEFAULT 60 | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * internal data for audio management | ||||||
|  |  */ | ||||||
|  | struct gsm_audio_support { | ||||||
|  | 	u_int8_t hr  : 1, | ||||||
|  | 		 ver : 7; | ||||||
|  | }; | ||||||
|  |  | ||||||
| struct gsm_network { | struct gsm_network { | ||||||
| 	/* global parameters */ | 	/* global parameters */ | ||||||
| 	u_int16_t country_code; | 	u_int16_t country_code; | ||||||
| @@ -576,6 +612,11 @@ struct gsm_network { | |||||||
|  |  | ||||||
| 	struct gsmnet_stats stats; | 	struct gsmnet_stats stats; | ||||||
|  |  | ||||||
|  | 	struct gsm_audio_support **audio_support; | ||||||
|  | 	int audio_length; | ||||||
|  | 	int rtp_payload; | ||||||
|  | 	int rtp_base_port; | ||||||
|  |  | ||||||
| 	/* layer 4 */ | 	/* layer 4 */ | ||||||
| 	int (*mncc_recv) (struct gsm_network *net, int msg_type, void *arg); | 	int (*mncc_recv) (struct gsm_network *net, int msg_type, void *arg); | ||||||
| 	struct llist_head upqueue; | 	struct llist_head upqueue; | ||||||
| @@ -601,6 +642,18 @@ struct gsm_network { | |||||||
| 	struct { | 	struct { | ||||||
| 		enum rrlp_mode mode; | 		enum rrlp_mode mode; | ||||||
| 	} rrlp; | 	} rrlp; | ||||||
|  |  | ||||||
|  | 	enum gsm_chan_t ctype_by_chreq[16]; | ||||||
|  |  | ||||||
|  | 	/* Use a TCH for handling requests of type paging any */ | ||||||
|  | 	int pag_any_tch; | ||||||
|  |  | ||||||
|  | 	/* a hack for On Waves. It must be signed */ | ||||||
|  | 	int32_t core_country_code; | ||||||
|  | 	int32_t core_network_code; | ||||||
|  |  | ||||||
|  | 	/* a simple token for this network... */ | ||||||
|  | 	char *bsc_token; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| #define SMS_HDR_SIZE	128 | #define SMS_HDR_SIZE	128 | ||||||
|   | |||||||
| @@ -81,6 +81,8 @@ struct gsm_subscriber *subscr_get_by_extension(struct gsm_network *net, | |||||||
| 					       const char *ext); | 					       const char *ext); | ||||||
| struct gsm_subscriber *subscr_get_by_id(struct gsm_network *net, | struct gsm_subscriber *subscr_get_by_id(struct gsm_network *net, | ||||||
| 					unsigned long long id); | 					unsigned long long id); | ||||||
|  | struct gsm_subscriber *subscr_get_or_create(struct gsm_network *net, | ||||||
|  | 					    const char *imsi); | ||||||
| int subscr_update(struct gsm_subscriber *s, struct gsm_bts *bts, int reason); | int subscr_update(struct gsm_subscriber *s, struct gsm_bts *bts, int reason); | ||||||
| void subscr_put_channel(struct gsm_lchan *lchan); | void subscr_put_channel(struct gsm_lchan *lchan); | ||||||
| void subscr_get_channel(struct gsm_subscriber *subscr, | void subscr_get_channel(struct gsm_subscriber *subscr, | ||||||
|   | |||||||
| @@ -29,7 +29,6 @@ | |||||||
| #include <arpa/inet.h> | #include <arpa/inet.h> | ||||||
|  |  | ||||||
| #define RTP_PORT_DEFAULT 4000 | #define RTP_PORT_DEFAULT 4000 | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Calculate the RTP audio port for the given multiplex |  * Calculate the RTP audio port for the given multiplex | ||||||
|  * and the direction. This allows a semi static endpoint |  * and the direction. This allows a semi static endpoint | ||||||
| @@ -77,14 +76,17 @@ struct mgcp_config; | |||||||
|  |  | ||||||
| typedef int (*mgcp_change)(struct mgcp_config *cfg, int endpoint, int state, int local_rtp); | typedef int (*mgcp_change)(struct mgcp_config *cfg, int endpoint, int state, int local_rtp); | ||||||
| typedef int (*mgcp_policy)(struct mgcp_config *cfg, int endpoint, int state, const char *transactio_id); | typedef int (*mgcp_policy)(struct mgcp_config *cfg, int endpoint, int state, const char *transactio_id); | ||||||
|  | typedef int (*mgcp_reset)(struct mgcp_config *cfg); | ||||||
|  |  | ||||||
| struct mgcp_config { | struct mgcp_config { | ||||||
|  | 	/* common configuration */ | ||||||
| 	int source_port; | 	int source_port; | ||||||
| 	char *local_ip; | 	char *local_ip; | ||||||
| 	char *source_addr; | 	char *source_addr; | ||||||
| 	unsigned int number_endpoints; |  | ||||||
| 	char *bts_ip; | 	char *bts_ip; | ||||||
|  | 	char *call_agent_addr; | ||||||
|  |  | ||||||
|  | 	/* default endpoint data */ | ||||||
| 	struct in_addr bts_in; | 	struct in_addr bts_in; | ||||||
| 	char *audio_name; | 	char *audio_name; | ||||||
| 	int audio_payload; | 	int audio_payload; | ||||||
| @@ -92,15 +94,24 @@ struct mgcp_config { | |||||||
| 	int early_bind; | 	int early_bind; | ||||||
| 	int rtp_base_port; | 	int rtp_base_port; | ||||||
|  |  | ||||||
|  | 	/* only used in forward mode */ | ||||||
| 	char *forward_ip; | 	char *forward_ip; | ||||||
| 	int forward_port; | 	int forward_port; | ||||||
|  |  | ||||||
|  | 	unsigned int last_call_id; | ||||||
|  |  | ||||||
|  | 	/* endpoint configuration */ | ||||||
|  | 	unsigned int number_endpoints; | ||||||
|  | 	struct mgcp_endpoint *endpoints; | ||||||
|  |  | ||||||
|  | 	/* spec handling */ | ||||||
|  | 	int force_realloc; | ||||||
|  |  | ||||||
|  | 	/* callback functionality */ | ||||||
| 	mgcp_change change_cb; | 	mgcp_change change_cb; | ||||||
| 	mgcp_policy policy_cb; | 	mgcp_policy policy_cb; | ||||||
|  | 	mgcp_reset reset_cb; | ||||||
| 	void *data; | 	void *data; | ||||||
|  |  | ||||||
| 	struct mgcp_endpoint *endpoints; |  | ||||||
| 	unsigned int last_call_id; |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
| /* config management */ | /* config management */ | ||||||
| @@ -115,8 +126,15 @@ void mgcp_free_endp(struct mgcp_endpoint *endp); | |||||||
|  * format helper functions |  * format helper functions | ||||||
|  */ |  */ | ||||||
| struct msgb *mgcp_handle_message(struct mgcp_config *cfg, struct msgb *msg); | struct msgb *mgcp_handle_message(struct mgcp_config *cfg, struct msgb *msg); | ||||||
| struct msgb *mgcp_create_rsip(void); |  | ||||||
| struct msgb *mgcp_create_response_with_data(int code, const char *msg, const char *trans, const char *data); | struct msgb *mgcp_create_response_with_data(int code, const char *msg, const char *trans, const char *data); | ||||||
|  |  | ||||||
|  | /* adc helper */ | ||||||
|  | static inline int mgcp_timeslot_to_endpoint(int multiplex, int timeslot) | ||||||
|  | { | ||||||
|  | 	if (timeslot == 0) | ||||||
|  | 		timeslot = 1; | ||||||
|  | 	return timeslot + (31 * multiplex); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
| #endif | #endif | ||||||
|   | |||||||
| @@ -57,8 +57,21 @@ struct mgcp_endpoint { | |||||||
|  |  | ||||||
| 	/* backpointer */ | 	/* backpointer */ | ||||||
| 	struct mgcp_config *cfg; | 	struct mgcp_config *cfg; | ||||||
|  |  | ||||||
|  | 	/* statistics */ | ||||||
|  | 	unsigned int in_bts; | ||||||
|  | 	unsigned int in_remote; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| #define ENDPOINT_NUMBER(endp) abs(endp - endp->cfg->endpoints) | #define ENDPOINT_NUMBER(endp) abs(endp - endp->cfg->endpoints) | ||||||
|  |  | ||||||
|  | struct mgcp_msg_ptr { | ||||||
|  | 	unsigned int start; | ||||||
|  | 	unsigned int length; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | int mgcp_analyze_header(struct mgcp_config *cfg, struct msgb *msg, | ||||||
|  | 			struct mgcp_msg_ptr *ptr, int size, | ||||||
|  | 			const char **transaction_id, struct mgcp_endpoint **endp); | ||||||
|  |  | ||||||
| #endif | #endif | ||||||
|   | |||||||
| @@ -94,7 +94,7 @@ struct sccp_connection { | |||||||
|  *   call sccp_system_incoming for incoming data (from the network) |  *   call sccp_system_incoming for incoming data (from the network) | ||||||
|  *   sccp will call outgoing whenever outgoing data exists |  *   sccp will call outgoing whenever outgoing data exists | ||||||
|  */ |  */ | ||||||
| int sccp_system_init(int (*outgoing)(struct msgb *data, void *ctx), void *context); | int sccp_system_init(void (*outgoing)(struct msgb *data, void *ctx), void *context); | ||||||
| int sccp_system_incoming(struct msgb *data); | int sccp_system_incoming(struct msgb *data); | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -105,6 +105,11 @@ int sccp_connection_send_it(struct sccp_connection *connection); | |||||||
| int sccp_connection_close(struct sccp_connection *connection, int cause); | int sccp_connection_close(struct sccp_connection *connection, int cause); | ||||||
| int sccp_connection_free(struct sccp_connection *connection); | int sccp_connection_free(struct sccp_connection *connection); | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * internal..  | ||||||
|  |  */ | ||||||
|  | int sccp_connection_force_free(struct sccp_connection *conn); | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Create a new socket. Set your callbacks and then call bind to open |  * Create a new socket. Set your callbacks and then call bind to open | ||||||
|  * the connection. |  * the connection. | ||||||
|   | |||||||
| @@ -107,6 +107,8 @@ enum node_type { | |||||||
| 	TS_NODE, | 	TS_NODE, | ||||||
| 	SUBSCR_NODE, | 	SUBSCR_NODE, | ||||||
| 	MGCP_NODE, | 	MGCP_NODE, | ||||||
|  | 	NAT_NODE, | ||||||
|  | 	BSC_NODE, | ||||||
| }; | }; | ||||||
|  |  | ||||||
| /* Node which has some commands and prompt string and configuration | /* Node which has some commands and prompt string and configuration | ||||||
|   | |||||||
| @@ -3,7 +3,8 @@ AM_CFLAGS=-Wall $(LIBOSMOCORE_CFLAGS) | |||||||
| AM_LDFLAGS = $(LIBOSMOCORE_LIBS) | AM_LDFLAGS = $(LIBOSMOCORE_LIBS) | ||||||
|  |  | ||||||
| sbin_PROGRAMS = bsc_hack bs11_config ipaccess-find ipaccess-config \ | sbin_PROGRAMS = bsc_hack bs11_config ipaccess-find ipaccess-config \ | ||||||
|                 isdnsync bsc_mgcp ipaccess-proxy |                 isdnsync bsc_mgcp ipaccess-proxy \ | ||||||
|  |                 bsc_msc_ip bsc_nat | ||||||
| noinst_LIBRARIES = libbsc.a libmsc.a libvty.a | noinst_LIBRARIES = libbsc.a libmsc.a libvty.a | ||||||
| noinst_HEADERS = vty/cardshell.h | noinst_HEADERS = vty/cardshell.h | ||||||
|  |  | ||||||
| @@ -17,12 +18,12 @@ libbsc_a_SOURCES = abis_rsl.c abis_nm.c gsm_data.c gsm_04_08_utils.c \ | |||||||
| 		input/misdn.c input/ipaccess.c \ | 		input/misdn.c input/ipaccess.c \ | ||||||
| 		talloc_ctx.c system_information.c rest_octets.c \ | 		talloc_ctx.c system_information.c rest_octets.c \ | ||||||
| 		rtp_proxy.c bts_siemens_bs11.c bts_ipaccess_nanobts.c \ | 		rtp_proxy.c bts_siemens_bs11.c bts_ipaccess_nanobts.c \ | ||||||
| 		bts_unknown.c bsc_version.c bsc_api.c vty_interface_cmds.c | 		bts_unknown.c meas_rep.c telnet_interface.c bsc_version.c bsc_api.c vty_interface_cmds.c  | ||||||
|  |  | ||||||
| libmsc_a_SOURCES = gsm_subscriber.c db.c telnet_interface.c \ | libmsc_a_SOURCES = gsm_subscriber.c db.c \ | ||||||
| 		mncc.c gsm_04_08.c gsm_04_11.c transaction.c \ | 		mncc.c gsm_04_08.c gsm_04_11.c transaction.c \ | ||||||
| 		token_auth.c rrlp.c gsm_04_80.c ussd.c silent_call.c \ | 		token_auth.c rrlp.c gsm_04_80.c ussd.c silent_call.c \ | ||||||
| 		handover_logic.c handover_decision.c meas_rep.c | 		handover_logic.c handover_decision.c | ||||||
|  |  | ||||||
| libvty_a_SOURCES = vty/buffer.c vty/command.c vty/vector.c vty/vty.c | libvty_a_SOURCES = vty/buffer.c vty/command.c vty/vector.c vty/vty.c | ||||||
|  |  | ||||||
| @@ -33,6 +34,10 @@ bsc_hack_LDADD = libmsc.a libbsc.a libmsc.a libvty.a -ldl -ldbi $(LIBCRYPT) | |||||||
|  |  | ||||||
| bs11_config_SOURCES = bs11_config.c abis_nm.c gsm_data.c debug.c \ | bs11_config_SOURCES = bs11_config.c abis_nm.c gsm_data.c debug.c \ | ||||||
| 		      rs232.c bts_siemens_bs11.c | 		      rs232.c bts_siemens_bs11.c | ||||||
|  | bsc_msc_ip_SOURCES = bssap.c bsc_msc_ip.c bsc_init.c vty_interface.c vty_interface_bsc.c \ | ||||||
|  | 		bsc_msc.c | ||||||
|  | bsc_msc_ip_LDADD = libbsc.a libvty.a libsccp.a | ||||||
|  |  | ||||||
|  |  | ||||||
| ipaccess_find_SOURCES = ipaccess/ipaccess-find.c | ipaccess_find_SOURCES = ipaccess/ipaccess-find.c | ||||||
|  |  | ||||||
| @@ -42,7 +47,13 @@ ipaccess_config_LDADD = libbsc.a libmsc.a libbsc.a libvty.a -ldl -ldbi $(LIBCRYP | |||||||
| isdnsync_SOURCES = isdnsync.c | isdnsync_SOURCES = isdnsync.c | ||||||
|  |  | ||||||
| bsc_mgcp_SOURCES = mgcp/mgcp_main.c mgcp/mgcp_protocol.c mgcp/mgcp_network.c mgcp/mgcp_vty.c \ | bsc_mgcp_SOURCES = mgcp/mgcp_main.c mgcp/mgcp_protocol.c mgcp/mgcp_network.c mgcp/mgcp_vty.c \ | ||||||
| 		   debug.c telnet_interface.c | 		   debug.c telnet_interface.c vty_interface_cmds.c | ||||||
| bsc_mgcp_LDADD = libvty.a | bsc_mgcp_LDADD = libvty.a | ||||||
|  |  | ||||||
| ipaccess_proxy_SOURCES = ipaccess/ipaccess-proxy.c debug.c | ipaccess_proxy_SOURCES = ipaccess/ipaccess-proxy.c debug.c | ||||||
|  |  | ||||||
|  | bsc_nat_SOURCES = nat/bsc_nat.c nat/bsc_filter.c nat/bsc_sccp.c \ | ||||||
|  | 		  nat/bsc_nat_utils.c nat/bsc_nat_vty.c nat/bsc_mgcp_utils.c \ | ||||||
|  | 		  mgcp/mgcp_protocol.c mgcp/mgcp_network.c mgcp/mgcp_vty.c \ | ||||||
|  | 		  bsc_msc.c bssap.c | ||||||
|  | bsc_nat_LDADD = libvty.a libbsc.a libsccp.a | ||||||
|   | |||||||
| @@ -568,12 +568,33 @@ int rsl_deact_sacch(struct gsm_lchan *lchan) | |||||||
| 	return abis_rsl_sendmsg(msg); | 	return abis_rsl_sendmsg(msg); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | static void error_timeout_cb(void *data) | ||||||
|  | { | ||||||
|  | 	struct gsm_lchan *lchan = data; | ||||||
|  | 	if (lchan->state != LCHAN_S_REL_ERR) { | ||||||
|  | 		LOGP(DRSL, LOGL_ERROR, "%s error timeout but not in error state: %d\n", | ||||||
|  | 		     gsm_lchan_name(lchan), lchan->state); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* go back to the none state */ | ||||||
|  | 	LOGP(DRSL, LOGL_NOTICE, "%s is back in operation.\n", gsm_lchan_name(lchan)); | ||||||
|  | 	rsl_lchan_set_state(lchan, LCHAN_S_NONE); | ||||||
|  | } | ||||||
|  |  | ||||||
| /* Chapter 8.4.14 / 4.7: Tell BTS to release the radio channel */ | /* Chapter 8.4.14 / 4.7: Tell BTS to release the radio channel */ | ||||||
| int rsl_rf_chan_release(struct gsm_lchan *lchan) | static int rsl_rf_chan_release(struct gsm_lchan *lchan, int error) | ||||||
| { | { | ||||||
| 	struct abis_rsl_dchan_hdr *dh; | 	struct abis_rsl_dchan_hdr *dh; | ||||||
| 	struct msgb *msg = rsl_msgb_alloc(); | 	struct msgb *msg; | ||||||
|  |  | ||||||
|  | 	if (lchan->state == LCHAN_S_REL_ERR) { | ||||||
|  | 		LOGP(DRSL, LOGL_NOTICE, "%s is in error state not sending release.\n", | ||||||
|  | 		     gsm_lchan_name(lchan)); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	msg = rsl_msgb_alloc(); | ||||||
| 	dh = (struct abis_rsl_dchan_hdr *) msgb_put(msg, sizeof(*dh)); | 	dh = (struct abis_rsl_dchan_hdr *) msgb_put(msg, sizeof(*dh)); | ||||||
| 	init_dchan_hdr(dh, RSL_MT_RF_CHAN_REL); | 	init_dchan_hdr(dh, RSL_MT_RF_CHAN_REL); | ||||||
| 	dh->chan_nr = lchan2chan_nr(lchan); | 	dh->chan_nr = lchan2chan_nr(lchan); | ||||||
| @@ -581,7 +602,20 @@ int rsl_rf_chan_release(struct gsm_lchan *lchan) | |||||||
| 	msg->lchan = lchan; | 	msg->lchan = lchan; | ||||||
| 	msg->trx = lchan->ts->trx; | 	msg->trx = lchan->ts->trx; | ||||||
|  |  | ||||||
| 	DEBUGP(DRSL, "%s RF Channel Release CMD\n", gsm_lchan_name(lchan)); | 	DEBUGP(DRSL, "%s RF Channel Release CMD due error %d\n", gsm_lchan_name(lchan), error); | ||||||
|  |  | ||||||
|  | 	if (error) { | ||||||
|  | 		/* | ||||||
|  | 		 * the nanoBTS sends RLL release indications after the channel release. This can | ||||||
|  | 		 * be a problem when we have reassigned the channel to someone else and then can | ||||||
|  | 		 * not figure out who used this channel. | ||||||
|  | 		 */ | ||||||
|  | 		rsl_lchan_set_state(lchan, LCHAN_S_REL_ERR); | ||||||
|  | 		lchan->error_timer.data = lchan; | ||||||
|  | 		lchan->error_timer.cb = error_timeout_cb; | ||||||
|  | 		bsc_schedule_timer(&lchan->error_timer, | ||||||
|  | 				   msg->trx->bts->network->T3111 + 2, 0); | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	/* BTS will respond by RF CHAN REL ACK */ | 	/* BTS will respond by RF CHAN REL ACK */ | ||||||
| 	return abis_rsl_sendmsg(msg); | 	return abis_rsl_sendmsg(msg); | ||||||
| @@ -718,16 +752,16 @@ int rsl_establish_request(struct gsm_lchan *lchan, u_int8_t link_id) | |||||||
|    RELEASE CONFIRM, which we in turn use to trigger RSL CHANNEL RELEASE, |    RELEASE CONFIRM, which we in turn use to trigger RSL CHANNEL RELEASE, | ||||||
|    which in turn is acknowledged by RSL CHANNEL RELEASE ACK, which calls |    which in turn is acknowledged by RSL CHANNEL RELEASE ACK, which calls | ||||||
|    lchan_free() */ |    lchan_free() */ | ||||||
| int rsl_release_request(struct gsm_lchan *lchan, u_int8_t link_id) | int rsl_release_request(struct gsm_lchan *lchan, u_int8_t link_id, u_int8_t reason) | ||||||
| { | { | ||||||
|  |  | ||||||
| 	struct msgb *msg; | 	struct msgb *msg; | ||||||
|  |  | ||||||
| 	msg = rsl_rll_simple(RSL_MT_REL_REQ, lchan2chan_nr(lchan), | 	msg = rsl_rll_simple(RSL_MT_REL_REQ, lchan2chan_nr(lchan), | ||||||
| 			     link_id, 0); | 			     link_id, 0); | ||||||
| 	msgb_tv_put(msg, RSL_IE_RELEASE_MODE, 0);	/* normal release */ |         /* 0 is normal release, 1 is local end */ | ||||||
|  | 	msgb_tv_put(msg, RSL_IE_RELEASE_MODE, reason); | ||||||
|  |  | ||||||
| 	lchan->state = LCHAN_S_REL_REQ; |  | ||||||
| 	/* FIXME: start some timer in case we don't receive a REL ACK ? */ | 	/* FIXME: start some timer in case we don't receive a REL ACK ? */ | ||||||
|  |  | ||||||
| 	msg->trx = lchan->ts->trx; | 	msg->trx = lchan->ts->trx; | ||||||
| @@ -735,6 +769,12 @@ int rsl_release_request(struct gsm_lchan *lchan, u_int8_t link_id) | |||||||
| 	return abis_rsl_sendmsg(msg); | 	return abis_rsl_sendmsg(msg); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | int rsl_lchan_set_state(struct gsm_lchan *lchan, int state) | ||||||
|  | { | ||||||
|  | 	lchan->state = state; | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
| /* Chapter 8.4.2: Channel Activate Acknowledge */ | /* Chapter 8.4.2: Channel Activate Acknowledge */ | ||||||
| static int rsl_rx_chan_act_ack(struct msgb *msg) | static int rsl_rx_chan_act_ack(struct msgb *msg) | ||||||
| { | { | ||||||
| @@ -749,7 +789,7 @@ static int rsl_rx_chan_act_ack(struct msgb *msg) | |||||||
| 		LOGP(DRSL, LOGL_NOTICE, "%s CHAN ACT ACK, but state %s\n", | 		LOGP(DRSL, LOGL_NOTICE, "%s CHAN ACT ACK, but state %s\n", | ||||||
| 			gsm_lchan_name(msg->lchan), | 			gsm_lchan_name(msg->lchan), | ||||||
| 			gsm_lchans_name(msg->lchan->state)); | 			gsm_lchans_name(msg->lchan->state)); | ||||||
| 	msg->lchan->state = LCHAN_S_ACTIVE; | 	rsl_lchan_set_state(msg->lchan, LCHAN_S_ACTIVE); | ||||||
|  |  | ||||||
| 	dispatch_signal(SS_LCHAN, S_LCHAN_ACTIVATE_ACK, msg->lchan); | 	dispatch_signal(SS_LCHAN, S_LCHAN_ACTIVATE_ACK, msg->lchan); | ||||||
|  |  | ||||||
| @@ -775,9 +815,9 @@ static int rsl_rx_chan_act_nack(struct msgb *msg) | |||||||
| 		print_rsl_cause(LOGL_ERROR, cause, | 		print_rsl_cause(LOGL_ERROR, cause, | ||||||
| 				TLVP_LEN(&tp, RSL_IE_CAUSE)); | 				TLVP_LEN(&tp, RSL_IE_CAUSE)); | ||||||
| 		if (*cause != RSL_ERR_RCH_ALR_ACTV_ALLOC) | 		if (*cause != RSL_ERR_RCH_ALR_ACTV_ALLOC) | ||||||
| 			msg->lchan->state = LCHAN_S_NONE; | 			rsl_lchan_set_state(msg->lchan, LCHAN_S_NONE); | ||||||
| 	} else | 	} else | ||||||
| 		msg->lchan->state = LCHAN_S_NONE; | 		rsl_lchan_set_state(msg->lchan, LCHAN_S_NONE); | ||||||
|  |  | ||||||
| 	LOGPC(DRSL, LOGL_ERROR, "\n"); | 	LOGPC(DRSL, LOGL_ERROR, "\n"); | ||||||
|  |  | ||||||
| @@ -805,7 +845,7 @@ static int rsl_rx_conn_fail(struct msgb *msg) | |||||||
|  |  | ||||||
| 	LOGPC(DRSL, LOGL_NOTICE, "\n"); | 	LOGPC(DRSL, LOGL_NOTICE, "\n"); | ||||||
| 	/* FIXME: only free it after channel release ACK */ | 	/* FIXME: only free it after channel release ACK */ | ||||||
| 	return rsl_rf_chan_release(msg->lchan); | 	return rsl_rf_chan_release(msg->lchan, 1); | ||||||
| } | } | ||||||
|  |  | ||||||
| static void print_meas_rep_uni(struct gsm_meas_rep_unidir *mru, | static void print_meas_rep_uni(struct gsm_meas_rep_unidir *mru, | ||||||
| @@ -977,11 +1017,14 @@ static int abis_rsl_rx_dchan(struct msgb *msg) | |||||||
| 		break; | 		break; | ||||||
| 	case RSL_MT_RF_CHAN_REL_ACK: | 	case RSL_MT_RF_CHAN_REL_ACK: | ||||||
| 		DEBUGP(DRSL, "%s RF CHANNEL RELEASE ACK\n", ts_name); | 		DEBUGP(DRSL, "%s RF CHANNEL RELEASE ACK\n", ts_name); | ||||||
| 		if (msg->lchan->state != LCHAN_S_REL_REQ) | 		if (msg->lchan->state != LCHAN_S_REL_REQ && msg->lchan->state != LCHAN_S_REL_ERR) | ||||||
| 			LOGP(DRSL, LOGL_NOTICE, "%s CHAN REL ACK but state %s\n", | 			LOGP(DRSL, LOGL_NOTICE, "%s CHAN REL ACK but state %s\n", | ||||||
| 				gsm_lchan_name(msg->lchan), | 				gsm_lchan_name(msg->lchan), | ||||||
| 				gsm_lchans_name(msg->lchan->state)); | 				gsm_lchans_name(msg->lchan->state)); | ||||||
| 		msg->lchan->state = LCHAN_S_NONE; | 		bsc_del_timer(&msg->lchan->T3111); | ||||||
|  | 		/* we have an error timer pending to release that */ | ||||||
|  | 		if (msg->lchan->state != LCHAN_S_REL_ERR) | ||||||
|  | 			rsl_lchan_set_state(msg->lchan, LCHAN_S_NONE); | ||||||
| 		lchan_free(msg->lchan); | 		lchan_free(msg->lchan); | ||||||
| 		break; | 		break; | ||||||
| 	case RSL_MT_MODE_MODIFY_ACK: | 	case RSL_MT_MODE_MODIFY_ACK: | ||||||
| @@ -1073,7 +1116,15 @@ static void t3101_expired(void *data) | |||||||
| { | { | ||||||
| 	struct gsm_lchan *lchan = data; | 	struct gsm_lchan *lchan = data; | ||||||
|  |  | ||||||
| 	rsl_rf_chan_release(lchan); | 	rsl_rf_chan_release(lchan, 1); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* If T3111 expires, we will send the RF Channel Request */ | ||||||
|  | static void t3111_expired(void *data) | ||||||
|  | { | ||||||
|  | 	struct gsm_lchan *lchan = data; | ||||||
|  |  | ||||||
|  | 	rsl_rf_chan_release(lchan, 0); | ||||||
| } | } | ||||||
|  |  | ||||||
| /* MS has requested a channel on the RACH */ | /* MS has requested a channel on the RACH */ | ||||||
| @@ -1105,8 +1156,8 @@ static int rsl_rx_chan_rqd(struct msgb *msg) | |||||||
|  |  | ||||||
| 	/* determine channel type (SDCCH/TCH_F/TCH_H) based on | 	/* determine channel type (SDCCH/TCH_F/TCH_H) based on | ||||||
| 	 * request reference RA */ | 	 * request reference RA */ | ||||||
| 	lctype = get_ctype_by_chreq(bts, rqd_ref->ra, bts->network->neci); | 	lctype = get_ctype_by_chreq(bts->network, rqd_ref->ra); | ||||||
| 	chreq_reason = get_reason_by_chreq(bts, rqd_ref->ra, bts->network->neci); | 	chreq_reason = get_reason_by_chreq(rqd_ref->ra, bts->network->neci); | ||||||
|  |  | ||||||
| 	counter_inc(bts->network->stats.chreq.total); | 	counter_inc(bts->network->stats.chreq.total); | ||||||
|  |  | ||||||
| @@ -1124,7 +1175,7 @@ static int rsl_rx_chan_rqd(struct msgb *msg) | |||||||
| 		LOGP(DRSL, LOGL_NOTICE, "%s lchan_alloc() returned channel " | 		LOGP(DRSL, LOGL_NOTICE, "%s lchan_alloc() returned channel " | ||||||
| 		     "in state %s\n", gsm_lchan_name(lchan), | 		     "in state %s\n", gsm_lchan_name(lchan), | ||||||
| 		     gsm_lchans_name(lchan->state)); | 		     gsm_lchans_name(lchan->state)); | ||||||
| 	lchan->state = LCHAN_S_ACT_REQ; | 	rsl_lchan_set_state(lchan, LCHAN_S_ACT_REQ); | ||||||
|  |  | ||||||
| 	ts_number = lchan->ts->nr; | 	ts_number = lchan->ts->nr; | ||||||
| 	arfcn = lchan->ts->trx->arfcn; | 	arfcn = lchan->ts->trx->arfcn; | ||||||
| @@ -1241,11 +1292,37 @@ static int rsl_rx_rll_err_ind(struct msgb *msg) | |||||||
| 	rll_indication(msg->lchan, rllh->link_id, BSC_RLLR_IND_ERR_IND); | 	rll_indication(msg->lchan, rllh->link_id, BSC_RLLR_IND_ERR_IND); | ||||||
|  |  | ||||||
| 	if (rlm_cause[1] == RLL_CAUSE_T200_EXPIRED) | 	if (rlm_cause[1] == RLL_CAUSE_T200_EXPIRED) | ||||||
| 		return rsl_rf_chan_release(msg->lchan); | 		return rsl_rf_chan_release(msg->lchan, 1); | ||||||
|  |  | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | static void rsl_handle_release(struct gsm_lchan *lchan) | ||||||
|  | { | ||||||
|  | 	int sapi; | ||||||
|  | 	struct gsm_bts *bts; | ||||||
|  |  | ||||||
|  | 	/* maybe we have only brought down one RLL */ | ||||||
|  | 	if (lchan->state != LCHAN_S_REL_REQ) | ||||||
|  | 		return; | ||||||
|  |  | ||||||
|  | 	for (sapi = 0; sapi < ARRAY_SIZE(lchan->sapis); ++sapi) { | ||||||
|  | 		if (lchan->sapis[sapi] == LCHAN_SAPI_UNUSED) | ||||||
|  | 			continue; | ||||||
|  | 		LOGP(DRSL, LOGL_NOTICE, "%s waiting for SAPI=%d to be released.\n", | ||||||
|  | 		     gsm_lchan_name(lchan), sapi); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 			 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	/* wait a bit to send the RF Channel Release */ | ||||||
|  | 	lchan->T3111.cb = t3111_expired; | ||||||
|  | 	lchan->T3111.data = lchan; | ||||||
|  | 	bts = lchan->ts->trx->bts; | ||||||
|  | 	bsc_schedule_timer(&lchan->T3111, bts->network->T3111, 0); | ||||||
|  | } | ||||||
|  |  | ||||||
| /*	ESTABLISH INDICATION, LOCATION AREA UPDATE REQUEST | /*	ESTABLISH INDICATION, LOCATION AREA UPDATE REQUEST | ||||||
| 	0x02, 0x06, | 	0x02, 0x06, | ||||||
| 	0x01, 0x20, | 	0x01, 0x20, | ||||||
| @@ -1297,20 +1374,16 @@ static int abis_rsl_rx_rll(struct msgb *msg) | |||||||
| 		msg->lchan->sapis[rllh->link_id & 0x7] = LCHAN_SAPI_UNUSED; | 		msg->lchan->sapis[rllh->link_id & 0x7] = LCHAN_SAPI_UNUSED; | ||||||
| 		rll_indication(msg->lchan, rllh->link_id, | 		rll_indication(msg->lchan, rllh->link_id, | ||||||
| 				  BSC_RLLR_IND_REL_IND); | 				  BSC_RLLR_IND_REL_IND); | ||||||
| 		/* we can now releae the channel on the BTS/Abis side */ | 		rsl_handle_release(msg->lchan); | ||||||
| 		/* FIXME: officially we need to start T3111 and wait for | 		rsl_lchan_rll_release(msg->lchan, rllh->link_id); | ||||||
| 		 * some grace period */ |  | ||||||
| 		rsl_rf_chan_release(msg->lchan); |  | ||||||
| 		break; | 		break; | ||||||
| 	case RSL_MT_REL_CONF: | 	case RSL_MT_REL_CONF: | ||||||
| 		/* BTS informs us of having received UA from MS, | 		/* BTS informs us of having received UA from MS, | ||||||
| 		 * in response to DISC that we've sent earlier */ | 		 * in response to DISC that we've sent earlier */ | ||||||
| 		DEBUGPC(DRLL, "RELEASE CONFIRMATION\n"); | 		DEBUGPC(DRLL, "RELEASE CONFIRMATION\n"); | ||||||
| 		msg->lchan->sapis[rllh->link_id & 0x7] = LCHAN_SAPI_UNUSED; | 		msg->lchan->sapis[rllh->link_id & 0x7] = LCHAN_SAPI_UNUSED; | ||||||
| 		/* we can now releae the channel on the BTS/Abis side */ | 		rsl_handle_release(msg->lchan); | ||||||
| 		/* FIXME: officially we need to start T3111 and wait for | 		rsl_lchan_rll_release(msg->lchan, rllh->link_id); | ||||||
| 		 * some grace period */ |  | ||||||
| 		rsl_rf_chan_release(msg->lchan); |  | ||||||
| 		break; | 		break; | ||||||
| 	case RSL_MT_ERROR_IND: | 	case RSL_MT_ERROR_IND: | ||||||
| 		rc = rsl_rx_rll_err_ind(msg); | 		rc = rsl_rx_rll_err_ind(msg); | ||||||
| @@ -1330,31 +1403,11 @@ static u_int8_t ipa_smod_s_for_lchan(struct gsm_lchan *lchan) | |||||||
| { | { | ||||||
| 	switch (lchan->tch_mode) { | 	switch (lchan->tch_mode) { | ||||||
| 	case GSM48_CMODE_SPEECH_V1: | 	case GSM48_CMODE_SPEECH_V1: | ||||||
| 		switch (lchan->type) { | 		return 0x00; | ||||||
| 		case GSM_LCHAN_TCH_F: |  | ||||||
| 			return 0x00; |  | ||||||
| 		case GSM_LCHAN_TCH_H: |  | ||||||
| 			return 0x03; |  | ||||||
| 		default: |  | ||||||
| 			break; |  | ||||||
| 		} |  | ||||||
| 	case GSM48_CMODE_SPEECH_EFR: | 	case GSM48_CMODE_SPEECH_EFR: | ||||||
| 		switch (lchan->type) { | 		return 0x01; | ||||||
| 		case GSM_LCHAN_TCH_F: |  | ||||||
| 			return 0x01; |  | ||||||
| 		/* there's no half-rate EFR */ |  | ||||||
| 		default: |  | ||||||
| 			break; |  | ||||||
| 		} |  | ||||||
| 	case GSM48_CMODE_SPEECH_AMR: | 	case GSM48_CMODE_SPEECH_AMR: | ||||||
| 		switch (lchan->type) { | 		return 0x02; | ||||||
| 		case GSM_LCHAN_TCH_F: |  | ||||||
| 			return 0x02; |  | ||||||
| 		case GSM_LCHAN_TCH_H: |  | ||||||
| 			return 0x05; |  | ||||||
| 		default: |  | ||||||
| 			break; |  | ||||||
| 		} |  | ||||||
| 	default: | 	default: | ||||||
| 		break; | 		break; | ||||||
| 	} | 	} | ||||||
| @@ -1652,9 +1705,21 @@ static int abis_rsl_rx_ipacc(struct msgb *msg) | |||||||
| /* Entry-point where L2 RSL from BTS enters */ | /* Entry-point where L2 RSL from BTS enters */ | ||||||
| int abis_rsl_rcvmsg(struct msgb *msg) | int abis_rsl_rcvmsg(struct msgb *msg) | ||||||
| { | { | ||||||
| 	struct abis_rsl_common_hdr *rslh = msgb_l2(msg)	; | 	struct abis_rsl_common_hdr *rslh; | ||||||
| 	int rc = 0; | 	int rc = 0; | ||||||
|  |  | ||||||
|  | 	if (!msg) { | ||||||
|  | 		DEBUGP(DRSL, "Empty RSL msg?..\n"); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if (msgb_l2len(msg) < sizeof(*rslh)) { | ||||||
|  | 		DEBUGP(DRSL, "Truncated RSL message with l2len: %u\n", msgb_l2len(msg)); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	rslh = msgb_l2(msg); | ||||||
|  |  | ||||||
| 	switch (rslh->msg_discr & 0xfe) { | 	switch (rslh->msg_discr & 0xfe) { | ||||||
| 	case ABIS_RSL_MDISC_RLL: | 	case ABIS_RSL_MDISC_RLL: | ||||||
| 		rc = abis_rsl_rx_rll(msg); | 		rc = abis_rsl_rx_rll(msg); | ||||||
|   | |||||||
| @@ -870,3 +870,8 @@ int main(int argc, char **argv) | |||||||
|  |  | ||||||
| 	exit(0); | 	exit(0); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /* dummy to be able to compile */ | ||||||
|  | void gsm_net_update_ctype(struct gsm_network *net) | ||||||
|  | { | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										24
									
								
								openbsc/src/bsc-nat.cfg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								openbsc/src/bsc-nat.cfg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  | ! | ||||||
|  | ! BSC NAT configuration hand edited | ||||||
|  | !   ! | ||||||
|  | password foo | ||||||
|  | ! | ||||||
|  | line vty | ||||||
|  |  no login | ||||||
|  | ! | ||||||
|  | nat | ||||||
|  |  bsc 0 | ||||||
|  |    token zecke | ||||||
|  |    location_area_code 3 | ||||||
|  |  bsc 1 | ||||||
|  |    token roch | ||||||
|  |    location_area_code 4 | ||||||
|  | mgcp | ||||||
|  |   local ip 10.0.0.23 | ||||||
|  |   bts ip 0.0.0.0 | ||||||
|  |   bind ip 127.0.0.1 | ||||||
|  |   bind port 2427 | ||||||
|  |   bind early 1 | ||||||
|  |   rtp base 4000 | ||||||
|  |   number endpoints 31 | ||||||
|  |   call agent ip 127.0.0.1 | ||||||
| @@ -426,6 +426,7 @@ int nm_state_event(enum nm_evt evt, u_int8_t obj_class, void *obj, | |||||||
| 	case NM_OC_BTS: | 	case NM_OC_BTS: | ||||||
| 		bts = obj; | 		bts = obj; | ||||||
| 		if (new_state->availability == NM_AVSTATE_DEPENDENCY) { | 		if (new_state->availability == NM_AVSTATE_DEPENDENCY) { | ||||||
|  | 			printf("STARTING BTS...\n"); | ||||||
| 			patch_nm_tables(bts); | 			patch_nm_tables(bts); | ||||||
| 			abis_nm_set_bts_attr(bts, nanobts_attr_bts, | 			abis_nm_set_bts_attr(bts, nanobts_attr_bts, | ||||||
| 					     sizeof(nanobts_attr_bts)); | 					     sizeof(nanobts_attr_bts)); | ||||||
| @@ -441,6 +442,7 @@ int nm_state_event(enum nm_evt evt, u_int8_t obj_class, void *obj, | |||||||
| 		trx = ts->trx; | 		trx = ts->trx; | ||||||
| 		if (new_state->operational == 1 && | 		if (new_state->operational == 1 && | ||||||
| 		    new_state->availability == NM_AVSTATE_DEPENDENCY) { | 		    new_state->availability == NM_AVSTATE_DEPENDENCY) { | ||||||
|  | 			printf("STARTING OC Channel...\n"); | ||||||
| 			patch_nm_tables(trx->bts); | 			patch_nm_tables(trx->bts); | ||||||
| 			enum abis_nm_chan_comb ccomb = | 			enum abis_nm_chan_comb ccomb = | ||||||
| 						abis_nm_chcomb4pchan(ts->pchan); | 						abis_nm_chcomb4pchan(ts->pchan); | ||||||
|   | |||||||
							
								
								
									
										229
									
								
								openbsc/src/bsc_msc.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										229
									
								
								openbsc/src/bsc_msc.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,229 @@ | |||||||
|  | /* Routines to talk to the MSC using the IPA Protocol */ | ||||||
|  | /* | ||||||
|  |  * (C) 2010 by Holger Hans Peter Freyther <zecke@selfish.org> | ||||||
|  |  * (C) 2010 by On-Waves | ||||||
|  |  * All Rights Reserved | ||||||
|  |  * | ||||||
|  |  * This program is free software; you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation; either version 2 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License along | ||||||
|  |  * with this program; if not, write to the Free Software Foundation, Inc., | ||||||
|  |  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #include <openbsc/bsc_msc.h> | ||||||
|  | #include <openbsc/debug.h> | ||||||
|  |  | ||||||
|  | #include <osmocore/write_queue.h> | ||||||
|  | #include <osmocore/talloc.h> | ||||||
|  |  | ||||||
|  | #include <arpa/inet.h> | ||||||
|  | #include <sys/socket.h> | ||||||
|  | #include <errno.h> | ||||||
|  | #include <fcntl.h> | ||||||
|  | #include <stdio.h> | ||||||
|  | #include <string.h> | ||||||
|  | #include <unistd.h> | ||||||
|  |  | ||||||
|  | static void connection_loss(struct bsc_msc_connection *con) | ||||||
|  | { | ||||||
|  | 	struct bsc_fd *fd; | ||||||
|  |  | ||||||
|  | 	fd = &con->write_queue.bfd; | ||||||
|  |  | ||||||
|  | 	close(fd->fd); | ||||||
|  | 	fd->fd = -1; | ||||||
|  | 	fd->cb = write_queue_bfd_cb; | ||||||
|  | 	fd->when = 0; | ||||||
|  |  | ||||||
|  | 	con->is_connected = 0; | ||||||
|  | 	con->connection_loss(con); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static int bsc_msc_except(struct bsc_fd *bfd) | ||||||
|  | { | ||||||
|  | 	struct write_queue *wrt; | ||||||
|  | 	struct bsc_msc_connection *con; | ||||||
|  |  | ||||||
|  | 	LOGP(DMSC, LOGL_ERROR, "Exception on the BFD. Closing down.\n"); | ||||||
|  |  | ||||||
|  | 	wrt = container_of(bfd, struct write_queue, bfd); | ||||||
|  | 	con = container_of(wrt, struct bsc_msc_connection, write_queue); | ||||||
|  |  | ||||||
|  | 	connection_loss(con); | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* called in the case of a non blocking connect */ | ||||||
|  | static int msc_connection_connect(struct bsc_fd *fd, unsigned int what) | ||||||
|  | { | ||||||
|  | 	int rc; | ||||||
|  | 	int val; | ||||||
|  | 	struct bsc_msc_connection *con; | ||||||
|  | 	struct write_queue *queue; | ||||||
|  |  | ||||||
|  | 	socklen_t len = sizeof(val); | ||||||
|  |  | ||||||
|  | 	if ((what & BSC_FD_WRITE) == 0) { | ||||||
|  | 		LOGP(DMSC, LOGL_ERROR, "Callback but not readable.\n"); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	queue = container_of(fd, struct write_queue, bfd); | ||||||
|  | 	con = container_of(queue, struct bsc_msc_connection, write_queue); | ||||||
|  |  | ||||||
|  | 	/* check the socket state */ | ||||||
|  | 	rc = getsockopt(fd->fd, SOL_SOCKET, SO_ERROR, &val, &len); | ||||||
|  | 	if (rc != 0) { | ||||||
|  | 		LOGP(DMSC, LOGL_ERROR, "getsockopt for the MSC socket failed.\n"); | ||||||
|  | 		goto error; | ||||||
|  | 	} | ||||||
|  | 	if (val != 0) { | ||||||
|  | 		LOGP(DMSC, LOGL_ERROR, "Not connected to the MSC: %d\n", val); | ||||||
|  | 		goto error; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	/* go to full operation */ | ||||||
|  | 	fd->cb = write_queue_bfd_cb; | ||||||
|  | 	fd->when = BSC_FD_READ | BSC_FD_EXCEPT; | ||||||
|  |  | ||||||
|  | 	con->is_connected = 1; | ||||||
|  | 	LOGP(DMSC, LOGL_NOTICE, "(Re)Connected to the MSC.\n"); | ||||||
|  | 	if (con->connected) | ||||||
|  | 		con->connected(con); | ||||||
|  | 	return 0; | ||||||
|  |  | ||||||
|  | error: | ||||||
|  | 	bsc_unregister_fd(fd); | ||||||
|  | 	connection_loss(con); | ||||||
|  | 	return -1; | ||||||
|  | } | ||||||
|  | static void setnonblocking(struct bsc_fd *fd) | ||||||
|  | { | ||||||
|  | 	int flags; | ||||||
|  |  | ||||||
|  | 	flags = fcntl(fd->fd, F_GETFL); | ||||||
|  | 	if (flags < 0) { | ||||||
|  | 		perror("fcntl get failed"); | ||||||
|  | 		close(fd->fd); | ||||||
|  | 		fd->fd = -1; | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	flags |= O_NONBLOCK; | ||||||
|  | 	flags = fcntl(fd->fd, F_SETFL, flags); | ||||||
|  | 	if (flags < 0) { | ||||||
|  | 		perror("fcntl get failed"); | ||||||
|  | 		close(fd->fd); | ||||||
|  | 		fd->fd = -1; | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int bsc_msc_connect(struct bsc_msc_connection *con) | ||||||
|  | { | ||||||
|  | 	struct bsc_fd *fd; | ||||||
|  | 	struct sockaddr_in sin; | ||||||
|  | 	int on = 1, ret; | ||||||
|  |  | ||||||
|  | 	LOGP(DMSC, LOGL_NOTICE, "Attempting to connect MSC at %s:%d\n", con->ip, con->port); | ||||||
|  |  | ||||||
|  | 	con->is_connected = 0; | ||||||
|  |  | ||||||
|  | 	fd = &con->write_queue.bfd; | ||||||
|  | 	fd->fd = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); | ||||||
|  | 	fd->data = NULL; | ||||||
|  | 	fd->priv_nr = 1; | ||||||
|  |  | ||||||
|  | 	if (fd->fd < 0) { | ||||||
|  | 		perror("Creating TCP socket failed"); | ||||||
|  | 		return fd->fd; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* make it non blocking */ | ||||||
|  | 	setnonblocking(fd); | ||||||
|  |  | ||||||
|  | 	memset(&sin, 0, sizeof(sin)); | ||||||
|  | 	sin.sin_family = AF_INET; | ||||||
|  | 	sin.sin_port = htons(con->port); | ||||||
|  | 	inet_aton(con->ip, &sin.sin_addr); | ||||||
|  |  | ||||||
|  | 	setsockopt(fd->fd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)); | ||||||
|  | 	ret = connect(fd->fd, (struct sockaddr *) &sin, sizeof(sin)); | ||||||
|  |  | ||||||
|  | 	if (ret == -1 && errno == EINPROGRESS) { | ||||||
|  | 		LOGP(DMSC, LOGL_ERROR, "MSC Connection in progress\n"); | ||||||
|  | 		fd->when = BSC_FD_WRITE; | ||||||
|  | 		fd->cb = msc_connection_connect; | ||||||
|  | 	} else if (ret < 0) { | ||||||
|  | 		perror("Connection failed"); | ||||||
|  | 		connection_loss(con); | ||||||
|  | 		return ret; | ||||||
|  | 	} else { | ||||||
|  | 		fd->when = BSC_FD_READ | BSC_FD_EXCEPT; | ||||||
|  | 		fd->cb = write_queue_bfd_cb; | ||||||
|  | 		con->is_connected = 1; | ||||||
|  | 		if (con->connected) | ||||||
|  | 			con->connected(con); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	ret = bsc_register_fd(fd); | ||||||
|  | 	if (ret < 0) { | ||||||
|  | 		perror("Registering the fd failed"); | ||||||
|  | 		close(fd->fd); | ||||||
|  | 		return ret; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | struct bsc_msc_connection *bsc_msc_create(const char *ip, int port) | ||||||
|  | { | ||||||
|  | 	struct bsc_msc_connection *con; | ||||||
|  |  | ||||||
|  | 	con = talloc_zero(NULL, struct bsc_msc_connection); | ||||||
|  | 	if (!con) { | ||||||
|  | 		LOGP(DMSC, LOGL_FATAL, "Failed to create the MSC connection.\n"); | ||||||
|  | 		return NULL; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	con->ip = ip; | ||||||
|  | 	con->port = port; | ||||||
|  | 	write_queue_init(&con->write_queue, 100); | ||||||
|  | 	con->write_queue.except_cb = bsc_msc_except; | ||||||
|  | 	return con; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void bsc_msc_lost(struct bsc_msc_connection *con) | ||||||
|  | { | ||||||
|  | 	write_queue_clear(&con->write_queue); | ||||||
|  | 	bsc_unregister_fd(&con->write_queue.bfd); | ||||||
|  | 	connection_loss(con); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void reconnect_msc(void *_msc) | ||||||
|  | { | ||||||
|  | 	struct bsc_msc_connection *con = _msc; | ||||||
|  |  | ||||||
|  | 	LOGP(DMSC, LOGL_NOTICE, "Attempting to reconnect to the MSC.\n"); | ||||||
|  | 	bsc_msc_connect(con); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void bsc_msc_schedule_connect(struct bsc_msc_connection *con) | ||||||
|  | { | ||||||
|  | 	LOGP(DMSC, LOGL_NOTICE, "Attempting to reconnect to the MSC.\n"); | ||||||
|  | 	con->reconnect_timer.cb = reconnect_msc; | ||||||
|  | 	con->reconnect_timer.data = con; | ||||||
|  | 	bsc_schedule_timer(&con->reconnect_timer, 5, 0); | ||||||
|  | } | ||||||
							
								
								
									
										1094
									
								
								openbsc/src/bsc_msc_ip.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1094
									
								
								openbsc/src/bsc_msc_ip.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -55,7 +55,7 @@ static void complete_rllr(struct bsc_rll_req *rllr, enum bsc_rllr_ind type) | |||||||
|  |  | ||||||
| 	conn = &rllr->lchan->conn; | 	conn = &rllr->lchan->conn; | ||||||
| 	llist_del(&rllr->list); | 	llist_del(&rllr->list); | ||||||
| 	put_subscr_con(conn); | 	put_subscr_con(conn, 0); | ||||||
| 	rllr->cb(rllr->lchan, rllr->link_id, rllr->data, type); | 	rllr->cb(rllr->lchan, rllr->link_id, rllr->data, type); | ||||||
| 	talloc_free(rllr); | 	talloc_free(rllr); | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										1365
									
								
								openbsc/src/bssap.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1365
									
								
								openbsc/src/bssap.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -33,8 +33,6 @@ | |||||||
| #include <openbsc/debug.h> | #include <openbsc/debug.h> | ||||||
| #include <openbsc/signal.h> | #include <openbsc/signal.h> | ||||||
|  |  | ||||||
| static void auto_release_channel(void *_lchan); |  | ||||||
|  |  | ||||||
| static int ts_is_usable(struct gsm_bts_trx_ts *ts) | static int ts_is_usable(struct gsm_bts_trx_ts *ts) | ||||||
| { | { | ||||||
| 	/* FIXME: How does this behave for BS-11 ? */ | 	/* FIXME: How does this behave for BS-11 ? */ | ||||||
| @@ -268,16 +266,13 @@ struct gsm_lchan *lchan_alloc(struct gsm_bts *bts, enum gsm_chan_t type) | |||||||
| 		/* clear multi rate config */ | 		/* clear multi rate config */ | ||||||
| 		memset(&lchan->mr_conf, 0, sizeof(lchan->mr_conf)); | 		memset(&lchan->mr_conf, 0, sizeof(lchan->mr_conf)); | ||||||
|  |  | ||||||
|  | 		/* clear any msc reference */ | ||||||
|  | 		lchan->msc_data = NULL; | ||||||
|  |  | ||||||
| 		/* clear per MSC/BSC data */ | 		/* clear per MSC/BSC data */ | ||||||
| 		memset(&lchan->conn, 0, sizeof(lchan->conn)); | 		memset(&lchan->conn, 0, sizeof(lchan->conn)); | ||||||
|  |  | ||||||
| 		/* Configure the time and start it so it will be closed */ |  | ||||||
| 		lchan->conn.lchan = lchan; | 		lchan->conn.lchan = lchan; | ||||||
| 		lchan->conn.bts = lchan->ts->trx->bts; | 		lchan->conn.bts = lchan->ts->trx->bts; | ||||||
| 		lchan->conn.release_timer.cb = auto_release_channel; |  | ||||||
| 		lchan->conn.release_timer.data = lchan; |  | ||||||
| 		bsc_schedule_timer(&lchan->conn.release_timer, LCHAN_RELEASE_TIMEOUT); |  | ||||||
|  |  | ||||||
| 	} else { | 	} else { | ||||||
| 		struct challoc_signal_data sig; | 		struct challoc_signal_data sig; | ||||||
| 		sig.bts = bts; | 		sig.bts = bts; | ||||||
| @@ -307,8 +302,6 @@ void lchan_free(struct gsm_lchan *lchan) | |||||||
| 		lchan->conn.use_count = 0; | 		lchan->conn.use_count = 0; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	/* stop the timer */ |  | ||||||
| 	bsc_del_timer(&lchan->conn.release_timer); |  | ||||||
| 	bsc_del_timer(&lchan->T3101); | 	bsc_del_timer(&lchan->T3101); | ||||||
|  |  | ||||||
| 	/* clear cached measuement reports */ | 	/* clear cached measuement reports */ | ||||||
| @@ -319,7 +312,6 @@ void lchan_free(struct gsm_lchan *lchan) | |||||||
| 	} | 	} | ||||||
| 	for (i = 0; i < ARRAY_SIZE(lchan->neigh_meas); i++) | 	for (i = 0; i < ARRAY_SIZE(lchan->neigh_meas); i++) | ||||||
| 		lchan->neigh_meas[i].arfcn = 0; | 		lchan->neigh_meas[i].arfcn = 0; | ||||||
|  |  | ||||||
| 	lchan->conn.silent_call = 0; | 	lchan->conn.silent_call = 0; | ||||||
|  |  | ||||||
| 	sig.lchan = lchan; | 	sig.lchan = lchan; | ||||||
| @@ -330,16 +322,36 @@ void lchan_free(struct gsm_lchan *lchan) | |||||||
| 	 * channel using it */ | 	 * channel using it */ | ||||||
| } | } | ||||||
|  |  | ||||||
| /* Consider releasing the channel now */ | static int _lchan_release_next_sapi(struct gsm_lchan *lchan) | ||||||
| int lchan_auto_release(struct gsm_lchan *lchan) |  | ||||||
| { | { | ||||||
| 	if (lchan->conn.use_count > 0) { | 	int sapi; | ||||||
|  |  | ||||||
|  | 	for (sapi = 1; sapi < ARRAY_SIZE(lchan->sapis); ++sapi) { | ||||||
|  | 		u_int8_t link_id; | ||||||
|  | 		if (lchan->sapis[sapi] == LCHAN_SAPI_UNUSED) | ||||||
|  | 			continue; | ||||||
|  |  | ||||||
|  | 		link_id = sapi; | ||||||
|  | 		if (lchan->type == GSM_LCHAN_TCH_F || lchan->type == GSM_LCHAN_TCH_H) | ||||||
|  | 			link_id |= 0x40; | ||||||
|  | 		rsl_release_request(lchan, link_id, lchan->release_reason); | ||||||
| 		return 0; | 		return 0; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	return 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void _lchan_handle_release(struct gsm_lchan *lchan) | ||||||
|  | { | ||||||
|  | 	/* Ask for SAPI != 0 to be freed first and stop if we need to wait */ | ||||||
|  | 	if (_lchan_release_next_sapi(lchan) == 0) | ||||||
|  | 		return; | ||||||
|  |  | ||||||
| 	/* Assume we have GSM04.08 running and send a release */ | 	/* Assume we have GSM04.08 running and send a release */ | ||||||
| 	if (lchan->conn.subscr) { | 	if (lchan->conn.subscr) { | ||||||
|  | 		++lchan->conn.use_count; | ||||||
| 		gsm48_send_rr_release(lchan); | 		gsm48_send_rr_release(lchan); | ||||||
|  | 		--lchan->conn.use_count; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	/* spoofed? message */ | 	/* spoofed? message */ | ||||||
| @@ -347,18 +359,43 @@ int lchan_auto_release(struct gsm_lchan *lchan) | |||||||
| 		LOGP(DRLL, LOGL_ERROR, "Channel count is negative: %d\n", | 		LOGP(DRLL, LOGL_ERROR, "Channel count is negative: %d\n", | ||||||
| 			lchan->conn.use_count); | 			lchan->conn.use_count); | ||||||
|  |  | ||||||
| 	DEBUGP(DRLL, "%s Recycling Channel\n", gsm_lchan_name(lchan)); | 	rsl_release_request(lchan, 0, lchan->release_reason); | ||||||
| 	rsl_release_request(lchan, 0); |  | ||||||
| 	return 1; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| /* Auto release the channel when the use count is zero */ | /* called from abis rsl */ | ||||||
| static void auto_release_channel(void *_lchan) | int rsl_lchan_rll_release(struct gsm_lchan *lchan, u_int8_t link_id) | ||||||
| { | { | ||||||
| 	struct gsm_lchan *lchan = _lchan; | 	if (lchan->state != LCHAN_S_REL_REQ) | ||||||
|  | 		return -1; | ||||||
|  |  | ||||||
| 	if (!lchan_auto_release(lchan)) | 	if ((link_id & 0x7) != 0) | ||||||
| 		bsc_schedule_timer(&lchan->conn.release_timer, LCHAN_RELEASE_TIMEOUT); | 		_lchan_handle_release(lchan); | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Start the channel release procedure now. We will start by shutting | ||||||
|  |  * down SAPI!=0, then we will deactivate the SACCH and finish by releasing | ||||||
|  |  * the last SAPI at which point the RSL code will send the channel release | ||||||
|  |  * for us. We should guard the whole shutdown by T3109 or similiar and then | ||||||
|  |  * update the fixme inside gsm_04_08_utils.c | ||||||
|  |  * When we request to release the RLL and we don't get an answer within T200 | ||||||
|  |  * the BTS will send us an Error indication which we will handle by closing | ||||||
|  |  * the channel and be done. | ||||||
|  |  */ | ||||||
|  | int _lchan_release(struct gsm_lchan *lchan, u_int8_t release_reason) | ||||||
|  | { | ||||||
|  | 	if (lchan->conn.use_count > 0) { | ||||||
|  | 		DEBUGP(DRLL, "BUG: _lchan_release called without zero use_count.\n"); | ||||||
|  | 		return 0; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	DEBUGP(DRLL, "%s Recycling Channel\n", gsm_lchan_name(lchan)); | ||||||
|  | 	rsl_lchan_set_state(lchan, LCHAN_S_REL_REQ); | ||||||
|  | 	lchan->release_reason = release_reason; | ||||||
|  | 	_lchan_handle_release(lchan); | ||||||
|  | 	return 1; | ||||||
| } | } | ||||||
|  |  | ||||||
| struct gsm_lchan* lchan_find(struct gsm_bts *bts, struct gsm_subscriber *subscr) { | struct gsm_lchan* lchan_find(struct gsm_bts *bts, struct gsm_subscriber *subscr) { | ||||||
|   | |||||||
| @@ -146,6 +146,11 @@ static const struct log_info_cat default_categories[] = { | |||||||
| 		.description = "Reference Counting", | 		.description = "Reference Counting", | ||||||
| 		.enabled = 0, .loglevel = LOGL_NOTICE, | 		.enabled = 0, .loglevel = LOGL_NOTICE, | ||||||
| 	}, | 	}, | ||||||
|  | 	[DNAT] = { | ||||||
|  | 		.name = "DNAT", | ||||||
|  | 		.description = "BSC MUX/NAT", | ||||||
|  | 		.enabled = 1, .loglevel = LOGL_NOTICE, | ||||||
|  | 	}, | ||||||
| }; | }; | ||||||
|  |  | ||||||
| enum log_ctxt { | enum log_ctxt { | ||||||
|   | |||||||
| @@ -104,12 +104,12 @@ static void release_loc_updating_req(struct gsm_subscriber_connection *conn) | |||||||
| 	bsc_del_timer(&conn->loc_operation->updating_timer); | 	bsc_del_timer(&conn->loc_operation->updating_timer); | ||||||
| 	talloc_free(conn->loc_operation); | 	talloc_free(conn->loc_operation); | ||||||
| 	conn->loc_operation = 0; | 	conn->loc_operation = 0; | ||||||
| 	put_subscr_con(conn); | 	put_subscr_con(conn, 0); | ||||||
| } | } | ||||||
|  |  | ||||||
| static void allocate_loc_updating_req(struct gsm_subscriber_connection *conn) | static void allocate_loc_updating_req(struct gsm_subscriber_connection *conn) | ||||||
| { | { | ||||||
| 	use_subscr_con(conn) | 	use_subscr_con(conn); | ||||||
| 	release_loc_updating_req(conn); | 	release_loc_updating_req(conn); | ||||||
|  |  | ||||||
| 	conn->loc_operation = talloc_zero(tall_locop_ctx, | 	conn->loc_operation = talloc_zero(tall_locop_ctx, | ||||||
| @@ -122,7 +122,6 @@ static int gsm0408_authorize(struct gsm_subscriber_connection *conn, struct msgb | |||||||
| 		int rc; | 		int rc; | ||||||
|  |  | ||||||
| 		db_subscriber_alloc_tmsi(conn->subscr); | 		db_subscriber_alloc_tmsi(conn->subscr); | ||||||
| 		release_loc_updating_req(conn); |  | ||||||
| 		rc = gsm0408_loc_upd_acc(msg->lchan, conn->subscr->tmsi); | 		rc = gsm0408_loc_upd_acc(msg->lchan, conn->subscr->tmsi); | ||||||
| 		if (msg->lchan->ts->trx->bts->network->send_mm_info) { | 		if (msg->lchan->ts->trx->bts->network->send_mm_info) { | ||||||
| 			/* send MM INFO with network name */ | 			/* send MM INFO with network name */ | ||||||
| @@ -134,9 +133,7 @@ static int gsm0408_authorize(struct gsm_subscriber_connection *conn, struct msgb | |||||||
| 		 * trigger further action like SMS delivery */ | 		 * trigger further action like SMS delivery */ | ||||||
| 		subscr_update(conn->subscr, msg->trx->bts, | 		subscr_update(conn->subscr, msg->trx->bts, | ||||||
| 			      GSM_SUBSCRIBER_UPDATE_ATTACHED); | 			      GSM_SUBSCRIBER_UPDATE_ATTACHED); | ||||||
|  | 		release_loc_updating_req(conn); | ||||||
| 		/* try to close channel ASAP */ |  | ||||||
| 		lchan_auto_release(conn->lchan); |  | ||||||
| 		return rc; | 		return rc; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -298,9 +295,8 @@ static void loc_upd_rej_cb(void *data) | |||||||
| 	struct gsm_lchan *lchan = conn->lchan; | 	struct gsm_lchan *lchan = conn->lchan; | ||||||
| 	struct gsm_bts *bts = lchan->ts->trx->bts; | 	struct gsm_bts *bts = lchan->ts->trx->bts; | ||||||
|  |  | ||||||
| 	release_loc_updating_req(conn); |  | ||||||
| 	gsm0408_loc_upd_rej(lchan, bts->network->reject_cause); | 	gsm0408_loc_upd_rej(lchan, bts->network->reject_cause); | ||||||
| 	lchan_auto_release(lchan); | 	release_loc_updating_req(conn); | ||||||
| } | } | ||||||
|  |  | ||||||
| static void schedule_reject(struct gsm_subscriber_connection *conn) | static void schedule_reject(struct gsm_subscriber_connection *conn) | ||||||
| @@ -722,8 +718,6 @@ static int gsm48_rx_mm_imsi_detach_ind(struct msgb *msg) | |||||||
| 	 * imagine an IMSI DETACH happening during an active call! */ | 	 * imagine an IMSI DETACH happening during an active call! */ | ||||||
|  |  | ||||||
| 	/* subscriber is detached: should we release lchan? */ | 	/* subscriber is detached: should we release lchan? */ | ||||||
| 	lchan_auto_release(msg->lchan); |  | ||||||
|  |  | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -2070,7 +2064,6 @@ static int gsm48_cc_rx_release_compl(struct gsm_trans *trans, struct msgb *msg) | |||||||
| 					  MNCC_REL_CNF, &rel); | 					  MNCC_REL_CNF, &rel); | ||||||
| 			/* FIXME: in case of multiple calls, we can't simply | 			/* FIXME: in case of multiple calls, we can't simply | ||||||
| 			 * hang up here ! */ | 			 * hang up here ! */ | ||||||
| 			lchan_auto_release(msg->lchan); |  | ||||||
| 			break; | 			break; | ||||||
| 		default: | 		default: | ||||||
| 			rc = mncc_recvmsg(trans->subscr->net, trans, | 			rc = mncc_recvmsg(trans->subscr->net, trans, | ||||||
|   | |||||||
| @@ -164,13 +164,46 @@ static const enum gsm_chreq_reason_t reason_by_chreq[] = { | |||||||
| 	[CHREQ_T_RESERVED_IGNORE]	= GSM_CHREQ_REASON_OTHER, | 	[CHREQ_T_RESERVED_IGNORE]	= GSM_CHREQ_REASON_OTHER, | ||||||
| }; | }; | ||||||
|  |  | ||||||
| enum gsm_chan_t get_ctype_by_chreq(struct gsm_bts *bts, u_int8_t ra, int neci) | /* verify that the two tables match */ | ||||||
|  | static_assert(sizeof(ctype_by_chreq) == | ||||||
|  | 	      sizeof(((struct gsm_network *) NULL)->ctype_by_chreq), assert_size); | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Update channel types for request based on policy. E.g. in the | ||||||
|  |  * case of a TCH/H network/bsc use TCH/H for the emergency calls, | ||||||
|  |  * for early assignment assign a SDCCH and some other options. | ||||||
|  |  */ | ||||||
|  | void gsm_net_update_ctype(struct gsm_network *network) | ||||||
|  | { | ||||||
|  | 	/* copy over the data */ | ||||||
|  | 	memcpy(network->ctype_by_chreq, ctype_by_chreq, sizeof(ctype_by_chreq)); | ||||||
|  |  | ||||||
|  | 	/* | ||||||
|  | 	 * Use TCH/H for emergency calls when this cell allows TCH/H. Maybe it | ||||||
|  | 	 * is better to iterate over the BTS/TRX and check if no TCH/F is available | ||||||
|  | 	 * and then set it to TCH/H. | ||||||
|  | 	 */ | ||||||
|  | 	if (network->neci) | ||||||
|  | 		network->ctype_by_chreq[CHREQ_T_EMERG_CALL] = GSM_LCHAN_TCH_H; | ||||||
|  |  | ||||||
|  | 	if (network->pag_any_tch) { | ||||||
|  | 		if (network->neci) { | ||||||
|  | 			network->ctype_by_chreq[CHREQ_T_PAG_R_ANY_NECI0] = GSM_LCHAN_TCH_H; | ||||||
|  | 			network->ctype_by_chreq[CHREQ_T_PAG_R_ANY_NECI1] = GSM_LCHAN_TCH_H; | ||||||
|  | 		} else { | ||||||
|  | 			network->ctype_by_chreq[CHREQ_T_PAG_R_ANY_NECI0] = GSM_LCHAN_TCH_F; | ||||||
|  | 			network->ctype_by_chreq[CHREQ_T_PAG_R_ANY_NECI1] = GSM_LCHAN_TCH_F; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | enum gsm_chan_t get_ctype_by_chreq(struct gsm_network *network, u_int8_t ra) | ||||||
| { | { | ||||||
| 	int i; | 	int i; | ||||||
| 	int length; | 	int length; | ||||||
| 	const struct chreq *chreq; | 	const struct chreq *chreq; | ||||||
|  |  | ||||||
| 	if (neci) { | 	if (network->neci) { | ||||||
| 		chreq = chreq_type_neci1; | 		chreq = chreq_type_neci1; | ||||||
| 		length = ARRAY_SIZE(chreq_type_neci1); | 		length = ARRAY_SIZE(chreq_type_neci1); | ||||||
| 	} else { | 	} else { | ||||||
| @@ -182,13 +215,13 @@ enum gsm_chan_t get_ctype_by_chreq(struct gsm_bts *bts, u_int8_t ra, int neci) | |||||||
| 	for (i = 0; i < length; i++) { | 	for (i = 0; i < length; i++) { | ||||||
| 		const struct chreq *chr = &chreq[i]; | 		const struct chreq *chr = &chreq[i]; | ||||||
| 		if ((ra & chr->mask) == chr->val) | 		if ((ra & chr->mask) == chr->val) | ||||||
| 			return ctype_by_chreq[chr->type]; | 			return network->ctype_by_chreq[chr->type]; | ||||||
| 	} | 	} | ||||||
| 	LOGP(DRR, LOGL_ERROR, "Unknown CHANNEL REQUEST RQD 0x%02x\n", ra); | 	LOGP(DRR, LOGL_ERROR, "Unknown CHANNEL REQUEST RQD 0x%02x\n", ra); | ||||||
| 	return GSM_LCHAN_SDCCH; | 	return GSM_LCHAN_SDCCH; | ||||||
| } | } | ||||||
|  |  | ||||||
| enum gsm_chreq_reason_t get_reason_by_chreq(struct gsm_bts *bts, u_int8_t ra, int neci) | enum gsm_chreq_reason_t get_reason_by_chreq(u_int8_t ra, int neci) | ||||||
| { | { | ||||||
| 	int i; | 	int i; | ||||||
| 	int length; | 	int length; | ||||||
|   | |||||||
| @@ -757,7 +757,7 @@ static int gsm411_rx_rp_ack(struct msgb *msg, struct gsm_trans *trans, | |||||||
| 	/* release channel if done */ | 	/* release channel if done */ | ||||||
| #warning "BROKEN. The SAPI will be released automatically by the BSC" | #warning "BROKEN. The SAPI will be released automatically by the BSC" | ||||||
| 	if (!sms) | 	if (!sms) | ||||||
| 		rsl_release_request(msg->lchan, trans->sms.link_id); | 		rsl_release_request(msg->lchan, trans->sms.link_id, 0); | ||||||
|  |  | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| @@ -833,7 +833,7 @@ static int gsm411_rx_rp_smma(struct msgb *msg, struct gsm_trans *trans, | |||||||
| 	if (sms) | 	if (sms) | ||||||
| 		gsm411_send_sms_lchan(trans->conn, sms); | 		gsm411_send_sms_lchan(trans->conn, sms); | ||||||
| 	else | 	else | ||||||
| 		rsl_release_request(msg->lchan, trans->sms.link_id); | 		rsl_release_request(msg->lchan, trans->sms.link_id, 0); | ||||||
| #warning "BROKEN: The SAPI=3 will be released automatically by the BSC" | #warning "BROKEN: The SAPI=3 will be released automatically by the BSC" | ||||||
|  |  | ||||||
| 	return rc; | 	return rc; | ||||||
|   | |||||||
| @@ -88,6 +88,7 @@ static const struct value_string lchan_s_names[] = { | |||||||
| 	{ LCHAN_S_ACTIVE,	"ACTIVE" }, | 	{ LCHAN_S_ACTIVE,	"ACTIVE" }, | ||||||
| 	{ LCHAN_S_INACTIVE,	"INACTIVE" }, | 	{ LCHAN_S_INACTIVE,	"INACTIVE" }, | ||||||
| 	{ LCHAN_S_REL_REQ,	"RELEASE REQUESTED" }, | 	{ LCHAN_S_REL_REQ,	"RELEASE REQUESTED" }, | ||||||
|  | 	{ LCHAN_S_REL_ERR,	"RELEASE DUE ERROR" }, | ||||||
| 	{ 0,			NULL } | 	{ 0,			NULL } | ||||||
| }; | }; | ||||||
|  |  | ||||||
| @@ -283,6 +284,12 @@ struct gsm_network *gsm_network_init(u_int16_t country_code, u_int16_t network_c | |||||||
|  |  | ||||||
| 	net->mncc_recv = mncc_recv; | 	net->mncc_recv = mncc_recv; | ||||||
|  |  | ||||||
|  | 	gsm_net_update_ctype(net); | ||||||
|  |  | ||||||
|  | 	net->core_country_code = -1; | ||||||
|  | 	net->core_network_code = -1; | ||||||
|  | 	net->rtp_base_port = 4000; | ||||||
|  |  | ||||||
| 	return net; | 	return net; | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -31,6 +31,7 @@ | |||||||
| #include <openbsc/gsm_subscriber.h> | #include <openbsc/gsm_subscriber.h> | ||||||
| #include <openbsc/paging.h> | #include <openbsc/paging.h> | ||||||
| #include <openbsc/debug.h> | #include <openbsc/debug.h> | ||||||
|  | #include <openbsc/chan_alloc.h> | ||||||
|  |  | ||||||
| LLIST_HEAD(active_subscribers); | LLIST_HEAD(active_subscribers); | ||||||
| void *tall_subscr_ctx; | void *tall_subscr_ctx; | ||||||
| @@ -88,6 +89,7 @@ static int subscr_paging_cb(unsigned int hooknum, unsigned int event, | |||||||
| 	request->cbfn(hooknum, event, msg, data, request->param); | 	request->cbfn(hooknum, event, msg, data, request->param); | ||||||
| 	subscr->in_callback = 0; | 	subscr->in_callback = 0; | ||||||
|  |  | ||||||
|  | 	subscr_put(request->subscr); | ||||||
| 	talloc_free(request); | 	talloc_free(request); | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| @@ -165,7 +167,7 @@ void subscr_get_channel(struct gsm_subscriber *subscr, | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	memset(request, 0, sizeof(*request)); | 	memset(request, 0, sizeof(*request)); | ||||||
| 	request->subscr = subscr; | 	request->subscr = subscr_get(subscr); | ||||||
| 	request->channel_type = type; | 	request->channel_type = type; | ||||||
| 	request->cbfn = cbfn; | 	request->cbfn = cbfn; | ||||||
| 	request->param = param; | 	request->param = param; | ||||||
| @@ -206,9 +208,28 @@ void subscr_put_channel(struct gsm_lchan *lchan) | |||||||
| 	 * will listen to the paging requests before we timeout | 	 * will listen to the paging requests before we timeout | ||||||
| 	 */ | 	 */ | ||||||
|  |  | ||||||
| 	put_subscr_con(conn); | 	put_subscr_con(conn, 0); | ||||||
|  |  | ||||||
| 	if (lchan->conn.subscr && !llist_empty(&lchan->conn.subscr->requests)) | 	if (lchan->conn.subscr && !llist_empty(&lchan->conn.subscr->requests)) | ||||||
| 		subscr_send_paging_request(lchan->conn.subscr); | 		subscr_send_paging_request(lchan->conn.subscr); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | struct gsm_subscriber *subscr_get_or_create(struct gsm_network *net, | ||||||
|  | 					    const char *imsi) | ||||||
|  | { | ||||||
|  | 	struct gsm_subscriber *subscr; | ||||||
|  |  | ||||||
|  | 	llist_for_each_entry(subscr, subscr_bsc_active_subscriber(), entry) { | ||||||
|  | 		if (strcmp(subscr->imsi, imsi) == 0 && subscr->net == net) | ||||||
|  | 			return subscr_get(subscr); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	subscr = subscr_alloc(); | ||||||
|  | 	if (!subscr) | ||||||
|  | 		return NULL; | ||||||
|  |  | ||||||
|  | 	strcpy(subscr->imsi, imsi); | ||||||
|  | 	subscr->net = net; | ||||||
|  | 	return subscr; | ||||||
|  | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -134,6 +134,7 @@ int bsc_handover_start(struct gsm_lchan *old_lchan, struct gsm_bts *bts) | |||||||
| 		return rc; | 		return rc; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	rsl_lchan_set_state(new_lchan, LCHAN_S_ACT_REQ); | ||||||
| 	llist_add(&ho->list, &bsc_handovers); | 	llist_add(&ho->list, &bsc_handovers); | ||||||
| 	/* we continue in the SS_LCHAN handler / ho_chan_activ_ack */ | 	/* we continue in the SS_LCHAN handler / ho_chan_activ_ack */ | ||||||
|  |  | ||||||
| @@ -230,7 +231,6 @@ static int ho_gsm48_ho_compl(struct gsm_lchan *new_lchan) | |||||||
| 	trans_lchan_change(&ho->old_lchan->conn, &new_lchan->conn); | 	trans_lchan_change(&ho->old_lchan->conn, &new_lchan->conn); | ||||||
|  |  | ||||||
| 	ho->old_lchan->state = LCHAN_S_INACTIVE; | 	ho->old_lchan->state = LCHAN_S_INACTIVE; | ||||||
| 	lchan_auto_release(ho->old_lchan); |  | ||||||
|  |  | ||||||
| 	/* do something to re-route the actual speech frames ! */ | 	/* do something to re-route the actual speech frames ! */ | ||||||
|  |  | ||||||
| @@ -258,7 +258,7 @@ static int ho_gsm48_ho_fail(struct gsm_lchan *old_lchan) | |||||||
| 	bsc_del_timer(&ho->T3103); | 	bsc_del_timer(&ho->T3103); | ||||||
| 	llist_del(&ho->list); | 	llist_del(&ho->list); | ||||||
| 	conn = &ho->new_lchan->conn; | 	conn = &ho->new_lchan->conn; | ||||||
| 	put_subscr_con(conn); | 	put_subscr_con(conn, 0); | ||||||
| 	talloc_free(ho); | 	talloc_free(ho); | ||||||
|  |  | ||||||
| 	return 0; | 	return 0; | ||||||
|   | |||||||
| @@ -57,7 +57,7 @@ struct ia_e1_handle { | |||||||
| static struct ia_e1_handle *e1h; | static struct ia_e1_handle *e1h; | ||||||
|  |  | ||||||
|  |  | ||||||
| #define TS1_ALLOC_SIZE	300 | #define TS1_ALLOC_SIZE	900 | ||||||
|  |  | ||||||
| static const u_int8_t pong[] = { 0, 1, IPAC_PROTO_IPACCESS, IPAC_MSGT_PONG }; | static const u_int8_t pong[] = { 0, 1, IPAC_PROTO_IPACCESS, IPAC_MSGT_PONG }; | ||||||
| static const u_int8_t id_ack[] = { 0, 1, IPAC_PROTO_IPACCESS, IPAC_MSGT_ID_ACK }; | static const u_int8_t id_ack[] = { 0, 1, IPAC_PROTO_IPACCESS, IPAC_MSGT_ID_ACK }; | ||||||
|   | |||||||
| @@ -38,7 +38,11 @@ | |||||||
| #include <openbsc/gsm_data.h> | #include <openbsc/gsm_data.h> | ||||||
| #include <osmocore/select.h> | #include <osmocore/select.h> | ||||||
| #include <openbsc/mgcp.h> | #include <openbsc/mgcp.h> | ||||||
|  | #include <openbsc/mgcp_internal.h> | ||||||
| #include <openbsc/telnet_interface.h> | #include <openbsc/telnet_interface.h> | ||||||
|  | #include <openbsc/vty.h> | ||||||
|  |  | ||||||
|  | #include <vty/command.h> | ||||||
|  |  | ||||||
| #include "../../bscconfig.h" | #include "../../bscconfig.h" | ||||||
|  |  | ||||||
| @@ -51,8 +55,9 @@ void subscr_put() { abort(); } | |||||||
| #warning "Make use of the rtp proxy code" | #warning "Make use of the rtp proxy code" | ||||||
|  |  | ||||||
| static struct bsc_fd bfd; | static struct bsc_fd bfd; | ||||||
| static int first_request = 1; |  | ||||||
| static struct mgcp_config *cfg; | static struct mgcp_config *cfg; | ||||||
|  | static int reset_endpoints = 0; | ||||||
|  |  | ||||||
| const char *openbsc_version = "OpenBSC MGCP " PACKAGE_VERSION; | const char *openbsc_version = "OpenBSC MGCP " PACKAGE_VERSION; | ||||||
| const char *openbsc_copyright = | const char *openbsc_copyright = | ||||||
| 	"Copyright (C) 2009-2010 Holger Freyther and On-Waves\n" | 	"Copyright (C) 2009-2010 Holger Freyther and On-Waves\n" | ||||||
| @@ -74,10 +79,10 @@ static void print_help() | |||||||
| 	printf(" -c --config-file filename The config file to use.\n"); | 	printf(" -c --config-file filename The config file to use.\n"); | ||||||
| } | } | ||||||
|  |  | ||||||
| static void print_version() | static void print_mgcp_version() | ||||||
| { | { | ||||||
| 	printf("%s\n\n", openbsc_version); | 	printf("%s\n\n", openbsc_version); | ||||||
| 	printf(openbsc_copyright); | 	printf("%s", openbsc_copyright); | ||||||
| } | } | ||||||
|  |  | ||||||
| static void handle_options(int argc, char** argv) | static void handle_options(int argc, char** argv) | ||||||
| @@ -105,7 +110,7 @@ static void handle_options(int argc, char** argv) | |||||||
| 			config_file = talloc_strdup(tall_bsc_ctx, optarg); | 			config_file = talloc_strdup(tall_bsc_ctx, optarg); | ||||||
| 			break; | 			break; | ||||||
| 		case 'V': | 		case 'V': | ||||||
| 			print_version(); | 			print_mgcp_version(); | ||||||
| 			exit(0); | 			exit(0); | ||||||
| 			break; | 			break; | ||||||
| 		default: | 		default: | ||||||
| @@ -115,12 +120,21 @@ static void handle_options(int argc, char** argv) | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /* simply remember this */ | ||||||
|  | static int mgcp_rsip_cb(struct mgcp_config *cfg) | ||||||
|  | { | ||||||
|  | 	reset_endpoints = 1; | ||||||
|  |  | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
| static int read_call_agent(struct bsc_fd *fd, unsigned int what) | static int read_call_agent(struct bsc_fd *fd, unsigned int what) | ||||||
| { | { | ||||||
| 	struct sockaddr_in addr; | 	struct sockaddr_in addr; | ||||||
| 	socklen_t slen = sizeof(addr); | 	socklen_t slen = sizeof(addr); | ||||||
| 	struct msgb *msg; | 	struct msgb *msg; | ||||||
| 	struct msgb *resp; | 	struct msgb *resp; | ||||||
|  | 	int i; | ||||||
|  |  | ||||||
| 	msg = (struct msgb *) fd->data; | 	msg = (struct msgb *) fd->data; | ||||||
|  |  | ||||||
| @@ -136,18 +150,6 @@ static int read_call_agent(struct bsc_fd *fd, unsigned int what) | |||||||
| 		return -1; | 		return -1; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if (first_request) { |  | ||||||
| 		first_request = 0; |  | ||||||
| 		resp = mgcp_create_rsip(); |  | ||||||
|  |  | ||||||
| 		if (resp) { |  | ||||||
| 			sendto(bfd.fd, resp->l2h, msgb_l2len(resp), 0, |  | ||||||
| 				(struct sockaddr *) &addr, sizeof(addr)); |  | ||||||
| 			msgb_free(resp); |  | ||||||
| 		} |  | ||||||
| 		return 0; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
| 	/* handle message now */ | 	/* handle message now */ | ||||||
| 	msg->l2h = msgb_put(msg, rc); | 	msg->l2h = msgb_put(msg, rc); | ||||||
| 	resp = mgcp_handle_message(cfg, msg); | 	resp = mgcp_handle_message(cfg, msg); | ||||||
| @@ -157,6 +159,16 @@ static int read_call_agent(struct bsc_fd *fd, unsigned int what) | |||||||
| 		sendto(bfd.fd, resp->l2h, msgb_l2len(resp), 0, (struct sockaddr *) &addr, sizeof(addr)); | 		sendto(bfd.fd, resp->l2h, msgb_l2len(resp), 0, (struct sockaddr *) &addr, sizeof(addr)); | ||||||
| 		msgb_free(resp); | 		msgb_free(resp); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	if (reset_endpoints) { | ||||||
|  | 		LOGP(DMGCP, LOGL_NOTICE, "Asked to reset endpoints.\n"); | ||||||
|  | 		reset_endpoints = 0; | ||||||
|  |  | ||||||
|  | 		/* is checking in_addr.s_addr == INADDR_LOOPBACK making it more secure? */ | ||||||
|  | 		for (i = 1; i < cfg->number_endpoints; ++i) | ||||||
|  | 			mgcp_free_endp(&cfg->endpoints[i]); | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -186,6 +198,8 @@ int main(int argc, char** argv) | |||||||
| 	if (rc < 0) | 	if (rc < 0) | ||||||
| 		return rc; | 		return rc; | ||||||
|  |  | ||||||
|  | 	/* set some callbacks */ | ||||||
|  | 	cfg->reset_cb = mgcp_rsip_cb; | ||||||
|  |  | ||||||
|         /* we need to bind a socket */ |         /* we need to bind a socket */ | ||||||
|         if (rc == 0) { |         if (rc == 0) { | ||||||
| @@ -217,11 +231,11 @@ int main(int argc, char** argv) | |||||||
|  |  | ||||||
|  |  | ||||||
| 		if (bsc_register_fd(&bfd) != 0) { | 		if (bsc_register_fd(&bfd) != 0) { | ||||||
| 			DEBUGP(DMGCP, "Failed to register the fd\n"); | 			LOGP(DMGCP, LOGL_FATAL, "Failed to register the fd\n"); | ||||||
| 			return -1; | 			return -1; | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		DEBUGP(DMGCP, "Configured for MGCP.\n"); | 		LOGP(DMGCP, LOGL_NOTICE, "Configured for MGCP.\n"); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	/* initialisation */ | 	/* initialisation */ | ||||||
| @@ -235,3 +249,15 @@ int main(int argc, char** argv) | |||||||
|  |  | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | struct gsm_network; | ||||||
|  | int bsc_vty_init(struct gsm_network *dummy) | ||||||
|  | { | ||||||
|  | 	cmd_init(1); | ||||||
|  | 	vty_init(); | ||||||
|  |  | ||||||
|  | 	openbsc_vty_add_cmds(); | ||||||
|  |         mgcp_vty_init(); | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -25,6 +25,7 @@ | |||||||
| #include <string.h> | #include <string.h> | ||||||
| #include <unistd.h> | #include <unistd.h> | ||||||
| #include <endian.h> | #include <endian.h> | ||||||
|  | #include <errno.h> | ||||||
|  |  | ||||||
| #include <sys/socket.h> | #include <sys/socket.h> | ||||||
| #include <arpa/inet.h> | #include <arpa/inet.h> | ||||||
| @@ -90,6 +91,9 @@ static void patch_payload(int payload, char *data, int len) | |||||||
| 	if (len < sizeof(*rtp_hdr)) | 	if (len < sizeof(*rtp_hdr)) | ||||||
| 		return; | 		return; | ||||||
|  |  | ||||||
|  | 	if (payload < 0) | ||||||
|  | 		return; | ||||||
|  |  | ||||||
| 	rtp_hdr = (struct rtp_hdr *) data; | 	rtp_hdr = (struct rtp_hdr *) data; | ||||||
| 	rtp_hdr->payload_type = payload; | 	rtp_hdr->payload_type = payload; | ||||||
| } | } | ||||||
| @@ -119,14 +123,14 @@ static int rtp_data_cb(struct bsc_fd *fd, unsigned int what) | |||||||
| 	rc = recvfrom(fd->fd, &buf, sizeof(buf), 0, | 	rc = recvfrom(fd->fd, &buf, sizeof(buf), 0, | ||||||
| 			    (struct sockaddr *) &addr, &slen); | 			    (struct sockaddr *) &addr, &slen); | ||||||
| 	if (rc < 0) { | 	if (rc < 0) { | ||||||
| 		LOGP(DMGCP, LOGL_ERROR, "Failed to receive message on: 0x%x\n", | 		LOGP(DMGCP, LOGL_ERROR, "Failed to receive message on: 0x%x errno: %d/%s\n", | ||||||
| 			ENDPOINT_NUMBER(endp)); | 			ENDPOINT_NUMBER(endp), errno, strerror(errno)); | ||||||
| 		return -1; | 		return -1; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	/* do not forward aynthing... maybe there is a packet from the bts */ | 	/* do not forward aynthing... maybe there is a packet from the bts */ | ||||||
| 	if (endp->ci == CI_UNUSED) { | 	if (endp->ci == CI_UNUSED) { | ||||||
| 		LOGP(DMGCP, LOGL_ERROR, "Unknown message on endpoint: 0x%x\n", ENDPOINT_NUMBER(endp)); | 		LOGP(DMGCP, LOGL_DEBUG, "Unknown message on endpoint: 0x%x\n", ENDPOINT_NUMBER(endp)); | ||||||
| 		return -1; | 		return -1; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -146,7 +150,9 @@ static int rtp_data_cb(struct bsc_fd *fd, unsigned int what) | |||||||
| 	/* We have no idea who called us, maybe it is the BTS. */ | 	/* We have no idea who called us, maybe it is the BTS. */ | ||||||
| 	if (dest == DEST_NETWORK && (endp->bts_rtp == 0 || cfg->forward_ip)) { | 	if (dest == DEST_NETWORK && (endp->bts_rtp == 0 || cfg->forward_ip)) { | ||||||
| 		/* it was the BTS... */ | 		/* it was the BTS... */ | ||||||
| 		if (!cfg->bts_ip || memcmp(&addr.sin_addr, &cfg->bts_in, sizeof(cfg->bts_in)) == 0) { | 		if (!cfg->bts_ip | ||||||
|  | 		    || memcmp(&addr.sin_addr, &cfg->bts_in, sizeof(cfg->bts_in)) == 0 | ||||||
|  | 		    || memcmp(&addr.sin_addr, &endp->bts, sizeof(endp->bts)) == 0) { | ||||||
| 			if (fd == &endp->local_rtp) { | 			if (fd == &endp->local_rtp) { | ||||||
| 				endp->bts_rtp = addr.sin_port; | 				endp->bts_rtp = addr.sin_port; | ||||||
| 			} else { | 			} else { | ||||||
| @@ -159,6 +165,12 @@ static int rtp_data_cb(struct bsc_fd *fd, unsigned int what) | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	/* do this before the loop handling */ | ||||||
|  | 	if (dest == DEST_NETWORK) | ||||||
|  | 		++endp->in_bts; | ||||||
|  | 	else | ||||||
|  | 		++endp->in_remote; | ||||||
|  |  | ||||||
| 	/* dispatch */ | 	/* dispatch */ | ||||||
| 	if (cfg->audio_loop) | 	if (cfg->audio_loop) | ||||||
| 		dest = !dest; | 		dest = !dest; | ||||||
|   | |||||||
| @@ -80,11 +80,6 @@ enum mgcp_connection_mode { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  |  | ||||||
| struct mgcp_msg_ptr { |  | ||||||
| 	unsigned int start; |  | ||||||
| 	unsigned int length; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| struct mgcp_request { | struct mgcp_request { | ||||||
| 	char *name; | 	char *name; | ||||||
| 	struct msgb *(*handle_request) (struct mgcp_config *cfg, struct msgb *msg); | 	struct msgb *(*handle_request) (struct mgcp_config *cfg, struct msgb *msg); | ||||||
| @@ -98,6 +93,7 @@ static struct msgb *handle_audit_endpoint(struct mgcp_config *cfg, struct msgb * | |||||||
| static struct msgb *handle_create_con(struct mgcp_config *cfg, struct msgb *msg); | static struct msgb *handle_create_con(struct mgcp_config *cfg, struct msgb *msg); | ||||||
| static struct msgb *handle_delete_con(struct mgcp_config *cfg, struct msgb *msg); | static struct msgb *handle_delete_con(struct mgcp_config *cfg, struct msgb *msg); | ||||||
| static struct msgb *handle_modify_con(struct mgcp_config *cfg, struct msgb *msg); | static struct msgb *handle_modify_con(struct mgcp_config *cfg, struct msgb *msg); | ||||||
|  | static struct msgb *handle_rsip(struct mgcp_config *cfg, struct msgb *msg); | ||||||
|  |  | ||||||
| static int generate_call_id(struct mgcp_config *cfg) | static int generate_call_id(struct mgcp_config *cfg) | ||||||
| { | { | ||||||
| @@ -119,12 +115,6 @@ static int generate_call_id(struct mgcp_config *cfg) | |||||||
| 	return cfg->last_call_id; | 	return cfg->last_call_id; | ||||||
| } | } | ||||||
|  |  | ||||||
| /* FIXIME/TODO: need to have a list of pending transactions and check that */ |  | ||||||
| static unsigned int generate_transaction_id() |  | ||||||
| { |  | ||||||
| 	return abs(rand()); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /* | /* | ||||||
|  * array of function pointers for handling various |  * array of function pointers for handling various | ||||||
|  * messages. In the future this might be binary sorted |  * messages. In the future this might be binary sorted | ||||||
| @@ -135,6 +125,9 @@ static const struct mgcp_request mgcp_requests [] = { | |||||||
| 	MGCP_REQUEST("CRCX", handle_create_con, "CreateConnection") | 	MGCP_REQUEST("CRCX", handle_create_con, "CreateConnection") | ||||||
| 	MGCP_REQUEST("DLCX", handle_delete_con, "DeleteConnection") | 	MGCP_REQUEST("DLCX", handle_delete_con, "DeleteConnection") | ||||||
| 	MGCP_REQUEST("MDCX", handle_modify_con, "ModifiyConnection") | 	MGCP_REQUEST("MDCX", handle_modify_con, "ModifiyConnection") | ||||||
|  |  | ||||||
|  | 	/* SPEC extension */ | ||||||
|  | 	MGCP_REQUEST("RSIP", handle_rsip, "ReSetInProgress") | ||||||
| }; | }; | ||||||
|  |  | ||||||
| static struct msgb *mgcp_msgb_alloc(void) | static struct msgb *mgcp_msgb_alloc(void) | ||||||
| @@ -194,23 +187,6 @@ static struct msgb *create_response_with_sdp(struct mgcp_endpoint *endp, | |||||||
| 	return mgcp_create_response_with_data(200, msg, trans_id, sdp_record); | 	return mgcp_create_response_with_data(200, msg, trans_id, sdp_record); | ||||||
| } | } | ||||||
|  |  | ||||||
| /* send a static record */ |  | ||||||
| struct msgb *mgcp_create_rsip(void) |  | ||||||
| { |  | ||||||
| 	struct msgb *msg; |  | ||||||
| 	int len; |  | ||||||
|  |  | ||||||
| 	msg = mgcp_msgb_alloc(); |  | ||||||
| 	if (!msg) |  | ||||||
| 		return NULL; |  | ||||||
|  |  | ||||||
| 	len = snprintf((char *) msg->data, 2048, |  | ||||||
| 			"RSIP %u *@mgw MGCP 1.0\n" |  | ||||||
| 			"RM: restart\n", generate_transaction_id()); |  | ||||||
| 	msg->l2h = msgb_put(msg, len); |  | ||||||
| 	return msg; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /* | /* | ||||||
|  * handle incoming messages: |  * handle incoming messages: | ||||||
|  *   - this can be a command (four letters, space, transaction id) |  *   - this can be a command (four letters, space, transaction id) | ||||||
| @@ -221,25 +197,25 @@ struct msgb *mgcp_handle_message(struct mgcp_config *cfg, struct msgb *msg) | |||||||
|         int code; |         int code; | ||||||
| 	struct msgb *resp = NULL; | 	struct msgb *resp = NULL; | ||||||
|  |  | ||||||
| 	if (msg->len < 4) { | 	if (msgb_l2len(msg) < 4) { | ||||||
| 		LOGP(DMGCP, LOGL_ERROR, "mgs too short: %d\n", msg->len); | 		LOGP(DMGCP, LOGL_ERROR, "mgs too short: %d\n", msg->len); | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|         /* attempt to treat it as a response */ |         /* attempt to treat it as a response */ | ||||||
|         if (sscanf((const char *)&msg->data[0], "%3d %*s", &code) == 1) { |         if (sscanf((const char *)&msg->l2h[0], "%3d %*s", &code) == 1) { | ||||||
| 		LOGP(DMGCP, LOGL_DEBUG, "Response: Code: %d\n", code); | 		LOGP(DMGCP, LOGL_DEBUG, "Response: Code: %d\n", code); | ||||||
| 	} else { | 	} else { | ||||||
| 		int i, handled = 0; | 		int i, handled = 0; | ||||||
| 		msg->l3h = &msg->l2h[4]; | 		msg->l3h = &msg->l2h[4]; | ||||||
| 		for (i = 0; i < ARRAY_SIZE(mgcp_requests); ++i) | 		for (i = 0; i < ARRAY_SIZE(mgcp_requests); ++i) | ||||||
| 			if (strncmp(mgcp_requests[i].name, (const char *) &msg->data[0], 4) == 0) { | 			if (strncmp(mgcp_requests[i].name, (const char *) &msg->l2h[0], 4) == 0) { | ||||||
| 				handled = 1; | 				handled = 1; | ||||||
| 				resp = mgcp_requests[i].handle_request(cfg, msg); | 				resp = mgcp_requests[i].handle_request(cfg, msg); | ||||||
| 				break; | 				break; | ||||||
| 			} | 			} | ||||||
| 		if (!handled) { | 		if (!handled) { | ||||||
| 			LOGP(DMGCP, LOGL_NOTICE, "MSG with type: '%.4s' not handled\n", &msg->data[0]); | 			LOGP(DMGCP, LOGL_NOTICE, "MSG with type: '%.4s' not handled\n", &msg->l2h[0]); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -296,9 +272,9 @@ static struct mgcp_endpoint *find_endpoint(struct mgcp_config *cfg, const char * | |||||||
| 	return &cfg->endpoints[gw]; | 	return &cfg->endpoints[gw]; | ||||||
| } | } | ||||||
|  |  | ||||||
| static int analyze_header(struct mgcp_config *cfg, struct msgb *msg, | int mgcp_analyze_header(struct mgcp_config *cfg, struct msgb *msg, | ||||||
| 			  struct mgcp_msg_ptr *ptr, int size, | 			struct mgcp_msg_ptr *ptr, int size, | ||||||
| 			  const char **transaction_id, struct mgcp_endpoint **endp) | 			const char **transaction_id, struct mgcp_endpoint **endp) | ||||||
| { | { | ||||||
| 	int found; | 	int found; | ||||||
|  |  | ||||||
| @@ -334,8 +310,11 @@ static int analyze_header(struct mgcp_config *cfg, struct msgb *msg, | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	*transaction_id = (const char *)&msg->l3h[ptr[0].start]; | 	*transaction_id = (const char *)&msg->l3h[ptr[0].start]; | ||||||
| 	*endp = find_endpoint(cfg, (const char *)&msg->l3h[ptr[1].start]); | 	if (endp) { | ||||||
| 	return *endp == NULL; | 		*endp = find_endpoint(cfg, (const char *)&msg->l3h[ptr[1].start]); | ||||||
|  | 		return *endp == NULL; | ||||||
|  | 	} | ||||||
|  | 	return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| static int verify_call_id(const struct mgcp_endpoint *endp, | static int verify_call_id(const struct mgcp_endpoint *endp, | ||||||
| @@ -369,7 +348,7 @@ static struct msgb *handle_audit_endpoint(struct mgcp_config *cfg, struct msgb * | |||||||
| 	const char *trans_id; | 	const char *trans_id; | ||||||
| 	struct mgcp_endpoint *endp; | 	struct mgcp_endpoint *endp; | ||||||
|  |  | ||||||
| 	found = analyze_header(cfg, msg, data_ptrs, ARRAY_SIZE(data_ptrs), &trans_id, &endp); | 	found = mgcp_analyze_header(cfg, msg, data_ptrs, ARRAY_SIZE(data_ptrs), &trans_id, &endp); | ||||||
| 	if (found != 0) | 	if (found != 0) | ||||||
| 	    response = 500; | 	    response = 500; | ||||||
| 	else | 	else | ||||||
| @@ -402,13 +381,19 @@ static struct msgb *handle_create_con(struct mgcp_config *cfg, struct msgb *msg) | |||||||
| 	int error_code = 500; | 	int error_code = 500; | ||||||
| 	int port; | 	int port; | ||||||
|  |  | ||||||
| 	found = analyze_header(cfg, msg, data_ptrs, ARRAY_SIZE(data_ptrs), &trans_id, &endp); | 	found = mgcp_analyze_header(cfg, msg, data_ptrs, ARRAY_SIZE(data_ptrs), &trans_id, &endp); | ||||||
| 	if (found != 0) | 	if (found != 0) | ||||||
| 		return create_response(500, "CRCX", trans_id); | 		return create_response(500, "CRCX", trans_id); | ||||||
|  |  | ||||||
| 	if (endp->ci != CI_UNUSED) { | 	if (endp->ci != CI_UNUSED) { | ||||||
| 		LOGP(DMGCP, LOGL_ERROR, "Endpoint is already used. 0x%x\n", ENDPOINT_NUMBER(endp)); | 		if (cfg->force_realloc) { | ||||||
| 		return create_response(500, "CRCX", trans_id); | 			LOGP(DMGCP, LOGL_NOTICE, "Endpoint 0x%x already allocated. Forcing realloc.\n", | ||||||
|  | 			    ENDPOINT_NUMBER(endp)); | ||||||
|  | 		} else { | ||||||
|  | 			LOGP(DMGCP, LOGL_ERROR, "Endpoint is already used. 0x%x\n", | ||||||
|  | 			     ENDPOINT_NUMBER(endp)); | ||||||
|  | 			return create_response(500, "CRCX", trans_id); | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	/* parse CallID C: and LocalParameters L: */ | 	/* parse CallID C: and LocalParameters L: */ | ||||||
| @@ -501,7 +486,7 @@ static struct msgb *handle_modify_con(struct mgcp_config *cfg, struct msgb *msg) | |||||||
| 	struct mgcp_endpoint *endp; | 	struct mgcp_endpoint *endp; | ||||||
| 	int error_code = 500; | 	int error_code = 500; | ||||||
|  |  | ||||||
| 	found = analyze_header(cfg, msg, data_ptrs, ARRAY_SIZE(data_ptrs), &trans_id, &endp); | 	found = mgcp_analyze_header(cfg, msg, data_ptrs, ARRAY_SIZE(data_ptrs), &trans_id, &endp); | ||||||
| 	if (found != 0) | 	if (found != 0) | ||||||
| 		return create_response(error_code, "MDCX", trans_id); | 		return create_response(error_code, "MDCX", trans_id); | ||||||
|  |  | ||||||
| @@ -614,7 +599,7 @@ static struct msgb *handle_delete_con(struct mgcp_config *cfg, struct msgb *msg) | |||||||
| 	struct mgcp_endpoint *endp; | 	struct mgcp_endpoint *endp; | ||||||
| 	int error_code = 500; | 	int error_code = 500; | ||||||
|  |  | ||||||
| 	found = analyze_header(cfg, msg, data_ptrs, ARRAY_SIZE(data_ptrs), &trans_id, &endp); | 	found = mgcp_analyze_header(cfg, msg, data_ptrs, ARRAY_SIZE(data_ptrs), &trans_id, &endp); | ||||||
| 	if (found != 0) | 	if (found != 0) | ||||||
| 		return create_response(error_code, "DLCX", trans_id); | 		return create_response(error_code, "DLCX", trans_id); | ||||||
|  |  | ||||||
| @@ -678,6 +663,13 @@ error3: | |||||||
| 	return create_response(error_code, "DLCX", trans_id); | 	return create_response(error_code, "DLCX", trans_id); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | static struct msgb *handle_rsip(struct mgcp_config *cfg, struct msgb *msg) | ||||||
|  | { | ||||||
|  | 	if (cfg->reset_cb) | ||||||
|  | 		cfg->reset_cb(cfg); | ||||||
|  | 	return NULL; | ||||||
|  | } | ||||||
|  |  | ||||||
| struct mgcp_config *mgcp_config_alloc(void) | struct mgcp_config *mgcp_config_alloc(void) | ||||||
| { | { | ||||||
| 	struct mgcp_config *cfg; | 	struct mgcp_config *cfg; | ||||||
| @@ -722,7 +714,7 @@ int mgcp_endpoints_allocate(struct mgcp_config *cfg) | |||||||
|  |  | ||||||
| void mgcp_free_endp(struct mgcp_endpoint *endp) | void mgcp_free_endp(struct mgcp_endpoint *endp) | ||||||
| { | { | ||||||
| 	LOGP(DMGCP, LOGL_NOTICE, "Deleting endpoint on: 0x%x\n", ENDPOINT_NUMBER(endp)); | 	LOGP(DMGCP, LOGL_DEBUG, "Deleting endpoint on: 0x%x\n", ENDPOINT_NUMBER(endp)); | ||||||
| 	endp->ci= CI_UNUSED; | 	endp->ci= CI_UNUSED; | ||||||
|  |  | ||||||
| 	if (endp->callid) { | 	if (endp->callid) { | ||||||
| @@ -732,7 +724,7 @@ void mgcp_free_endp(struct mgcp_endpoint *endp) | |||||||
|  |  | ||||||
| 	if (endp->local_options) { | 	if (endp->local_options) { | ||||||
| 		talloc_free(endp->local_options); | 		talloc_free(endp->local_options); | ||||||
| 		endp->callid = NULL; | 		endp->local_options = NULL; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if (!endp->cfg->early_bind) { | 	if (!endp->cfg->early_bind) { | ||||||
| @@ -742,4 +734,7 @@ void mgcp_free_endp(struct mgcp_endpoint *endp) | |||||||
|  |  | ||||||
| 	endp->net_rtp = endp->net_rtcp = endp->bts_rtp = endp->bts_rtcp = 0; | 	endp->net_rtp = endp->net_rtcp = endp->bts_rtp = endp->bts_rtcp = 0; | ||||||
| 	endp->net_payload_type = endp->bts_payload_type = -1; | 	endp->net_payload_type = endp->bts_payload_type = -1; | ||||||
|  | 	endp->in_bts = endp->in_remote = 0; | ||||||
|  | 	memset(&endp->remote, 0, sizeof(endp->remote)); | ||||||
|  | 	memset(&endp->bts, 0, sizeof(endp->bts)); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -63,6 +63,8 @@ static int config_write_mgcp(struct vty *vty) | |||||||
| 		vty_out(vty, " forward audio ip %s%s", g_cfg->forward_ip, VTY_NEWLINE); | 		vty_out(vty, " forward audio ip %s%s", g_cfg->forward_ip, VTY_NEWLINE); | ||||||
| 	if (g_cfg->forward_port != 0) | 	if (g_cfg->forward_port != 0) | ||||||
| 		vty_out(vty, " forward audio port %d%s", g_cfg->forward_port, VTY_NEWLINE); | 		vty_out(vty, " forward audio port %d%s", g_cfg->forward_port, VTY_NEWLINE); | ||||||
|  | 	if (g_cfg->call_agent_addr) | ||||||
|  | 		vty_out(vty, " call agent ip %s%s", g_cfg->call_agent_addr, VTY_NEWLINE); | ||||||
|  |  | ||||||
| 	return CMD_SUCCESS; | 	return CMD_SUCCESS; | ||||||
| } | } | ||||||
| @@ -75,10 +77,12 @@ DEFUN(show_mcgp, show_mgcp_cmd, "show mgcp", | |||||||
| 	vty_out(vty, "MGCP is up and running with %u endpoints:%s", g_cfg->number_endpoints - 1, VTY_NEWLINE); | 	vty_out(vty, "MGCP is up and running with %u endpoints:%s", g_cfg->number_endpoints - 1, VTY_NEWLINE); | ||||||
| 	for (i = 1; i < g_cfg->number_endpoints; ++i) { | 	for (i = 1; i < g_cfg->number_endpoints; ++i) { | ||||||
| 		struct mgcp_endpoint *endp = &g_cfg->endpoints[i]; | 		struct mgcp_endpoint *endp = &g_cfg->endpoints[i]; | ||||||
| 		vty_out(vty, " Endpoint 0x%.2x: CI: %d net: %u/%u bts: %u/%u%s", | 		vty_out(vty, " Endpoint 0x%.2x: CI: %d net: %u/%u bts: %u/%u on %s traffic in :%u/%u%s", | ||||||
| 			i, endp->ci, | 			i, endp->ci, | ||||||
| 			ntohs(endp->net_rtp), ntohs(endp->net_rtcp), | 			ntohs(endp->net_rtp), ntohs(endp->net_rtcp), | ||||||
| 			ntohs(endp->bts_rtp), ntohs(endp->bts_rtcp), VTY_NEWLINE); | 			ntohs(endp->bts_rtp), ntohs(endp->bts_rtcp), | ||||||
|  | 			inet_ntoa(endp->bts), endp->in_bts, endp->in_remote, | ||||||
|  | 			VTY_NEWLINE); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return CMD_SUCCESS; | 	return CMD_SUCCESS; | ||||||
| @@ -237,6 +241,17 @@ DEFUN(cfg_mgcp_forward_port, | |||||||
| 	return CMD_SUCCESS; | 	return CMD_SUCCESS; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | DEFUN(cfg_mgcp_agent_addr, | ||||||
|  |       cfg_mgcp_agent_addr_cmd, | ||||||
|  |       "call agent ip IP", | ||||||
|  |       "Set the address of the call agent.") | ||||||
|  | { | ||||||
|  | 	if (g_cfg->call_agent_addr) | ||||||
|  | 		talloc_free(g_cfg->call_agent_addr); | ||||||
|  | 	g_cfg->call_agent_addr = talloc_strdup(g_cfg, argv[0]); | ||||||
|  | 	return CMD_SUCCESS; | ||||||
|  | } | ||||||
|  |  | ||||||
| int mgcp_vty_init(void) | int mgcp_vty_init(void) | ||||||
| { | { | ||||||
| 	install_element(VIEW_NODE, &show_mgcp_cmd); | 	install_element(VIEW_NODE, &show_mgcp_cmd); | ||||||
| @@ -256,6 +271,7 @@ int mgcp_vty_init(void) | |||||||
| 	install_element(MGCP_NODE, &cfg_mgcp_number_endp_cmd); | 	install_element(MGCP_NODE, &cfg_mgcp_number_endp_cmd); | ||||||
| 	install_element(MGCP_NODE, &cfg_mgcp_forward_ip_cmd); | 	install_element(MGCP_NODE, &cfg_mgcp_forward_ip_cmd); | ||||||
| 	install_element(MGCP_NODE, &cfg_mgcp_forward_port_cmd); | 	install_element(MGCP_NODE, &cfg_mgcp_forward_port_cmd); | ||||||
|  | 	install_element(MGCP_NODE, &cfg_mgcp_agent_addr_cmd); | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -274,6 +290,11 @@ int mgcp_parse_config(const char *config_file, struct mgcp_config *cfg) | |||||||
| 	if (!g_cfg->bts_ip) | 	if (!g_cfg->bts_ip) | ||||||
| 		fprintf(stderr, "No BTS ip address specified. This will allow everyone to connect.\n"); | 		fprintf(stderr, "No BTS ip address specified. This will allow everyone to connect.\n"); | ||||||
|  |  | ||||||
|  | 	if (!g_cfg->source_addr) { | ||||||
|  | 		fprintf(stderr, "You need to specify a bind address.\n"); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	if (mgcp_endpoints_allocate(g_cfg) != 0) { | 	if (mgcp_endpoints_allocate(g_cfg) != 0) { | ||||||
| 		fprintf(stderr, "Failed to allocate endpoints: %d. Quitting.\n", g_cfg->number_endpoints); | 		fprintf(stderr, "Failed to allocate endpoints: %d. Quitting.\n", g_cfg->number_endpoints); | ||||||
| 		return -1; | 		return -1; | ||||||
| @@ -327,13 +348,3 @@ int mgcp_parse_config(const char *config_file, struct mgcp_config *cfg) | |||||||
| 	return !!g_cfg->forward_ip; | 	return !!g_cfg->forward_ip; | ||||||
| } | } | ||||||
|  |  | ||||||
| struct gsm_network; |  | ||||||
| int bsc_vty_init(struct gsm_network *dummy) |  | ||||||
| { |  | ||||||
| 	cmd_init(1); |  | ||||||
| 	vty_init(); |  | ||||||
|  |  | ||||||
|         mgcp_vty_init(); |  | ||||||
| 	return 0; |  | ||||||
| } |  | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										216
									
								
								openbsc/src/nat/bsc_filter.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										216
									
								
								openbsc/src/nat/bsc_filter.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,216 @@ | |||||||
|  | /* BSC Multiplexer/NAT */ | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * (C) 2010 by Holger Hans Peter Freyther <zecke@selfish.org> | ||||||
|  |  * (C) 2010 by On-Waves | ||||||
|  |  * All Rights Reserved | ||||||
|  |  * | ||||||
|  |  * This program is free software; you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation; either version 2 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License along | ||||||
|  |  * with this program; if not, write to the Free Software Foundation, Inc., | ||||||
|  |  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #include <openbsc/bsc_nat.h> | ||||||
|  | #include <openbsc/bssap.h> | ||||||
|  | #include <openbsc/ipaccess.h> | ||||||
|  | #include <openbsc/debug.h> | ||||||
|  |  | ||||||
|  | #include <osmocore/talloc.h> | ||||||
|  |  | ||||||
|  | #include <sccp/sccp.h> | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * The idea is to have a simple struct describing a IPA packet with | ||||||
|  |  * SCCP SSN and the GSM 08.08 payload and decide. We will both have | ||||||
|  |  * a white and a blacklist of packets we want to handle. | ||||||
|  |  * | ||||||
|  |  * TODO: Implement a "NOT" in the filter language. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #define ALLOW_ANY -1 | ||||||
|  |  | ||||||
|  | #define FILTER_TO_BSC	1 | ||||||
|  | #define FILTER_TO_MSC	2 | ||||||
|  | #define FILTER_TO_BOTH	3 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | struct bsc_pkt_filter { | ||||||
|  | 	int ipa_proto; | ||||||
|  | 	int dest_ssn; | ||||||
|  | 	int bssap; | ||||||
|  | 	int gsm; | ||||||
|  | 	int filter_dir; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | static struct bsc_pkt_filter black_list[] = { | ||||||
|  | 	/* filter reset messages to the MSC */ | ||||||
|  | 	{ IPAC_PROTO_SCCP, SCCP_SSN_BSSAP, 0, BSS_MAP_MSG_RESET, FILTER_TO_MSC }, | ||||||
|  |  | ||||||
|  | 	/* filter reset ack messages to the BSC */ | ||||||
|  | 	{ IPAC_PROTO_SCCP, SCCP_SSN_BSSAP, 0, BSS_MAP_MSG_RESET_ACKNOWLEDGE, FILTER_TO_BSC }, | ||||||
|  |  | ||||||
|  | 	/* filter ip access */ | ||||||
|  | 	{ IPAC_PROTO_IPACCESS, ALLOW_ANY, ALLOW_ANY, ALLOW_ANY, FILTER_TO_MSC }, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | static struct bsc_pkt_filter white_list[] = { | ||||||
|  | 	/* allow IPAC_PROTO_SCCP messages to both sides */ | ||||||
|  | 	{ IPAC_PROTO_SCCP, ALLOW_ANY, ALLOW_ANY, ALLOW_ANY, FILTER_TO_BOTH }, | ||||||
|  |  | ||||||
|  | 	/* allow MGCP messages to both sides */ | ||||||
|  | 	{ NAT_IPAC_PROTO_MGCP, ALLOW_ANY, ALLOW_ANY, ALLOW_ANY, FILTER_TO_BOTH }, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | struct bsc_nat_parsed* bsc_nat_parse(struct msgb *msg) | ||||||
|  | { | ||||||
|  | 	struct sccp_parse_result result; | ||||||
|  | 	struct bsc_nat_parsed *parsed; | ||||||
|  | 	struct ipaccess_head *hh; | ||||||
|  |  | ||||||
|  | 	/* quick fail */ | ||||||
|  | 	if (msg->len < 4) | ||||||
|  | 		return NULL; | ||||||
|  |  | ||||||
|  | 	parsed = talloc_zero(msg, struct bsc_nat_parsed); | ||||||
|  | 	if (!parsed) | ||||||
|  | 		return NULL; | ||||||
|  |  | ||||||
|  | 	/* more init */ | ||||||
|  | 	parsed->ipa_proto = parsed->called_ssn = parsed->calling_ssn = -1; | ||||||
|  | 	parsed->sccp_type = parsed->bssap = parsed->gsm_type = -1; | ||||||
|  |  | ||||||
|  | 	/* start parsing */ | ||||||
|  | 	hh = (struct ipaccess_head *) msg->data; | ||||||
|  | 	parsed->ipa_proto = hh->proto; | ||||||
|  |  | ||||||
|  | 	msg->l2h = &hh->data[0]; | ||||||
|  |  | ||||||
|  | 	/* do a size check on the input */ | ||||||
|  | 	if (ntohs(hh->len) != msgb_l2len(msg)) { | ||||||
|  | 		LOGP(DINP, LOGL_ERROR, "Wrong input length?\n"); | ||||||
|  | 		talloc_free(parsed); | ||||||
|  | 		return NULL; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* analyze sccp down here */ | ||||||
|  | 	if (parsed->ipa_proto == IPAC_PROTO_SCCP) { | ||||||
|  | 		memset(&result, 0, sizeof(result)); | ||||||
|  | 		if (sccp_parse_header(msg, &result) != 0) { | ||||||
|  | 			talloc_free(parsed); | ||||||
|  | 			return 0; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (msg->l3h && msgb_l3len(msg) < 3) { | ||||||
|  | 			LOGP(DNAT, LOGL_ERROR, "Not enough space or GSM payload\n"); | ||||||
|  | 			talloc_free(parsed); | ||||||
|  | 			return 0; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		parsed->sccp_type = sccp_determine_msg_type(msg); | ||||||
|  | 		parsed->src_local_ref = result.source_local_reference; | ||||||
|  | 		parsed->dest_local_ref = result.destination_local_reference; | ||||||
|  | 		parsed->called_ssn = result.called.ssn; | ||||||
|  | 		parsed->calling_ssn = result.calling.ssn; | ||||||
|  |  | ||||||
|  | 		/* in case of connection confirm we have no payload */ | ||||||
|  | 		if (msg->l3h) { | ||||||
|  | 			parsed->bssap = msg->l3h[0]; | ||||||
|  | 			parsed->gsm_type = msg->l3h[2]; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return parsed; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int bsc_nat_filter_ipa(int dir, struct msgb *msg, struct bsc_nat_parsed *parsed) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  |  | ||||||
|  | 	/* go through the blacklist now */ | ||||||
|  | 	for (i = 0; i < ARRAY_SIZE(black_list); ++i) { | ||||||
|  | 		/* ignore the rule? */ | ||||||
|  | 		if (black_list[i].filter_dir != FILTER_TO_BOTH | ||||||
|  | 		    && black_list[i].filter_dir != dir) | ||||||
|  | 			continue; | ||||||
|  |  | ||||||
|  | 		/* the proto is not blacklisted */ | ||||||
|  | 		if (black_list[i].ipa_proto != ALLOW_ANY | ||||||
|  | 		    && black_list[i].ipa_proto != parsed->ipa_proto) | ||||||
|  | 			continue; | ||||||
|  |  | ||||||
|  | 		if (parsed->ipa_proto == IPAC_PROTO_SCCP) { | ||||||
|  | 			/* the SSN is not blacklisted */ | ||||||
|  | 			if (black_list[i].dest_ssn != ALLOW_ANY | ||||||
|  | 			    && black_list[i].dest_ssn != parsed->called_ssn) | ||||||
|  | 				continue; | ||||||
|  |  | ||||||
|  | 			/* bssap */ | ||||||
|  | 			if (black_list[i].bssap != ALLOW_ANY | ||||||
|  | 			    && black_list[i].bssap != parsed->bssap) | ||||||
|  | 				continue; | ||||||
|  |  | ||||||
|  | 			/* gsm */ | ||||||
|  | 			if (black_list[i].gsm != ALLOW_ANY | ||||||
|  | 			    && black_list[i].gsm != parsed->gsm_type) | ||||||
|  | 				continue; | ||||||
|  |  | ||||||
|  | 			/* blacklisted */ | ||||||
|  | 			LOGP(DNAT, LOGL_INFO, "Blacklisted with rule %d\n", i); | ||||||
|  | 			return 1; | ||||||
|  | 		} else { | ||||||
|  | 			/* blacklisted, we have no content sniffing yet */ | ||||||
|  | 			LOGP(DNAT, LOGL_INFO, "Blacklisted with rule %d\n", i); | ||||||
|  | 			return 1; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* go through the whitelust now */ | ||||||
|  | 	for (i = 0; i < ARRAY_SIZE(white_list); ++i) { | ||||||
|  | 		/* ignore the rule? */ | ||||||
|  | 		if (white_list[i].filter_dir != FILTER_TO_BOTH | ||||||
|  | 		    && white_list[i].filter_dir != dir) | ||||||
|  | 			continue; | ||||||
|  |  | ||||||
|  | 		/* the proto is not whitelisted */ | ||||||
|  | 		if (white_list[i].ipa_proto != ALLOW_ANY | ||||||
|  | 		    && white_list[i].ipa_proto != parsed->ipa_proto) | ||||||
|  | 			continue; | ||||||
|  |  | ||||||
|  | 		if (parsed->ipa_proto == IPAC_PROTO_SCCP) { | ||||||
|  | 			/* the SSN is not whitelisted */ | ||||||
|  | 			if (white_list[i].dest_ssn != ALLOW_ANY | ||||||
|  | 			    && white_list[i].dest_ssn != parsed->called_ssn) | ||||||
|  | 				continue; | ||||||
|  |  | ||||||
|  | 			/* bssap */ | ||||||
|  | 			if (white_list[i].bssap != ALLOW_ANY | ||||||
|  | 			    && white_list[i].bssap != parsed->bssap) | ||||||
|  | 				continue; | ||||||
|  |  | ||||||
|  | 			/* gsm */ | ||||||
|  | 			if (white_list[i].gsm != ALLOW_ANY | ||||||
|  | 			    && white_list[i].gsm != parsed->gsm_type) | ||||||
|  | 				continue; | ||||||
|  |  | ||||||
|  | 			/* whitelisted */ | ||||||
|  | 			LOGP(DNAT, LOGL_INFO, "Whitelisted with rule %d\n", i); | ||||||
|  | 			return 0; | ||||||
|  | 		} else { | ||||||
|  | 			/* whitelisted */ | ||||||
|  | 			return 0; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return 1; | ||||||
|  | } | ||||||
							
								
								
									
										484
									
								
								openbsc/src/nat/bsc_mgcp_utils.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										484
									
								
								openbsc/src/nat/bsc_mgcp_utils.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,484 @@ | |||||||
|  | /* | ||||||
|  |  * (C) 2010 by Holger Hans Peter Freyther <zecke@selfish.org> | ||||||
|  |  * (C) 2010 by On-Waves | ||||||
|  |  * All Rights Reserved | ||||||
|  |  * | ||||||
|  |  * This program is free software; you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation; either version 2 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License along | ||||||
|  |  * with this program; if not, write to the Free Software Foundation, Inc., | ||||||
|  |  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #include <openbsc/bsc_nat.h> | ||||||
|  | #include <openbsc/gsm_data.h> | ||||||
|  | #include <openbsc/bssap.h> | ||||||
|  | #include <openbsc/debug.h> | ||||||
|  | #include <openbsc/mgcp.h> | ||||||
|  | #include <openbsc/mgcp_internal.h> | ||||||
|  |  | ||||||
|  | #include <osmocore/talloc.h> | ||||||
|  |  | ||||||
|  | #include <netinet/in.h> | ||||||
|  | #include <arpa/inet.h> | ||||||
|  |  | ||||||
|  | #include <errno.h> | ||||||
|  | #include <unistd.h> | ||||||
|  |  | ||||||
|  | int bsc_mgcp_assign(struct sccp_connections *con, struct msgb *msg) | ||||||
|  | { | ||||||
|  | 	struct tlv_parsed tp; | ||||||
|  | 	u_int16_t cic; | ||||||
|  | 	u_int8_t timeslot; | ||||||
|  | 	u_int8_t multiplex; | ||||||
|  |  | ||||||
|  | 	if (!msg->l3h) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Assignment message should have l3h pointer.\n"); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if (msgb_l3len(msg) < 3) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Assignment message has not enough space for GSM0808.\n"); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	tlv_parse(&tp, gsm0808_att_tlvdef(), msg->l3h + 3, msgb_l3len(msg) - 3, 0, 0); | ||||||
|  | 	if (!TLVP_PRESENT(&tp, GSM0808_IE_CIRCUIT_IDENTITY_CODE)) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Circuit identity code not found in assignment message.\n"); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	cic = ntohs(*(u_int16_t *)TLVP_VAL(&tp, GSM0808_IE_CIRCUIT_IDENTITY_CODE)); | ||||||
|  | 	timeslot = cic & 0x1f; | ||||||
|  | 	multiplex = (cic & ~0x1f) >> 5; | ||||||
|  |  | ||||||
|  | 	con->msc_timeslot = (32 * multiplex) + timeslot; | ||||||
|  | 	con->bsc_timeslot = con->msc_timeslot; | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void bsc_mgcp_clear(struct sccp_connections *con) | ||||||
|  | { | ||||||
|  | 	con->msc_timeslot = -1; | ||||||
|  | 	con->bsc_timeslot = -1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void bsc_mgcp_free_endpoint(struct bsc_nat *nat, int i) | ||||||
|  | { | ||||||
|  | 	if (nat->bsc_endpoints[i].transaction_id) { | ||||||
|  | 		talloc_free(nat->bsc_endpoints[i].transaction_id); | ||||||
|  | 		nat->bsc_endpoints[i].transaction_id = NULL; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	nat->bsc_endpoints[i].bsc = NULL; | ||||||
|  | 	mgcp_free_endp(&nat->mgcp_cfg->endpoints[i]); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void bsc_mgcp_free_endpoints(struct bsc_nat *nat) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  |  | ||||||
|  | 	for (i = 1; i < nat->mgcp_cfg->number_endpoints; ++i) | ||||||
|  | 		bsc_mgcp_free_endpoint(nat, i); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | struct bsc_connection *bsc_mgcp_find_con(struct bsc_nat *nat, int endpoint) | ||||||
|  | { | ||||||
|  | 	struct sccp_connections *sccp; | ||||||
|  |  | ||||||
|  | 	llist_for_each_entry(sccp, &nat->sccp_connections, list_entry) { | ||||||
|  | 		if (sccp->msc_timeslot == -1) | ||||||
|  | 			continue; | ||||||
|  | 		if (mgcp_timeslot_to_endpoint(0, sccp->msc_timeslot) != endpoint) | ||||||
|  | 			continue; | ||||||
|  |  | ||||||
|  | 		return sccp->bsc; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	LOGP(DMGCP, LOGL_ERROR, "Failed to find the connection.\n"); | ||||||
|  | 	return NULL; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int bsc_mgcp_policy_cb(struct mgcp_config *cfg, int endpoint, int state, const char *transaction_id) | ||||||
|  | { | ||||||
|  | 	struct bsc_nat *nat; | ||||||
|  | 	struct bsc_endpoint *bsc_endp; | ||||||
|  | 	struct bsc_connection *bsc_con; | ||||||
|  | 	struct mgcp_endpoint *mgcp_endp; | ||||||
|  | 	struct msgb *bsc_msg; | ||||||
|  |  | ||||||
|  | 	nat = cfg->data; | ||||||
|  | 	bsc_endp = &nat->bsc_endpoints[endpoint]; | ||||||
|  | 	mgcp_endp = &nat->mgcp_cfg->endpoints[endpoint]; | ||||||
|  |  | ||||||
|  | 	bsc_con = bsc_mgcp_find_con(nat, endpoint); | ||||||
|  |  | ||||||
|  | 	if (!bsc_con) { | ||||||
|  | 		LOGP(DMGCP, LOGL_ERROR, "Did not find BSC for a new connection on 0x%x for %d\n", endpoint, state); | ||||||
|  |  | ||||||
|  | 		switch (state) { | ||||||
|  | 		case MGCP_ENDP_CRCX: | ||||||
|  | 			return MGCP_POLICY_REJECT; | ||||||
|  | 			break; | ||||||
|  | 		case MGCP_ENDP_DLCX: | ||||||
|  | 			return MGCP_POLICY_CONT; | ||||||
|  | 			break; | ||||||
|  | 		case MGCP_ENDP_MDCX: | ||||||
|  | 			return MGCP_POLICY_CONT; | ||||||
|  | 			break; | ||||||
|  | 		default: | ||||||
|  | 			LOGP(DMGCP, LOGL_FATAL, "Unhandled state: %d\n", state); | ||||||
|  | 			return MGCP_POLICY_CONT; | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if (bsc_endp->transaction_id) { | ||||||
|  | 		LOGP(DMGCP, LOGL_ERROR, "One transaction with id '%s' on 0x%x\n", | ||||||
|  | 		     bsc_endp->transaction_id, endpoint); | ||||||
|  | 		talloc_free(bsc_endp->transaction_id); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* we need to generate a new and patched message */ | ||||||
|  | 	bsc_msg = bsc_mgcp_rewrite((char *) nat->mgcp_msg, nat->mgcp_length, | ||||||
|  | 				   nat->mgcp_cfg->source_addr, mgcp_endp->rtp_port); | ||||||
|  | 	if (!bsc_msg) { | ||||||
|  | 		LOGP(DMGCP, LOGL_ERROR, "Failed to patch the msg.\n"); | ||||||
|  | 		return MGCP_POLICY_CONT; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	bsc_endp->transaction_id = talloc_strdup(nat, transaction_id); | ||||||
|  | 	bsc_endp->bsc = bsc_con; | ||||||
|  | 	bsc_endp->pending_delete = 0; | ||||||
|  |  | ||||||
|  | 	/* we need to update some bits */ | ||||||
|  | 	if (state == MGCP_ENDP_CRCX) { | ||||||
|  | 		struct sockaddr_in sock; | ||||||
|  | 		socklen_t len = sizeof(sock); | ||||||
|  | 		if (getpeername(bsc_con->write_queue.bfd.fd, (struct sockaddr *) &sock, &len) != 0) { | ||||||
|  | 			LOGP(DMGCP, LOGL_ERROR, "Can not get the peername...%d/%s\n", | ||||||
|  | 			      errno, strerror(errno)); | ||||||
|  | 		} else { | ||||||
|  | 			mgcp_endp->bts = sock.sin_addr; | ||||||
|  | 		} | ||||||
|  | 	} else if (state == MGCP_ENDP_DLCX) { | ||||||
|  | 		/* we will free the endpoint now in case the BSS does not respond */ | ||||||
|  | 		bsc_endp->pending_delete = 1; | ||||||
|  | 		mgcp_free_endp(mgcp_endp); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	bsc_write(bsc_con, bsc_msg, NAT_IPAC_PROTO_MGCP); | ||||||
|  | 	return MGCP_POLICY_DEFER; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * We have received a msg from the BSC. We will see if we know | ||||||
|  |  * this transaction and if it belongs to the BSC. Then we will | ||||||
|  |  * need to patch the content to point to the local network and we | ||||||
|  |  * need to update the I: that was assigned by the BSS. | ||||||
|  |  */ | ||||||
|  | void bsc_mgcp_forward(struct bsc_connection *bsc, struct msgb *msg) | ||||||
|  | { | ||||||
|  | 	struct msgb *output; | ||||||
|  | 	struct bsc_endpoint *bsc_endp = NULL; | ||||||
|  | 	struct mgcp_endpoint *endp = NULL; | ||||||
|  | 	int i, code; | ||||||
|  | 	char transaction_id[60]; | ||||||
|  |  | ||||||
|  | 	/* Some assumption that our buffer is big enough.. and null terminate */ | ||||||
|  | 	if (msgb_l2len(msg) > 2000) { | ||||||
|  | 		LOGP(DMGCP, LOGL_ERROR, "MGCP message too long.\n"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	msg->l2h[msgb_l2len(msg)] = '\0'; | ||||||
|  |  | ||||||
|  | 	if (bsc_mgcp_parse_response((const char *) msg->l2h, &code, transaction_id) != 0) { | ||||||
|  | 		LOGP(DMGCP, LOGL_ERROR, "Failed to parse response code.\n"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for (i = 1; i < bsc->nat->mgcp_cfg->number_endpoints; ++i) { | ||||||
|  | 		if (bsc->nat->bsc_endpoints[i].bsc != bsc) | ||||||
|  | 			continue; | ||||||
|  | 		/* no one listening? a bug? */ | ||||||
|  | 		if (!bsc->nat->bsc_endpoints[i].transaction_id) | ||||||
|  | 			continue; | ||||||
|  | 		if (strcmp(transaction_id, bsc->nat->bsc_endpoints[i].transaction_id) != 0) | ||||||
|  | 			continue; | ||||||
|  |  | ||||||
|  | 		endp = &bsc->nat->mgcp_cfg->endpoints[i]; | ||||||
|  | 		bsc_endp = &bsc->nat->bsc_endpoints[i]; | ||||||
|  | 		break; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if (!bsc_endp) { | ||||||
|  | 		LOGP(DMGCP, LOGL_ERROR, "Could not find active endpoint: %s for msg: '%s'\n", | ||||||
|  | 		     transaction_id, (const char *) msg->l2h); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* make it point to our endpoint if it was not deleted */ | ||||||
|  | 	if (bsc_endp->pending_delete) { | ||||||
|  | 		bsc_endp->bsc = NULL; | ||||||
|  | 		bsc_endp->pending_delete = 0; | ||||||
|  | 	} else { | ||||||
|  | 		endp->ci = bsc_mgcp_extract_ci((const char *) msg->l2h); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* free some stuff */ | ||||||
|  | 	talloc_free(bsc_endp->transaction_id); | ||||||
|  | 	bsc_endp->transaction_id = NULL; | ||||||
|  |  | ||||||
|  | 	/* | ||||||
|  | 	 * rewrite the information. In case the endpoint was deleted | ||||||
|  | 	 * there should be nothing for us to rewrite so putting endp->rtp_port | ||||||
|  | 	 * with the value of 0 should be no problem. | ||||||
|  | 	 */ | ||||||
|  | 	output = bsc_mgcp_rewrite((char * ) msg->l2h, msgb_l2len(msg), | ||||||
|  | 				  bsc->nat->mgcp_cfg->source_addr, endp->rtp_port); | ||||||
|  |  | ||||||
|  | 	if (!output) { | ||||||
|  | 		LOGP(DMGCP, LOGL_ERROR, "Failed to rewrite MGCP msg.\n"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if (write_queue_enqueue(&bsc->nat->mgcp_queue, output) != 0) { | ||||||
|  | 		LOGP(DMGCP, LOGL_ERROR, "Failed to queue MGCP msg.\n"); | ||||||
|  | 		msgb_free(output); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int bsc_mgcp_parse_response(const char *str, int *code, char transaction[60]) | ||||||
|  | { | ||||||
|  | 	/* we want to parse two strings */ | ||||||
|  | 	return sscanf(str, "%3d %59s\n", code, transaction) != 2; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int bsc_mgcp_extract_ci(const char *str) | ||||||
|  | { | ||||||
|  | 	int ci; | ||||||
|  | 	char *res = strstr(str, "I: "); | ||||||
|  | 	if (!res) | ||||||
|  | 		return CI_UNUSED; | ||||||
|  |  | ||||||
|  | 	if (sscanf(res, "I: %d", &ci) != 1) | ||||||
|  | 		return CI_UNUSED; | ||||||
|  | 	return ci; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* we need to replace some strings... */ | ||||||
|  | struct msgb *bsc_mgcp_rewrite(char *input, int length, const char *ip, int port) | ||||||
|  | { | ||||||
|  | 	static const char *ip_str = "c=IN IP4 "; | ||||||
|  | 	static const char *aud_str = "m=audio "; | ||||||
|  |  | ||||||
|  | 	char buf[128]; | ||||||
|  | 	char *running, *token; | ||||||
|  | 	struct msgb *output; | ||||||
|  |  | ||||||
|  | 	if (length > 4096 - 128) { | ||||||
|  | 		LOGP(DMGCP, LOGL_ERROR, "Input is too long.\n"); | ||||||
|  | 		return NULL; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	output = msgb_alloc_headroom(4096, 128, "MGCP rewritten"); | ||||||
|  | 	if (!output) { | ||||||
|  | 		LOGP(DMGCP, LOGL_ERROR, "Failed to allocate new MGCP msg.\n"); | ||||||
|  | 		return NULL; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	running = input; | ||||||
|  | 	output->l2h = output->data; | ||||||
|  | 	for (token = strsep(&running, "\n"); running; token = strsep(&running, "\n")) { | ||||||
|  | 		int len = strlen(token); | ||||||
|  | 		int cr = len > 0 && token[len - 1] == '\r'; | ||||||
|  |  | ||||||
|  | 		if (strncmp(ip_str, token, (sizeof ip_str) - 1) == 0) { | ||||||
|  | 			output->l3h = msgb_put(output, strlen(ip_str)); | ||||||
|  | 			memcpy(output->l3h, ip_str, strlen(ip_str)); | ||||||
|  | 			output->l3h = msgb_put(output, strlen(ip)); | ||||||
|  | 			memcpy(output->l3h, ip, strlen(ip)); | ||||||
|  |  | ||||||
|  | 			if (cr) { | ||||||
|  | 				output->l3h = msgb_put(output, 2); | ||||||
|  | 				output->l3h[0] = '\r'; | ||||||
|  | 				output->l3h[1] = '\n'; | ||||||
|  | 			} else { | ||||||
|  | 				output->l3h = msgb_put(output, 1); | ||||||
|  | 				output->l3h[0] = '\n'; | ||||||
|  | 			} | ||||||
|  | 		} else if (strncmp(aud_str, token, (sizeof aud_str) - 1) == 0) { | ||||||
|  | 			int payload; | ||||||
|  | 			if (sscanf(token, "m=audio %*d RTP/AVP %d", &payload) != 1) { | ||||||
|  | 				LOGP(DMGCP, LOGL_ERROR, "Could not parsed audio line.\n"); | ||||||
|  | 				msgb_free(output); | ||||||
|  | 				return NULL; | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			snprintf(buf, sizeof(buf)-1, "m=audio %d RTP/AVP %d%s", | ||||||
|  | 				 port, payload, cr ? "\r\n" : "\n"); | ||||||
|  | 			buf[sizeof(buf)-1] = '\0'; | ||||||
|  |  | ||||||
|  | 			output->l3h = msgb_put(output, strlen(buf)); | ||||||
|  | 			memcpy(output->l3h, buf, strlen(buf)); | ||||||
|  | 		} else { | ||||||
|  | 			output->l3h = msgb_put(output, len + 1); | ||||||
|  | 			memcpy(output->l3h, token, len); | ||||||
|  | 			output->l3h[len] = '\n'; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return output; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static int mgcp_do_read(struct bsc_fd *fd) | ||||||
|  | { | ||||||
|  | 	struct bsc_nat *nat; | ||||||
|  | 	struct msgb *msg, *resp; | ||||||
|  | 	int rc; | ||||||
|  |  | ||||||
|  | 	nat = fd->data; | ||||||
|  |  | ||||||
|  | 	rc = read(fd->fd, nat->mgcp_msg, sizeof(nat->mgcp_msg) - 1); | ||||||
|  | 	if (rc <= 0) { | ||||||
|  | 		LOGP(DMGCP, LOGL_ERROR, "Failed to read errno: %d\n", errno); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	nat->mgcp_msg[rc] = '\0'; | ||||||
|  | 	nat->mgcp_length = rc; | ||||||
|  |  | ||||||
|  | 	msg = msgb_alloc(sizeof(nat->mgcp_msg), "MGCP GW Read"); | ||||||
|  | 	if (!msg) { | ||||||
|  | 		LOGP(DMGCP, LOGL_ERROR, "Failed to create buffer.\n"); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	msg->l2h = msgb_put(msg, rc); | ||||||
|  | 	memcpy(msg->l2h, nat->mgcp_msg, msgb_l2len(msg)); | ||||||
|  | 	resp = mgcp_handle_message(nat->mgcp_cfg, msg); | ||||||
|  | 	msgb_free(msg); | ||||||
|  |  | ||||||
|  | 	/* we do have a direct answer... e.g. AUEP */ | ||||||
|  | 	if (resp) { | ||||||
|  | 		if (write_queue_enqueue(&nat->mgcp_queue, resp) != 0) { | ||||||
|  | 			LOGP(DMGCP, LOGL_ERROR, "Failed to enqueue msg.\n"); | ||||||
|  | 			msgb_free(resp); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static int mgcp_do_write(struct bsc_fd *bfd, struct msgb *msg) | ||||||
|  | { | ||||||
|  | 	int rc; | ||||||
|  |  | ||||||
|  | 	rc = write(bfd->fd, msg->data, msg->len); | ||||||
|  |  | ||||||
|  | 	if (rc != msg->len) { | ||||||
|  | 		LOGP(DMGCP, LOGL_ERROR, "Failed to write msg to MGCP CallAgent.\n"); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return rc; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int bsc_mgcp_init(struct bsc_nat *nat) | ||||||
|  | { | ||||||
|  | 	int on; | ||||||
|  | 	struct sockaddr_in addr; | ||||||
|  |  | ||||||
|  | 	if (!nat->mgcp_cfg->call_agent_addr) { | ||||||
|  | 		LOGP(DMGCP, LOGL_ERROR, "The BSC nat requires the call agent ip to be set.\n"); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if (nat->mgcp_cfg->bts_ip) { | ||||||
|  | 		LOGP(DMGCP, LOGL_ERROR, "Do not set the BTS ip for the nat.\n"); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	nat->mgcp_queue.bfd.fd = socket(AF_INET, SOCK_DGRAM, 0); | ||||||
|  | 	if (nat->mgcp_queue.bfd.fd < 0) { | ||||||
|  | 		LOGP(DMGCP, LOGL_ERROR, "Failed to create MGCP socket. errno: %d\n", errno); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	on = 1; | ||||||
|  | 	setsockopt(nat->mgcp_queue.bfd.fd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)); | ||||||
|  |  | ||||||
|  | 	addr.sin_family = AF_INET; | ||||||
|  | 	addr.sin_port = htons(nat->mgcp_cfg->source_port); | ||||||
|  | 	inet_aton(nat->mgcp_cfg->source_addr, &addr.sin_addr); | ||||||
|  |  | ||||||
|  | 	if (bind(nat->mgcp_queue.bfd.fd, (struct sockaddr *) &addr, sizeof(addr)) < 0) { | ||||||
|  | 		LOGP(DMGCP, LOGL_ERROR, "Failed to bind. errno: %d\n", errno); | ||||||
|  | 		close(nat->mgcp_queue.bfd.fd); | ||||||
|  | 		nat->mgcp_queue.bfd.fd = -1; | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	addr.sin_port = htons(2727); | ||||||
|  | 	inet_aton(nat->mgcp_cfg->call_agent_addr, &addr.sin_addr); | ||||||
|  | 	if (connect(nat->mgcp_queue.bfd.fd, (struct sockaddr *) &addr, sizeof(addr)) < 0) { | ||||||
|  | 		LOGP(DMGCP, LOGL_ERROR, "Failed to connect to: '%s'. errno: %d\n", | ||||||
|  | 		     nat->mgcp_cfg->call_agent_addr, errno); | ||||||
|  | 		close(nat->mgcp_queue.bfd.fd); | ||||||
|  | 		nat->mgcp_queue.bfd.fd = -1; | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	write_queue_init(&nat->mgcp_queue, 10); | ||||||
|  | 	nat->mgcp_queue.bfd.when = BSC_FD_READ; | ||||||
|  | 	nat->mgcp_queue.bfd.data = nat; | ||||||
|  | 	nat->mgcp_queue.read_cb = mgcp_do_read; | ||||||
|  | 	nat->mgcp_queue.write_cb = mgcp_do_write; | ||||||
|  |  | ||||||
|  | 	if (bsc_register_fd(&nat->mgcp_queue.bfd) != 0) { | ||||||
|  | 		LOGP(DMGCP, LOGL_ERROR, "Failed to register MGCP fd.\n"); | ||||||
|  | 		close(nat->mgcp_queue.bfd.fd); | ||||||
|  | 		nat->mgcp_queue.bfd.fd = -1; | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* some more MGCP config handling */ | ||||||
|  | 	nat->mgcp_cfg->audio_payload = -1; | ||||||
|  | 	nat->mgcp_cfg->data = nat; | ||||||
|  | 	nat->mgcp_cfg->policy_cb = bsc_mgcp_policy_cb; | ||||||
|  | 	nat->mgcp_cfg->force_realloc = 1; | ||||||
|  | 	nat->bsc_endpoints = talloc_zero_array(nat, | ||||||
|  | 					       struct bsc_endpoint, | ||||||
|  | 					       nat->mgcp_cfg->number_endpoints + 1); | ||||||
|  |  | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void bsc_mgcp_clear_endpoints_for(struct bsc_connection *bsc) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 	for (i = 1; i < bsc->nat->mgcp_cfg->number_endpoints; ++i) { | ||||||
|  | 		struct bsc_endpoint *bsc_endp = &bsc->nat->bsc_endpoints[i]; | ||||||
|  |  | ||||||
|  | 		if (bsc_endp->bsc != bsc) | ||||||
|  | 			continue; | ||||||
|  |  | ||||||
|  | 		bsc_endp->bsc = NULL; | ||||||
|  | 		bsc_endp->pending_delete = 0; | ||||||
|  | 		if (bsc_endp->transaction_id) | ||||||
|  | 			talloc_free(bsc_endp->transaction_id); | ||||||
|  | 		bsc_endp->transaction_id = NULL; | ||||||
|  | 		mgcp_free_endp(&bsc->nat->mgcp_cfg->endpoints[i]); | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										865
									
								
								openbsc/src/nat/bsc_nat.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										865
									
								
								openbsc/src/nat/bsc_nat.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,865 @@ | |||||||
|  | /* BSC Multiplexer/NAT */ | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * (C) 2010 by Holger Hans Peter Freyther <zecke@selfish.org> | ||||||
|  |  * (C) 2010 by On-Waves | ||||||
|  |  * (C) 2009 by Harald Welte <laforge@gnumonks.org> | ||||||
|  |  * All Rights Reserved | ||||||
|  |  * | ||||||
|  |  * This program is free software; you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation; either version 2 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License along | ||||||
|  |  * with this program; if not, write to the Free Software Foundation, Inc., | ||||||
|  |  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  | #include <sys/socket.h> | ||||||
|  | #include <netinet/in.h> | ||||||
|  | #include <arpa/inet.h> | ||||||
|  |  | ||||||
|  | #include <errno.h> | ||||||
|  | #include <signal.h> | ||||||
|  | #include <stdio.h> | ||||||
|  | #include <stdlib.h> | ||||||
|  | #include <time.h> | ||||||
|  | #include <unistd.h> | ||||||
|  |  | ||||||
|  | #define _GNU_SOURCE | ||||||
|  | #include <getopt.h> | ||||||
|  |  | ||||||
|  | #include <openbsc/debug.h> | ||||||
|  | #include <openbsc/bsc_msc.h> | ||||||
|  | #include <openbsc/bsc_nat.h> | ||||||
|  | #include <openbsc/bssap.h> | ||||||
|  | #include <openbsc/ipaccess.h> | ||||||
|  | #include <openbsc/abis_nm.h> | ||||||
|  | #include <openbsc/telnet_interface.h> | ||||||
|  |  | ||||||
|  | #include <osmocore/talloc.h> | ||||||
|  |  | ||||||
|  | #include <vty/vty.h> | ||||||
|  |  | ||||||
|  | #include <sccp/sccp.h> | ||||||
|  |  | ||||||
|  | struct log_target *stderr_target; | ||||||
|  | static const char *config_file = "bsc-nat.cfg"; | ||||||
|  | static char *msc_address = "127.0.0.1"; | ||||||
|  | static struct in_addr local_addr; | ||||||
|  | static struct bsc_msc_connection *msc_con; | ||||||
|  | static struct bsc_fd bsc_listen; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | static struct bsc_nat *nat; | ||||||
|  | static void bsc_send_data(struct bsc_connection *bsc, const u_int8_t *data, unsigned int length, int); | ||||||
|  | static void remove_bsc_connection(struct bsc_connection *connection); | ||||||
|  | static void msc_send_reset(struct bsc_msc_connection *con); | ||||||
|  |  | ||||||
|  | struct bsc_config *bsc_config_num(struct bsc_nat *nat, int num) | ||||||
|  | { | ||||||
|  | 	struct bsc_config *conf; | ||||||
|  |  | ||||||
|  | 	llist_for_each_entry(conf, &nat->bsc_configs, entry) | ||||||
|  | 		if (conf->nr == num) | ||||||
|  | 			return conf; | ||||||
|  |  | ||||||
|  | 	return NULL; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * below are stubs we need to link | ||||||
|  |  */ | ||||||
|  | int nm_state_event(enum nm_evt evt, u_int8_t obj_class, void *obj, | ||||||
|  | 		   struct gsm_nm_state *old_state, struct gsm_nm_state *new_state) | ||||||
|  | { | ||||||
|  | 	return -1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void input_event(int event, enum e1inp_sign_type type, struct gsm_bts_trx *trx) | ||||||
|  | {} | ||||||
|  |  | ||||||
|  | int gsm0408_rcvmsg(struct msgb *msg, u_int8_t link_id) | ||||||
|  | { | ||||||
|  | 	return -1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void send_reset_ack(struct bsc_connection *bsc) | ||||||
|  | { | ||||||
|  | 	static const u_int8_t gsm_reset_ack[] = { | ||||||
|  | 		0x09, 0x00, 0x03, 0x07, 0x0b, 0x04, 0x43, 0x01, | ||||||
|  | 		0x00, 0xfe, 0x04, 0x43, 0x5c, 0x00, 0xfe, 0x03, | ||||||
|  | 		0x00, 0x01, 0x31, | ||||||
|  | 	}; | ||||||
|  |  | ||||||
|  | 	bsc_send_data(bsc, gsm_reset_ack, sizeof(gsm_reset_ack), IPAC_PROTO_SCCP); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void send_id_ack(struct bsc_connection *bsc) | ||||||
|  | { | ||||||
|  | 	static const u_int8_t id_ack[] = { | ||||||
|  | 		IPAC_MSGT_ID_ACK | ||||||
|  | 	}; | ||||||
|  |  | ||||||
|  | 	bsc_send_data(bsc, id_ack, sizeof(id_ack), IPAC_PROTO_IPACCESS); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void send_id_req(struct bsc_connection *bsc) | ||||||
|  | { | ||||||
|  | 	static const u_int8_t id_req[] = { | ||||||
|  | 		IPAC_MSGT_ID_GET, | ||||||
|  | 		0x01, IPAC_IDTAG_UNIT, | ||||||
|  | 		0x01, IPAC_IDTAG_MACADDR, | ||||||
|  | 		0x01, IPAC_IDTAG_LOCATION1, | ||||||
|  | 		0x01, IPAC_IDTAG_LOCATION2, | ||||||
|  | 		0x01, IPAC_IDTAG_EQUIPVERS, | ||||||
|  | 		0x01, IPAC_IDTAG_SWVERSION, | ||||||
|  | 		0x01, IPAC_IDTAG_UNITNAME, | ||||||
|  | 		0x01, IPAC_IDTAG_SERNR, | ||||||
|  | 	}; | ||||||
|  |  | ||||||
|  | 	bsc_send_data(bsc, id_req, sizeof(id_req), IPAC_PROTO_IPACCESS); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void nat_send_rlsd(struct sccp_connections *conn) | ||||||
|  | { | ||||||
|  | 	struct sccp_connection_released *rel; | ||||||
|  | 	struct msgb *msg; | ||||||
|  |  | ||||||
|  | 	msg = msgb_alloc_headroom(4096, 128, "rlsd"); | ||||||
|  | 	if (!msg) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Failed to allocate clear command.\n"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	msg->l2h = msgb_put(msg, sizeof(*rel)); | ||||||
|  | 	rel = (struct sccp_connection_released *) msg->l2h; | ||||||
|  | 	rel->type = SCCP_MSG_TYPE_RLSD; | ||||||
|  | 	rel->release_cause = SCCP_RELEASE_CAUSE_SCCP_FAILURE; | ||||||
|  | 	rel->destination_local_reference = conn->remote_ref; | ||||||
|  | 	rel->source_local_reference = conn->patched_ref; | ||||||
|  |  | ||||||
|  | 	ipaccess_prepend_header(msg, IPAC_PROTO_SCCP); | ||||||
|  |  | ||||||
|  | 	if (write_queue_enqueue(&msc_con->write_queue, msg) != 0) { | ||||||
|  | 		LOGP(DINP, LOGL_ERROR, "Failed to enqueue the write.\n"); | ||||||
|  | 		msgb_free(msg); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void nat_send_rlc(struct sccp_source_reference *src, | ||||||
|  | 			 struct sccp_source_reference *dst) | ||||||
|  | { | ||||||
|  | 	struct sccp_connection_release_complete *rlc; | ||||||
|  | 	struct msgb *msg; | ||||||
|  |  | ||||||
|  | 	msg = msgb_alloc_headroom(4096, 128, "rlc"); | ||||||
|  | 	if (!msg) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Failed to allocate clear command.\n"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	msg->l2h = msgb_put(msg, sizeof(*rlc)); | ||||||
|  | 	rlc = (struct sccp_connection_release_complete *) msg->l2h; | ||||||
|  | 	rlc->type = SCCP_MSG_TYPE_RLC; | ||||||
|  | 	rlc->destination_local_reference = *dst; | ||||||
|  | 	rlc->source_local_reference = *src; | ||||||
|  |  | ||||||
|  | 	ipaccess_prepend_header(msg, IPAC_PROTO_SCCP); | ||||||
|  |  | ||||||
|  | 	if (write_queue_enqueue(&msc_con->write_queue, msg) != 0) { | ||||||
|  | 		LOGP(DINP, LOGL_ERROR, "Failed to enqueue the write.\n"); | ||||||
|  | 		msgb_free(msg); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void send_mgcp_reset(struct bsc_connection *bsc) | ||||||
|  | { | ||||||
|  | 	static const u_int8_t mgcp_reset[] = { | ||||||
|  | 	    "RSIP 1 13@mgw MGCP 1.0\r\n" | ||||||
|  | 	}; | ||||||
|  |  | ||||||
|  | 	bsc_write_mgcp(bsc, mgcp_reset, sizeof mgcp_reset - 1); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Below is the handling of messages coming | ||||||
|  |  * from the MSC and need to be forwarded to | ||||||
|  |  * a real BSC. | ||||||
|  |  */ | ||||||
|  | static void initialize_msc_if_needed() | ||||||
|  | { | ||||||
|  | 	if (nat->first_contact) | ||||||
|  | 		return; | ||||||
|  |  | ||||||
|  | 	nat->first_contact = 1; | ||||||
|  | 	msc_send_reset(msc_con); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Currently we are lacking refcounting so we need to copy each message. | ||||||
|  |  */ | ||||||
|  | static void bsc_send_data(struct bsc_connection *bsc, const u_int8_t *data, unsigned int length, int proto) | ||||||
|  | { | ||||||
|  | 	struct msgb *msg; | ||||||
|  |  | ||||||
|  | 	if (length > 4096 - 128) { | ||||||
|  | 		LOGP(DINP, LOGL_ERROR, "Can not send message of that size.\n"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	msg = msgb_alloc_headroom(4096, 128, "to-bsc"); | ||||||
|  | 	if (!msg) { | ||||||
|  | 		LOGP(DINP, LOGL_ERROR, "Failed to allocate memory for BSC msg.\n"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	msg->l2h = msgb_put(msg, length); | ||||||
|  | 	memcpy(msg->data, data, length); | ||||||
|  |  | ||||||
|  | 	bsc_write(bsc, msg, proto); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static int forward_sccp_to_bts(struct msgb *msg) | ||||||
|  | { | ||||||
|  | 	struct sccp_connections *con; | ||||||
|  | 	struct bsc_connection *bsc; | ||||||
|  | 	struct bsc_nat_parsed *parsed; | ||||||
|  | 	int proto; | ||||||
|  |  | ||||||
|  | 	/* filter, drop, patch the message? */ | ||||||
|  | 	parsed = bsc_nat_parse(msg); | ||||||
|  | 	if (!parsed) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Can not parse msg from BSC.\n"); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if (bsc_nat_filter_ipa(DIR_BSC, msg, parsed)) | ||||||
|  | 		goto exit; | ||||||
|  |  | ||||||
|  | 	proto = parsed->ipa_proto; | ||||||
|  |  | ||||||
|  | 	/* Route and modify the SCCP packet */ | ||||||
|  | 	if (proto == IPAC_PROTO_SCCP) { | ||||||
|  | 		switch (parsed->sccp_type) { | ||||||
|  | 		case SCCP_MSG_TYPE_UDT: | ||||||
|  | 			/* forward UDT messages to every BSC */ | ||||||
|  | 			goto send_to_all; | ||||||
|  | 			break; | ||||||
|  | 		case SCCP_MSG_TYPE_RLSD: | ||||||
|  | 		case SCCP_MSG_TYPE_CREF: | ||||||
|  | 		case SCCP_MSG_TYPE_DT1: | ||||||
|  | 		case SCCP_MSG_TYPE_IT: | ||||||
|  | 			con = patch_sccp_src_ref_to_bsc(msg, parsed, nat); | ||||||
|  | 			if (parsed->gsm_type == BSS_MAP_MSG_ASSIGMENT_RQST) { | ||||||
|  | 				if (con) { | ||||||
|  | 					if (bsc_mgcp_assign(con, msg) != 0) | ||||||
|  | 						LOGP(DNAT, LOGL_ERROR, "Failed to assign...\n"); | ||||||
|  | 				} else | ||||||
|  | 					LOGP(DNAT, LOGL_ERROR, "Assignment command but no BSC.\n"); | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		case SCCP_MSG_TYPE_CC: | ||||||
|  | 			con = patch_sccp_src_ref_to_bsc(msg, parsed, nat); | ||||||
|  | 			if (!con || update_sccp_src_ref(con, parsed) != 0) | ||||||
|  | 				goto exit; | ||||||
|  | 			break; | ||||||
|  | 		case SCCP_MSG_TYPE_RLC: | ||||||
|  | 			LOGP(DNAT, LOGL_ERROR, "Unexpected release complete from MSC.\n"); | ||||||
|  | 			goto exit; | ||||||
|  | 			break; | ||||||
|  | 		case SCCP_MSG_TYPE_CR: | ||||||
|  | 			/* MSC never opens a SCCP connection, fall through */ | ||||||
|  | 		default: | ||||||
|  | 			goto exit; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (!con && parsed->sccp_type == SCCP_MSG_TYPE_RLSD) { | ||||||
|  | 			LOGP(DNAT, LOGL_NOTICE, "Sending fake RLC on RLSD message to network.\n"); | ||||||
|  | 			/* Exchange src/dest for the reply */ | ||||||
|  | 			nat_send_rlc(parsed->dest_local_ref, parsed->src_local_ref); | ||||||
|  | 		} else if (!con) | ||||||
|  | 			LOGP(DNAT, LOGL_ERROR, "Unknown connection for msg type: 0x%x.\n", parsed->sccp_type); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	talloc_free(parsed); | ||||||
|  | 	if (!con) | ||||||
|  | 		return -1; | ||||||
|  | 	if (!con->bsc->authenticated) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Selected BSC not authenticated.\n"); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	bsc_send_data(con->bsc, msg->l2h, msgb_l2len(msg), proto); | ||||||
|  | 	return 0; | ||||||
|  |  | ||||||
|  | send_to_all: | ||||||
|  | 	/* | ||||||
|  | 	 * Filter Paging from the network. We do not want to send a PAGING | ||||||
|  | 	 * Command to every BSC in our network. We will analys the PAGING | ||||||
|  | 	 * message and then send it to the authenticated messages... | ||||||
|  | 	 */ | ||||||
|  | 	if (parsed->ipa_proto == IPAC_PROTO_SCCP && parsed->gsm_type == BSS_MAP_MSG_PAGING) { | ||||||
|  | 		bsc = bsc_nat_find_bsc(nat, msg); | ||||||
|  | 		if (bsc) | ||||||
|  | 			bsc_send_data(bsc, msg->l2h, msgb_l2len(msg), parsed->ipa_proto); | ||||||
|  | 		else | ||||||
|  | 			LOGP(DNAT, LOGL_ERROR, "Could not determine BSC for paging.\n"); | ||||||
|  |  | ||||||
|  | 		goto exit; | ||||||
|  | 	} | ||||||
|  | 	/* currently send this to every BSC connected */ | ||||||
|  | 	llist_for_each_entry(bsc, &nat->bsc_connections, list_entry) { | ||||||
|  | 		if (!bsc->authenticated) | ||||||
|  | 			continue; | ||||||
|  |  | ||||||
|  | 		bsc_send_data(bsc, msg->l2h, msgb_l2len(msg), parsed->ipa_proto); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | exit: | ||||||
|  | 	talloc_free(parsed); | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void msc_connection_was_lost(struct bsc_msc_connection *con) | ||||||
|  | { | ||||||
|  | 	struct bsc_connection *bsc, *tmp; | ||||||
|  |  | ||||||
|  | 	LOGP(DMSC, LOGL_ERROR, "Closing all connections downstream.\n"); | ||||||
|  | 	llist_for_each_entry_safe(bsc, tmp, &nat->bsc_connections, list_entry) | ||||||
|  | 		remove_bsc_connection(bsc); | ||||||
|  |  | ||||||
|  | 	nat->first_contact = 0; | ||||||
|  | 	bsc_mgcp_free_endpoints(nat); | ||||||
|  | 	bsc_msc_schedule_connect(con); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void msc_send_reset(struct bsc_msc_connection *msc_con) | ||||||
|  | { | ||||||
|  | 	static const u_int8_t reset[] = { | ||||||
|  | 		0x00, 0x12, 0xfd, | ||||||
|  | 		0x09, 0x00, 0x03, 0x05, 0x07, 0x02, 0x42, 0xfe, | ||||||
|  | 		0x02, 0x42, 0xfe, 0x06, 0x00, 0x04, 0x30, 0x04, | ||||||
|  | 		0x01, 0x20 | ||||||
|  | 	}; | ||||||
|  |  | ||||||
|  | 	struct msgb *msg; | ||||||
|  |  | ||||||
|  | 	msg = msgb_alloc_headroom(4096, 128, "08.08 reset"); | ||||||
|  | 	if (!msg) { | ||||||
|  | 		LOGP(DMSC, LOGL_ERROR, "Failed to allocate reset msg.\n"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	msg->l2h = msgb_put(msg, sizeof(reset)); | ||||||
|  | 	memcpy(msg->l2h, reset, msgb_l2len(msg)); | ||||||
|  |  | ||||||
|  | 	if (write_queue_enqueue(&msc_con->write_queue, msg) != 0) { | ||||||
|  | 		LOGP(DMSC, LOGL_ERROR, "Failed to enqueue reset msg.\n"); | ||||||
|  | 		msgb_free(msg); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	LOGP(DMSC, LOGL_NOTICE, "Scheduled GSM0808 reset msg for the MSC.\n"); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static int ipaccess_msc_read_cb(struct bsc_fd *bfd) | ||||||
|  | { | ||||||
|  | 	int error; | ||||||
|  | 	struct msgb *msg = ipaccess_read_msg(bfd, &error); | ||||||
|  | 	struct ipaccess_head *hh; | ||||||
|  |  | ||||||
|  | 	if (!msg) { | ||||||
|  | 		if (error == 0) { | ||||||
|  | 			LOGP(DNAT, LOGL_FATAL, "The connection the MSC was lost, exiting\n"); | ||||||
|  | 			bsc_msc_lost(msc_con); | ||||||
|  | 			return -1; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Failed to parse ip access message: %d\n", error); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	LOGP(DNAT, LOGL_DEBUG, "MSG from MSC: %s proto: %d\n", hexdump(msg->data, msg->len), msg->l2h[0]); | ||||||
|  |  | ||||||
|  | 	/* handle base message handling */ | ||||||
|  | 	hh = (struct ipaccess_head *) msg->data; | ||||||
|  | 	ipaccess_rcvmsg_base(msg, bfd); | ||||||
|  |  | ||||||
|  | 	/* initialize the networking. This includes sending a GSM08.08 message */ | ||||||
|  | 	if (hh->proto == IPAC_PROTO_IPACCESS && msg->l2h[0] == IPAC_MSGT_ID_ACK) | ||||||
|  | 		initialize_msc_if_needed(); | ||||||
|  | 	else if (hh->proto == IPAC_PROTO_SCCP) | ||||||
|  | 		forward_sccp_to_bts(msg); | ||||||
|  |  | ||||||
|  | 	msgb_free(msg); | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static int ipaccess_msc_write_cb(struct bsc_fd *bfd, struct msgb *msg) | ||||||
|  | { | ||||||
|  | 	int rc; | ||||||
|  | 	rc = write(bfd->fd, msg->data, msg->len); | ||||||
|  |  | ||||||
|  | 	if (rc != msg->len) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Failed to write MSG to MSC.\n"); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return rc; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Below is the handling of messages coming | ||||||
|  |  * from the BSC and need to be forwarded to | ||||||
|  |  * a real BSC. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Remove the connection from the connections list, | ||||||
|  |  * remove it from the patching of SCCP header lists | ||||||
|  |  * as well. Maybe in the future even close connection.. | ||||||
|  |  */ | ||||||
|  | static void remove_bsc_connection(struct bsc_connection *connection) | ||||||
|  | { | ||||||
|  | 	struct sccp_connections *sccp_patch, *tmp; | ||||||
|  | 	bsc_unregister_fd(&connection->write_queue.bfd); | ||||||
|  | 	close(connection->write_queue.bfd.fd); | ||||||
|  | 	write_queue_clear(&connection->write_queue); | ||||||
|  | 	llist_del(&connection->list_entry); | ||||||
|  |  | ||||||
|  | 	/* stop the timeout timer */ | ||||||
|  | 	bsc_del_timer(&connection->id_timeout); | ||||||
|  |  | ||||||
|  | 	/* remove all SCCP connections */ | ||||||
|  | 	llist_for_each_entry_safe(sccp_patch, tmp, &nat->sccp_connections, list_entry) { | ||||||
|  | 		if (sccp_patch->bsc != connection) | ||||||
|  | 			continue; | ||||||
|  |  | ||||||
|  | 		nat_send_rlsd(sccp_patch); | ||||||
|  | 		sccp_connection_destroy(sccp_patch); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* close endpoints allocated by this BSC */ | ||||||
|  | 	bsc_mgcp_clear_endpoints_for(connection); | ||||||
|  |  | ||||||
|  | 	talloc_free(connection); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void ipaccess_close_bsc(void *data) | ||||||
|  | { | ||||||
|  | 	struct bsc_connection *conn = data; | ||||||
|  |  | ||||||
|  | 	LOGP(DNAT, LOGL_ERROR, "BSC didn't respond to identity request. Closing.\n"); | ||||||
|  | 	remove_bsc_connection(conn); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void ipaccess_auth_bsc(struct tlv_parsed *tvp, struct bsc_connection *bsc) | ||||||
|  | { | ||||||
|  | 	struct bsc_config *conf; | ||||||
|  | 	const char* token = (const char *) TLVP_VAL(tvp, IPAC_IDTAG_UNITNAME); | ||||||
|  |  | ||||||
|  | 	llist_for_each_entry(conf, &bsc->nat->bsc_configs, entry) { | ||||||
|  | 		if (strcmp(conf->token, token) == 0) { | ||||||
|  | 			bsc->authenticated = 1; | ||||||
|  | 			bsc->cfg = conf; | ||||||
|  | 			bsc_del_timer(&bsc->id_timeout); | ||||||
|  | 			LOGP(DNAT, LOGL_NOTICE, "Authenticated bsc nr: %d lac: %d\n", conf->nr, conf->lac); | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static int forward_sccp_to_msc(struct bsc_connection *bsc, struct msgb *msg) | ||||||
|  | { | ||||||
|  | 	struct sccp_connections *con; | ||||||
|  | 	struct bsc_nat_parsed *parsed; | ||||||
|  |  | ||||||
|  | 	/* Parse and filter messages */ | ||||||
|  | 	parsed = bsc_nat_parse(msg); | ||||||
|  | 	if (!parsed) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Can not parse msg from BSC.\n"); | ||||||
|  | 		msgb_free(msg); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if (bsc_nat_filter_ipa(DIR_MSC, msg, parsed)) | ||||||
|  | 		goto exit; | ||||||
|  |  | ||||||
|  | 	/* | ||||||
|  | 	 * check authentication after filtering to not reject auth | ||||||
|  | 	 * responses coming from the BSC. We have to make sure that | ||||||
|  | 	 * nothing from the exit path will forward things to the MSC | ||||||
|  | 	 */ | ||||||
|  | 	if (!bsc->authenticated) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "BSC is not authenticated.\n"); | ||||||
|  | 		msgb_free(msg); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	/* modify the SCCP entries */ | ||||||
|  | 	if (parsed->ipa_proto == IPAC_PROTO_SCCP) { | ||||||
|  | 		switch (parsed->sccp_type) { | ||||||
|  | 		case SCCP_MSG_TYPE_CR: | ||||||
|  | 			if (create_sccp_src_ref(bsc, msg, parsed) != 0) | ||||||
|  | 				goto exit2; | ||||||
|  | 			con = patch_sccp_src_ref_to_msc(msg, parsed, nat); | ||||||
|  | 			break; | ||||||
|  | 		case SCCP_MSG_TYPE_RLSD: | ||||||
|  | 		case SCCP_MSG_TYPE_CREF: | ||||||
|  | 		case SCCP_MSG_TYPE_DT1: | ||||||
|  | 		case SCCP_MSG_TYPE_CC: | ||||||
|  | 		case SCCP_MSG_TYPE_IT: | ||||||
|  | 			con = patch_sccp_src_ref_to_msc(msg, parsed, nat); | ||||||
|  | 			break; | ||||||
|  | 		case SCCP_MSG_TYPE_RLC: | ||||||
|  | 			con = patch_sccp_src_ref_to_msc(msg, parsed, nat); | ||||||
|  | 			remove_sccp_src_ref(bsc, msg, parsed); | ||||||
|  | 			break; | ||||||
|  | 		case SCCP_MSG_TYPE_UDT: | ||||||
|  | 			/* simply forward everything */ | ||||||
|  | 			con = NULL; | ||||||
|  | 			break; | ||||||
|  | 		default: | ||||||
|  | 			LOGP(DNAT, LOGL_ERROR, "Not forwarding to msc sccp type: 0x%x\n", parsed->sccp_type); | ||||||
|  | 			con = NULL; | ||||||
|  | 			goto exit2; | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  |         } else if (parsed->ipa_proto == NAT_IPAC_PROTO_MGCP) { | ||||||
|  |                 bsc_mgcp_forward(bsc, msg); | ||||||
|  |                 goto exit2; | ||||||
|  | 	} else { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Not forwarding unknown stream id: 0x%x\n", parsed->ipa_proto); | ||||||
|  | 		goto exit2; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if (con && con->bsc != bsc) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Found the wrong entry.\n"); | ||||||
|  | 		goto exit2; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* send the non-filtered but maybe modified msg */ | ||||||
|  | 	if (write_queue_enqueue(&msc_con->write_queue, msg) != 0) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Can not queue message for the MSC.\n"); | ||||||
|  | 		msgb_free(msg); | ||||||
|  | 	} | ||||||
|  | 	talloc_free(parsed); | ||||||
|  | 	return 0; | ||||||
|  |  | ||||||
|  | exit: | ||||||
|  | 	/* if we filter out the reset send an ack to the BSC */ | ||||||
|  | 	if (parsed->bssap == 0 && parsed->gsm_type == BSS_MAP_MSG_RESET) { | ||||||
|  | 		send_reset_ack(bsc); | ||||||
|  | 		send_reset_ack(bsc); | ||||||
|  | 	} else if (parsed->ipa_proto == IPAC_PROTO_IPACCESS) { | ||||||
|  | 		/* do we know who is handling this? */ | ||||||
|  | 		if (msg->l2h[0] == IPAC_MSGT_ID_RESP) { | ||||||
|  | 			struct tlv_parsed tvp; | ||||||
|  | 			ipaccess_idtag_parse(&tvp, | ||||||
|  | 					     (unsigned char *) msg->l2h + 2, | ||||||
|  | 					     msgb_l2len(msg) - 2); | ||||||
|  | 			if (TLVP_PRESENT(&tvp, IPAC_IDTAG_UNITNAME)) | ||||||
|  | 				ipaccess_auth_bsc(&tvp, bsc); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		goto exit2; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | exit2: | ||||||
|  | 	talloc_free(parsed); | ||||||
|  | 	msgb_free(msg); | ||||||
|  | 	return -1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static int ipaccess_bsc_read_cb(struct bsc_fd *bfd) | ||||||
|  | { | ||||||
|  | 	int error; | ||||||
|  | 	struct bsc_connection *bsc = bfd->data; | ||||||
|  | 	struct msgb *msg = ipaccess_read_msg(bfd, &error); | ||||||
|  |  | ||||||
|  | 	if (!msg) { | ||||||
|  | 		if (error == 0) { | ||||||
|  | 			LOGP(DNAT, LOGL_ERROR,	"The connection to the BSC was lost. Cleaning it\n"); | ||||||
|  | 			remove_bsc_connection(bsc); | ||||||
|  | 		} else { | ||||||
|  | 			LOGP(DNAT, LOGL_ERROR, "Failed to parse ip access message: %d\n", error); | ||||||
|  | 		} | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	LOGP(DNAT, LOGL_DEBUG, "MSG from BSC: %s proto: %d\n", hexdump(msg->data, msg->len), msg->l2h[0]); | ||||||
|  |  | ||||||
|  | 	/* Handle messages from the BSC */ | ||||||
|  | 	/* FIXME: Currently no PONG is sent to the BSC */ | ||||||
|  | 	/* FIXME: Currently no ID ACK is sent to the BSC */ | ||||||
|  | 	forward_sccp_to_msc(bsc, msg); | ||||||
|  |  | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static int ipaccess_bsc_write_cb(struct bsc_fd *bfd, struct msgb *msg) | ||||||
|  | { | ||||||
|  | 	int rc; | ||||||
|  |  | ||||||
|  | 	rc = write(bfd->fd, msg->data, msg->len); | ||||||
|  | 	if (rc != msg->len) | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Failed to write message to the BSC.\n"); | ||||||
|  |  | ||||||
|  | 	return rc; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static int ipaccess_listen_bsc_cb(struct bsc_fd *bfd, unsigned int what) | ||||||
|  | { | ||||||
|  | 	struct bsc_connection *bsc; | ||||||
|  | 	int ret; | ||||||
|  | 	struct sockaddr_in sa; | ||||||
|  | 	socklen_t sa_len = sizeof(sa); | ||||||
|  |  | ||||||
|  | 	if (!(what & BSC_FD_READ)) | ||||||
|  | 		return 0; | ||||||
|  |  | ||||||
|  | 	ret = accept(bfd->fd, (struct sockaddr *) &sa, &sa_len); | ||||||
|  | 	if (ret < 0) { | ||||||
|  | 		perror("accept"); | ||||||
|  | 		return ret; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* | ||||||
|  | 	 * if we are not connected to a msc... just close the socket | ||||||
|  | 	 */ | ||||||
|  | 	if (!msc_con->is_connected) { | ||||||
|  | 		LOGP(DNAT, LOGL_NOTICE, "Disconnecting BSC due lack of MSC connection.\n"); | ||||||
|  | 		close(ret); | ||||||
|  | 		return 0; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* todo... do something with the connection */ | ||||||
|  | 	/* todo... use GNUtls to see if we want to trust this as a BTS */ | ||||||
|  |  | ||||||
|  | 	/* | ||||||
|  | 	 * | ||||||
|  | 	 */ | ||||||
|  | 	bsc = bsc_connection_alloc(nat); | ||||||
|  | 	if (!bsc) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Failed to allocate BSC struct.\n"); | ||||||
|  | 		close(ret); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	write_queue_init(&bsc->write_queue, 100); | ||||||
|  | 	bsc->write_queue.bfd.data = bsc; | ||||||
|  | 	bsc->write_queue.bfd.fd = ret; | ||||||
|  | 	bsc->write_queue.read_cb = ipaccess_bsc_read_cb; | ||||||
|  | 	bsc->write_queue.write_cb = ipaccess_bsc_write_cb; | ||||||
|  | 	bsc->write_queue.bfd.when = BSC_FD_READ; | ||||||
|  | 	if (bsc_register_fd(&bsc->write_queue.bfd) < 0) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Failed to register BSC fd.\n"); | ||||||
|  | 		close(ret); | ||||||
|  | 		talloc_free(bsc); | ||||||
|  | 		return -2; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	LOGP(DNAT, LOGL_NOTICE, "Registered new BSC\n"); | ||||||
|  | 	llist_add(&bsc->list_entry, &nat->bsc_connections); | ||||||
|  | 	send_id_ack(bsc); | ||||||
|  | 	send_id_req(bsc); | ||||||
|  | 	send_mgcp_reset(bsc); | ||||||
|  |  | ||||||
|  | 	/* | ||||||
|  | 	 * start the hangup timer | ||||||
|  | 	 */ | ||||||
|  | 	bsc->id_timeout.data = bsc; | ||||||
|  | 	bsc->id_timeout.cb = ipaccess_close_bsc; | ||||||
|  | 	bsc_schedule_timer(&bsc->id_timeout, 2, 0); | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static int listen_for_bsc(struct bsc_fd *bfd, struct in_addr *in_addr, int port) | ||||||
|  | { | ||||||
|  | 	struct sockaddr_in addr; | ||||||
|  | 	int ret, on = 1; | ||||||
|  |  | ||||||
|  | 	bfd->fd = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); | ||||||
|  | 	bfd->cb = ipaccess_listen_bsc_cb; | ||||||
|  | 	bfd->when = BSC_FD_READ; | ||||||
|  |  | ||||||
|  | 	memset(&addr, 0, sizeof(addr)); | ||||||
|  | 	addr.sin_family = AF_INET; | ||||||
|  | 	addr.sin_port = htons(port); | ||||||
|  | 	addr.sin_addr.s_addr = in_addr->s_addr; | ||||||
|  |  | ||||||
|  | 	setsockopt(bfd->fd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)); | ||||||
|  |  | ||||||
|  | 	ret = bind(bfd->fd, (struct sockaddr *) &addr, sizeof(addr)); | ||||||
|  | 	if (ret < 0) { | ||||||
|  | 		fprintf(stderr, "Could not bind the BSC socket %s\n", | ||||||
|  | 			strerror(errno)); | ||||||
|  | 		return -EIO; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	ret = listen(bfd->fd, 1); | ||||||
|  | 	if (ret < 0) { | ||||||
|  | 		perror("listen"); | ||||||
|  | 		return ret; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	ret = bsc_register_fd(bfd); | ||||||
|  | 	if (ret < 0) { | ||||||
|  | 		perror("register_listen_fd"); | ||||||
|  | 		return ret; | ||||||
|  | 	} | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void print_usage() | ||||||
|  | { | ||||||
|  | 	printf("Usage: bsc_nat\n"); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void print_help() | ||||||
|  | { | ||||||
|  | 	printf("  Some useful help...\n"); | ||||||
|  | 	printf("  -h --help this text\n"); | ||||||
|  | 	printf("  -d option --debug=DRLL:DCC:DMM:DRR:DRSL:DNM enable debugging\n"); | ||||||
|  | 	printf("  -s --disable-color\n"); | ||||||
|  | 	printf("  -c --config-file filename The config file to use.\n"); | ||||||
|  | 	printf("  -m --msc=IP. The address of the MSC.\n"); | ||||||
|  | 	printf("  -l --local=IP. The local address of this BSC.\n"); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void handle_options(int argc, char** argv) | ||||||
|  | { | ||||||
|  | 	while (1) { | ||||||
|  | 		int option_index = 0, c; | ||||||
|  | 		static struct option long_options[] = { | ||||||
|  | 			{"help", 0, 0, 'h'}, | ||||||
|  | 			{"debug", 1, 0, 'd'}, | ||||||
|  | 			{"config-file", 1, 0, 'c'}, | ||||||
|  | 			{"disable-color", 0, 0, 's'}, | ||||||
|  | 			{"timestamp", 0, 0, 'T'}, | ||||||
|  | 			{"msc", 1, 0, 'm'}, | ||||||
|  | 			{"local", 1, 0, 'l'}, | ||||||
|  | 			{0, 0, 0, 0} | ||||||
|  | 		}; | ||||||
|  |  | ||||||
|  | 		c = getopt_long(argc, argv, "hd:sTPc:m:l:", | ||||||
|  | 				long_options, &option_index); | ||||||
|  | 		if (c == -1) | ||||||
|  | 			break; | ||||||
|  |  | ||||||
|  | 		switch (c) { | ||||||
|  | 		case 'h': | ||||||
|  | 			print_usage(); | ||||||
|  | 			print_help(); | ||||||
|  | 			exit(0); | ||||||
|  | 		case 's': | ||||||
|  | 			log_set_use_color(stderr_target, 0); | ||||||
|  | 			break; | ||||||
|  | 		case 'd': | ||||||
|  | 			log_parse_category_mask(stderr_target, optarg); | ||||||
|  | 			break; | ||||||
|  | 		case 'c': | ||||||
|  | 			config_file = strdup(optarg); | ||||||
|  | 			break; | ||||||
|  | 		case 'T': | ||||||
|  | 			log_set_print_timestamp(stderr_target, 1); | ||||||
|  | 			break; | ||||||
|  | 		case 'm': | ||||||
|  | 			msc_address = strdup(optarg); | ||||||
|  | 			break; | ||||||
|  | 		case 'l': | ||||||
|  | 			inet_aton(optarg, &local_addr); | ||||||
|  | 			break; | ||||||
|  | 		default: | ||||||
|  | 			/* ignore */ | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void signal_handler(int signal) | ||||||
|  | { | ||||||
|  | 	switch (signal) { | ||||||
|  | 	case SIGABRT: | ||||||
|  | 		/* in case of abort, we want to obtain a talloc report | ||||||
|  | 		 * and then return to the caller, who will abort the process */ | ||||||
|  | 	case SIGUSR1: | ||||||
|  | 		talloc_report_full(tall_bsc_ctx, stderr); | ||||||
|  | 		break; | ||||||
|  | 	default: | ||||||
|  | 		break; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int main(int argc, char** argv) | ||||||
|  | { | ||||||
|  | 	log_init(&log_info); | ||||||
|  | 	stderr_target = log_target_create_stderr(); | ||||||
|  | 	log_add_target(stderr_target); | ||||||
|  | 	log_set_all_filter(stderr_target, 1); | ||||||
|  |  | ||||||
|  | 	/* parse options */ | ||||||
|  | 	local_addr.s_addr = INADDR_ANY; | ||||||
|  | 	handle_options(argc, argv); | ||||||
|  |  | ||||||
|  | 	nat = bsc_nat_alloc(); | ||||||
|  | 	if (!nat) { | ||||||
|  | 		fprintf(stderr, "Failed to allocate the BSC nat.\n"); | ||||||
|  | 		return -4; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	nat->mgcp_cfg = talloc_zero(nat, struct mgcp_config); | ||||||
|  |  | ||||||
|  | 	/* init vty and parse */ | ||||||
|  | 	bsc_nat_vty_init(nat); | ||||||
|  | 	telnet_init(NULL, 4244); | ||||||
|  | 	if (mgcp_parse_config(config_file, nat->mgcp_cfg) < 0) { | ||||||
|  | 		fprintf(stderr, "Failed to parse the config file: '%s'\n", config_file); | ||||||
|  | 		return -3; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* seed the PRNG */ | ||||||
|  | 	srand(time(NULL)); | ||||||
|  |  | ||||||
|  | 	/* | ||||||
|  | 	 * Setup the MGCP code.. | ||||||
|  | 	 */ | ||||||
|  | 	if (bsc_mgcp_init(nat) != 0) | ||||||
|  | 		return -4; | ||||||
|  |  | ||||||
|  | 	/* connect to the MSC */ | ||||||
|  | 	msc_con = bsc_msc_create(msc_address, 5000); | ||||||
|  | 	if (!msc_con) { | ||||||
|  | 		fprintf(stderr, "Creating a bsc_msc_connection failed.\n"); | ||||||
|  | 		exit(1); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	msc_con->connection_loss = msc_connection_was_lost; | ||||||
|  | 	msc_con->write_queue.read_cb = ipaccess_msc_read_cb; | ||||||
|  | 	msc_con->write_queue.write_cb = ipaccess_msc_write_cb;; | ||||||
|  | 	bsc_msc_connect(msc_con); | ||||||
|  |  | ||||||
|  | 	/* wait for the BSC */ | ||||||
|  | 	if (listen_for_bsc(&bsc_listen, &local_addr, 5000) < 0) { | ||||||
|  | 		fprintf(stderr, "Failed to listen for BSC.\n"); | ||||||
|  | 		exit(1); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	signal(SIGABRT, &signal_handler); | ||||||
|  | 	signal(SIGUSR1, &signal_handler); | ||||||
|  | 	signal(SIGPIPE, SIG_IGN); | ||||||
|  |  | ||||||
|  | 	while (1) { | ||||||
|  | 		bsc_select_main(0); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
							
								
								
									
										165
									
								
								openbsc/src/nat/bsc_nat_utils.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										165
									
								
								openbsc/src/nat/bsc_nat_utils.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,165 @@ | |||||||
|  |  | ||||||
|  | /* BSC Multiplexer/NAT Utilities */ | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * (C) 2010 by Holger Hans Peter Freyther <zecke@selfish.org> | ||||||
|  |  * (C) 2010 by On-Waves | ||||||
|  |  * All Rights Reserved | ||||||
|  |  * | ||||||
|  |  * This program is free software; you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation; either version 2 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License along | ||||||
|  |  * with this program; if not, write to the Free Software Foundation, Inc., | ||||||
|  |  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #include <openbsc/bsc_nat.h> | ||||||
|  | #include <openbsc/gsm_data.h> | ||||||
|  | #include <openbsc/bssap.h> | ||||||
|  | #include <openbsc/debug.h> | ||||||
|  | #include <openbsc/ipaccess.h> | ||||||
|  |  | ||||||
|  | #include <osmocore/linuxlist.h> | ||||||
|  | #include <osmocore/talloc.h> | ||||||
|  |  | ||||||
|  | #include <sccp/sccp.h> | ||||||
|  |  | ||||||
|  | #include <netinet/in.h> | ||||||
|  | #include <arpa/inet.h> | ||||||
|  |  | ||||||
|  | struct bsc_nat *bsc_nat_alloc(void) | ||||||
|  | { | ||||||
|  | 	struct bsc_nat *nat = talloc_zero(tall_bsc_ctx, struct bsc_nat); | ||||||
|  | 	if (!nat) | ||||||
|  | 		return NULL; | ||||||
|  |  | ||||||
|  | 	INIT_LLIST_HEAD(&nat->sccp_connections); | ||||||
|  | 	INIT_LLIST_HEAD(&nat->bsc_connections); | ||||||
|  | 	INIT_LLIST_HEAD(&nat->bsc_configs); | ||||||
|  | 	return nat; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | struct bsc_connection *bsc_connection_alloc(struct bsc_nat *nat) | ||||||
|  | { | ||||||
|  | 	struct bsc_connection *con = talloc_zero(nat, struct bsc_connection); | ||||||
|  | 	if (!con) | ||||||
|  | 		return NULL; | ||||||
|  |  | ||||||
|  | 	con->nat = nat; | ||||||
|  | 	return con; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | struct bsc_config *bsc_config_alloc(struct bsc_nat *nat, const char *token, unsigned int lac) | ||||||
|  | { | ||||||
|  | 	struct bsc_config *conf = talloc_zero(nat, struct bsc_config); | ||||||
|  | 	if (!conf) | ||||||
|  | 		return NULL; | ||||||
|  |  | ||||||
|  | 	conf->token = talloc_strdup(conf, token); | ||||||
|  | 	conf->lac = lac; | ||||||
|  | 	conf->nr = nat->num_bsc; | ||||||
|  | 	conf->nat = nat; | ||||||
|  |  | ||||||
|  | 	llist_add(&conf->entry, &nat->bsc_configs); | ||||||
|  | 	++nat->num_bsc; | ||||||
|  |  | ||||||
|  | 	return conf; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void sccp_connection_destroy(struct sccp_connections *conn) | ||||||
|  | { | ||||||
|  | 	LOGP(DNAT, LOGL_DEBUG, "Destroy 0x%x <-> 0x%x mapping for con %p\n", | ||||||
|  | 	     sccp_src_ref_to_int(&conn->real_ref), | ||||||
|  | 	     sccp_src_ref_to_int(&conn->patched_ref), conn->bsc); | ||||||
|  | 	bsc_mgcp_clear(conn); | ||||||
|  | 	llist_del(&conn->list_entry); | ||||||
|  | 	talloc_free(conn); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | struct bsc_connection *bsc_nat_find_bsc(struct bsc_nat *nat, struct msgb *msg) | ||||||
|  | { | ||||||
|  | 	struct bsc_connection *bsc; | ||||||
|  | 	int data_length; | ||||||
|  | 	const u_int8_t *data; | ||||||
|  | 	struct tlv_parsed tp; | ||||||
|  | 	int i = 0; | ||||||
|  |  | ||||||
|  | 	if (!msg->l3h || msgb_l3len(msg) < 3) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Paging message is too short.\n"); | ||||||
|  | 		return NULL; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	tlv_parse(&tp, gsm0808_att_tlvdef(), msg->l3h + 3, msgb_l3len(msg) - 3, 0, 0); | ||||||
|  | 	if (!TLVP_PRESENT(&tp, GSM0808_IE_CELL_IDENTIFIER_LIST)) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "No CellIdentifier List inside paging msg.\n"); | ||||||
|  | 		return NULL; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	data_length = TLVP_LEN(&tp, GSM0808_IE_CELL_IDENTIFIER_LIST); | ||||||
|  | 	data = TLVP_VAL(&tp, GSM0808_IE_CELL_IDENTIFIER_LIST); | ||||||
|  | 	if (data[0] !=  CELL_IDENT_LAC) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Unhandled cell ident discrminator: %d\n", data[0]); | ||||||
|  | 		return NULL; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* Currently we only handle one BSC */ | ||||||
|  | 	for (i = 1; i < data_length - 1; i += 2) { | ||||||
|  | 		unsigned int _lac = ntohs(*(unsigned int *) &data[i]); | ||||||
|  | 		llist_for_each_entry(bsc, &nat->bsc_connections, list_entry) { | ||||||
|  | 			if (!bsc->cfg) | ||||||
|  | 				continue; | ||||||
|  | 			if (!bsc->authenticated || _lac != bsc->cfg->lac) | ||||||
|  | 				continue; | ||||||
|  |  | ||||||
|  | 			return bsc; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return NULL; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int bsc_write_mgcp(struct bsc_connection *bsc, const u_int8_t *data, unsigned int length) | ||||||
|  | { | ||||||
|  | 	struct msgb *msg; | ||||||
|  |  | ||||||
|  | 	if (length > 4096 - 128) { | ||||||
|  | 		LOGP(DINP, LOGL_ERROR, "Can not send message of that size.\n"); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	msg = msgb_alloc_headroom(4096, 128, "to-bsc"); | ||||||
|  | 	if (!msg) { | ||||||
|  | 		LOGP(DINP, LOGL_ERROR, "Failed to allocate memory for BSC msg.\n"); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* copy the data */ | ||||||
|  | 	msg->l3h = msgb_put(msg, length); | ||||||
|  | 	memcpy(msg->l3h, data, length); | ||||||
|  |  | ||||||
|  |         return bsc_write(bsc, msg, NAT_IPAC_PROTO_MGCP); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int bsc_write(struct bsc_connection *bsc, struct msgb *msg, int proto) | ||||||
|  | { | ||||||
|  | 	/* prepend the header */ | ||||||
|  | 	ipaccess_prepend_header(msg, proto); | ||||||
|  |  | ||||||
|  | 	if (write_queue_enqueue(&bsc->write_queue, msg) != 0) { | ||||||
|  | 		LOGP(DINP, LOGL_ERROR, "Failed to enqueue the write.\n"); | ||||||
|  | 		msgb_free(msg); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
							
								
								
									
										233
									
								
								openbsc/src/nat/bsc_nat_vty.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										233
									
								
								openbsc/src/nat/bsc_nat_vty.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,233 @@ | |||||||
|  | /* OpenBSC NAT interface to quagga VTY */ | ||||||
|  | /* (C) 2010 by Holger Hans Peter Freyther | ||||||
|  |  * (C) 2010 by On-Waves | ||||||
|  |  * All Rights Reserved | ||||||
|  |  * | ||||||
|  |  * This program is free software; you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation; either version 2 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License along | ||||||
|  |  * with this program; if not, write to the Free Software Foundation, Inc., | ||||||
|  |  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #include <vty/command.h> | ||||||
|  | #include <vty/buffer.h> | ||||||
|  | #include <vty/vty.h> | ||||||
|  |  | ||||||
|  | #include <openbsc/bsc_nat.h> | ||||||
|  | #include <openbsc/gsm_04_08.h> | ||||||
|  | #include <openbsc/mgcp.h> | ||||||
|  | #include <openbsc/vty.h> | ||||||
|  |  | ||||||
|  | #include <osmocore/talloc.h> | ||||||
|  |  | ||||||
|  | #include <sccp/sccp.h> | ||||||
|  |  | ||||||
|  | #include <stdlib.h> | ||||||
|  |  | ||||||
|  | static struct bsc_nat *_nat; | ||||||
|  |  | ||||||
|  | static struct cmd_node nat_node = { | ||||||
|  | 	NAT_NODE, | ||||||
|  | 	"%s(nat)#", | ||||||
|  | 	1, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | static struct cmd_node bsc_node = { | ||||||
|  | 	BSC_NODE, | ||||||
|  | 	"%s(bsc)#", | ||||||
|  | 	1, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | static int config_write_nat(struct vty *vty) | ||||||
|  | { | ||||||
|  | 	vty_out(vty, "nat%s", VTY_NEWLINE); | ||||||
|  | 	return CMD_SUCCESS; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void config_write_bsc_single(struct vty *vty, struct bsc_config *bsc) | ||||||
|  | { | ||||||
|  | 	vty_out(vty, " bsc %u%s", bsc->nr, VTY_NEWLINE); | ||||||
|  | 	vty_out(vty, "  token %s%s", bsc->token, VTY_NEWLINE); | ||||||
|  | 	vty_out(vty, "  lac %u%s", bsc->lac, VTY_NEWLINE); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static int config_write_bsc(struct vty *vty) | ||||||
|  | { | ||||||
|  | 	struct bsc_config *bsc; | ||||||
|  |  | ||||||
|  | 	llist_for_each_entry(bsc, &_nat->bsc_configs, entry) | ||||||
|  | 		config_write_bsc_single(vty, bsc); | ||||||
|  | 	return CMD_SUCCESS; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | DEFUN(show_sccp, show_sccp_cmd, "show connections sccp", | ||||||
|  |       SHOW_STR "Display information about current SCCP connections") | ||||||
|  | { | ||||||
|  | 	struct sccp_connections *con; | ||||||
|  | 	llist_for_each_entry(con, &_nat->sccp_connections, list_entry) { | ||||||
|  | 		vty_out(vty, "SCCP for BSC: Nr: %d lac: %d BSC ref: 0x%x Local ref: 0x%x MSC/BSC mux: 0x%x/0x%x%s", | ||||||
|  | 			con->bsc->cfg ? con->bsc->cfg->nr : -1, | ||||||
|  | 			con->bsc->cfg ? con->bsc->cfg->lac : -1, | ||||||
|  | 			sccp_src_ref_to_int(&con->real_ref), | ||||||
|  | 			sccp_src_ref_to_int(&con->patched_ref), | ||||||
|  | 			con->msc_timeslot, con->bsc_timeslot, | ||||||
|  | 			VTY_NEWLINE); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return CMD_SUCCESS; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | DEFUN(show_bsc, show_bsc_cmd, "show connections bsc", | ||||||
|  |       SHOW_STR "Display information about current BSCs") | ||||||
|  | { | ||||||
|  | 	struct bsc_connection *con; | ||||||
|  | 	struct sockaddr_in sock; | ||||||
|  | 	socklen_t len = sizeof(sock); | ||||||
|  |  | ||||||
|  | 	llist_for_each_entry(con, &_nat->bsc_connections, list_entry) { | ||||||
|  | 		getpeername(con->write_queue.bfd.fd, (struct sockaddr *) &sock, &len); | ||||||
|  | 		vty_out(vty, "BSC lac: %d, %d auth: %d fd: %d peername: %s%s", | ||||||
|  | 			con->cfg ? con->cfg->nr : -1, | ||||||
|  | 			con->cfg ? con->cfg->lac : -1, | ||||||
|  | 			con->authenticated, con->write_queue.bfd.fd, | ||||||
|  | 			inet_ntoa(sock.sin_addr), VTY_NEWLINE); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return CMD_SUCCESS; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | DEFUN(show_bsc_cfg, show_bsc_cfg_cmd, "show bsc config", | ||||||
|  |       SHOW_STR "Display information about known BSC configs") | ||||||
|  | { | ||||||
|  | 	struct bsc_config *conf; | ||||||
|  | 	llist_for_each_entry(conf, &_nat->bsc_configs, entry) { | ||||||
|  | 		vty_out(vty, "BSC token: '%s' lac: %u nr: %u%s", | ||||||
|  | 			conf->token, conf->lac, conf->nr, VTY_NEWLINE); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return CMD_SUCCESS; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | DEFUN(cfg_nat, cfg_nat_cmd, "nat", "Configute the NAT") | ||||||
|  | { | ||||||
|  | 	vty->index = _nat; | ||||||
|  | 	vty->node = NAT_NODE; | ||||||
|  |  | ||||||
|  | 	return CMD_SUCCESS; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* per BSC configuration */ | ||||||
|  | DEFUN(cfg_bsc, cfg_bsc_cmd, "bsc BSC_NR", "Select a BSC to configure\n") | ||||||
|  | { | ||||||
|  | 	int bsc_nr = atoi(argv[0]); | ||||||
|  | 	struct bsc_config *bsc; | ||||||
|  |  | ||||||
|  | 	if (bsc_nr > _nat->num_bsc) { | ||||||
|  | 		vty_out(vty, "%% The next unused BSC number is %u%s", | ||||||
|  | 			_nat->num_bsc, VTY_NEWLINE); | ||||||
|  | 		return CMD_WARNING; | ||||||
|  | 	} else if (bsc_nr == _nat->num_bsc) { | ||||||
|  | 		/* allocate a new one */ | ||||||
|  | 		bsc = bsc_config_alloc(_nat, "unknown", 0); | ||||||
|  | 	} else | ||||||
|  | 		bsc = bsc_config_num(_nat, bsc_nr); | ||||||
|  |  | ||||||
|  | 	if (!bsc) | ||||||
|  | 		return CMD_WARNING; | ||||||
|  |  | ||||||
|  | 	vty->index = bsc; | ||||||
|  | 	vty->node = BSC_NODE; | ||||||
|  |  | ||||||
|  | 	return CMD_SUCCESS; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | DEFUN(cfg_bsc_token, cfg_bsc_token_cmd, "token TOKEN", "Set the token") | ||||||
|  | { | ||||||
|  | 	struct bsc_config *conf = vty->index; | ||||||
|  |  | ||||||
|  | 	if (conf->token) | ||||||
|  | 	    talloc_free(conf->token); | ||||||
|  | 	conf->token = talloc_strdup(conf, argv[0]); | ||||||
|  | 	return CMD_SUCCESS; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | DEFUN(cfg_bsc_lac, cfg_bsc_lac_cmd, "location_area_code <0-65535>", | ||||||
|  |       "Set the Location Area Code (LAC) of this BSC\n") | ||||||
|  | { | ||||||
|  | 	struct bsc_config *tmp; | ||||||
|  | 	struct bsc_config *conf = vty->index; | ||||||
|  |  | ||||||
|  | 	int lac = atoi(argv[0]); | ||||||
|  |  | ||||||
|  | 	if (lac < 0 || lac > 0xffff) { | ||||||
|  | 		vty_out(vty, "%% LAC %d is not in the valid range (0-65535)%s", | ||||||
|  | 			lac, VTY_NEWLINE); | ||||||
|  | 		return CMD_WARNING; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if (lac == GSM_LAC_RESERVED_DETACHED || lac == GSM_LAC_RESERVED_ALL_BTS) { | ||||||
|  | 		vty_out(vty, "%% LAC %d is reserved by GSM 04.08%s", | ||||||
|  | 			lac, VTY_NEWLINE); | ||||||
|  | 		return CMD_WARNING; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* verify that the LACs are unique */ | ||||||
|  | 	llist_for_each_entry(tmp, &_nat->bsc_configs, entry) { | ||||||
|  | 		if (tmp->lac == lac) { | ||||||
|  | 			vty_out(vty, "%% LAC %d is already used.%s", lac, VTY_NEWLINE); | ||||||
|  | 			return CMD_ERR_INCOMPLETE; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	conf->lac = lac; | ||||||
|  |  | ||||||
|  | 	return CMD_SUCCESS; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int bsc_nat_vty_init(struct bsc_nat *nat) | ||||||
|  | { | ||||||
|  | 	_nat = nat; | ||||||
|  |  | ||||||
|  | 	cmd_init(1); | ||||||
|  | 	vty_init(); | ||||||
|  |  | ||||||
|  | 	/* show commands */ | ||||||
|  | 	install_element(VIEW_NODE, &show_sccp_cmd); | ||||||
|  | 	install_element(VIEW_NODE, &show_bsc_cmd); | ||||||
|  | 	install_element(VIEW_NODE, &show_bsc_cfg_cmd); | ||||||
|  |  | ||||||
|  | 	openbsc_vty_add_cmds(); | ||||||
|  |  | ||||||
|  | 	/* nat group */ | ||||||
|  | 	install_element(CONFIG_NODE, &cfg_nat_cmd); | ||||||
|  | 	install_node(&nat_node, config_write_nat); | ||||||
|  | 	install_default(NAT_NODE); | ||||||
|  |  | ||||||
|  | 	/* BSC subgroups */ | ||||||
|  | 	install_element(NAT_NODE, &cfg_bsc_cmd); | ||||||
|  | 	install_node(&bsc_node, config_write_bsc); | ||||||
|  | 	install_default(BSC_NODE); | ||||||
|  | 	install_element(BSC_NODE, &cfg_bsc_token_cmd); | ||||||
|  | 	install_element(BSC_NODE, &cfg_bsc_lac_cmd); | ||||||
|  |  | ||||||
|  | 	mgcp_vty_init(); | ||||||
|  |  | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | /* called by the telnet interface... we have our own init above */ | ||||||
|  | void bsc_vty_init() | ||||||
|  | {} | ||||||
							
								
								
									
										203
									
								
								openbsc/src/nat/bsc_sccp.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										203
									
								
								openbsc/src/nat/bsc_sccp.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,203 @@ | |||||||
|  | /* SCCP patching and handling routines */ | ||||||
|  | /* | ||||||
|  |  * (C) 2010 by Holger Hans Peter Freyther <zecke@selfish.org> | ||||||
|  |  * (C) 2010 by On-Waves | ||||||
|  |  * All Rights Reserved | ||||||
|  |  * | ||||||
|  |  * This program is free software; you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation; either version 2 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License along | ||||||
|  |  * with this program; if not, write to the Free Software Foundation, Inc., | ||||||
|  |  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #include <openbsc/debug.h> | ||||||
|  | #include <openbsc/bsc_nat.h> | ||||||
|  |  | ||||||
|  | #include <sccp/sccp.h> | ||||||
|  |  | ||||||
|  | #include <osmocore/talloc.h> | ||||||
|  |  | ||||||
|  | #include <string.h> | ||||||
|  |  | ||||||
|  | static int equal(struct sccp_source_reference *ref1, struct sccp_source_reference *ref2) | ||||||
|  | { | ||||||
|  | 	return memcmp(ref1, ref2, sizeof(*ref1)) == 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * SCCP patching below | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | /* check if we are using this ref for patched already */ | ||||||
|  | static int sccp_ref_is_free(struct sccp_source_reference *ref, struct bsc_nat *nat) | ||||||
|  | { | ||||||
|  | 	struct sccp_connections *conn; | ||||||
|  |  | ||||||
|  | 	llist_for_each_entry(conn, &nat->sccp_connections, list_entry) { | ||||||
|  | 		if (memcmp(ref, &conn->patched_ref, sizeof(*ref)) == 0) | ||||||
|  | 			return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* copied from sccp.c */ | ||||||
|  | static int assign_src_local_reference(struct sccp_source_reference *ref, struct bsc_nat *nat) | ||||||
|  | { | ||||||
|  | 	static u_int32_t last_ref = 0x50000; | ||||||
|  | 	int wrapped = 0; | ||||||
|  |  | ||||||
|  | 	do { | ||||||
|  | 		struct sccp_source_reference reference; | ||||||
|  | 		reference.octet1 = (last_ref >>  0) & 0xff; | ||||||
|  | 		reference.octet2 = (last_ref >>  8) & 0xff; | ||||||
|  | 		reference.octet3 = (last_ref >> 16) & 0xff; | ||||||
|  |  | ||||||
|  | 		++last_ref; | ||||||
|  | 		/* do not use the reversed word and wrap around */ | ||||||
|  | 		if ((last_ref & 0x00FFFFFF) == 0x00FFFFFF) { | ||||||
|  | 			LOGP(DNAT, LOGL_NOTICE, "Wrapped searching for a free code\n"); | ||||||
|  | 			last_ref = 0; | ||||||
|  | 			++wrapped; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (sccp_ref_is_free(&reference, nat) == 0) { | ||||||
|  | 			*ref = reference; | ||||||
|  | 			return 0; | ||||||
|  | 		} | ||||||
|  | 	} while (wrapped != 2); | ||||||
|  |  | ||||||
|  | 	LOGP(DNAT, LOGL_ERROR, "Finding a free reference failed\n"); | ||||||
|  | 	return -1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int create_sccp_src_ref(struct bsc_connection *bsc, struct msgb *msg, struct bsc_nat_parsed *parsed) | ||||||
|  | { | ||||||
|  | 	struct sccp_connections *conn; | ||||||
|  |  | ||||||
|  | 	conn = talloc_zero(bsc->nat, struct sccp_connections); | ||||||
|  | 	if (!conn) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Memory allocation failure.\n"); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	conn->bsc = bsc; | ||||||
|  | 	conn->real_ref = *parsed->src_local_ref; | ||||||
|  | 	if (assign_src_local_reference(&conn->patched_ref, bsc->nat) != 0) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "Failed to assign a ref.\n"); | ||||||
|  | 		talloc_free(conn); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	llist_add(&conn->list_entry, &bsc->nat->sccp_connections); | ||||||
|  |  | ||||||
|  | 	LOGP(DNAT, LOGL_DEBUG, "Created 0x%x <-> 0x%x mapping for con %p\n", | ||||||
|  | 	     sccp_src_ref_to_int(&conn->real_ref), | ||||||
|  | 	     sccp_src_ref_to_int(&conn->patched_ref), bsc); | ||||||
|  |  | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int update_sccp_src_ref(struct sccp_connections *sccp, struct bsc_nat_parsed *parsed) | ||||||
|  | { | ||||||
|  | 	if (!parsed->dest_local_ref || !parsed->src_local_ref) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "CC MSG should contain both local and dest address.\n"); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	sccp->remote_ref = *parsed->src_local_ref; | ||||||
|  | 	LOGP(DNAT, LOGL_DEBUG, "Updating 0x%x to remote 0x%x on %p\n", | ||||||
|  | 	     sccp_src_ref_to_int(&sccp->patched_ref), | ||||||
|  | 	     sccp_src_ref_to_int(&sccp->remote_ref), sccp->bsc); | ||||||
|  |  | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void remove_sccp_src_ref(struct bsc_connection *bsc, struct msgb *msg, struct bsc_nat_parsed *parsed) | ||||||
|  | { | ||||||
|  | 	struct sccp_connections *conn; | ||||||
|  |  | ||||||
|  | 	llist_for_each_entry(conn, &bsc->nat->sccp_connections, list_entry) { | ||||||
|  | 		if (memcmp(parsed->src_local_ref, | ||||||
|  | 			   &conn->patched_ref, sizeof(conn->patched_ref)) == 0) { | ||||||
|  |  | ||||||
|  | 			sccp_connection_destroy(conn); | ||||||
|  | 			return; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	LOGP(DNAT, LOGL_ERROR, "Can not remove connection: 0x%x\n", | ||||||
|  | 	     sccp_src_ref_to_int(parsed->src_local_ref)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * We have a message from the MSC to the BSC. The MSC is using | ||||||
|  |  * an address that was assigned by the MUX, we need to update the | ||||||
|  |  * dest reference to the real network. | ||||||
|  |  */ | ||||||
|  | struct sccp_connections *patch_sccp_src_ref_to_bsc(struct msgb *msg, | ||||||
|  | 						   struct bsc_nat_parsed *parsed, | ||||||
|  | 						   struct bsc_nat *nat) | ||||||
|  | { | ||||||
|  | 	struct sccp_connections *conn; | ||||||
|  |  | ||||||
|  | 	if (!parsed->dest_local_ref) { | ||||||
|  | 		LOGP(DNAT, LOGL_ERROR, "MSG should contain dest_local_ref.\n"); | ||||||
|  | 		return NULL; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	llist_for_each_entry(conn, &nat->sccp_connections, list_entry) { | ||||||
|  | 		if (!equal(parsed->dest_local_ref, &conn->patched_ref)) | ||||||
|  | 			continue; | ||||||
|  |  | ||||||
|  | 		/* Change the dest address to the real one */ | ||||||
|  | 		*parsed->dest_local_ref = conn->real_ref; | ||||||
|  | 		return conn; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return NULL; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * These are message to the MSC. We will need to find the BSC | ||||||
|  |  * Connection by either the SRC or the DST local reference. | ||||||
|  |  * | ||||||
|  |  * In case of a CR we need to work by the SRC local reference | ||||||
|  |  * in all other cases we need to work by the destination local | ||||||
|  |  * reference.. | ||||||
|  |  */ | ||||||
|  | struct sccp_connections *patch_sccp_src_ref_to_msc(struct msgb *msg, | ||||||
|  | 						   struct bsc_nat_parsed *parsed, | ||||||
|  | 						   struct bsc_nat *nat) | ||||||
|  | { | ||||||
|  | 	struct sccp_connections *conn; | ||||||
|  |  | ||||||
|  | 	llist_for_each_entry(conn, &nat->sccp_connections, list_entry) { | ||||||
|  | 		if (parsed->src_local_ref) { | ||||||
|  | 			if (equal(parsed->src_local_ref, &conn->real_ref)) { | ||||||
|  | 				*parsed->src_local_ref = conn->patched_ref; | ||||||
|  | 				return conn; | ||||||
|  | 			} | ||||||
|  | 		} else if (parsed->dest_local_ref) { | ||||||
|  | 			if (equal(parsed->dest_local_ref, &conn->remote_ref)) | ||||||
|  | 				return conn; | ||||||
|  | 		} else { | ||||||
|  | 			LOGP(DNAT, LOGL_ERROR, "Header has neither loc/dst ref.\n"); | ||||||
|  | 			return NULL; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return NULL; | ||||||
|  | } | ||||||
|  |  | ||||||
| @@ -160,7 +160,7 @@ static void paging_handle_pending_requests(struct gsm_bts_paging_state *paging_b | |||||||
| 	} while (paging_bts->available_slots > 0 | 	} while (paging_bts->available_slots > 0 | ||||||
| 		    &&  initial_request != current_request); | 		    &&  initial_request != current_request); | ||||||
|  |  | ||||||
| 	bsc_schedule_timer(&paging_bts->work_timer, 1, 0); | 	bsc_schedule_timer(&paging_bts->work_timer, 2, 0); | ||||||
| } | } | ||||||
|  |  | ||||||
| static void paging_worker(void *data) | static void paging_worker(void *data) | ||||||
| @@ -208,11 +208,11 @@ static void paging_T3113_expired(void *data) | |||||||
| 	sig_data.lchan	= NULL; | 	sig_data.lchan	= NULL; | ||||||
|  |  | ||||||
| 	/* must be destroyed before calling cbfn, to prevent double free */ | 	/* must be destroyed before calling cbfn, to prevent double free */ | ||||||
|  | 	counter_inc(req->bts->network->stats.paging.expired); | ||||||
| 	cbfn_param = req->cbfn_param; | 	cbfn_param = req->cbfn_param; | ||||||
| 	cbfn = req->cbfn; | 	cbfn = req->cbfn; | ||||||
| 	paging_remove_request(&req->bts->paging, req); | 	paging_remove_request(&req->bts->paging, req); | ||||||
|  |  | ||||||
| 	counter_inc(req->bts->network->stats.paging.expired); |  | ||||||
|  |  | ||||||
| 	dispatch_signal(SS_PAGING, S_PAGING_EXPIRED, &sig_data); | 	dispatch_signal(SS_PAGING, S_PAGING_EXPIRED, &sig_data); | ||||||
| 	if (cbfn) | 	if (cbfn) | ||||||
| @@ -245,7 +245,7 @@ static int _paging_request(struct gsm_bts *bts, struct gsm_subscriber *subscr, | |||||||
| 	llist_add_tail(&req->entry, &bts_entry->pending_requests); | 	llist_add_tail(&req->entry, &bts_entry->pending_requests); | ||||||
|  |  | ||||||
| 	if (!bsc_timer_pending(&bts_entry->work_timer)) | 	if (!bsc_timer_pending(&bts_entry->work_timer)) | ||||||
| 		bsc_schedule_timer(&bts_entry->work_timer, 1, 0); | 		bsc_schedule_timer(&bts_entry->work_timer, 2, 0); | ||||||
|  |  | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -45,7 +45,7 @@ const struct sockaddr_sccp sccp_ssn_bssap = { | |||||||
|  |  | ||||||
| struct sccp_system { | struct sccp_system { | ||||||
| 	/* layer3 -> layer2 */ | 	/* layer3 -> layer2 */ | ||||||
| 	int (*write_data)(struct msgb *data, void *context); | 	void (*write_data)(struct msgb *data, void *context); | ||||||
| 	void *write_context; | 	void *write_context; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| @@ -91,9 +91,9 @@ static struct sccp_data_callback *_find_ssn(u_int8_t ssn) | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| static int _send_msg(struct msgb *msg) | static void _send_msg(struct msgb *msg) | ||||||
| { | { | ||||||
| 	return sccp_system.write_data(msg, sccp_system.write_context); | 	sccp_system.write_data(msg, sccp_system.write_context); | ||||||
| } | } | ||||||
|  |  | ||||||
| /* | /* | ||||||
| @@ -499,7 +499,6 @@ static int _sccp_send_data(int class, const struct sockaddr_sccp *in, | |||||||
| { | { | ||||||
| 	struct sccp_data_unitdata *udt; | 	struct sccp_data_unitdata *udt; | ||||||
| 	u_int8_t *data; | 	u_int8_t *data; | ||||||
| 	int ret; |  | ||||||
|  |  | ||||||
| 	if (msgb_l3len(payload) > 256) { | 	if (msgb_l3len(payload) > 256) { | ||||||
| 		DEBUGP(DSCCP, "The payload is too big for one udt\n"); | 		DEBUGP(DSCCP, "The payload is too big for one udt\n"); | ||||||
| @@ -533,10 +532,8 @@ static int _sccp_send_data(int class, const struct sockaddr_sccp *in, | |||||||
| 	data[0] = msgb_l3len(payload); | 	data[0] = msgb_l3len(payload); | ||||||
| 	memcpy(&data[1], payload->l3h, msgb_l3len(payload)); | 	memcpy(&data[1], payload->l3h, msgb_l3len(payload)); | ||||||
|  |  | ||||||
| 	ret = _send_msg(msg); | 	_send_msg(msg); | ||||||
| 	msgb_free(msg); | 	return 0; | ||||||
|  |  | ||||||
| 	return ret; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| static int _sccp_handle_read(struct msgb *msgb) | static int _sccp_handle_read(struct msgb *msgb) | ||||||
| @@ -627,7 +624,6 @@ static int _sccp_send_refuse(struct sccp_source_reference *src_ref, int cause) | |||||||
| 	struct msgb *msgb; | 	struct msgb *msgb; | ||||||
| 	struct sccp_connection_refused *ref; | 	struct sccp_connection_refused *ref; | ||||||
| 	u_int8_t *data; | 	u_int8_t *data; | ||||||
| 	int ret; |  | ||||||
|  |  | ||||||
| 	msgb = msgb_alloc_headroom(SCCP_MSG_SIZE, | 	msgb = msgb_alloc_headroom(SCCP_MSG_SIZE, | ||||||
| 				   SCCP_MSG_HEADROOM, "sccp ref"); | 				   SCCP_MSG_HEADROOM, "sccp ref"); | ||||||
| @@ -643,9 +639,8 @@ static int _sccp_send_refuse(struct sccp_source_reference *src_ref, int cause) | |||||||
| 	data = msgb_put(msgb, 1); | 	data = msgb_put(msgb, 1); | ||||||
| 	data[0] = SCCP_PNC_END_OF_OPTIONAL; | 	data[0] = SCCP_PNC_END_OF_OPTIONAL; | ||||||
|  |  | ||||||
| 	ret = _send_msg(msgb); | 	_send_msg(msgb); | ||||||
| 	msgb_free(msgb); | 	return 0; | ||||||
| 	return ret; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| static int _sccp_send_connection_confirm(struct sccp_connection *connection) | static int _sccp_send_connection_confirm(struct sccp_connection *connection) | ||||||
| @@ -653,7 +648,6 @@ static int _sccp_send_connection_confirm(struct sccp_connection *connection) | |||||||
| 	struct msgb *response; | 	struct msgb *response; | ||||||
| 	struct sccp_connection_confirm *confirm; | 	struct sccp_connection_confirm *confirm; | ||||||
| 	u_int8_t *optional_data; | 	u_int8_t *optional_data; | ||||||
| 	int ret; |  | ||||||
|  |  | ||||||
| 	if (assign_source_local_reference(connection) != 0) | 	if (assign_source_local_reference(connection) != 0) | ||||||
| 		return -1; | 		return -1; | ||||||
| @@ -677,11 +671,9 @@ static int _sccp_send_connection_confirm(struct sccp_connection *connection) | |||||||
| 	optional_data = (u_int8_t *) msgb_put(response, 1); | 	optional_data = (u_int8_t *) msgb_put(response, 1); | ||||||
| 	optional_data[0] = SCCP_PNC_END_OF_OPTIONAL; | 	optional_data[0] = SCCP_PNC_END_OF_OPTIONAL; | ||||||
|  |  | ||||||
| 	ret = _send_msg(response); | 	_send_msg(response); | ||||||
| 	msgb_free(response); |  | ||||||
|  |  | ||||||
| 	_sccp_set_connection_state(connection, SCCP_CONNECTION_STATE_ESTABLISHED); | 	_sccp_set_connection_state(connection, SCCP_CONNECTION_STATE_ESTABLISHED); | ||||||
| 	return ret; | 	return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| static int _sccp_send_connection_request(struct sccp_connection *connection, | static int _sccp_send_connection_request(struct sccp_connection *connection, | ||||||
| @@ -691,7 +683,6 @@ static int _sccp_send_connection_request(struct sccp_connection *connection, | |||||||
| 	struct sccp_connection_request *req; | 	struct sccp_connection_request *req; | ||||||
| 	u_int8_t *data; | 	u_int8_t *data; | ||||||
| 	u_int8_t extra_size = 3 + 1; | 	u_int8_t extra_size = 3 + 1; | ||||||
| 	int ret; |  | ||||||
|  |  | ||||||
|  |  | ||||||
| 	if (msg && (msgb_l3len(msg) < 3 || msgb_l3len(msg) > 130)) { | 	if (msg && (msgb_l3len(msg) < 3 || msgb_l3len(msg) > 130)) { | ||||||
| @@ -741,10 +732,8 @@ static int _sccp_send_connection_request(struct sccp_connection *connection, | |||||||
| 	llist_add_tail(&connection->list, &sccp_connections); | 	llist_add_tail(&connection->list, &sccp_connections); | ||||||
| 	_sccp_set_connection_state(connection, SCCP_CONNECTION_STATE_REQUEST); | 	_sccp_set_connection_state(connection, SCCP_CONNECTION_STATE_REQUEST); | ||||||
|  |  | ||||||
| 	ret = _send_msg(request); | 	_send_msg(request); | ||||||
| 	msgb_free(request); | 	return 0; | ||||||
|  |  | ||||||
| 	return ret; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| static int _sccp_send_connection_data(struct sccp_connection *conn, struct msgb *_data) | static int _sccp_send_connection_data(struct sccp_connection *conn, struct msgb *_data) | ||||||
| @@ -753,7 +742,6 @@ static int _sccp_send_connection_data(struct sccp_connection *conn, struct msgb | |||||||
| 	struct sccp_data_form1 *dt1; | 	struct sccp_data_form1 *dt1; | ||||||
| 	u_int8_t *data; | 	u_int8_t *data; | ||||||
| 	int extra_size; | 	int extra_size; | ||||||
| 	int ret; |  | ||||||
|  |  | ||||||
| 	if (msgb_l3len(_data) < 2 || msgb_l3len(_data) > 256) { | 	if (msgb_l3len(_data) < 2 || msgb_l3len(_data) > 256) { | ||||||
| 		DEBUGP(DSCCP, "data size too big, segmenting unimplemented.\n"); | 		DEBUGP(DSCCP, "data size too big, segmenting unimplemented.\n"); | ||||||
| @@ -777,17 +765,14 @@ static int _sccp_send_connection_data(struct sccp_connection *conn, struct msgb | |||||||
| 	data[0] = extra_size - 1; | 	data[0] = extra_size - 1; | ||||||
| 	memcpy(&data[1], _data->l3h, extra_size - 1); | 	memcpy(&data[1], _data->l3h, extra_size - 1); | ||||||
|  |  | ||||||
| 	ret = _send_msg(msgb); | 	_send_msg(msgb); | ||||||
| 	msgb_free(msgb); | 	return 0; | ||||||
|  |  | ||||||
| 	return ret; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| static int _sccp_send_connection_it(struct sccp_connection *conn) | static int _sccp_send_connection_it(struct sccp_connection *conn) | ||||||
| { | { | ||||||
| 	struct msgb *msgb; | 	struct msgb *msgb; | ||||||
| 	struct sccp_data_it *it; | 	struct sccp_data_it *it; | ||||||
| 	int ret; |  | ||||||
|  |  | ||||||
| 	msgb = msgb_alloc_headroom(SCCP_MSG_SIZE, | 	msgb = msgb_alloc_headroom(SCCP_MSG_SIZE, | ||||||
| 				   SCCP_MSG_HEADROOM, "sccp it"); | 				   SCCP_MSG_HEADROOM, "sccp it"); | ||||||
| @@ -803,9 +788,8 @@ static int _sccp_send_connection_it(struct sccp_connection *conn) | |||||||
| 	it->sequencing[0] = it->sequencing[1] = 0; | 	it->sequencing[0] = it->sequencing[1] = 0; | ||||||
| 	it->credit = 0; | 	it->credit = 0; | ||||||
|  |  | ||||||
| 	ret = _send_msg(msgb); | 	_send_msg(msgb); | ||||||
| 	msgb_free(msgb); | 	return 0; | ||||||
| 	return ret; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| static int _sccp_send_connection_released(struct sccp_connection *conn, int cause) | static int _sccp_send_connection_released(struct sccp_connection *conn, int cause) | ||||||
| @@ -813,7 +797,6 @@ static int _sccp_send_connection_released(struct sccp_connection *conn, int caus | |||||||
| 	struct msgb *msg; | 	struct msgb *msg; | ||||||
| 	struct sccp_connection_released *rel; | 	struct sccp_connection_released *rel; | ||||||
| 	u_int8_t *data; | 	u_int8_t *data; | ||||||
| 	int ret; |  | ||||||
|  |  | ||||||
| 	msg = msgb_alloc_headroom(SCCP_MSG_SIZE, SCCP_MSG_HEADROOM, | 	msg = msgb_alloc_headroom(SCCP_MSG_SIZE, SCCP_MSG_HEADROOM, | ||||||
| 				  "sccp: connection released"); | 				  "sccp: connection released"); | ||||||
| @@ -832,10 +815,8 @@ static int _sccp_send_connection_released(struct sccp_connection *conn, int caus | |||||||
| 	data[0] = SCCP_PNC_END_OF_OPTIONAL; | 	data[0] = SCCP_PNC_END_OF_OPTIONAL; | ||||||
|  |  | ||||||
| 	_sccp_set_connection_state(conn, SCCP_CONNECTION_STATE_RELEASE); | 	_sccp_set_connection_state(conn, SCCP_CONNECTION_STATE_RELEASE); | ||||||
| 	ret = _send_msg(msg); | 	_send_msg(msg); | ||||||
| 	msgb_free(msg); | 	return 0; | ||||||
|  |  | ||||||
| 	return ret; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| /* | /* | ||||||
| @@ -982,7 +963,6 @@ static int _sccp_send_connection_release_complete(struct sccp_connection *connec | |||||||
| { | { | ||||||
| 	struct msgb *msgb; | 	struct msgb *msgb; | ||||||
| 	struct sccp_connection_release_complete *rlc; | 	struct sccp_connection_release_complete *rlc; | ||||||
| 	int ret; |  | ||||||
|  |  | ||||||
| 	msgb = msgb_alloc_headroom(SCCP_MSG_SIZE, | 	msgb = msgb_alloc_headroom(SCCP_MSG_SIZE, | ||||||
| 				   SCCP_MSG_HEADROOM, "sccp rlc"); | 				   SCCP_MSG_HEADROOM, "sccp rlc"); | ||||||
| @@ -995,8 +975,7 @@ static int _sccp_send_connection_release_complete(struct sccp_connection *connec | |||||||
| 	memcpy(&rlc->source_local_reference, | 	memcpy(&rlc->source_local_reference, | ||||||
| 	       &connection->source_local_reference, sizeof(struct sccp_source_reference)); | 	       &connection->source_local_reference, sizeof(struct sccp_source_reference)); | ||||||
|  |  | ||||||
| 	ret = _send_msg(msgb); | 	_send_msg(msgb); | ||||||
| 	msgb_free(msgb); |  | ||||||
|  |  | ||||||
| 	/* | 	/* | ||||||
| 	 * Remove from the list of active connections and set the state. User code | 	 * Remove from the list of active connections and set the state. User code | ||||||
| @@ -1004,8 +983,7 @@ static int _sccp_send_connection_release_complete(struct sccp_connection *connec | |||||||
| 	 */ | 	 */ | ||||||
| 	llist_del(&connection->list); | 	llist_del(&connection->list); | ||||||
| 	_sccp_set_connection_state(connection, SCCP_CONNECTION_STATE_RELEASE_COMPLETE); | 	_sccp_set_connection_state(connection, SCCP_CONNECTION_STATE_RELEASE_COMPLETE); | ||||||
|  | 	return 0; | ||||||
| 	return ret; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| /* connection released, send a released confirm */ | /* connection released, send a released confirm */ | ||||||
| @@ -1118,7 +1096,7 @@ found: | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| int sccp_system_init(int (*outgoing)(struct msgb *data, void *ctx), void *ctx) | int sccp_system_init(void (*outgoing)(struct msgb *data, void *ctx), void *ctx) | ||||||
| { | { | ||||||
| 	sccp_system.write_data = outgoing; | 	sccp_system.write_data = outgoing; | ||||||
| 	sccp_system.write_context = ctx; | 	sccp_system.write_context = ctx; | ||||||
| @@ -1220,6 +1198,17 @@ int sccp_connection_free(struct sccp_connection *connection) | |||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | int sccp_connection_force_free(struct sccp_connection *con) | ||||||
|  | { | ||||||
|  | 	if (con->connection_state > SCCP_CONNECTION_STATE_NONE && | ||||||
|  | 	    con->connection_state < SCCP_CONNECTION_STATE_RELEASE_COMPLETE) | ||||||
|  | 		llist_del(&con->list); | ||||||
|  |  | ||||||
|  | 	con->connection_state = SCCP_CONNECTION_STATE_REFUSED; | ||||||
|  | 	sccp_connection_free(con); | ||||||
|  |         return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
| struct sccp_connection *sccp_connection_socket(void) | struct sccp_connection *sccp_connection_socket(void) | ||||||
| { | { | ||||||
| 	return talloc_zero(tall_sccp_ctx, struct sccp_connection); | 	return talloc_zero(tall_sccp_ctx, struct sccp_connection); | ||||||
|   | |||||||
| @@ -140,7 +140,7 @@ int gsm_silent_call_stop(struct gsm_subscriber *subscr) | |||||||
| 	if (!conn->silent_call) | 	if (!conn->silent_call) | ||||||
| 		return -EINVAL; | 		return -EINVAL; | ||||||
|  |  | ||||||
| 	put_subscr_con(conn); | 	put_subscr_con(conn, 0); | ||||||
|  |  | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -28,6 +28,7 @@ | |||||||
| #include <openbsc/gsm_04_08.h> | #include <openbsc/gsm_04_08.h> | ||||||
| #include <openbsc/mncc.h> | #include <openbsc/mncc.h> | ||||||
| #include <openbsc/paging.h> | #include <openbsc/paging.h> | ||||||
|  | #include <openbsc/chan_alloc.h> | ||||||
|  |  | ||||||
| void *tall_trans_ctx; | void *tall_trans_ctx; | ||||||
|  |  | ||||||
| @@ -95,14 +96,14 @@ void trans_free(struct gsm_trans *trans) | |||||||
| 		break; | 		break; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if (trans->conn) |  | ||||||
| 		put_subscr_con(trans->conn); |  | ||||||
|  |  | ||||||
| 	if (!trans->conn && trans->subscr && trans->subscr->net) { | 	if (!trans->conn && trans->subscr && trans->subscr->net) { | ||||||
| 		/* Stop paging on all bts' */ | 		/* Stop paging on all bts' */ | ||||||
| 		paging_request_stop(NULL, trans->subscr, NULL); | 		paging_request_stop(NULL, trans->subscr, NULL); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	if (trans->conn) | ||||||
|  | 		put_subscr_con(trans->conn, 0); | ||||||
|  |  | ||||||
| 	if (trans->subscr) | 	if (trans->subscr) | ||||||
| 		subscr_put(trans->subscr); | 		subscr_put(trans->subscr); | ||||||
|  |  | ||||||
| @@ -159,7 +160,7 @@ int trans_lchan_change(struct gsm_subscriber_connection *conn_old, | |||||||
| 		if (trans->conn == conn_old) { | 		if (trans->conn == conn_old) { | ||||||
|  |  | ||||||
| 			/* drop old channel use count */ | 			/* drop old channel use count */ | ||||||
| 			put_subscr_con(conn_old); | 			put_subscr_con(conn_old, 0); | ||||||
| 			/* assign new channel */ | 			/* assign new channel */ | ||||||
| 			trans->conn = conn_new; | 			trans->conn = conn_new; | ||||||
| 			/* bump new channel use count */ | 			/* bump new channel use count */ | ||||||
|   | |||||||
| @@ -47,6 +47,7 @@ Boston, MA 02111-1307, USA.  */ | |||||||
|  |  | ||||||
| #include <openbsc/gsm_data.h> | #include <openbsc/gsm_data.h> | ||||||
| #include <openbsc/gsm_subscriber.h> | #include <openbsc/gsm_subscriber.h> | ||||||
|  | #include <openbsc/bsc_nat.h> | ||||||
| #include <osmocore/talloc.h> | #include <osmocore/talloc.h> | ||||||
|  |  | ||||||
| void *tall_vty_cmd_ctx; | void *tall_vty_cmd_ctx; | ||||||
| @@ -1949,6 +1950,13 @@ enum node_type vty_go_parent(struct vty *vty) | |||||||
| 		subscr_put(vty->index); | 		subscr_put(vty->index); | ||||||
| 		vty->index = NULL; | 		vty->index = NULL; | ||||||
| 		break; | 		break; | ||||||
|  | 	case BSC_NODE: | ||||||
|  | 		vty->node = NAT_NODE; | ||||||
|  | 		{ | ||||||
|  | 			struct bsc_config *bsc = vty->index; | ||||||
|  | 			vty->index = bsc->nat; | ||||||
|  | 		} | ||||||
|  | 		break; | ||||||
| 	default: | 	default: | ||||||
| 		vty->node = CONFIG_NODE; | 		vty->node = CONFIG_NODE; | ||||||
| 	} | 	} | ||||||
| @@ -2362,6 +2370,18 @@ DEFUN(config_exit, | |||||||
| 	case VTY_NODE: | 	case VTY_NODE: | ||||||
| 		vty->node = CONFIG_NODE; | 		vty->node = CONFIG_NODE; | ||||||
| 		break; | 		break; | ||||||
|  | 	case MGCP_NODE: | ||||||
|  | 		vty->node = CONFIG_NODE; | ||||||
|  | 		vty->index = NULL; | ||||||
|  | 	case NAT_NODE: | ||||||
|  | 		vty->node = CONFIG_NODE; | ||||||
|  | 		vty->index = NULL; | ||||||
|  | 		break; | ||||||
|  | 	case BSC_NODE: | ||||||
|  | 		vty->node = NAT_NODE; | ||||||
|  | 		vty->index = NULL; | ||||||
|  | 		break; | ||||||
|  |  | ||||||
| 	default: | 	default: | ||||||
| 		break; | 		break; | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -100,6 +100,7 @@ static void dump_pchan_load_vty(struct vty *vty, char *prefix, | |||||||
|  |  | ||||||
| static void net_dump_vty(struct vty *vty, struct gsm_network *net) | static void net_dump_vty(struct vty *vty, struct gsm_network *net) | ||||||
| { | { | ||||||
|  | 	int i; | ||||||
| 	struct pchan_load pl; | 	struct pchan_load pl; | ||||||
|  |  | ||||||
| 	vty_out(vty, "BSC is on Country Code %u, Network Code %u " | 	vty_out(vty, "BSC is on Country Code %u, Network Code %u " | ||||||
| @@ -117,6 +118,8 @@ static void net_dump_vty(struct vty *vty, struct gsm_network *net) | |||||||
| 		VTY_NEWLINE); | 		VTY_NEWLINE); | ||||||
| 	vty_out(vty, "  NECI (TCH/H): %u%s", net->neci, | 	vty_out(vty, "  NECI (TCH/H): %u%s", net->neci, | ||||||
| 		VTY_NEWLINE); | 		VTY_NEWLINE); | ||||||
|  | 	vty_out(vty, "  Use TCH for Paging any: %d%s", net->pag_any_tch, | ||||||
|  | 		VTY_NEWLINE); | ||||||
| 	vty_out(vty, "  RRLP Mode: %s%s", rrlp_mode_name(net->rrlp.mode), | 	vty_out(vty, "  RRLP Mode: %s%s", rrlp_mode_name(net->rrlp.mode), | ||||||
| 		VTY_NEWLINE); | 		VTY_NEWLINE); | ||||||
| 	vty_out(vty, "  MM Info: %s%s", net->send_mm_info ? "On" : "Off", | 	vty_out(vty, "  MM Info: %s%s", net->send_mm_info ? "On" : "Off", | ||||||
| @@ -126,6 +129,12 @@ static void net_dump_vty(struct vty *vty, struct gsm_network *net) | |||||||
| 	network_chan_load(&pl, net); | 	network_chan_load(&pl, net); | ||||||
| 	vty_out(vty, "  Current Channel Load:%s", VTY_NEWLINE); | 	vty_out(vty, "  Current Channel Load:%s", VTY_NEWLINE); | ||||||
| 	dump_pchan_load_vty(vty, "    ", &pl); | 	dump_pchan_load_vty(vty, "    ", &pl); | ||||||
|  |  | ||||||
|  | 	vty_out(vty, "  Allowed Audio Codecs: "); | ||||||
|  | 	for (i = 0; i < net->audio_length; ++i) | ||||||
|  | 		vty_out(vty, "hr: %d ver: %d, ", | ||||||
|  | 			net->audio_support[i]->hr, net->audio_support[i]->ver); | ||||||
|  | 	vty_out(vty, "%s", VTY_NEWLINE); | ||||||
| } | } | ||||||
|  |  | ||||||
| DEFUN(show_net, show_net_cmd, "show network", | DEFUN(show_net, show_net_cmd, "show network", | ||||||
| @@ -358,7 +367,11 @@ static int config_write_net(struct vty *vty) | |||||||
| { | { | ||||||
| 	vty_out(vty, "network%s", VTY_NEWLINE); | 	vty_out(vty, "network%s", VTY_NEWLINE); | ||||||
| 	vty_out(vty, " network country code %u%s", gsmnet->country_code, VTY_NEWLINE); | 	vty_out(vty, " network country code %u%s", gsmnet->country_code, VTY_NEWLINE); | ||||||
|  | 	if (gsmnet->core_country_code > 0) | ||||||
|  | 		vty_out(vty, " core network country code %u%s", gsmnet->core_country_code, VTY_NEWLINE); | ||||||
| 	vty_out(vty, " mobile network code %u%s", gsmnet->network_code, VTY_NEWLINE); | 	vty_out(vty, " mobile network code %u%s", gsmnet->network_code, VTY_NEWLINE); | ||||||
|  | 	if (gsmnet->core_network_code > 0) | ||||||
|  | 		vty_out(vty, " core mobile network code %u%s", gsmnet->core_network_code, VTY_NEWLINE); | ||||||
| 	vty_out(vty, " short name %s%s", gsmnet->name_short, VTY_NEWLINE); | 	vty_out(vty, " short name %s%s", gsmnet->name_short, VTY_NEWLINE); | ||||||
| 	vty_out(vty, " long name %s%s", gsmnet->name_long, VTY_NEWLINE); | 	vty_out(vty, " long name %s%s", gsmnet->name_long, VTY_NEWLINE); | ||||||
| 	vty_out(vty, " auth policy %s%s", gsm_auth_policy_name(gsmnet->auth_policy), VTY_NEWLINE); | 	vty_out(vty, " auth policy %s%s", gsm_auth_policy_name(gsmnet->auth_policy), VTY_NEWLINE); | ||||||
| @@ -366,6 +379,7 @@ static int config_write_net(struct vty *vty) | |||||||
| 		gsmnet->reject_cause, VTY_NEWLINE); | 		gsmnet->reject_cause, VTY_NEWLINE); | ||||||
| 	vty_out(vty, " encryption a5 %u%s", gsmnet->a5_encryption, VTY_NEWLINE); | 	vty_out(vty, " encryption a5 %u%s", gsmnet->a5_encryption, VTY_NEWLINE); | ||||||
| 	vty_out(vty, " neci %u%s", gsmnet->neci, VTY_NEWLINE); | 	vty_out(vty, " neci %u%s", gsmnet->neci, VTY_NEWLINE); | ||||||
|  | 	vty_out(vty, " paging any use tch %d%s", gsmnet->pag_any_tch, VTY_NEWLINE); | ||||||
| 	vty_out(vty, " rrlp mode %s%s", rrlp_mode_name(gsmnet->rrlp.mode), | 	vty_out(vty, " rrlp mode %s%s", rrlp_mode_name(gsmnet->rrlp.mode), | ||||||
| 		VTY_NEWLINE); | 		VTY_NEWLINE); | ||||||
| 	vty_out(vty, " mm info %u%s", gsmnet->send_mm_info, VTY_NEWLINE); | 	vty_out(vty, " mm info %u%s", gsmnet->send_mm_info, VTY_NEWLINE); | ||||||
| @@ -393,6 +407,28 @@ static int config_write_net(struct vty *vty) | |||||||
| 	vty_out(vty, " timer t3117 %u%s", gsmnet->T3117, VTY_NEWLINE); | 	vty_out(vty, " timer t3117 %u%s", gsmnet->T3117, VTY_NEWLINE); | ||||||
| 	vty_out(vty, " timer t3119 %u%s", gsmnet->T3119, VTY_NEWLINE); | 	vty_out(vty, " timer t3119 %u%s", gsmnet->T3119, VTY_NEWLINE); | ||||||
| 	vty_out(vty, " timer t3141 %u%s", gsmnet->T3141, VTY_NEWLINE); | 	vty_out(vty, " timer t3141 %u%s", gsmnet->T3141, VTY_NEWLINE); | ||||||
|  | 	vty_out(vty, " ipacc rtp_payload %u%s", gsmnet->rtp_payload, VTY_NEWLINE); | ||||||
|  | 	vty_out(vty, " rtp base %u%s", gsmnet->rtp_base_port, VTY_NEWLINE); | ||||||
|  |  | ||||||
|  | 	if (gsmnet->audio_length != 0) { | ||||||
|  | 		int i; | ||||||
|  |  | ||||||
|  | 		vty_out(vty, " codec_list "); | ||||||
|  | 		for (i = 0; i < gsmnet->audio_length; ++i) { | ||||||
|  | 			printf("I... %d %d\n", i, gsmnet->audio_length); | ||||||
|  | 			if (i != 0) | ||||||
|  | 				vty_out(vty, ", "); | ||||||
|  |  | ||||||
|  | 			if (gsmnet->audio_support[i]->hr) | ||||||
|  | 				vty_out(vty, "hr%.1u", gsmnet->audio_support[i]->ver); | ||||||
|  | 			else | ||||||
|  | 				vty_out(vty, "fr%.1u", gsmnet->audio_support[i]->ver); | ||||||
|  | 		} | ||||||
|  | 		vty_out(vty, "%s", VTY_NEWLINE); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if (gsmnet->bsc_token) | ||||||
|  | 		vty_out(vty, " bsc_token %s%s", gsmnet->bsc_token, VTY_NEWLINE); | ||||||
|  |  | ||||||
| 	return CMD_SUCCESS; | 	return CMD_SUCCESS; | ||||||
| } | } | ||||||
| @@ -616,7 +652,7 @@ static void lchan_dump_vty(struct vty *vty, struct gsm_lchan *lchan) | |||||||
| 		vty_out(vty, "  No Subscriber%s", VTY_NEWLINE); | 		vty_out(vty, "  No Subscriber%s", VTY_NEWLINE); | ||||||
| 	if (is_ipaccess_bts(lchan->ts->trx->bts)) { | 	if (is_ipaccess_bts(lchan->ts->trx->bts)) { | ||||||
| 		struct in_addr ia; | 		struct in_addr ia; | ||||||
| 		ia.s_addr = lchan->abis_ip.bound_ip; | 		ia.s_addr = htonl(lchan->abis_ip.bound_ip); | ||||||
| 		vty_out(vty, "  Bound IP: %s Port %u RTP_TYPE2=%u CONN_ID=%u%s", | 		vty_out(vty, "  Bound IP: %s Port %u RTP_TYPE2=%u CONN_ID=%u%s", | ||||||
| 			inet_ntoa(ia), lchan->abis_ip.bound_port, | 			inet_ntoa(ia), lchan->abis_ip.bound_port, | ||||||
| 			lchan->abis_ip.rtp_payload2, lchan->abis_ip.conn_id, | 			lchan->abis_ip.rtp_payload2, lchan->abis_ip.conn_id, | ||||||
| @@ -919,6 +955,50 @@ DEFUN(show_stats, | |||||||
| 	return CMD_SUCCESS; | 	return CMD_SUCCESS; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | DEFUN(drop_bts, | ||||||
|  |       drop_bts_cmd, | ||||||
|  |       "drop bts connection [nr] (oml|rsl)", | ||||||
|  |       SHOW_STR "Debug/Simulation command to drop ipaccess BTS\n") | ||||||
|  | { | ||||||
|  | 	struct gsm_bts_trx *trx; | ||||||
|  | 	struct gsm_bts *bts; | ||||||
|  | 	unsigned int bts_nr; | ||||||
|  |  | ||||||
|  | 	bts_nr = atoi(argv[0]); | ||||||
|  | 	if (bts_nr >= gsmnet->num_bts) { | ||||||
|  | 		vty_out(vty, "BTS number must be between 0 and %d. It was %d.%s", | ||||||
|  | 			gsmnet->num_bts, bts_nr, VTY_NEWLINE); | ||||||
|  | 		return CMD_WARNING; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	bts = gsm_bts_num(gsmnet, bts_nr); | ||||||
|  | 	if (!bts) { | ||||||
|  | 		vty_out(vty, "BTS Nr. %d could not be found.%s", bts_nr, VTY_NEWLINE); | ||||||
|  | 		return CMD_WARNING; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if (!is_ipaccess_bts(bts)) { | ||||||
|  | 		vty_out(vty, "This command only works for ipaccess.%s", VTY_NEWLINE); | ||||||
|  | 		return CMD_WARNING; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	/* close all connections */ | ||||||
|  | 	if (strcmp(argv[1], "oml") == 0) { | ||||||
|  | 		close(bts->oml_link->ts->driver.ipaccess.fd.fd); | ||||||
|  | 	} else if (strcmp(argv[1], "rsl") == 0) { | ||||||
|  | 		/* close all rsl connections */ | ||||||
|  | 		llist_for_each_entry(trx, &bts->trx_list, list) { | ||||||
|  | 			close(trx->rsl_link->ts->driver.ipaccess.fd.fd); | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		vty_out(vty, "Argument must be 'oml# or 'rsl'.%s", VTY_NEWLINE); | ||||||
|  | 		return CMD_WARNING; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return CMD_SUCCESS; | ||||||
|  | } | ||||||
|  |  | ||||||
| DEFUN(cfg_net, | DEFUN(cfg_net, | ||||||
|       cfg_net_cmd, |       cfg_net_cmd, | ||||||
|       "network", |       "network", | ||||||
| @@ -941,6 +1021,16 @@ DEFUN(cfg_net_ncc, | |||||||
| 	return CMD_SUCCESS; | 	return CMD_SUCCESS; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | DEFUN(cfg_core_net_ncc, | ||||||
|  |       cfg_core_net_ncc_cmd, | ||||||
|  |       "core network country code <1-999>", | ||||||
|  |       "Set the GSM country code to be used in the MSC connection") | ||||||
|  | { | ||||||
|  | 	gsmnet->core_country_code = atoi(argv[0]); | ||||||
|  |  | ||||||
|  | 	return CMD_SUCCESS; | ||||||
|  | } | ||||||
|  |  | ||||||
| DEFUN(cfg_net_mnc, | DEFUN(cfg_net_mnc, | ||||||
|       cfg_net_mnc_cmd, |       cfg_net_mnc_cmd, | ||||||
|       "mobile network code <1-999>", |       "mobile network code <1-999>", | ||||||
| @@ -951,6 +1041,16 @@ DEFUN(cfg_net_mnc, | |||||||
| 	return CMD_SUCCESS; | 	return CMD_SUCCESS; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | DEFUN(cfg_core_net_mnc, | ||||||
|  |       cfg_core_net_mnc_cmd, | ||||||
|  |       "core mobile network code <1-999>", | ||||||
|  |       "Set the GSM mobile network code to be used in the MSC connection") | ||||||
|  | { | ||||||
|  | 	gsmnet->core_network_code = atoi(argv[0]); | ||||||
|  |  | ||||||
|  | 	return CMD_SUCCESS; | ||||||
|  | } | ||||||
|  |  | ||||||
| DEFUN(cfg_net_name_short, | DEFUN(cfg_net_name_short, | ||||||
|       cfg_net_name_short_cmd, |       cfg_net_name_short_cmd, | ||||||
|       "short name NAME", |       "short name NAME", | ||||||
| @@ -1015,6 +1115,7 @@ DEFUN(cfg_net_neci, | |||||||
|       "Set if NECI of cell selection is to be set") |       "Set if NECI of cell selection is to be set") | ||||||
| { | { | ||||||
| 	gsmnet->neci = atoi(argv[0]); | 	gsmnet->neci = atoi(argv[0]); | ||||||
|  | 	gsm_net_update_ctype(gsmnet); | ||||||
| 	return CMD_SUCCESS; | 	return CMD_SUCCESS; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -1101,6 +1202,112 @@ DEFUN(cfg_net_ho_max_distance, cfg_net_ho_max_distance_cmd, | |||||||
| 	return CMD_SUCCESS; | 	return CMD_SUCCESS; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | DEFUN(cfg_net_supported_codecs, | ||||||
|  |       cfg_net_supported_codecs_cmd, | ||||||
|  |       "codec_list .LIST", | ||||||
|  |       "Set the three preferred audio codecs.\n" | ||||||
|  |       "Codec List") | ||||||
|  | { | ||||||
|  | 	int saw_fr, saw_hr; | ||||||
|  | 	int i; | ||||||
|  |  | ||||||
|  | 	saw_fr = saw_hr = 0; | ||||||
|  |  | ||||||
|  | 	/* free the old list... if it exists */ | ||||||
|  | 	if (gsmnet->audio_support) { | ||||||
|  | 		talloc_free(gsmnet->audio_support); | ||||||
|  | 		gsmnet->audio_support = NULL; | ||||||
|  | 		gsmnet->audio_length = 0; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* create a new array */ | ||||||
|  | 	gsmnet->audio_support = | ||||||
|  | 			talloc_zero_array(gsmnet, struct gsm_audio_support *, argc); | ||||||
|  | 	gsmnet->audio_length = argc; | ||||||
|  |  | ||||||
|  | 	for (i = 0; i < argc; ++i) { | ||||||
|  | 		/* check for hrX or frX */ | ||||||
|  | 		if (strlen(argv[i]) != 3 | ||||||
|  | 		    || argv[i][1] != 'r' | ||||||
|  | 		    || (argv[i][0] != 'h' && argv[i][0] != 'f') | ||||||
|  | 		    || argv[i][2] < 0x30 | ||||||
|  | 		    || argv[i][2] > 0x39) | ||||||
|  | 			goto error; | ||||||
|  |  | ||||||
|  | 		gsmnet->audio_support[i] = talloc_zero(gsmnet->audio_support, | ||||||
|  | 						       struct gsm_audio_support); | ||||||
|  | 		gsmnet->audio_support[i]->ver = atoi(argv[i] + 2); | ||||||
|  |  | ||||||
|  | 		if (strncmp("hr", argv[i], 2) == 0) { | ||||||
|  | 			gsmnet->audio_support[i]->hr = 1; | ||||||
|  | 			saw_hr = 1; | ||||||
|  | 		} else if (strncmp("fr", argv[i], 2) == 0) { | ||||||
|  | 			gsmnet->audio_support[i]->hr = 0; | ||||||
|  | 			saw_fr = 1; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (saw_hr && saw_fr) { | ||||||
|  | 			vty_out(vty, "Can not have full-rate and half-rate codec.%s", | ||||||
|  | 				VTY_NEWLINE); | ||||||
|  | 			return CMD_ERR_INCOMPLETE; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return CMD_SUCCESS; | ||||||
|  |  | ||||||
|  | error: | ||||||
|  | 	vty_out(vty, "Codec name must be hrX or frX. Was '%s'%s", | ||||||
|  | 		argv[i], VTY_NEWLINE); | ||||||
|  | 	return CMD_ERR_INCOMPLETE; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | DEFUN(cfg_net_ipacc_rtp_payload, | ||||||
|  |       cfg_net_ipacc_rtp_payload_cmd, | ||||||
|  |       "ipacc rtp_payload <0-256>", | ||||||
|  |       "Override the RTP payload to use") | ||||||
|  | { | ||||||
|  | 	gsmnet->rtp_payload = atoi(argv[0]) & 0xff; | ||||||
|  |  | ||||||
|  | 	return CMD_SUCCESS; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | DEFUN(cfg_net_rtp_base_port, | ||||||
|  |       cfg_net_rtp_base_port_cmd, | ||||||
|  |       "rtp base <0-65534>", | ||||||
|  |       "Base port to use for MGCP RTP") | ||||||
|  | { | ||||||
|  | 	unsigned int port = atoi(argv[0]); | ||||||
|  | 	if (port > 65534) { | ||||||
|  | 		vty_out(vty, "%% wrong base port '%s'%s", argv[0], VTY_NEWLINE); | ||||||
|  | 		return CMD_WARNING; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	gsmnet->rtp_base_port = port; | ||||||
|  | 	return CMD_SUCCESS; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | DEFUN(cfg_net_bsc_token, | ||||||
|  |       cfg_net_bsc_token_cmd, | ||||||
|  |       "bsc_token TOKEN", | ||||||
|  |       "A token for the BSC to be sent to the MSC") | ||||||
|  | { | ||||||
|  | 	if (gsmnet->bsc_token) | ||||||
|  | 		talloc_free(gsmnet->bsc_token); | ||||||
|  | 	gsmnet->bsc_token = talloc_strdup(gsmnet, argv[0]); | ||||||
|  |  | ||||||
|  | 	return CMD_SUCCESS; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | DEFUN(cfg_net_pag_any_tch, | ||||||
|  |       cfg_net_pag_any_tch_cmd, | ||||||
|  |       "paging any use tch (0|1)", | ||||||
|  |       "Assign a TCH when receiving a Paging Any request") | ||||||
|  | { | ||||||
|  | 	gsmnet->pag_any_tch = atoi(argv[0]); | ||||||
|  | 	gsm_net_update_ctype(gsmnet); | ||||||
|  | 	return CMD_SUCCESS; | ||||||
|  | } | ||||||
|  |  | ||||||
| #define DECLARE_TIMER(number, doc) \ | #define DECLARE_TIMER(number, doc) \ | ||||||
|     DEFUN(cfg_net_T##number,					\ |     DEFUN(cfg_net_T##number,					\ | ||||||
|       cfg_net_T##number##_cmd,					\ |       cfg_net_T##number##_cmd,					\ | ||||||
| @@ -1111,7 +1318,7 @@ DEFUN(cfg_net_ho_max_distance, cfg_net_ho_max_distance_cmd, | |||||||
| 								\ | 								\ | ||||||
| 	if (value < 0 || value > 65535) {			\ | 	if (value < 0 || value > 65535) {			\ | ||||||
| 		vty_out(vty, "Timer value %s out of range.%s",	\ | 		vty_out(vty, "Timer value %s out of range.%s",	\ | ||||||
| 		        argv[0], VTY_NEWLINE);			\ | 			argv[0], VTY_NEWLINE);			\ | ||||||
| 		return CMD_WARNING;				\ | 		return CMD_WARNING;				\ | ||||||
| 	}							\ | 	}							\ | ||||||
| 								\ | 								\ | ||||||
| @@ -1124,14 +1331,13 @@ DECLARE_TIMER(3103, "Set the timeout value for HANDOVER.") | |||||||
| DECLARE_TIMER(3105, "Currently not used.") | DECLARE_TIMER(3105, "Currently not used.") | ||||||
| DECLARE_TIMER(3107, "Currently not used.") | DECLARE_TIMER(3107, "Currently not used.") | ||||||
| DECLARE_TIMER(3109, "Currently not used.") | DECLARE_TIMER(3109, "Currently not used.") | ||||||
| DECLARE_TIMER(3111, "Currently not used.") | DECLARE_TIMER(3111, "Set the RSL timeout to wait before releasing the RF Channel.") | ||||||
| DECLARE_TIMER(3113, "Set the time to try paging a subscriber.") | DECLARE_TIMER(3113, "Set the time to try paging a subscriber.") | ||||||
| DECLARE_TIMER(3115, "Currently not used.") | DECLARE_TIMER(3115, "Currently not used.") | ||||||
| DECLARE_TIMER(3117, "Currently not used.") | DECLARE_TIMER(3117, "Currently not used.") | ||||||
| DECLARE_TIMER(3119, "Currently not used.") | DECLARE_TIMER(3119, "Currently not used.") | ||||||
| DECLARE_TIMER(3141, "Currently not used.") | DECLARE_TIMER(3141, "Currently not used.") | ||||||
|  |  | ||||||
|  |  | ||||||
| /* per-BTS configuration */ | /* per-BTS configuration */ | ||||||
| DEFUN(cfg_bts, | DEFUN(cfg_bts, | ||||||
|       cfg_bts_cmd, |       cfg_bts_cmd, | ||||||
| @@ -1760,13 +1966,17 @@ int bsc_vty_init(struct gsm_network *net) | |||||||
| 	install_element(VIEW_NODE, &show_paging_cmd); | 	install_element(VIEW_NODE, &show_paging_cmd); | ||||||
| 	install_element(VIEW_NODE, &show_stats_cmd); | 	install_element(VIEW_NODE, &show_stats_cmd); | ||||||
|  |  | ||||||
| 	openbsc_vty_add_cmds(); | 	install_element(VIEW_NODE, &drop_bts_cmd); | ||||||
|  |  | ||||||
|  | 	openbsc_vty_add_cmds(); | ||||||
|  |          | ||||||
| 	install_element(CONFIG_NODE, &cfg_net_cmd); | 	install_element(CONFIG_NODE, &cfg_net_cmd); | ||||||
| 	install_node(&net_node, config_write_net); | 	install_node(&net_node, config_write_net); | ||||||
| 	install_default(GSMNET_NODE); | 	install_default(GSMNET_NODE); | ||||||
| 	install_element(GSMNET_NODE, &cfg_net_ncc_cmd); | 	install_element(GSMNET_NODE, &cfg_net_ncc_cmd); | ||||||
|  | 	install_element(GSMNET_NODE, &cfg_core_net_ncc_cmd); | ||||||
| 	install_element(GSMNET_NODE, &cfg_net_mnc_cmd); | 	install_element(GSMNET_NODE, &cfg_net_mnc_cmd); | ||||||
|  | 	install_element(GSMNET_NODE, &cfg_core_net_mnc_cmd); | ||||||
| 	install_element(GSMNET_NODE, &cfg_net_name_short_cmd); | 	install_element(GSMNET_NODE, &cfg_net_name_short_cmd); | ||||||
| 	install_element(GSMNET_NODE, &cfg_net_name_long_cmd); | 	install_element(GSMNET_NODE, &cfg_net_name_long_cmd); | ||||||
| 	install_element(GSMNET_NODE, &cfg_net_auth_policy_cmd); | 	install_element(GSMNET_NODE, &cfg_net_auth_policy_cmd); | ||||||
| @@ -1782,6 +1992,9 @@ int bsc_vty_init(struct gsm_network *net) | |||||||
| 	install_element(GSMNET_NODE, &cfg_net_ho_pwr_interval_cmd); | 	install_element(GSMNET_NODE, &cfg_net_ho_pwr_interval_cmd); | ||||||
| 	install_element(GSMNET_NODE, &cfg_net_ho_pwr_hysteresis_cmd); | 	install_element(GSMNET_NODE, &cfg_net_ho_pwr_hysteresis_cmd); | ||||||
| 	install_element(GSMNET_NODE, &cfg_net_ho_max_distance_cmd); | 	install_element(GSMNET_NODE, &cfg_net_ho_max_distance_cmd); | ||||||
|  | 	install_element(GSMNET_NODE, &cfg_net_supported_codecs_cmd); | ||||||
|  | 	install_element(GSMNET_NODE, &cfg_net_ipacc_rtp_payload_cmd); | ||||||
|  | 	install_element(GSMNET_NODE, &cfg_net_rtp_base_port_cmd); | ||||||
| 	install_element(GSMNET_NODE, &cfg_net_T3101_cmd); | 	install_element(GSMNET_NODE, &cfg_net_T3101_cmd); | ||||||
| 	install_element(GSMNET_NODE, &cfg_net_T3103_cmd); | 	install_element(GSMNET_NODE, &cfg_net_T3103_cmd); | ||||||
| 	install_element(GSMNET_NODE, &cfg_net_T3105_cmd); | 	install_element(GSMNET_NODE, &cfg_net_T3105_cmd); | ||||||
| @@ -1793,6 +2006,8 @@ int bsc_vty_init(struct gsm_network *net) | |||||||
| 	install_element(GSMNET_NODE, &cfg_net_T3117_cmd); | 	install_element(GSMNET_NODE, &cfg_net_T3117_cmd); | ||||||
| 	install_element(GSMNET_NODE, &cfg_net_T3119_cmd); | 	install_element(GSMNET_NODE, &cfg_net_T3119_cmd); | ||||||
| 	install_element(GSMNET_NODE, &cfg_net_T3141_cmd); | 	install_element(GSMNET_NODE, &cfg_net_T3141_cmd); | ||||||
|  | 	install_element(GSMNET_NODE, &cfg_net_bsc_token_cmd); | ||||||
|  | 	install_element(GSMNET_NODE, &cfg_net_pag_any_tch_cmd); | ||||||
|  |  | ||||||
| 	install_element(GSMNET_NODE, &cfg_bts_cmd); | 	install_element(GSMNET_NODE, &cfg_bts_cmd); | ||||||
| 	install_node(&bts_node, config_write_bts); | 	install_node(&bts_node, config_write_bts); | ||||||
|   | |||||||
							
								
								
									
										48
									
								
								openbsc/src/vty_interface_bsc.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								openbsc/src/vty_interface_bsc.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,48 @@ | |||||||
|  | /* OpenBSC interface to quagga VTY - BSC options */ | ||||||
|  | /* (C) 2009 by Harald Welte <laforge@gnumonks.org> | ||||||
|  |  * All Rights Reserved | ||||||
|  |  * | ||||||
|  |  * This program is free software; you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation; either version 2 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License along | ||||||
|  |  * with this program; if not, write to the Free Software Foundation, Inc., | ||||||
|  |  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #include <stdlib.h> | ||||||
|  | #include <unistd.h> | ||||||
|  | #include <sys/types.h> | ||||||
|  |  | ||||||
|  | #include <vty/command.h> | ||||||
|  | #include <vty/buffer.h> | ||||||
|  | #include <vty/vty.h> | ||||||
|  |  | ||||||
|  | #include <openbsc/gsm_data.h> | ||||||
|  |  | ||||||
|  | static struct gsmnet *gsmnet = NULL; | ||||||
|  |  | ||||||
|  | DEFUN(show_bsc, show_bsc_cmd, "show bsc", | ||||||
|  | 	SHOW_STR "Display information about the BSC\n") | ||||||
|  | { | ||||||
|  | 	vty_out(vty, "BSC... not implemented yet%s", VTY_NEWLINE); | ||||||
|  | 	return CMD_SUCCESS; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int bsc_vty_init_extra(struct gsm_network *net) | ||||||
|  | { | ||||||
|  | 	gsmnet = net; | ||||||
|  |  | ||||||
|  | 	/* get runtime information */ | ||||||
|  | 	install_element(VIEW_NODE, &show_bsc_cmd); | ||||||
|  |  | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
| @@ -1 +1 @@ | |||||||
| SUBDIRS = debug gsm0408 db channel sccp | SUBDIRS = debug gsm0408 db channel sccp bsc-nat | ||||||
|   | |||||||
							
								
								
									
										17
									
								
								openbsc/tests/bsc-nat/Makefile.am
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								openbsc/tests/bsc-nat/Makefile.am
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | |||||||
|  | INCLUDES = $(all_includes) -I$(top_srcdir)/include | ||||||
|  | AM_CFLAGS=-Wall -ggdb3 $(LIBOSMOCORE_CFLAGS) | ||||||
|  |  | ||||||
|  | EXTRA_DIST = bsc_data.c | ||||||
|  |  | ||||||
|  | noinst_PROGRAMS = bsc_nat_test | ||||||
|  |  | ||||||
|  | bsc_nat_test_SOURCES = bsc_nat_test.c \ | ||||||
|  | 			$(top_srcdir)/src/nat/bsc_filter.c \ | ||||||
|  | 			$(top_srcdir)/src/nat/bsc_sccp.c \ | ||||||
|  | 			$(top_srcdir)/src/nat/bsc_nat_utils.c \ | ||||||
|  | 			$(top_srcdir)/src/nat/bsc_mgcp_utils.c \ | ||||||
|  | 			$(top_srcdir)/src/mgcp/mgcp_protocol.c \ | ||||||
|  | 			$(top_srcdir)/src/mgcp/mgcp_network.c \ | ||||||
|  | 			$(top_srcdir)/src/bssap.c | ||||||
|  | bsc_nat_test_LDADD = $(top_builddir)/src/libbsc.a $(top_builddir)/src/libsccp.a $(LIBOSMOCORE_LIBS) | ||||||
|  |  | ||||||
							
								
								
									
										154
									
								
								openbsc/tests/bsc-nat/bsc_data.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										154
									
								
								openbsc/tests/bsc-nat/bsc_data.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,154 @@ | |||||||
|  | /* test data */ | ||||||
|  |  | ||||||
|  | /* BSC -> MSC, CR */ | ||||||
|  | static const u_int8_t bsc_cr[] = { | ||||||
|  | 0x00, 0x2e, 0xfd, | ||||||
|  | 0x01, 0x00, 0x00, 0x15, 0x02, 0x02, 0x04, 0x02, | ||||||
|  | 0x42, 0xfe, 0x0f, 0x21, 0x00, 0x1f, 0x57, 0x05, | ||||||
|  | 0x08, 0x00, 0x72, 0xf4, 0x80, 0x20, 0x1c, 0xc3, | ||||||
|  | 0x51, 0x17, 0x12, 0x05, 0x08, 0x20, 0x72, 0xf4, | ||||||
|  | 0x90, 0x20, 0x1d, 0x50, 0x08, 0x29, 0x47, 0x80, | ||||||
|  | 0x00, 0x00, 0x00, 0x00, 0x80, 0x00 }; | ||||||
|  |  | ||||||
|  | static const u_int8_t bsc_cr_patched[] = { | ||||||
|  | 0x00, 0x2e, 0xfd, | ||||||
|  | 0x01, 0x00, 0x00, 0x05, 0x02, 0x02, 0x04, 0x02, | ||||||
|  | 0x42, 0xfe, 0x0f, 0x21, 0x00, 0x1f, 0x57, 0x05, | ||||||
|  | 0x08, 0x00, 0x72, 0xf4, 0x80, 0x20, 0x1c, 0xc3, | ||||||
|  | 0x51, 0x17, 0x12, 0x05, 0x08, 0x20, 0x72, 0xf4, | ||||||
|  | 0x90, 0x20, 0x1d, 0x50, 0x08, 0x29, 0x47, 0x80, | ||||||
|  | 0x00, 0x00, 0x00, 0x00, 0x80, 0x00 }; | ||||||
|  |  | ||||||
|  | /* CC, MSC -> BSC */ | ||||||
|  | static const u_int8_t msc_cc[] = { | ||||||
|  | 0x00, 0x0a, 0xfd, | ||||||
|  | 0x02, 0x00, 0x00, 0x05, 0x01, 0x1f, 0xe4, 0x02, | ||||||
|  | 0x01, 0x00 }; | ||||||
|  | static const u_int8_t msc_cc_patched[] = { | ||||||
|  | 0x00, 0x0a, 0xfd, | ||||||
|  | 0x02, 0x00, 0x00, 0x15, 0x01, 0x1f, 0xe4, 0x02, | ||||||
|  | 0x01, 0x00 }; | ||||||
|  |  | ||||||
|  | /* Classmark, BSC -> MSC */ | ||||||
|  | static const u_int8_t bsc_dtap[] = { | ||||||
|  | 0x00, 0x17, 0xfd, | ||||||
|  | 0x06, 0x01, 0x1f, 0xe4, 0x00, 0x01, 0x10, 0x00, | ||||||
|  | 0x0e, 0x54, 0x12, 0x03, 0x50, 0x18, 0x93, 0x13, | ||||||
|  | 0x06, 0x60, 0x14, 0x45, 0x00, 0x81, 0x00 }; | ||||||
|  |  | ||||||
|  | static const u_int8_t bsc_dtap_patched[] = { | ||||||
|  | 0x00, 0x17, 0xfd, | ||||||
|  | 0x06, 0x01, 0x1f, 0xe4, 0x00, 0x01, 0x10, 0x00, | ||||||
|  | 0x0e, 0x54, 0x12, 0x03, 0x50, 0x18, 0x93, 0x13, | ||||||
|  | 0x06, 0x60, 0x14, 0x45, 0x00, 0x81, 0x00 }; | ||||||
|  |  | ||||||
|  | /* Clear command, MSC -> BSC */ | ||||||
|  | static const u_int8_t msc_dtap[] = { | ||||||
|  | 0x00, 0x0d, 0xfd, | ||||||
|  | 0x06, 0x00, 0x00, 0x05, 0x00, 0x01, 0x06, 0x00, | ||||||
|  | 0x04, 0x20, 0x04, 0x01, 0x09 }; | ||||||
|  | static const u_int8_t msc_dtap_patched[] = { | ||||||
|  | 0x00, 0x0d, 0xfd, | ||||||
|  | 0x06, 0x00, 0x00, 0x15, 0x00, 0x01, 0x06, 0x00, | ||||||
|  | 0x04, 0x20, 0x04, 0x01, 0x09 }; | ||||||
|  |  | ||||||
|  | /*RLSD, MSC -> BSC */ | ||||||
|  | static const u_int8_t msc_rlsd[] = { | ||||||
|  | 0x00, 0x0a, 0xfd, | ||||||
|  | 0x04, 0x00, 0x00, 0x05, 0x01, 0x1f, 0xe4, 0x00, | ||||||
|  | 0x01, 0x00 }; | ||||||
|  | static const u_int8_t msc_rlsd_patched[] = { | ||||||
|  | 0x00, 0x0a, 0xfd, | ||||||
|  | 0x04, 0x00, 0x00, 0x15, 0x01, 0x1f, 0xe4, 0x00, | ||||||
|  | 0x01, 0x00 }; | ||||||
|  |  | ||||||
|  | /* RLC, BSC -> MSC */ | ||||||
|  | static const u_int8_t bsc_rlc[] = { | ||||||
|  | 0x00, 0x07, 0xfd, | ||||||
|  | 0x05, 0x01, 0x1f, 0xe4, 0x00, 0x00, 0x15 }; | ||||||
|  |  | ||||||
|  | static const u_int8_t bsc_rlc_patched[] = { | ||||||
|  | 0x00, 0x07, 0xfd, | ||||||
|  | 0x05, 0x01, 0x1f, 0xe4, 0x00, 0x00, 0x05 }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | /* a paging command */ | ||||||
|  | static const u_int8_t paging_by_lac_cmd[] = { | ||||||
|  | 0x00, 0x22, 0xfd, 0x09, | ||||||
|  | 0x00, 0x03, 0x07, 0x0b, 0x04, 0x43, 0x02, 0x00, | ||||||
|  | 0xfe, 0x04, 0x43, 0x5c, 0x00, 0xfe, 0x12, 0x00, | ||||||
|  | 0x10, 0x52, 0x08, 0x08, 0x29, 0x47, 0x10, 0x02, | ||||||
|  | 0x01, 0x50, 0x02, 0x30, 0x1a, 0x03, 0x05, 0x20, | ||||||
|  | 0x15 }; | ||||||
|  |  | ||||||
|  | /* an assignment command */ | ||||||
|  | static const u_int8_t ass_cmd[] = { | ||||||
|  | 0x00, 0x12, 0xfd, 0x06, | ||||||
|  | 0x00, 0x00, 0x49, 0x00, 0x01, 0x0b, 0x00, 0x09, | ||||||
|  | 0x01, 0x0b, 0x03, 0x01, 0x0a, 0x11, 0x01, 0x00, | ||||||
|  | 0x15 }; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * MGCP messages | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | /* nothing to patch */ | ||||||
|  | static const char crcx[] = "CRCX 23265295 8@mgw MGCP 1.0\r\nC: 394b0439fb\r\nL: p:20, a:AMR, nt:IN\r\nM: recvonly\r\n"; | ||||||
|  | static const char crcx_patched[] = "CRCX 23265295 8@mgw MGCP 1.0\r\nC: 394b0439fb\r\nL: p:20, a:AMR, nt:IN\r\nM: recvonly\r\n"; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | /* patch the ip and port */ | ||||||
|  | static const char crcx_resp[] = "200 23265295\r\nI: 1\r\n\r\nv=0\r\nc=IN IP4 172.16.18.2\r\nm=audio 4002 RTP/AVP 98\r\na=rtpmap:98 AMR/8000\r\n"; | ||||||
|  | static const char crcx_resp_patched[] = "200 23265295\r\nI: 1\r\n\r\nv=0\r\nc=IN IP4 10.0.0.1\r\nm=audio 999 RTP/AVP 98\r\na=rtpmap:98 AMR/8000\r\n"; | ||||||
|  |  | ||||||
|  | /* patch the ip and port */ | ||||||
|  | static const char mdcx[] = " MDCX 23330829 8@mgw MGCP 1.0\r\nC: 394b0439fb\r\nI: 1\r\nL: p:20, a:AMR, nt:IN\r\nM: recvonly\r\n\r\nv=0\r\no=- 1049380491 0 IN IP4 172.16.18.2\r\ns=-\r\nc=IN IP4 172.16.18.2\r\nt=0 0\r\nm=audio 4410 RTP/AVP 126\r\na=rtpmap:126 AMR/8000/1\r\na=fmtp:126 mode-set=2;start-mode=0\r\na=ptime:20\r\na=recvonly\r\nm=image 4412 udptl t38\r\na=T38FaxVersion:0\r\na=T38MaxBitRate:14400\r\n"; | ||||||
|  | static const char mdcx_patched[] = " MDCX 23330829 8@mgw MGCP 1.0\r\nC: 394b0439fb\r\nI: 1\r\nL: p:20, a:AMR, nt:IN\r\nM: recvonly\r\n\r\nv=0\r\no=- 1049380491 0 IN IP4 172.16.18.2\r\ns=-\r\nc=IN IP4 10.0.0.23\r\nt=0 0\r\nm=audio 6666 RTP/AVP 126\r\na=rtpmap:126 AMR/8000/1\r\na=fmtp:126 mode-set=2;start-mode=0\r\na=ptime:20\r\na=recvonly\r\nm=image 4412 udptl t38\r\na=T38FaxVersion:0\r\na=T38MaxBitRate:14400\r\n"; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | static const char mdcx_resp[] = "200 23330829\r\n\r\nv=0\r\nc=IN IP4 172.16.18.2\r\nm=audio 4002 RTP/AVP 98\r\na=rtpmap:98 AMR/8000\r\n"; | ||||||
|  | static const char mdcx_resp_patched[] = "200 23330829\r\n\r\nv=0\r\nc=IN IP4 10.0.0.23\r\nm=audio 5555 RTP/AVP 98\r\na=rtpmap:98 AMR/8000\r\n"; | ||||||
|  |  | ||||||
|  | /* different line ending */ | ||||||
|  | static const char mdcx_resp2[] = "200 33330829\n\nv=0\nc=IN IP4 172.16.18.2\nm=audio 4002 RTP/AVP 98\na=rtpmap:98 AMR/8000\n"; | ||||||
|  | static const char mdcx_resp_patched2[] = "200 33330829\n\nv=0\nc=IN IP4 10.0.0.23\nm=audio 5555 RTP/AVP 98\na=rtpmap:98 AMR/8000\n"; | ||||||
|  |  | ||||||
|  | struct mgcp_patch_test { | ||||||
|  | 	const char *orig; | ||||||
|  | 	const char *patch; | ||||||
|  | 	const char *ip; | ||||||
|  | 	const int port; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | static const struct mgcp_patch_test mgcp_messages[] = { | ||||||
|  | 	{ | ||||||
|  | 		.orig = crcx, | ||||||
|  | 		.patch = crcx_patched, | ||||||
|  | 		.ip = "0.0.0.0", | ||||||
|  | 		.port = 2323, | ||||||
|  | 	}, | ||||||
|  | 	{ | ||||||
|  | 		.orig = crcx_resp, | ||||||
|  | 		.patch = crcx_resp_patched, | ||||||
|  | 		.ip = "10.0.0.1", | ||||||
|  | 		.port = 999, | ||||||
|  | 	}, | ||||||
|  | 	{ | ||||||
|  | 		.orig = mdcx, | ||||||
|  | 		.patch = mdcx_patched, | ||||||
|  | 		.ip = "10.0.0.23", | ||||||
|  | 		.port = 6666, | ||||||
|  | 	}, | ||||||
|  | 	{ | ||||||
|  | 		.orig = mdcx_resp, | ||||||
|  | 		.patch = mdcx_resp_patched, | ||||||
|  | 		.ip = "10.0.0.23", | ||||||
|  | 		.port = 5555, | ||||||
|  | 	}, | ||||||
|  | 	{ | ||||||
|  | 		.orig = mdcx_resp2, | ||||||
|  | 		.patch = mdcx_resp_patched2, | ||||||
|  | 		.ip = "10.0.0.23", | ||||||
|  | 		.port = 5555, | ||||||
|  | 	}, | ||||||
|  | }; | ||||||
							
								
								
									
										539
									
								
								openbsc/tests/bsc-nat/bsc_nat_test.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										539
									
								
								openbsc/tests/bsc-nat/bsc_nat_test.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,539 @@ | |||||||
|  | /* | ||||||
|  |  * BSC NAT Message filtering | ||||||
|  |  * | ||||||
|  |  * (C) 2010 by Holger Hans Peter Freyther <zecke@selfish.org> | ||||||
|  |  * (C) 2010 by On-Waves | ||||||
|  |  * | ||||||
|  |  * All Rights Reserved | ||||||
|  |  * | ||||||
|  |  * This program is free software; you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation; either version 2 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License along | ||||||
|  |  * with this program; if not, write to the Free Software Foundation, Inc., | ||||||
|  |  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #include <openbsc/debug.h> | ||||||
|  | #include <openbsc/gsm_data.h> | ||||||
|  | #include <openbsc/bsc_nat.h> | ||||||
|  |  | ||||||
|  | #include <osmocore/talloc.h> | ||||||
|  |  | ||||||
|  | #include <stdio.h> | ||||||
|  |  | ||||||
|  | /* test messages for ipa */ | ||||||
|  | static u_int8_t ipa_id[] = { | ||||||
|  | 	0x00, 0x01, 0xfe, 0x06, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /* SCCP messages are below */ | ||||||
|  | static u_int8_t gsm_reset[] = { | ||||||
|  | 	0x00, 0x12, 0xfd, | ||||||
|  | 	0x09, 0x00, 0x03, 0x05, 0x07, 0x02, 0x42, 0xfe, | ||||||
|  | 	0x02, 0x42, 0xfe, 0x06, 0x00, 0x04, 0x30, 0x04, | ||||||
|  | 	0x01, 0x20, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | static const u_int8_t gsm_reset_ack[] = { | ||||||
|  | 	0x00, 0x13, 0xfd, | ||||||
|  | 	0x09, 0x00, 0x03, 0x07, 0x0b, 0x04, 0x43, 0x01, | ||||||
|  | 	0x00, 0xfe, 0x04, 0x43, 0x5c, 0x00, 0xfe, 0x03, | ||||||
|  | 	0x00, 0x01, 0x31, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | static const u_int8_t gsm_paging[] = { | ||||||
|  | 	0x00, 0x20, 0xfd, | ||||||
|  | 	0x09, 0x00, 0x03, 0x07, 0x0b, 0x04, 0x43, 0x01, | ||||||
|  | 	0x00, 0xfe, 0x04, 0x43, 0x5c, 0x00, 0xfe, 0x10, | ||||||
|  | 	0x00, 0x0e, 0x52, 0x08, 0x08, 0x29, 0x47, 0x10, | ||||||
|  | 	0x02, 0x01, 0x31, 0x97, 0x61, 0x1a, 0x01, 0x06, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /* BSC -> MSC connection open */ | ||||||
|  | static const u_int8_t bssmap_cr[] = { | ||||||
|  | 	0x00, 0x2c, 0xfd, | ||||||
|  | 	0x01, 0x01, 0x02, 0x03, 0x02, 0x02, 0x04, 0x02, | ||||||
|  | 	0x42, 0xfe, 0x0f, 0x1f, 0x00, 0x1d, 0x57, 0x05, | ||||||
|  | 	0x08, 0x00, 0x72, 0xf4, 0x80, 0x20, 0x12, 0xc3, | ||||||
|  | 	0x50, 0x17, 0x10, 0x05, 0x24, 0x11, 0x03, 0x33, | ||||||
|  | 	0x19, 0xa2, 0x08, 0x29, 0x47, 0x10, 0x02, 0x01, | ||||||
|  | 	0x31, 0x97, 0x61, 0x00 | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /* MSC -> BSC connection confirm */ | ||||||
|  | static const u_int8_t bssmap_cc[] = { | ||||||
|  | 	0x00, 0x0a, 0xfd, | ||||||
|  | 	0x02, 0x01, 0x02, 0x03, 0x00, 0x00, 0x03, 0x02, 0x01, 0x00, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /* MSC -> BSC released */ | ||||||
|  | static const u_int8_t bssmap_released[] = { | ||||||
|  | 	0x00, 0x0e, 0xfd, | ||||||
|  | 	0x04, 0x00, 0x00, 0x03, 0x01, 0x02, 0x03, 0x00, 0x01, 0x0f, | ||||||
|  | 	0x02, 0x23, 0x42, 0x00, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /* BSC -> MSC released */ | ||||||
|  | static const u_int8_t bssmap_release_complete[] = { | ||||||
|  | 	0x00, 0x07, 0xfd, | ||||||
|  | 	0x05, 0x01, 0x02, 0x03, 0x00, 0x00, 0x03 | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /* both directions IT timer */ | ||||||
|  | static const u_int8_t connnection_it[] = { | ||||||
|  | 	0x00, 0x0b, 0xfd, | ||||||
|  | 	0x10, 0x01, 0x02, 0x03, 0x01, 0x02, 0x03, | ||||||
|  | 	0x00, 0x00, 0x00, 0x00, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /* MGCP wrap... */ | ||||||
|  | static const u_int8_t mgcp_msg[] = { | ||||||
|  | 	0x00, 0x03, 0xfc, | ||||||
|  | 	0x20, 0x20, 0x20, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | struct filter_result { | ||||||
|  | 	const u_int8_t *data; | ||||||
|  | 	const u_int16_t length; | ||||||
|  | 	const int dir; | ||||||
|  | 	const int result; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | static const struct filter_result results[] = { | ||||||
|  | 	{ | ||||||
|  | 		.data = ipa_id, | ||||||
|  | 		.length = ARRAY_SIZE(ipa_id), | ||||||
|  | 		.dir = DIR_MSC, | ||||||
|  | 		.result = 1, | ||||||
|  | 	}, | ||||||
|  | 	{ | ||||||
|  | 		.data = gsm_reset, | ||||||
|  | 		.length = ARRAY_SIZE(gsm_reset), | ||||||
|  | 		.dir = DIR_MSC, | ||||||
|  | 		.result = 1, | ||||||
|  | 	}, | ||||||
|  | 	{ | ||||||
|  | 		.data = gsm_reset_ack, | ||||||
|  | 		.length = ARRAY_SIZE(gsm_reset_ack), | ||||||
|  | 		.dir = DIR_BSC, | ||||||
|  | 		.result = 1, | ||||||
|  | 	}, | ||||||
|  | 	{ | ||||||
|  | 		.data = gsm_paging, | ||||||
|  | 		.length = ARRAY_SIZE(gsm_paging), | ||||||
|  | 		.dir = DIR_BSC, | ||||||
|  | 		.result = 0, | ||||||
|  | 	}, | ||||||
|  | 	{ | ||||||
|  | 		.data = bssmap_cr, | ||||||
|  | 		.length = ARRAY_SIZE(bssmap_cr), | ||||||
|  | 		.dir = DIR_MSC, | ||||||
|  | 		.result = 0, | ||||||
|  | 	}, | ||||||
|  | 	{ | ||||||
|  | 		.data = bssmap_cc, | ||||||
|  | 		.length = ARRAY_SIZE(bssmap_cc), | ||||||
|  | 		.dir = DIR_BSC, | ||||||
|  | 		.result = 0, | ||||||
|  | 	}, | ||||||
|  | 	{ | ||||||
|  | 		.data = bssmap_released, | ||||||
|  | 		.length = ARRAY_SIZE(bssmap_released), | ||||||
|  | 		.dir = DIR_MSC, | ||||||
|  | 		.result = 0, | ||||||
|  | 	}, | ||||||
|  | 	{ | ||||||
|  | 		.data = bssmap_release_complete, | ||||||
|  | 		.length = ARRAY_SIZE(bssmap_release_complete), | ||||||
|  | 		.dir = DIR_BSC, | ||||||
|  | 		.result = 0, | ||||||
|  | 	}, | ||||||
|  | 	{ | ||||||
|  | 		.data = mgcp_msg, | ||||||
|  | 		.length = ARRAY_SIZE(mgcp_msg), | ||||||
|  | 		.dir = DIR_MSC, | ||||||
|  | 		.result = 0, | ||||||
|  | 	}, | ||||||
|  | 	{ | ||||||
|  | 		.data = connnection_it, | ||||||
|  | 		.length = ARRAY_SIZE(connnection_it), | ||||||
|  | 		.dir = DIR_BSC, | ||||||
|  | 		.result = 0, | ||||||
|  | 	}, | ||||||
|  | 	{ | ||||||
|  | 		.data = connnection_it, | ||||||
|  | 		.length = ARRAY_SIZE(connnection_it), | ||||||
|  | 		.dir = DIR_MSC, | ||||||
|  | 		.result = 0, | ||||||
|  | 	}, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | static void test_filter(void) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	/* start testinh with proper messages */ | ||||||
|  | 	fprintf(stderr, "Testing BSS Filtering.\n"); | ||||||
|  | 	for (i = 0; i < ARRAY_SIZE(results); ++i) { | ||||||
|  | 		int result; | ||||||
|  | 		struct bsc_nat_parsed *parsed; | ||||||
|  | 		struct msgb *msg = msgb_alloc(4096, "test-message"); | ||||||
|  |  | ||||||
|  | 		fprintf(stderr, "Going to test item: %d\n", i); | ||||||
|  | 		memcpy(msg->data, results[i].data, results[i].length); | ||||||
|  | 		msg->l2h = msgb_put(msg, results[i].length); | ||||||
|  |  | ||||||
|  | 		parsed = bsc_nat_parse(msg); | ||||||
|  | 		if (!parsed) { | ||||||
|  | 			fprintf(stderr, "FAIL: Failed to parse the message\n"); | ||||||
|  | 			continue; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		result = bsc_nat_filter_ipa(results[i].dir, msg, parsed); | ||||||
|  | 		if (result != results[i].result) { | ||||||
|  | 			fprintf(stderr, "FAIL: Not the expected result got: %d wanted: %d\n", | ||||||
|  | 				result, results[i].result); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		msgb_free(msg); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #include "bsc_data.c" | ||||||
|  |  | ||||||
|  | static void copy_to_msg(struct msgb *msg, const u_int8_t *data, unsigned int length) | ||||||
|  | { | ||||||
|  | 	msgb_reset(msg); | ||||||
|  | 	msg->l2h = msgb_put(msg, length); | ||||||
|  | 	memcpy(msg->l2h, data, msgb_l2len(msg)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #define VERIFY(con_found, con, msg, ver, str) \ | ||||||
|  | 	if (!con_found || con_found->bsc != con) { \ | ||||||
|  | 		fprintf(stderr, "Failed to find the con: %p\n", con_found); \ | ||||||
|  | 		abort(); \ | ||||||
|  | 	} \ | ||||||
|  | 	if (memcmp(msg->data, ver, sizeof(ver)) != 0) { \ | ||||||
|  | 		fprintf(stderr, "Failed to patch the %s msg.\n", str); \ | ||||||
|  | 		abort(); \ | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | /* test conn tracking once */ | ||||||
|  | static void test_contrack() | ||||||
|  | { | ||||||
|  | 	int rc; | ||||||
|  | 	struct bsc_nat *nat; | ||||||
|  | 	struct bsc_connection *con; | ||||||
|  | 	struct sccp_connections *con_found; | ||||||
|  | 	struct bsc_nat_parsed *parsed; | ||||||
|  | 	struct msgb *msg; | ||||||
|  |  | ||||||
|  | 	fprintf(stderr, "Testing connection tracking.\n"); | ||||||
|  | 	nat = bsc_nat_alloc(); | ||||||
|  | 	con = bsc_connection_alloc(nat); | ||||||
|  | 	msg = msgb_alloc(4096, "test"); | ||||||
|  |  | ||||||
|  | 	/* 1.) create a connection */ | ||||||
|  | 	copy_to_msg(msg, bsc_cr, sizeof(bsc_cr)); | ||||||
|  | 	parsed = bsc_nat_parse(msg); | ||||||
|  | 	con_found = patch_sccp_src_ref_to_msc(msg, parsed, nat); | ||||||
|  | 	if (con_found != NULL) { | ||||||
|  | 		fprintf(stderr, "Con should not exist %p\n", con_found); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  | 	rc = create_sccp_src_ref(con, msg, parsed); | ||||||
|  | 	if (rc != 0) { | ||||||
|  | 		fprintf(stderr, "Failed to create a ref\n"); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  | 	con_found = patch_sccp_src_ref_to_msc(msg, parsed, nat); | ||||||
|  | 	if (!con_found || con_found->bsc != con) { | ||||||
|  | 		fprintf(stderr, "Failed to find the con: %p\n", con_found); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  | 	if (memcmp(msg->data, bsc_cr_patched, sizeof(bsc_cr_patched)) != 0) { | ||||||
|  | 		fprintf(stderr, "Failed to patch the BSC CR msg.\n"); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  | 	talloc_free(parsed); | ||||||
|  |  | ||||||
|  | 	/* 2.) get the cc */ | ||||||
|  | 	copy_to_msg(msg, msc_cc, sizeof(msc_cc)); | ||||||
|  | 	parsed = bsc_nat_parse(msg); | ||||||
|  | 	con_found = patch_sccp_src_ref_to_bsc(msg, parsed, nat); | ||||||
|  | 	VERIFY(con_found, con, msg, msc_cc_patched, "MSC CC"); | ||||||
|  | 	if (update_sccp_src_ref(con_found, parsed) != 0) { | ||||||
|  | 		fprintf(stderr, "Failed to update the SCCP con.\n"); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* 3.) send some data */ | ||||||
|  | 	copy_to_msg(msg, bsc_dtap, sizeof(bsc_dtap)); | ||||||
|  | 	parsed = bsc_nat_parse(msg); | ||||||
|  | 	con_found = patch_sccp_src_ref_to_msc(msg, parsed, nat); | ||||||
|  | 	VERIFY(con_found, con, msg, bsc_dtap_patched, "BSC DTAP"); | ||||||
|  |  | ||||||
|  | 	/* 4.) receive some data */ | ||||||
|  | 	copy_to_msg(msg, msc_dtap, sizeof(msc_dtap)); | ||||||
|  | 	parsed = bsc_nat_parse(msg); | ||||||
|  | 	con_found = patch_sccp_src_ref_to_bsc(msg, parsed, nat); | ||||||
|  | 	VERIFY(con_found, con, msg, msc_dtap_patched, "MSC DTAP"); | ||||||
|  |  | ||||||
|  | 	/* 5.) close the connection */ | ||||||
|  | 	copy_to_msg(msg, msc_rlsd, sizeof(msc_rlsd)); | ||||||
|  | 	parsed = bsc_nat_parse(msg); | ||||||
|  | 	con_found = patch_sccp_src_ref_to_bsc(msg, parsed, nat); | ||||||
|  | 	VERIFY(con_found, con, msg, msc_rlsd_patched, "MSC RLSD"); | ||||||
|  |  | ||||||
|  | 	/* 6.) confirm the connection close */ | ||||||
|  | 	copy_to_msg(msg, bsc_rlc, sizeof(bsc_rlc)); | ||||||
|  | 	parsed = bsc_nat_parse(msg); | ||||||
|  | 	con_found = patch_sccp_src_ref_to_msc(msg, parsed, nat); | ||||||
|  | 	if (!con_found || con_found->bsc != con) { | ||||||
|  | 		fprintf(stderr, "Failed to find the con: %p\n", con_found); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  | 	if (memcmp(msg->data, bsc_rlc_patched, sizeof(bsc_rlc_patched)) != 0) { | ||||||
|  | 		fprintf(stderr, "Failed to patch the BSC CR msg.\n"); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  | 	remove_sccp_src_ref(con, msg, parsed); | ||||||
|  | 	talloc_free(parsed); | ||||||
|  |  | ||||||
|  | 	copy_to_msg(msg, bsc_rlc, sizeof(bsc_rlc)); | ||||||
|  | 	parsed = bsc_nat_parse(msg); | ||||||
|  | 	con_found = patch_sccp_src_ref_to_msc(msg, parsed, nat); | ||||||
|  |  | ||||||
|  | 	/* verify that it is gone */ | ||||||
|  | 	if (con_found != NULL) { | ||||||
|  | 		fprintf(stderr, "Con should be gone. %p\n", con_found); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  | 	talloc_free(parsed); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	talloc_free(nat); | ||||||
|  | 	msgb_free(msg); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void test_paging(void) | ||||||
|  | { | ||||||
|  | 	struct bsc_nat *nat; | ||||||
|  | 	struct bsc_connection *con; | ||||||
|  | 	struct bsc_nat_parsed *parsed; | ||||||
|  | 	struct bsc_config cfg; | ||||||
|  | 	struct msgb *msg; | ||||||
|  |  | ||||||
|  | 	fprintf(stderr, "Testing paging by lac.\n"); | ||||||
|  |  | ||||||
|  | 	nat = bsc_nat_alloc(); | ||||||
|  | 	con = bsc_connection_alloc(nat); | ||||||
|  | 	con->cfg = &cfg; | ||||||
|  | 	cfg.lac = 23; | ||||||
|  | 	con->authenticated = 1; | ||||||
|  | 	llist_add(&con->list_entry, &nat->bsc_connections); | ||||||
|  | 	msg = msgb_alloc(4096, "test"); | ||||||
|  |  | ||||||
|  | 	/* Test completely bad input */ | ||||||
|  | 	copy_to_msg(msg, paging_by_lac_cmd, sizeof(paging_by_lac_cmd)); | ||||||
|  | 	if (bsc_nat_find_bsc(nat, msg) != 0) { | ||||||
|  | 		fprintf(stderr, "Should have not found anything.\n"); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* Test it by not finding it */ | ||||||
|  | 	copy_to_msg(msg, paging_by_lac_cmd, sizeof(paging_by_lac_cmd)); | ||||||
|  | 	parsed = bsc_nat_parse(msg); | ||||||
|  | 	if (bsc_nat_find_bsc(nat, msg) != 0) { | ||||||
|  | 		fprintf(stderr, "Should have not found aynthing.\n"); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  | 	talloc_free(parsed); | ||||||
|  |  | ||||||
|  | 	/* Test by finding it */ | ||||||
|  | 	cfg.lac = 8213; | ||||||
|  | 	copy_to_msg(msg, paging_by_lac_cmd, sizeof(paging_by_lac_cmd)); | ||||||
|  | 	parsed = bsc_nat_parse(msg); | ||||||
|  | 	if (bsc_nat_find_bsc(nat, msg) != con) { | ||||||
|  | 		fprintf(stderr, "Should have found it.\n"); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  | 	talloc_free(parsed); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void test_mgcp_ass_tracking(void) | ||||||
|  | { | ||||||
|  | 	struct sccp_connections con; | ||||||
|  | 	struct bsc_nat_parsed *parsed; | ||||||
|  | 	struct msgb *msg; | ||||||
|  |  | ||||||
|  | 	fprintf(stderr, "Testing MGCP.\n"); | ||||||
|  | 	memset(&con, 0, sizeof(con)); | ||||||
|  |  | ||||||
|  | 	msg = msgb_alloc(4096, "foo"); | ||||||
|  | 	copy_to_msg(msg, ass_cmd, sizeof(ass_cmd)); | ||||||
|  | 	parsed = bsc_nat_parse(msg); | ||||||
|  | 	if (bsc_mgcp_assign(&con, msg) != 0) { | ||||||
|  | 		fprintf(stderr, "Failed to handle assignment.\n"); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if (con.msc_timeslot != 21) { | ||||||
|  | 		fprintf(stderr, "Timeslot should be 21.\n"); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if (con.bsc_timeslot != 21) { | ||||||
|  | 		fprintf(stderr, "Assigned timeslot should have been 21.\n"); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  | 	talloc_free(parsed); | ||||||
|  |  | ||||||
|  | 	bsc_mgcp_clear(&con); | ||||||
|  | 	if (con.bsc_timeslot != -1 || con.msc_timeslot != -1) { | ||||||
|  | 		fprintf(stderr, "Clearing should remove the mapping.\n"); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* test the code to find a given connection */ | ||||||
|  | static void test_mgcp_find(void) | ||||||
|  | { | ||||||
|  | 	struct bsc_nat *nat; | ||||||
|  | 	struct bsc_connection *con; | ||||||
|  | 	struct sccp_connections *sccp_con; | ||||||
|  |  | ||||||
|  | 	fprintf(stderr, "Testing finding of a BSC Connection\n"); | ||||||
|  |  | ||||||
|  | 	nat = bsc_nat_alloc(); | ||||||
|  | 	con = bsc_connection_alloc(nat); | ||||||
|  | 	llist_add(&con->list_entry, &nat->bsc_connections); | ||||||
|  |  | ||||||
|  | 	sccp_con = talloc_zero(con, struct sccp_connections); | ||||||
|  | 	sccp_con->msc_timeslot = 12; | ||||||
|  | 	sccp_con->bsc_timeslot = 12; | ||||||
|  | 	sccp_con->bsc = con; | ||||||
|  | 	llist_add(&sccp_con->list_entry, &nat->sccp_connections); | ||||||
|  |  | ||||||
|  | 	if (bsc_mgcp_find_con(nat, 11) != NULL) { | ||||||
|  | 		fprintf(stderr, "Found the wrong connection.\n"); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if (bsc_mgcp_find_con(nat, 12) != con) { | ||||||
|  | 		fprintf(stderr, "Didn't find the connection\n"); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	sccp_con->msc_timeslot = 0; | ||||||
|  | 	sccp_con->bsc_timeslot = 0; | ||||||
|  | 	if (bsc_mgcp_find_con(nat, 1) != con) { | ||||||
|  | 		fprintf(stderr, "Didn't find the connection\n"); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* free everything */ | ||||||
|  | 	talloc_free(nat); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void test_mgcp_rewrite(void) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 	struct msgb *output; | ||||||
|  | 	fprintf(stderr, "Test rewriting MGCP messages.\n"); | ||||||
|  |  | ||||||
|  | 	for (i = 0; i < ARRAY_SIZE(mgcp_messages); ++i) { | ||||||
|  | 		const char *orig = mgcp_messages[i].orig; | ||||||
|  | 		const char *patc = mgcp_messages[i].patch; | ||||||
|  | 		const char *ip = mgcp_messages[i].ip; | ||||||
|  | 		const int port = mgcp_messages[i].port; | ||||||
|  |  | ||||||
|  | 		char *input = strdup(orig); | ||||||
|  |  | ||||||
|  | 		output = bsc_mgcp_rewrite(input, strlen(input), ip, port); | ||||||
|  | 		if (msgb_l2len(output) != strlen(patc)) { | ||||||
|  | 			fprintf(stderr, "Wrong sizes for test: %d  %d != %d != %d\n", i, msgb_l2len(output), strlen(patc), strlen(orig)); | ||||||
|  | 			fprintf(stderr, "String '%s' vs '%s'\n", (const char *) output->l2h, patc); | ||||||
|  | 			abort(); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (memcmp(output->l2h, patc, msgb_l2len(output)) != 0) { | ||||||
|  | 			fprintf(stderr, "Broken on %d msg: '%s'\n", i, (const char *) output->l2h); | ||||||
|  | 			abort(); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		msgb_free(output); | ||||||
|  | 		free(input); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void test_mgcp_parse(void) | ||||||
|  | { | ||||||
|  | 	int code, ci; | ||||||
|  | 	char transaction[60]; | ||||||
|  |  | ||||||
|  | 	fprintf(stderr, "Test MGCP response parsing.\n"); | ||||||
|  |  | ||||||
|  | 	if (bsc_mgcp_parse_response(crcx_resp, &code, transaction) != 0) { | ||||||
|  | 		fprintf(stderr, "Failed to parse CRCX resp.\n"); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if (code != 200) { | ||||||
|  | 		fprintf(stderr, "Failed to parse the CODE properly. Got: %d\n", code); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if (strcmp(transaction, "23265295") != 0) { | ||||||
|  | 		fprintf(stderr, "Failed to parse transaction id: '%s'\n", transaction); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	ci = bsc_mgcp_extract_ci(crcx_resp); | ||||||
|  | 	if (ci != 1) { | ||||||
|  | 		fprintf(stderr, "Failed to parse the CI. Got: %d\n", ci); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int main(int argc, char **argv) | ||||||
|  | { | ||||||
|  | 	struct log_target *stderr_target; | ||||||
|  |  | ||||||
|  | 	log_init(&log_info); | ||||||
|  | 	stderr_target = log_target_create_stderr(); | ||||||
|  | 	log_add_target(stderr_target); | ||||||
|  | 	log_set_all_filter(stderr_target, 1); | ||||||
|  |  | ||||||
|  | 	test_filter(); | ||||||
|  | 	test_contrack(); | ||||||
|  | 	test_paging(); | ||||||
|  | 	test_mgcp_ass_tracking(); | ||||||
|  | 	test_mgcp_find(); | ||||||
|  | 	test_mgcp_rewrite(); | ||||||
|  | 	test_mgcp_parse(); | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void input_event() | ||||||
|  | {} | ||||||
|  | int nm_state_event() | ||||||
|  | { | ||||||
|  | 	return -1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int gsm0408_rcvmsg(struct msgb *msg, u_int8_t link_id) | ||||||
|  | { | ||||||
|  | 	return -1; | ||||||
|  | } | ||||||
| @@ -76,6 +76,8 @@ int main(int argc, char** argv) | |||||||
| void nm_state_event() {} | void nm_state_event() {} | ||||||
| void input_event() {} | void input_event() {} | ||||||
| void sms_alloc() {} | void sms_alloc() {} | ||||||
|  | void _lchan_release() {} | ||||||
|  | void gsm_net_update_ctype(struct gsm_network *network) {} | ||||||
|  |  | ||||||
| struct tlv_definition nm_att_tlvdef; | struct tlv_definition nm_att_tlvdef; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -354,14 +354,14 @@ int sccp_read_cb(struct msgb *data, unsigned len, void *context) | |||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| int sccp_write_cb(struct msgb *data, void *ctx) | void sccp_write_cb(struct msgb *data, void *ctx) | ||||||
| { | { | ||||||
| 	int i = 0; | 	int i = 0; | ||||||
| 	const u_int8_t *got, *wanted; | 	const u_int8_t *got, *wanted; | ||||||
|  |  | ||||||
| 	if (test_data[current_test].response == NULL) { | 	if (test_data[current_test].response == NULL) { | ||||||
| 		FAIL("Didn't expect write callback\n"); | 		FAIL("Didn't expect write callback\n"); | ||||||
| 		return -1; | 		goto exit; | ||||||
| 	} else if (test_data[current_test].response_length != msgb_l2len(data)) { | 	} else if (test_data[current_test].response_length != msgb_l2len(data)) { | ||||||
| 		FAIL("Size does not match. Got: %d Wanted: %d\n", | 		FAIL("Size does not match. Got: %d Wanted: %d\n", | ||||||
| 		     msgb_l2len(data), test_data[current_test].response_length); | 		     msgb_l2len(data), test_data[current_test].response_length); | ||||||
| @@ -374,12 +374,14 @@ int sccp_write_cb(struct msgb *data, void *ctx) | |||||||
| 		if (got[i] != wanted[i]) { | 		if (got[i] != wanted[i]) { | ||||||
| 			FAIL("Failed to compare byte. Got: 0x%x Wanted: 0x%x at %d\n", | 			FAIL("Failed to compare byte. Got: 0x%x Wanted: 0x%x at %d\n", | ||||||
| 			     got[i], wanted[i], i); | 			     got[i], wanted[i], i); | ||||||
| 			return -1; | 			goto exit; | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	write_called = 1; | 	write_called = 1; | ||||||
| 	return 0; |  | ||||||
|  | exit: | ||||||
|  | 	msgb_free(data); | ||||||
| } | } | ||||||
|  |  | ||||||
| void sccp_c_read(struct sccp_connection *connection, struct msgb *msgb, unsigned int len) | void sccp_c_read(struct sccp_connection *connection, struct msgb *msgb, unsigned int len) | ||||||
| @@ -409,7 +411,7 @@ int sccp_accept_cb(struct sccp_connection *connection, void *user_data) | |||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| static int sccp_udt_write_cb(struct msgb *data, void *context) | static void sccp_udt_write_cb(struct msgb *data, void *context) | ||||||
| { | { | ||||||
| 	const u_int8_t *got, *wanted; | 	const u_int8_t *got, *wanted; | ||||||
| 	int i; | 	int i; | ||||||
| @@ -419,7 +421,7 @@ static int sccp_udt_write_cb(struct msgb *data, void *context) | |||||||
| 	if (send_data[current_test].length != msgb_l2len(data)) { | 	if (send_data[current_test].length != msgb_l2len(data)) { | ||||||
| 		FAIL("Size does not match. Got: %d Wanted: %d\n", | 		FAIL("Size does not match. Got: %d Wanted: %d\n", | ||||||
| 		     msgb_l2len(data), send_data[current_test].length); | 		     msgb_l2len(data), send_data[current_test].length); | ||||||
| 		return -1; | 		goto exit; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	got = &data->l2h[0]; | 	got = &data->l2h[0]; | ||||||
| @@ -429,12 +431,14 @@ static int sccp_udt_write_cb(struct msgb *data, void *context) | |||||||
| 		if (got[i] != wanted[i]) { | 		if (got[i] != wanted[i]) { | ||||||
| 			FAIL("Failed to compare byte. Got: 0x%x Wanted: 0x%x at %d\n", | 			FAIL("Failed to compare byte. Got: 0x%x Wanted: 0x%x at %d\n", | ||||||
| 			     got[i], wanted[i], i); | 			     got[i], wanted[i], i); | ||||||
| 			return -1; | 			goto exit; | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	matched = 1; | 	matched = 1; | ||||||
| 	return 0; |  | ||||||
|  | exit: | ||||||
|  | 	msgb_free(data); | ||||||
| } | } | ||||||
|  |  | ||||||
| static void test_sccp_system(void) | static void test_sccp_system(void) | ||||||
| @@ -504,11 +508,11 @@ static int sccp_udt_read(struct msgb *data, unsigned int len, void *context) | |||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| static int sccp_write_loop(struct msgb *data, void *context) | static void sccp_write_loop(struct msgb *data, void *context) | ||||||
| { | { | ||||||
| 	/* send it back to us */ | 	/* send it back to us */ | ||||||
| 	sccp_system_incoming(data); | 	sccp_system_incoming(data); | ||||||
| 	return 0; | 	msgb_free(data); | ||||||
| } | } | ||||||
|  |  | ||||||
| static void test_sccp_udt_communication(void) | static void test_sccp_udt_communication(void) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user