성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] Roll A Lisp In C - Reading ; https...
[정성태] Java - How to use the Foreign Funct...
[정성태] 제가 큰 실수를 했군요. ^^; Delegate를 통한 Bein...
[정성태] Working with Rust Libraries from C#...
[정성태] Detecting blocking calls using asyn...
[정성태] 아쉽게도, 커뮤니티는 아니고 개인 블로그입니다. ^^
[정성태] 질문이 잘 이해가 안 됩니다. 우선, 해당 소스코드에서 ILis...
[양승조
] var대신 dinamic으로 선언해서 해결은 했습니다. 맞는 해...
[양승조
] 또 막혔습니다. ㅠㅠ var list = props[i].Ge...
[양승조
] 아. 감사합니다. 어제는 안됐던것 같은데....정신을 차려야겠네...
글쓰기
제목
이름
암호
전자우편
HTML
홈페이지
유형
제니퍼 .NET
닷넷
COM 개체 관련
스크립트
VC++
VS.NET IDE
Windows
Team Foundation Server
디버깅 기술
오류 유형
개발 환경 구성
웹
기타
Linux
Java
DDK
Math
Phone
Graphics
사물인터넷
부모글 보이기/감추기
내용
<div style='display: inline'> <h1 style='font-family: Malgun Gothic, Consolas; font-size: 20pt; color: #006699; text-align: center; font-weight: bold'>C# - Environment.OSVersion의 문제점 및 윈도우 운영체제의 버전을 구하는 다양한 방법</h1> <p> 아래의 글에서도 잘 설명하고 있지만,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > WPF - 당신의 Environment.OSVersion은 거짓말을 하고 있다 ; <a target='tab' href='https://blog.gilbok.com/wpf-your-environment-dot-osversion-is-a-lier/'>https://blog.gilbok.com/wpf-your-environment-dot-osversion-is-a-lier/</a> </pre> <br /> Windows 10/2019에서조차 Environment.OSVersion.Version은 6.2.9200.0을 반환합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Console.WriteLine(Environment.OSVersion.Version); // Windows 10 / Server 2019 - 6.2.9200.0 // Windows Server 2016 - 6.2.9200.0 // Windows Server 2012 R2 - 6.2.9200.0 // Windows Server 2008 R2 - 6.1.7601.65536 </pre> <br /> 이 문제를 해결하기 위해 "<a target='tab' href='https://www.sysnet.pe.kr/2/0/11318'>Application Manifest File</a> (Windows only)" 유형의 파일을 추가하면 다음과 같은 내용을 갖는 app.manifest 파일이 추가되고,<br /> <br /> <pre style='height: 400px; margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > <?xml version="1.0" encoding="utf-8"?> <assembly manifestVersion="1.0" xmlns="urn:schemas-microsoft-com:asm.v1"> <assemblyIdentity version="1.0.0.0" name="MyApplication.app"/> <trustInfo xmlns="urn:schemas-microsoft-com:asm.v2"> <security> <requestedPrivileges xmlns="urn:schemas-microsoft-com:asm.v3"> <!-- UAC Manifest Options If you want to change the Windows User Account Control level replace the requestedExecutionLevel node with one of the following. <requestedExecutionLevel level="asInvoker" uiAccess="false" /> <requestedExecutionLevel level="requireAdministrator" uiAccess="false" /> <requestedExecutionLevel level="highestAvailable" uiAccess="false" /> Specifying requestedExecutionLevel element will disable file and registry virtualization. Remove this element if your application requires this virtualization for backwards compatibility. --> <requestedExecutionLevel level="asInvoker" uiAccess="false" /> </requestedPrivileges> </security> </trustInfo> <compatibility xmlns="urn:schemas-microsoft-com:compatibility.v1"> <application> <!-- A list of the Windows versions that this application has been tested on and is designed to work with. Uncomment the appropriate elements and Windows will automatically select the most compatible environment. --> <!-- Windows Vista --> <!--<supportedOS Id="{e2011457-1546-43c5-a5fe-008deee3d3f0}" />--> <!-- Windows 7 --> <!--<supportedOS Id="{35138b9a-5d96-4fbd-8e2d-a2440225f93a}" />--> <!-- Windows 8 --> <!--<supportedOS Id="{4a2f28e3-53b9-4441-ba9c-d69d4a4a6e38}" />--> <!-- Windows 8.1 --> <!--<supportedOS Id="{1f676c76-80e1-4239-95bb-83d0f6d0da78}" />--> <!-- Windows 10 --> <!--<supportedOS Id="{8e0f7a12-bfb3-4fe8-b9a5-48fd50a15a9a}" />--> </application> </compatibility> <!-- Indicates that the application is DPI-aware and will not be automatically scaled by Windows at higher DPIs. Windows Presentation Foundation (WPF) applications are automatically DPI-aware and do not need to opt in. Windows Forms applications targeting .NET Framework 4.6 that opt into this setting, should also set the 'EnableWindowsFormsHighDpiAutoResizing' setting to 'true' in their app.config. --> <!-- <application xmlns="urn:schemas-microsoft-com:asm.v3"> <windowsSettings> <dpiAware xmlns="http://schemas.microsoft.com/SMI/2005/WindowsSettings">true</dpiAware> </windowsSettings> </application> --> <!-- Enable themes for Windows common controls and dialogs (Windows XP and later) --> <!-- <dependency> <dependentAssembly> <assemblyIdentity type="win32" name="Microsoft.Windows.Common-Controls" version="6.0.0.0" processorArchitecture="*" publicKeyToken="6595b64144ccf1df" language="*" /> </dependentAssembly> </dependency> --> </assembly> </pre> <br /> 이중에서 compatibility 영역을 해당 닷넷 애플리케이션의 요구에 따라 적절하게 주석을 해제하면 됩니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > <compatibility xmlns="urn:schemas-microsoft-com:compatibility.v1"> <application> <!-- Windows Vista --> <span style='color: blue; font-weight: bold'><supportedOS Id="{e2011457-1546-43c5-a5fe-008deee3d3f0}" /></span> <!-- Windows 7 --> <span style='color: blue; font-weight: bold'><supportedOS Id="{35138b9a-5d96-4fbd-8e2d-a2440225f93a}" /></span> <!-- Windows 8 --> <span style='color: blue; font-weight: bold'><supportedOS Id="{4a2f28e3-53b9-4441-ba9c-d69d4a4a6e38}" /></span> <!-- Windows 8.1 --> <span style='color: blue; font-weight: bold'><supportedOS Id="{1f676c76-80e1-4239-95bb-83d0f6d0da78}" /></span> <!-- Windows 10 --> <span style='color: blue; font-weight: bold'><supportedOS Id="{8e0f7a12-bfb3-4fe8-b9a5-48fd50a15a9a}" /></span> </application> </compatibility> </pre> <br /> 이후 다시 빌드하면 다음과 같이 정상적으로 버전을 가져옵니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Console.WriteLine(Environment.OSVersion.Version); // Windows 10 20H2 - 10.0.19042.0 // Windows Server 2019 - 10.0.17763.0 // Windows Server 2016 - 10.0.14393.0 // Windows Server 2012 R2 - 6.3.9600.0 // Windows Server 2008 R2 - 6.1.7601.65536 </pre> <br /> <hr style='width: 50%' /><br /> <br /> 그런데, 만약 app.manifest를 임의로 추가할 수 없는 라이브러리에서의 코드라면 어떻게 해야 할까요?<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > How to get Windows Version - as in "Windows 10, version 1607"? ; <a target='tab' href='https://stackoverflow.com/questions/39778525/how-to-get-windows-version-as-in-windows-10-version-1607'>https://stackoverflow.com/questions/39778525/how-to-get-windows-version-as-in-windows-10-version-1607</a> </pre> <br /> 위의 글을 보면 추가로 고려할 수 있는 2가지 방법을 소개합니다.<br /> <br /> <ol> <li>레지스트리 키 이용 - HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows NT\CurrentVersion\ReleaseId</li> <li>WMIN 이용 - "Win32_OperatingSystem" / "Version"</li> </ol> <br /> 여기서 레지스트리의 경우 <a target='tab' href='https://en.wikipedia.org/wiki/Windows_10_version_history'>Windows 10의 YYMM 표기</a>로 된 버전을 반환합니다. <br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Windows 10 20H2 버전의 경우 - 2009 Windows Server 2019 - 1809 Windows Server 2016 - 1607 Windows Server 2012 R2 - (empty) Windows Server 2008 R2 - (empty) </pre> <br /> 반면 WMI의 경우 Environment.OSVersion.Version과 결과는 유사하지만 3자리 버전 번호를 반환한다는 차이가 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Windows 10 20H2 버전의 경우 - 10.0.19042 Windows Server 2019 - 10.0.17763 Windows Server 2016 - 10.0.14393 Windows Server 2012 R2 - 6.3.9600 Windows Server 2008 R2 - 6.1.7601 </pre> <br /> 한 가지 단점이라면 WMI의 특성상 최초 호출 시 다소 느리다는 점을 염두에 두어야 합니다.<br /> <br /> <hr style='width: 50%' /><br /> <br /> 만약, 버전을 구하는 것이 아닌, 특정 버전을 만족하는지에 대한 정보만 필요한 것이라면 P/Invoke를 활용하는 것도 방법일 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > IsWindowsVersionOrGreater function (versionhelpers.h) ; <a target='tab' href='https://learn.microsoft.com/en-us/windows/win32/api/versionhelpers/nf-versionhelpers-iswindowsversionorgreater'>https://learn.microsoft.com/en-us/windows/win32/api/versionhelpers/nf-versionhelpers-iswindowsversionorgreater</a> </pre> <br /> 단지 이 함수가 내부적으로는 versionhelpers.h에 다음과 같이 VerifyVersionInfo를 사용하는 함수로 정의되어 있으므로,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > VERSIONHELPERAPI IsWindowsVersionOrGreater(WORD wMajorVersion, WORD wMinorVersion, WORD wServicePackMajor) { OSVERSIONINFOEXW osvi = { sizeof(osvi), 0, 0, 0, 0, {0}, 0, 0 }; DWORDLONG const dwlConditionMask = VerSetConditionMask( VerSetConditionMask( VerSetConditionMask( 0, VER_MAJORVERSION, VER_GREATER_EQUAL), VER_MINORVERSION, VER_GREATER_EQUAL), VER_SERVICEPACKMAJOR, VER_GREATER_EQUAL); osvi.dwMajorVersion = wMajorVersion; osvi.dwMinorVersion = wMinorVersion; osvi.wServicePackMajor = wServicePackMajor; return VerifyVersionInfoW(&osvi, VER_MAJORVERSION | VER_MINORVERSION | VER_SERVICEPACKMAJOR, dwlConditionMask) != FALSE; } </pre> <br /> 결국 저 코드를 구현하는 C# 코드를 작성해야 합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > [DllImport("kernel32.dll", CharSet = CharSet.Auto)] extern static bool <a target='tab' href='https://learn.microsoft.com/en-us/windows/win32/api/winbase/nf-winbase-verifyversioninfow'>VerifyVersionInfoW</a>(ref OSVERSIONINFOEXW lpVersionInformation, int dwTypeMask, ulong dwlConditionMask); [DllImport("kernel32.dll", CharSet = CharSet.Auto)] extern static ulong <a target='tab' href='https://learn.microsoft.com/en-us/windows/win32/api/winnt/nf-winnt-versetconditionmask'>VerSetConditionMask</a>(ulong conditionMask, int typeMask, byte condition); static bool IsWindowsVersionOrGreater(ushort wMajorVersion, ushort wMinorVersion, ushort wServicePackMajor) { OSVERSIONINFOEXW osvi = new OSVERSIONINFOEXW(); osvi.dwOSVersionInfoSize = Marshal.SizeOf(osvi); ulong dwlConditionMask = VerSetConditionMask( VerSetConditionMask( VerSetConditionMask( 0, VER_MAJORVERSION, (byte)VER_GREATER_EQUAL), VER_MINORVERSION, (byte)VER_GREATER_EQUAL), VER_SERVICEPACKMAJOR, (byte)VER_GREATER_EQUAL); osvi.dwMajorVersion = wMajorVersion; osvi.dwMinorVersion = wMinorVersion; osvi.wServicePackMajor = wServicePackMajor; return VerifyVersionInfoW(ref osvi, VER_MAJORVERSION | VER_MINORVERSION | VER_SERVICEPACKMAJOR, dwlConditionMask) != false; } </pre> <br /> 그런데 "<a target='tab' href='https://learn.microsoft.com/en-us/windows/win32/api/versionhelpers/nf-versionhelpers-iswindowsversionorgreater'>IsWindowsVersionOrGreater</a>" 함수 도움말에는 나오지 않지만 저 코드가 동작하려면 app.manifest를 필요로 합니다. (유사하게 <a target='tab' href='https://learn.microsoft.com/en-us/windows/win32/api/versionhelpers/nf-versionhelpers-iswindows10orgreater'>IsWindows10OrGreater</a> 함수의 도움말에는 manifest가 필요하다는 설명이 있습니다.) 즉, 저걸 사용하느니 Environment.OSVersion으로 구현하는 것이 차라리 낫습니다.<br /> <br /> 게다가 위의 구현은 우리가 기대했던 버전 정보에 따른 결과를 반환하지도 않습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Console.WriteLine($">= Windows 8: {IsWindows8OrGreater()}"); Console.WriteLine($">= Windows 10: {IsWindows10OrGreater()}"); // if without manifest, false Console.WriteLine($">= 10.0.17063: {IsWindowsVersionOrGreater(10, 0, 17063)}"); // always false Console.WriteLine($">= 10.0.19042: {IsWindowsVersionOrGreater(10, 0, 19042)}"); // always false Console.WriteLine($">= 10.0.20000: {IsWindowsVersionOrGreater(10, 0, 20000)}"); // always false /* 출력 결과 Windows 10 20H2에서 테스트 >= Windows 8: True >= Windows 10: True >= 10.0.17063: False >= 10.0.19042: False >= 10.0.20000: False */ static bool IsWindows8OrGreater() { return IsWindowsVersionOrGreater(6, 2, 0); } static bool IsWindows10OrGreater() { return IsWindowsVersionOrGreater(10, 0, 0); } </pre> <br /> 이유를 알 수 없지만, versionhelpers.h에 담고 있는 C/C++ 코드가 이미 잘못된 부분이 있는데요, 우리가 알고 있는 3번째 버전 정보 - 예를 들어 "10.0.19042.0"에서 19042는 ServicePack 번호가 아닌 Build 번호이기 때문에 해당 코드를 다음과 같이 변경해야 합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > static bool IsWindowsVersionOrGreater(ushort wMajorVersion, ushort wMinorVersion, <span style='color: blue; font-weight: bold'>ushort wBuildVersion</span>) { OSVERSIONINFOEXW osvi = new OSVERSIONINFOEXW(); osvi.dwOSVersionInfoSize = Marshal.SizeOf(osvi); ulong dwlConditionMask = VerSetConditionMask( VerSetConditionMask( VerSetConditionMask( 0, VER_MAJORVERSION, (byte)VER_GREATER_EQUAL), VER_MINORVERSION, (byte)VER_GREATER_EQUAL), <span style='color: blue; font-weight: bold'>VER_BUILDNUMBER</span>, (byte)VER_GREATER_EQUAL); osvi.dwMajorVersion = wMajorVersion; osvi.dwMinorVersion = wMinorVersion; <span style='color: blue; font-weight: bold'>osvi.dwBuildNumber = wBuildVersion;</span> return VerifyVersionInfoW(ref osvi, VER_MAJORVERSION | VER_MINORVERSION | <span style='color: blue; font-weight: bold'>VER_BUILDNUMBER</span>, dwlConditionMask) != false; } Console.WriteLine($">= Windows 8: {IsWindows8OrGreater()}"); Console.WriteLine($">= Windows 10: {IsWindows10OrGreater()}"); Console.WriteLine($">= 10.0.17063: {IsWindowsVersionOrGreater(10, 0, 17063)}"); Console.WriteLine($">= 10.0.19042: {IsWindowsVersionOrGreater(10, 0, 19042)}"); Console.WriteLine($">= 10.0.20000: {IsWindowsVersionOrGreater(10, 0, 20000)}"); /* 출력 결과 Windows 10 20H2에서 테스트 >= Windows 8: True >= Windows 10: True >= 10.0.17063: True >= 10.0.19042: True >= 10.0.20000: False */ </pre> <br /> <hr style='width: 50%' /><br /> <br /> 마지막으로 ntdll.dll에서 제공하는 RtlGetVersion 함수를 보겠습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > RtlGetVersion function (wdm.h) ; <a target='tab' href='https://learn.microsoft.com/en-us/windows-hardware/drivers/ddi/wdm/nf-wdm-rtlgetversion'>https://learn.microsoft.com/en-us/windows-hardware/drivers/ddi/wdm/nf-wdm-rtlgetversion</a> </pre> <br /> 문서를 보면 "Minimum supported client"가 Windows 2000이기 때문에 사실상 현재 모든 운영체제에서 호출 가능한 함수입니다. 그리고 출력 결과도 WMI의 것과 완전히 같으며 app.manifest 없이도 잘 동작합니다. 단지 SDK 레벨에서 숨겨진 함수라는 점을 제외한다면 현재 닷넷 라이브러리 수준에서 사용할 수 있는 가장 최선의 후보가 RtlGetVersion입니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > [DllImport("ntdll.dll", CharSet = CharSet.Auto)] extern static int RtlGetVersion(ref OSVERSIONINFOEXW lpVersionInformation); Console.WriteLine($"RtlGetVersion: {GetRtlVersion()}"); static Version GetRtlVersion() { OSVERSIONINFOEXW info = new OSVERSIONINFOEXW(); info.dwOSVersionInfoSize = Marshal.SizeOf(info); RtlGetVersion(ref info); return new Version(info.dwMajorVersion, info.dwMinorVersion, info.dwBuildNumber); } </pre> <br /> 이 글의 예제 코드는 다음의 github repo에 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > DotNetSamples/WinConsole/OSVersionInfo ; <a target='tab' href='https://github.com/stjeong/DotNetSamples/tree/master/WinConsole/OSVersionInfo'>https://github.com/stjeong/DotNetSamples/tree/master/WinConsole/OSVersionInfo</a> </pre> <br /> <hr style='width: 50%' /><br /> <br /> 그나저나, 테스트하면서 알게 된 이상한 점이 하나 있는데요, w3wp.exe에서 호스팅하는 ASP.NET 응용 프로그램에서 Environment.OSVersion.Version을 구하면 정상적으로 운영체제 버전이 반환됩니다. 왜 그것이 이상하냐면, 바로 w3wp.exe는 manifest 파일을 포함하고 있지 않기 때문입니다.<br /> <br /> 도대체 w3wp.exe에는 어떤 마법이 숨겨져 있는 걸까요? 혹시 아시는 분은 덧글 부탁드립니다. ^^<br /> <br /> 이렇게 혼란스러운 와중에 한 가지 좋은 소식이라면, .NET Core의 경우 app.manifest 파일 없이 Environment.OSVersion.Version은 RtlGetVersion처럼 동작합니다.<br /> </p><br /> <br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
3769
(왼쪽의 숫자를 입력해야 합니다.)