Microsoft MVP성태의 닷넷 이야기
.NET Framework: 462. 커널 객체를 위한 null DACL 생성 방법 [링크 복사], [링크+제목 복사],
조회: 23621
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 3개 있습니다.)

커널 객체를 위한 null DACL 생성 방법

(비스타 운영체제 이상의 경우 세션 0번에 활성화되는) NT 서비스에서 생성한 커널 객체를 사용자 로그인 환경에서 실행한 응용 프로그램으로부터 접근하려는 경우 보안 오류가 발생합니다.

간단하게 재현을 해볼까요? ^^

NT 서비스로 테스트하면 번거로우니 psexec.exe를 이용해 Local SYSTEM 계정으로 실행해 보겠습니다.

Local SYSTEM 권한으로 코드를 실행하는 방법
; https://www.sysnet.pe.kr/2/0/1436

그래서 다음의 코드는,

using System;
using System.Threading;

class Program
{
    static void Main(string[] args)
    {
        EventWaitHandle ewh = new EventWaitHandle(false, EventResetMode.ManualReset, "my.event");

        Console.WriteLine("Wait for 'my.event'...");
        ewh.WaitOne();
    }
}

(관리자 권한으로 실행한 cmd.exe 창에서) psexec의 도움을 받아 SYSTEM 계정으로 실행되고 이벤트 객체는 그 권한에 영향을 받습니다.

c:\temp>psexec -s c:\temp\ConsoleApplication1.exe

PsExec v1.98 - Execute processes remotely
Copyright (C) 2001-2010 Mark Russinovich
Sysinternals - www.sysinternals.com

Wait for 'my.event'...

위와 같이 실행하면 ConsoleApplication1.exe 창은 보이지 않지만 작업관리자에서 Session 0번에 SYSTEM 권한으로 실행된 것을 확인할 수 있습니다.

이 상태에서 현재 윈도우에 로그온 한 사용자가 다음의 프로그램을 실행하면,

using System;
using System.Threading;

class Program
{
    static void Main(string[] args)
    {
        EventWaitHandle ewh = EventWaitHandle.OpenExisting("my.event");
        ewh.Set();
    }
}

OpenExisting에서 예외가 발생합니다.

Unhandled Exception: System.Threading.WaitHandleCannotBeOpenedException: No handle of the given name exists.
   at System.Threading.EventWaitHandle.OpenExisting(String name)
   at Program.Main(String[] args) in c:\...\ConsoleApplication2\Program.cs:line 8

그런데, 이상하군요. 접근 권한 에러가 아니라 아예 named kernel object 자체를 찾을 수 없다고 나옵니다. 왜냐하면 이름 공간이 다르기 때문인데요. 이럴 때는 Global 접두사를 붙여줘야 합니다.

// 처음 여는 쪽
EventWaitHandle ewh = new EventWaitHandle(false, EventResetMode.ManualReset, @"Global\my.event");

// 기존의 객체를 여는 쪽
EventWaitHandle ewh = EventWaitHandle.OpenExisting(@"Global\my.event");

그럼 해당 핸들을 찾을 수 있게 되지만, 예견했던대로 이제는 보안 오류가 발생합니다.

Unhandled Exception: System.UnauthorizedAccessException: Access to the path is denied.
   at System.IO.__Error.WinIOError(Int32 errorCode, String maybeFullPath)
   at System.Threading.EventWaitHandle.OpenExistingWorker(String name, EventWaitHandleRights rights, EventWaitHandle& result)
   at System.Threading.EventWaitHandle.OpenExisting(String name)
   at Program.Main(String[] args) in c:\...\ConsoleApplication2\Program.cs:line 9

자, 이 오류를 다시 벗어나기 위해서는 예전에 소개한 WorldSid를 부여해 권한을 열어주면 됩니다.

Named 동기화 개체 생성 시 System.UnauthorizedAccessException 예외 발생하는 경우
; https://www.sysnet.pe.kr/2/0/1170

EventWaitHandleSecurity mSec = new EventWaitHandleSecurity();

var everyoneSid = new SecurityIdentifier(WellKnownSidType.WorldSid, null);

EventWaitHandleAccessRule rule = new EventWaitHandleAccessRule(everyoneSid,
    EventWaitHandleRights.Synchronize | EventWaitHandleRights.Modify,
    AccessControlType.Allow);
mSec.AddAccessRule(rule);

bool createdNew;

string eventName = @"Global\my.event";
EventWaitHandle ewh = new EventWaitHandle(false, EventResetMode.ManualReset, eventName, out createdNew, mSec);

Console.WriteLine("Wait for 'my.event'...");
ewh.WaitOne();




