Q&A

  • 네트워크 공유팁을 해보고 나서
평소에 가끔 http://www.howto.pe.kr/ 에 들리는데,  

[네트웍/인터넷] 네트워크 공유 설정/공유 사용 권한/해제 하기 (Windows XP) 라는

팁을 보고 현재 제가 하고 있는 프로젝트에 적용을 시도했으나,

"폴더를 공유할 수 없습니다. 에라번호(2118)이 나오는데요"

무슨 연유인지 알고 싶어 메일을 보냅니다.



질문1) 에라번호 2118이 무엇을 의미하는지요.

질문2) 이 에라를 해결하고 네트워크 공유를 할려면은 어떻게 해야하는지요.





현재 특별히 소스를 고치것은 없고요.



function ShareResource(sServerName: String;
                       sNetName: String;
                       ShareType: DWORD;
                       sRemark: String;
                       Permission: DWORD;
                       Max_uses: DWORD;
                       Current_uses: DWORD;
                       sFilePath: String;
                       sPasswd: String): Integer;

함수의 마지막에

   LocalFree(Cardinal(ptr_acl)); 때문에 CPU  에라창이 보여서

이것만 //LocalFree(Cardinal(ptr_acl)); 마킹하였습니다.



제컴은

   펜티엄4    3.2Ghz이며 윈도우XP 프로페셔널을 사용중입니다.

  
아래는 김영대님 홈페이지에서 퍼온 팁소스입니다.

============================================================
unit Unit1;

interface

uses
  Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs,
  StdCtrls;

type
  TForm1 = class(TForm)
    Button1: TButton;
    Button2: TButton;
    procedure Button1Click(Sender: TObject);
    procedure Button2Click(Sender: TObject);
  private
    { Private declarations }
  public
    { Public declarations }
  end;

const
  MB_ERR_INVALID_CHARS = $00000008;
  ACL_REVISION = 2;
  
type
  // Windows 9x는 문자열을 ANSI로 사용하지만 Windows NT이상은 Unicode로 사용한다
  // Windows 9x에서 주로 SHARE_INFO_50 을 사용하지만 Windows NT 이상에서는 주로 SHARE_INFO_2를 사용한다
  // 특히 Windows NT 이상에서 공유사용권한을 주려면 SHARE_INFO_502를 사용해야 한다
  // Windows 9x는 Svrapi.dll에 의해 Windows NT는 Netapi32.dll 에 의해 서비스 받는다
  share_info_502 = packed record
    shi502_netname        : PWideChar;
    shi502_type           : DWORD;
    shi502_remark         : PWideChar;
    shi502_permissions    : DWORD;
    shi502_max_uses       : DWORD;
    shi502_current_uses   : DWORD;
    shi502_path           : PWideChar;
    shi502_passwd         : PWideChar;
    shi502_reserved       : DWORD;
    shi502_security_descriptor: PSECURITY_DESCRIPTOR ;
  end;
  pshare_info_502 = ^share_info_502;

var
  Form1: TForm1;

implementation
{$R *.DFM}

function NetShareAdd(ServerName: PWideChar;
                     ShareLevel: SmallInt;
                     Buffer: Pointer;
                     Error: PLongword): Integer; stdcall; external 'Netapi32.DLL';

function NetShareDel(ServerName: PWideChar;
                     NetName: PWideChar;
                     Reserved: Word): Integer; stdcall; external 'Netapi32.DLL';

procedure AddAccountNameToAcl(ptr_acl: PACL; AccountName: String; Access_Mask: DWORD);
var
  SidBuffer: array[1..100] of byte;
  ptr_sid: PSID;
  SidBufSz: dword;
  DomainNameBuf: String;
  DomainNameBufSz: dword;
  SNU: Sid_Name_Use;
