Microsoft MVP성태의 닷넷 이야기
.NET Framework: 2090. C# - UDP Datagram의 최대 크기 [링크 복사], [링크+제목 복사],
조회: 14122
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)

C# - UDP Datagram의 최대 크기

IP 프로토콜에서 datagram의 크기는 일반적으로 65,535 바이트까지 가능합니다. 그중에서 20바이트 IP 헤더와 8바이트 UDP 헤더를 제외하면 65,507을 데이터로 사용할 수 있습니다.

혹시, 그 65,507과 같은 크기를 구할 수 있을까요? Windows의 경우, getsockopt에 SO_MAX_MSG_SIZE 옵션을 제공하므로 다음과 같이 구하는 것이 가능합니다.

using System.Net.Sockets;
using System.Net;
using System.Text;

internal class Program
{
    public static int SOL_SOCKET
    {
        get
        {
            if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
            {
                return 0xffff;
            }

            return 1;
        }
    }

    public const int SO_MAX_MSG_SIZE = 0x2003; // 8195

    static void Main(string[] args)
    {
        using (UdpClient listener = new UdpClient(60900))
        {
            byte[] buffer = new byte[4];
            listener.Client.GetRawSocketOption(SOL_SOCKET, SO_MAX_MSG_SIZE, buffer);
            int result = BitConverter.ToInt32(buffer, 0);
            Console.WriteLine($"MSG_SIZE: {result}"); // 출력 결과: 65507
        }
    }
}

반면, 리눅스의 경우에는 이 값을 구하는 방법이 딱히 없습니다. 위의 C# 프로그램을 리눅스에서 실행하면 GetRawSocketOption 실행 시 "System.Net.Sockets.SocketException: 'Operation not supported'" 오류가 발생하는데요, 닷넷 BCL의 문제가 아닌 C/C++로 호출한 경우에도 이 옵션은 리눅스에서 제공하지 않는 것을 알 수 있습니다.

#include <cstdio>
#include <sys/socket.h>

int main()
{
    int value = 0;
    socklen_t len = sizeof(value);
    int fd = socket(AF_INET, SOCK_DGRAM, 0);
    int result = getsockopt(fd, SOL_SOCKET, 0x2003, &value, &len);
    // 위의 코드는 사실 의미가 없는데, Windows와 Linux는 옵션 값이 다르기 때문에 리눅스에서 SO_MAX_MSG_SIZE 옵션을 제공한다 해도 값이 0x2003이진 않을 것입니다.

    printf("%d, (%d)\n", value, result); // 출력 결과: 0, -1

    return 0;
}

어쨌든, 이 크기보다 큰 datagram을 보내려고 하면 예외가 발생합니다.

using (UdpClient client = new UdpClient())
{
    // 만약 아래의 IP 주소를 127.0.0.1로 하면 예외가 발생하지 않습니다.
    client.Connect(IPAddress.Parse("192.168.100.50"), 65000);
    client.Send(new byte[65507 + 1]);
}
/*
// Send에서 예외 발생 

// Windows 환경의 오류 메시지 (WSAEMSGSIZE 10040)
Unhandled exception. System.Net.Sockets.SocketException (10040): A message sent on a datagram socket was larger than the internal message buffer or some other network limit, or the buffer used to receive a datagram into was smaller than the datagram itself.
   at System.Net.Sockets.UdpClient.Send(ReadOnlySpan`1 datagram)
   at Program.Main(String[] args) in C:\temp\ConsoleApp1\ConsoleApp1\Program.cs:line 39

// WSL/Linux 환경의 오류 메시지
Unhandled exception. System.Net.Sockets.SocketException (90): Message too long
   at System.Net.Sockets.UdpClient.Send(ReadOnlySpan`1 datagram)
   at Program.Main(String[] args) in C:\temp\ConsoleApp1\ConsoleApp1\Program.cs:line 38
*/




위와 같이 얻은 UDP Datagram의 크기는 엄밀히 로컬 PC에서만 유효한 것입니다. 일단, 패킷이 네트워크를 타고 흐르면 각각의 네트워크 장비가 정한 규격으로 인해 65,507 바이트를 정상적으로 전송하지 못할 수 있습니다. 이에 대해 검색해 보면,

What is the largest Safe UDP Packet Size on the Internet
; https://stackoverflow.com/questions/1098897/what-is-the-largest-safe-udp-packet-size-on-the-internet

RFC 1122 표준 문서상으로는 "reassemble" 가능한 최소 크기를 576 바이트라고 명시했다고 합니다.