그런데, 이렇게 이름 있는 커널 객체에 AccessRule/Security 관련 객체들이 제공되는 것은 .NET 2.0부터입니다. 물론 요즘에는 1.x 버전을 사용할 일이 거의 없기 때문에 상관없지만 가끔은 .NET 2.0에서도 AccessRule/Security 객체들이 제공하지 않는 경우가 있어서 문제입니다.

바로 NamedPipe를 사용할 때가 그 한 예인데요. .NET에서 NamedPipe를 사용할 수 있는 NamedPipeServerStream / NamedPipeClientStream 객체는 .NET 3.5부터 제공되기 때문에 .NET 2.0 전체를 지원해야 하는 3rd-party 라이브러리 제작자들은 이 클래스를 사용하기가 살짝 난감합니다.

이 때문에 CreateNamedPipe Pinvoke 호출을 이용해 파이프를 생성해야 하는데, 그렇게 되면 보안 설정도 CreateNamedPipe의 마지막 인자에 전달되는 SECURITY_ATTRIBUTES를 이용해 null DACL을 만들 수밖에 없습니다. 다행히 ^^ 이에 대해서는 검색해 보니 친절하게 다음과 같이 설명하고 있습니다.

Creating a Null DACL in Managed Code 
; http://codemortem.blogspot.kr/2006/01/creating-null-dacl-in-managed-code.html

예를 들어, CreateFile에 (참고로 .NET 2.0부터는 FileStream에 FileSecurity가 제공되므로 그것을 쓰는 것이 좋습니다.) SECURITY_ATTRIBUTES를 null dacl로 채우고 직접 설정하는 코드는 다음과 같습니다.

using Microsoft.Win32.SafeHandles;
using System;
using System.IO;
using System.Runtime.InteropServices;
using System.Security.AccessControl;
using System.Security.Principal;
using System.Threading;

class Program
{
    [StructLayout(LayoutKind.Sequential)]
    public struct SECURITY_ATTRIBUTES
    {
        public uint nLength;
        public IntPtr lpSecurityDescriptor;
        public bool bInheritHandle;
    }

    [DllImport("kernel32.dll", SetLastError = true)]
    public static extern SafeFileHandle CreateFile(
       String pipeName,
       uint dwDesiredAccess,
       uint dwShareMode,
       ref SECURITY_ATTRIBUTES lpSecurityAttributes,
       uint dwCreationDisposition,
       uint dwFlagsAndAttributes,
       IntPtr hTemplate);

    static void Main(string[] args)
    {
        CreateFileWithLogonSecurity();
        CreateFileWithNullDacl();
    }
    
    private static void CreateFileWithLogonSecurity()
    {
        File.WriteAllText(@"c:\temp\current_system.txt", "test");
    }

    // http://codemortem.blogspot.kr/2006/01/creating-null-dacl-in-managed-code.html
    public static SECURITY_ATTRIBUTES GetNullDacl()
    {
        // Build NULL DACL (Allow everyone full access)
        RawSecurityDescriptor gsd = new RawSecurityDescriptor(ControlFlags.DiscretionaryAclPresent, null, null, null, null);

        // Construct SECURITY_ATTRIBUTES structure
        SECURITY_ATTRIBUTES sa = new SECURITY_ATTRIBUTES();
        sa.nLength = (uint)Marshal.SizeOf(typeof(SECURITY_ATTRIBUTES));
        sa.bInheritHandle = false;

        // Get binary form of the security descriptor and copy it into place
        byte[] desc = new byte[gsd.BinaryLength];
        gsd.GetBinaryForm(desc, 0);
        sa.lpSecurityDescriptor = Marshal.AllocHGlobal(desc.Length); // This Alloc is Freed by the Disposer or Finalizer
        Marshal.Copy(desc, 0, sa.lpSecurityDescriptor, desc.Length);

        return sa;
    }

    private static void CreateFileWithNullDacl()
    {
        SECURITY_ATTRIBUTES sa = GetNullDacl();
        SafeFileHandle pHandle = CreateFile(@"c:\temp\null_dacl.txt",
                  (uint)FileAccess.ReadWrite,
                  0,
                  ref sa,
                  (uint)2 ,
                  0,
                  IntPtr.Zero);
        int lastError = Marshal.GetLastWin32Error();
        if (pHandle.IsInvalid == true)
        {
            
            Console.WriteLine("Invalid: " + lastError);
        }

        pHandle.Close();
    }
}

"psexec -s" 옵션을 통해 위의 프로그램을 실행시킨 후 CreateFileWithNullDacl 메서드로 생성된 null_dacl.txt 파일을 탐색기에서 보안 설정을 확인하면 다음과 같이 특이하게 나옵니다.

null_dacl_1.png