begin
  ptr_sid  := @SidBuffer;
  SidBufSz := SizeOf(SidBuffer);
  SetLength(DomainNameBuf, 256);
  DomainNameBufSz := 256;

  // 사용자명 또는 그룹명이 시스템에 있는지 검사
  if not LookupAccountName(nil, PChar(AccountName), ptr_sid, SidBufSz,
                           PChar(DomainNameBuf), DomainNameBufSz, SNU) then
    ShowMessage('사용자 또는 그룹명이 없습니다: '+AccountName);

  SetLength(DomainNameBuf, DomainNameBufSz);
  
  // 사용자 또는 그룹에 접근 권한주기(모든 권한은 GENERIC_ALL 로 설정)
  Win32Check(AddAccessAllowedAce(ptr_acl^, ACL_REVISION,
                                 Access_Mask,
                                 ptr_sid));
end;
                    
function ShareResource(sServerName: String;
                       sNetName: String;
                       ShareType: DWORD;
                       sRemark: String;
                       Permission: DWORD;
                       Max_uses: DWORD;
                       Current_uses: DWORD;
                       sFilePath: String;
                       sPasswd: String): Integer;
var
  MyShare: share_info_502;
  PMyShare: pshare_info_502;
  bServerName, bNetName, bRemark, bFilePath, bPasswd: PWideChar;
  parm_err: DWORD;

  SD: TSecurityDescriptor;
  SA: TSecurityAttributes;
  SI: SECURITY_INFORMATION;
  AclBuf: array[1..1024] of byte;
  ptr_acl: PACL;  
begin
  try
    // 유니코드를 저장할 버퍼(double-byte)
    bServerName := GlobalAllocPtr(GHND, (length(sServerName)*2)+2);
    if bServerName = nil then
      raise Exception.Create('Out of memory');
    sServerName := UpperCase(sServerName);
    // 유니코드로 변환
    MultiByteToWideChar(CP_ACP, MB_ERR_INVALID_CHARS,
                        pchar(sServerName), length(sServerName),
                        bServerName, (length(sServerName)*2)+2);
                            
    bNetName := GlobalAllocPtr(GHND, (length(sNetName)*2)+2);
    if bNetName = nil then
      raise Exception.Create('Out of memory');
    sNetName := UpperCase(sNetName);
    MultiByteToWideChar(CP_ACP, MB_ERR_INVALID_CHARS,
                        pchar(sNetName), length(sNetName),
                        bNetName, (length(sNetName)*2)+2);

    bRemark := GlobalAllocPtr(GHND, (length(sRemark)*2)+2);
    if bRemark = nil then
      raise Exception.Create('Out of memory');
    MultiByteToWideChar(CP_ACP, MB_ERR_INVALID_CHARS,
                        pchar(sRemark), length(sRemark),
                        bRemark, (length(sRemark)*2)+2);

    bFilePath := GlobalAllocPtr(GHND, (length(sFilePath)*2)+2);
    if bFilePath = nil then
      raise Exception.Create('Out of memory');
    sFilePath := UpperCase(sFilePath);
    MultiByteToWideChar(CP_ACP, MB_ERR_INVALID_CHARS,
                        pchar(sFilePath), length(sFilePath),
                        bFilePath, (length(sFilePath)*2)+2);

    bPasswd := GlobalAllocPtr(GHND, (length(sPasswd)*2)+2);
    if bPasswd = nil then
      raise Exception.Create('Out of memory');
    MultiByteToWideChar(CP_ACP, MB_ERR_INVALID_CHARS,
                        pchar(sPasswd), length(sPasswd),
                        bPasswd, (length(sPasswd)*2)+2);

    // 권한 설정                        
    Win32Check(InitializeSecurityDescriptor(@SD, SECURITY_DESCRIPTOR_REVISION));
    ptr_acl:= @AclBuf;
    Win32Check(InitializeACL(ptr_acl^, SizeOf(AclBuf), ACL_REVISION));

    // 폴더에 권한을 줄 사용자 또는 그룹을 지정한다 (XP에서는 everyone 에 공유하는것만 가능하다)
    AddAccountNameToAcl(ptr_acl, 'everyone',    STANDARD_RIGHTS_READ); // 읽기
    //AddAccountNameToAcl(ptr_acl, 'Power Users', STANDARD_RIGHTS_ALL);  // 모든 권한
    Win32Check(SetSecurityDescriptorDacl(@SD, true, ptr_acl, false));

    // 공유 설정
    MyShare.shi502_netname      := bNetName;
    MyShare.shi502_type         := ShareType;
    MyShare.shi502_remark       := bRemark;
    MyShare.shi502_permissions  := Permission; // not support share-level security in Windows NT or higher
    MyShare.shi502_max_uses     := Max_uses;
    MyShare.shi502_current_uses := Current_uses;
    MyShare.shi502_path         := bFilePath;
    MyShare.shi502_passwd       := bPasswd;
    MyShare.shi502_security_descriptor := @SD;
    PMyShare := @MyShare;
    Result := NetShareAdd(bServerName, 502, PMyShare, @parm_err);

  finally
    if bServerName <> nil then
      GlobalFreePtr(bServerName);
    if bNetName <> nil then
      GlobalFreePtr(bNetName);
    if bRemark <> nil then
      GlobalFreePtr(bRemark);
    if bFilePath <> nil then
      GlobalFreePtr(bFilePath);
    if bPasswd <> nil then
      GlobalFreePtr(bPasswd);

    LocalFree(Cardinal(ptr_acl));
  end;