3.3.2  Reassembly

         The IP layer MUST implement reassembly of IP datagrams.

         We designate the largest datagram size that can be reassembled
         by EMTU_R ("Effective MTU to receive"); this is sometimes
         called the "reassembly buffer size".  EMTU_R MUST be greater
         than or equal to 576, SHOULD be either configurable or
         indefinite, and SHOULD be greater than or equal to the MTU of
         the connected network(s).

따라서, 어쩌면 인터넷에는 실제로 최소 규격만을 만족하는 장비가 있을 가능성을 배제할 수는 없으므로 가능한 모든 환경에서 테스트하는 것이 권장됩니다.




UDP datagram이 물론 프로토콜상으로는 메시지 단위로 한 번에 전송은 되지만, 하위 프로토콜로 전달되면서, 예를 들어 ethernet을 사용한다면 MTU 1500 바이트의 크기로 쪼개져서 전달됩니다. 따라서 만약 1개의 ehternet 패킷으로 UDP를 보내고 싶다면 1500에서 IP header 20바이트, UDP header 8바이트를 뺀 1472 크기가 sendto에 전송할 수 있는 실제 데이터 크기가 됩니다.

재미있는 건, 실제로 UDP datagram을 MTU 크기까지만, 즉 1개의 패킷으로만 전송하도록 강제하는 옵션이 있다는 점입니다.

public const int IPPROTO_IP = 0;

public static int IP_MTU_DISCOVER
{
    get
    {
        if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
        {
            return 71;
        }

        return 10;

    }
}

public static int IP_PMTUDISC_DO
{
    get
    {
        if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
        {
            return 1;
        }

        return 2;
    }
}

using (UdpClient client = new UdpClient())
{
    client.Connect(IPAddress.Parse("192.168.100.50"), 65000);

    byte[] buffer = BitConverter.GetBytes(IP_PMTUDISC_DO);
    client.Client.SetRawSocketOption(IPPROTO_IP, IP_MTU_DISCOVER, buffer);

    int sentBytes = client.Send(new byte[1473]); // 예외 발생
    Console.WriteLine(sentBytes);
}

import socket
import sys

length = 1473
text = 'a' * length

IP_MTU_DISCOVER = 10  # linux 10, windows 71
IP_PMTUDISC_DO = 2  # linux 2, windows 1

sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
sock.setsockopt(socket.IPPROTO_IP, IP_MTU_DISCOVER, IP_PMTUDISC_DO)

endpoint = ('172.28.96.1', 60900)
sent_bytes = sock.sendto(text.encode(), endpoint) 

/* 예외 발생
$ python test.py
Traceback (most recent call last):
  File "test.py", line 10, in <module>
    sent_bytes = sock.sendto(text.encode(), endpoint)
OSError: [Errno 90] Message too long
*/

위와 같이 IP_MTU_DISCOVER, IP_PMTUDISC_DO 옵션을 적용하면, 이제 UDP datagram의 크기는 반드시 1개의 패킷으로 전달되도록 강제가 됩니다. 따라서, 위의 코드를 수행하면 1473 바이트를 전송하므로 예외가 발생합니다.

(참고로, IP_PMTUDISC_DO를 이용해 IP MTU 크기만큼 제약을 해도 SO_MAX_MSG_SIZE로 반환하는 값은 여전히 65507로 나옵니다.)




닷넷의 경우, 위의 옵션을 간단하게 Socket 타입의 DontFragment 옵션을 설정하는 것으로도 가능합니다.

using (UdpClient client = new UdpClient())
{
    client.Connect(IPAddress.Parse("192.168.100.50"), 65000);
            
    client.DontFragment = true;
    // 또는 client.Client.DontFragment = true;

    sentBytes = client.Send(new byte[1473]); // 예외 발생
}

DontFragment 속성의 소스 코드를 분석하면 결국 이렇게도 구현할 수 있는 것입니다.

client.Client.SetSocketOption(SocketOptionLevel.IP, SocketOptionName.DontFragment, 1);

위의 소스 코드를 SetRawSocketOption으로는 이렇게 바꿀 수 있는데요,

// Windows에서만 동작
// https://learn.microsoft.com/en-us/troubleshoot/windows/win32/header-library-requirement-socket-ipproto-ip#ipproto_ip-level-socket-options-in-ws2tcpiph

byte[] buffer = BitConverter.GetBytes(1);
client.Client.SetRawSocketOption(IPPROTO_IP, 14, buffer);

