Обнаруживать, запускается ли приложение Swift из Xcode

Я хотел бы программно определить, запускается ли приложение iOS непосредственно из XCode (либо в симуляторе, либо на привязном устройстве). Я пробовал решение -D DEBUG, описанное здесь здесь, но когда я отсоединяюсь от Xcode и снова запускаю приложение, он все еще думает об этом в режиме отладки. Я думаю, что я ищу версию Swift эту функцию

#include <assert.h>
#include <stdbool.h>
#include <sys/types.h>
#include <unistd.h>
#include <sys/sysctl.h>

static bool AmIBeingDebugged(void)
    // Returns true if the current process is being debugged (either 
    // running under the debugger or has a debugger attached post facto).
{
    int                 junk;
    int                 mib[4];
    struct kinfo_proc   info;
    size_t              size;

    // Initialize the flags so that, if sysctl fails for some bizarre 
    // reason, we get a predictable result.

    info.kp_proc.p_flag = 0;

    // 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();

    // Call sysctl.
    size = sizeof(info);
    junk = sysctl(mib, sizeof(mib) / sizeof(*mib), &info, &size, NULL, 0);
    assert(junk == 0);

    // We're being debugged if the P_TRACED flag is set.
    return ( (info.kp_proc.p_flag & P_TRACED) != 0 );
}

Ответ 1

Вы можете просто сохранить функцию C и вызвать ее из Swift. Рецепты, приведенные в Как вызвать код Objective-C из Swift, применимы и к чистому C-коду.

Но на самом деле не слишком сложно перевести этот код в Swift:

func amIBeingDebugged() -> Bool {

    var info = kinfo_proc()
    var mib : [Int32] = [CTL_KERN, KERN_PROC, KERN_PROC_PID, getpid()]
    var size = strideofValue(info)
    let junk = sysctl(&mib, UInt32(mib.count), &info, &size, nil, 0)
    assert(junk == 0, "sysctl failed")
    return (info.kp_proc.p_flag & P_TRACED) != 0
}

Примечания:

  • kinfo_proc() создает полностью инициализированную структуру со всеми поля установлены на ноль, поэтому установка info.kp_proc.p_flag = 0 не требуется.
  • Тип C int Int32 - Swift.
  • sizeof(info) из кода C должен быть strideOfValue(info) в Swift, чтобы включить прокладку структуры. С sizeofValue(info) вышеуказанный код всегда возвращал false в Simulator для 64-разрядных устройств. Это была самая трудная часть, чтобы понять.

Обновление для Swift 3 (Xcode 8):

strideofValue и связанные функции уже не существуют, они были заменены на MemoryLayout:

func amIBeingDebugged() -> Bool {

    var info = kinfo_proc()
    var mib : [Int32] = [CTL_KERN, KERN_PROC, KERN_PROC_PID, getpid()]
    var size = MemoryLayout<kinfo_proc>.stride
    let junk = sysctl(&mib, UInt32(mib.count), &info, &size, nil, 0)
    assert(junk == 0, "sysctl failed")
    return (info.kp_proc.p_flag & P_TRACED) != 0
}