end;  

function DeleteShare(sServerName: String; sNetName: String): Integer;
var
  bServerName, bNetName: PWideChar;
begin
  try
    // 유니코드를 저장할 버퍼(double-byte)
    bServerName := GlobalAllocPtr(GHND, (length(sServerName)*2)+2);
    if bServerName = nil then
      raise Exception.Create('Out of memory');
    sServerName := UpperCase(sServerName);
    // 유니코드로 변환
    MultiByteToWideChar(CP_ACP, MB_ERR_INVALID_CHARS,
                        pchar(sServerName), length(sServerName),
                        bServerName, (length(sServerName)*2)+2);
                            
    bNetName := GlobalAllocPtr(GHND, (length(sNetName)*2)+2);
    if bNetName = nil then
      raise Exception.Create('Out of memory');
    sNetName := UpperCase(sNetName);
    MultiByteToWideChar(CP_ACP, MB_ERR_INVALID_CHARS,
                        pchar(sNetName), length(sNetName),
                        bNetName, (length(sNetName)*2)+2);

    // 공유 해제
    Result := NetShareDel(bServerName, bNetName, 0);
  finally
    if bServerName <> nil then
      GlobalFreePtr(bServerName);
    if bNetName <> nil then
      GlobalFreePtr(bNetName);
  end;
end;

procedure TForm1.Button1Click(Sender: TObject);
var
  rt: Integer;
begin
  rt := ShareResource('' ,         // 컴퓨터 이름 (nil 은 Local Machine을 의미합니다)
                      'KIMBIT',  // 공유이름
                      0,           // 0=Disk drive, 1=Print queue
                      '공유예제',  // 설명
                      0,           // 권한
                      4,           // 사용자 제한(최대한 허용은 DWORD(-1))
                      0,           //
                      'C:\DOWNLOAD',    // 공유할 폴더명 (전부 대문자로 지정)
                      '1111');     // 암호
  if rt = 0 then
    ShowMessage('폴더가 공유되었습니다')
  else
    ShowMessage('폴더를 공유할 수 없습니다.  에러번호('+IntToStr(rt)+')');
end;

procedure TForm1.Button2Click(Sender: TObject);
var
  rt: Integer;
begin
  rt := DeleteShare('', 'ShareZZZ'); // 컴퓨터 이름(nil 은 Local Machine을 의미)
  if rt = 0 then
    ShowMessage('폴더의 공유를 해제하였습니다')
  else
    ShowMessage('폴더의 공유를 해제할 수 없습니다.  에러번호('+IntToStr(rt)+')');
end;

end.

2  COMMENTS
  • Profile
    임똘 2005.10.19 03:04
    tt
  • Profile
    최용일 2005.10.10 09:15
    안녕하세요. 최용일입니다.

    2118 The name has already been shared.  NERR_DuplicateShare

    에러내용으로 보아 이미 공유가 된것을 또 공유하실려고 하는것 같네요...

    팁모아에서 검색해보시면 네트워크 에러코드들 목록이 있습니다. 참고하세요...

    ^^ 항상 즐코하세요...