아쉽게도 Linux에서는 동작하지 않습니다. netinet/in.h 헤더 파일을 보면,

// https://codebrowser.dev/glibc/glibc/sysdeps/unix/sysv/linux/bits/in.h.html
#define IPV6_DONTFRAG       62

IPv6에 대한 옵션은 있어도 IPv4에 해당하는 IP_DONTFRAG나 IPV4_DONTFRAG 상수를 찾을 수가 없습니다. 단지 몇몇 문서를 보면,

https://manpages.ubuntu.com/manpages/xenial/en/man4/rawip.4freebsd.html

IP_DONTFRAG가 있는데, 아마도 리눅스 종류에 따라 다른 듯합니다.




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 1/25/2023]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 106  107  108  109  110  111  112  113  114  115  116  [117]  118  119  120  ...
NoWriterDateCnt.TitleFile(s)
10999정성태7/16/201622015오류 유형: 341. .NET Framework 4.5.2가 설치 안 되는 경우
10998정성태7/16/201621810.NET Framework: 598. C# - Excel 시트에 윈도우 폼 기능을 추가하는 방법 [1]파일 다운로드1
10997정성태7/16/201621148오류 유형: 340. HTTP Error 500.23 - Internal Server Error파일 다운로드1
10996정성태7/14/201626723Windows: 118. 유선 접속 상태에서 재부팅하면 무선 연결이 자동 연결 안되는 문제 [4]파일 다운로드1
10995정성태6/27/201620897VS.NET IDE: 109. Visual Studio 유료 버전 사용자의 주기적인 온라인 인증을 없애는 방법
10994정성태6/23/201620302개발 환경 구성: 285. 알고스팟(https://algospot.com)을 위한 Visual C++ 답안 작성 요령파일 다운로드1
10993정성태6/23/201621088.NET Framework: 597. 닷넷 메타데이터에 struct/class(값/참조 형식)의 구분이 있을까요?
10992정성태6/13/201618272오류 유형: 339. vbs 스크립트 실행 시 항상 실행 여부를 묻는 질문 창이 뜬다면?
10991정성태6/13/201622538오류 유형: 338. octave-gui 실행 시 "octave-gui.exe has stopped working" 오류
10990정성태6/13/201624099오류 유형: 337. missing type specifier - [type] assumed. Note: C++ does not support default-[type]
10989정성태6/7/201620558.NET Framework: 596. C# - WCF wsDualHttpBinding의 ClientBaseAddress 속성 - 두 번째 이야기
10988정성태6/3/201621519기타: 57. Outlook blocked access to the following potentially unsafe attachments
10987정성태6/2/201622567.NET Framework: 595. XLL 파일에 포함된 .NET 어셈블리를 추출하는 방법
10986정성태6/1/201623021.NET Framework: 594. C# - WCF wsDualHttpBinding의 ClientBaseAddress 속성
10985정성태6/1/201621568오류 유형: 336. An error occurred while ejecting 'DVD RW drive ...'
10984정성태5/31/201627238.NET Framework: 593. C# - wsDualHttpBinding WCF 예제 프로그램파일 다운로드1
10983정성태5/30/201621403VC++: 97. C++ 템플릿 remove_pointer, enable_if, is_pointer 사용 예제파일 다운로드1
10982정성태5/26/201619715오류 유형: 335. SQL Server Management Studio - The database ... is not accessible.
10981정성태5/24/201624764.NET Framework: 592. C# - Lights Out 퍼즐 풀기 [2]파일 다운로드1
10980정성태5/24/201622012VS.NET IDE: 108. Visual Studio 2013/2015를 위한 "Macros for Visual Studio"
10979정성태5/23/201625262.NET Framework: 591. C# - 조합(Combination) 예제 코드 - 두 번째 이야기파일 다운로드1
10978정성태5/23/201623886.NET Framework: 590. C# - 모든 경우의 수를 조합하는 코드 (2)파일 다운로드1
10977정성태5/23/201628354.NET Framework: 589. C# - 모든 경우의 수를 조합하는 코드 (1)파일 다운로드1
10976정성태5/20/201622758Math: 18. C# - 오일러 공식을 이용한 복소수 값의 라디안 회전파일 다운로드1
10975정성태5/20/201623150Math: 17. C# - 복소수 타입의 승수를 지원하는 Power 메서드파일 다운로드1
10974정성태5/20/201623702.NET Framework: 588. C# - OxyPlot 라이브러리로 복소수 표현파일 다운로드1
... 106  107  108  109  110  111  112  113  114  115  116  [117]  118  119  120  ...