정리하자면, 보안을 풀어 세션 경계를 넘어서까지 공유하고 싶은 커널 객체가 있다면 각각의 커널 객체에 대한 ...Security/...AccessRule을 사용하고, 그것이 여의치 않을 때는 null dacl을 사용하면 됩니다.

(첨부 파일은 위의 코드를 포함합니다.)





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

[연관 글]






[최초 등록일: ]
[최종 수정일: 7/17/2021]

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

비밀번호

댓글 작성자
 




... 61  62  63  64  65  66  67  68  69  70  71  72  [73]  74  75  ...
NoWriterDateCnt.TitleFile(s)
12108정성태1/10/202017389오류 유형: 587. Kernel Driver 시작 시 127(The specified procedure could not be found.) 오류 메시지 발생
12107정성태1/10/202018522.NET Framework: 877. C# - 프로세스의 모든 핸들을 열람 - 두 번째 이야기
12106정성태1/8/202019604VC++: 136. C++ - OSR Driver Loader와 같은 Legacy 커널 드라이버 설치 프로그램 제작 [1]
12105정성태1/8/202018113디버깅 기술: 153. C# - PEB를 조작해 로드된 DLL을 숨기는 방법
12104정성태1/7/202019296DDK: 9. 커널 메모리를 읽고 쓰는 NT Legacy driver와 C# 클라이언트 프로그램 [4]
12103정성태1/7/202022390DDK: 8. Visual Studio 2019 + WDK Legacy Driver 제작- Hello World 예제 [1]파일 다운로드2
12102정성태1/6/202018765디버깅 기술: 152. User 권한(Ring 3)의 프로그램에서 _ETHREAD 주소(및 커널 메모리를 읽을 수 있다면 _EPROCESS 주소) 구하는 방법
12101정성태1/5/202018982.NET Framework: 876. C# - PEB(Process Environment Block)를 통해 로드된 모듈 목록 열람
12100정성태1/3/202016478.NET Framework: 875. .NET 3.5 이하에서 IntPtr.Add 사용
12099정성태1/3/202019295디버깅 기술: 151. Windows 10 - Process Explorer로 확인한 Handle 정보를 windbg에서 조회 [1]
12098정성태1/2/202019057.NET Framework: 874. C# - 커널 구조체의 Offset 값을 하드 코딩하지 않고 사용하는 방법 [3]
12097정성태1/2/202017127디버깅 기술: 150. windbg - Wow64, x86, x64에서의 커널 구조체(예: TEB) 구조체 확인
12096정성태12/30/201919848디버깅 기술: 149. C# - DbgEng.dll을 이용한 간단한 디버거 제작 [1]
12095정성태12/27/201921546VC++: 135. C++ - string_view의 동작 방식
12094정성태12/26/201919270.NET Framework: 873. C# - 코드를 통해 PDB 심벌 파일 다운로드 방법
12093정성태12/26/201918832.NET Framework: 872. C# - 로딩된 Native DLL의 export 함수 목록 출력파일 다운로드1
12092정성태12/25/201917654디버깅 기술: 148. cdb.exe를 이용해 (ntdll.dll 등에 정의된) 커널 구조체 출력하는 방법
12091정성태12/25/201919935디버깅 기술: 147. pdb 파일을 다운로드하기 위한 symchk.exe 실행에 필요한 최소 파일 [1]
12090정성태12/24/201920033.NET Framework: 871. .NET AnyCPU로 빌드된 PE 헤더의 로딩 전/후 차이점 [1]파일 다운로드1
12089정성태12/23/201918935디버깅 기술: 146. gflags와 _CrtIsMemoryBlock을 이용한 Heap 메모리 손상 여부 체크
12088정성태12/23/201917923Linux: 28. Linux - 윈도우의 "Run as different user" 기능을 shell에서 실행하는 방법
12087정성태12/21/201918380디버깅 기술: 145. windbg/sos - Dictionary의 entries 배열 내용을 모두 덤프하는 방법 (do_hashtable.py) [1]
12086정성태12/20/201920874디버깅 기술: 144. windbg - Marshal.FreeHGlobal에서 발생한 덤프 분석 사례
12085정성태12/20/201918833오류 유형: 586. iisreset - The data is invalid. (2147942413, 8007000d) 오류 발생 - 두 번째 이야기 [1]
12084정성태12/19/201919264디버깅 기술: 143. windbg/sos - Hashtable의 buckets 배열 내용을 모두 덤프하는 방법 (do_hashtable.py) [1]
12083정성태12/17/201922250Linux: 27. linux - lldb를 이용한 .NET Core 응용 프로그램의 메모리 덤프 분석 방법 [2]
... 61  62  63  64  65  66  67  68  69  70  71  72  [73]  74  75  ...