Как я могу узнать, сколько памяти использует мое приложение С++ на Mac

Некоторые операции в моем приложении используют больше памяти, чем, по моему мнению, должны, и я хотел бы регистрировать текущее использование памяти, чтобы определить, какие именно.

Есть ли системный вызов, который вернет объем используемой в данный момент памяти?


person David Sykes    schedule 09.10.2009    source источник


Ответы (4)


Следующая функция C возвращает время ЦП и резидентную память процесса pid. Чтобы получить ресурсы других процессов, вам нужны root-права. Вы также можете попробовать getrusage(), но я никогда не заставляю его правильно работать с использованием памяти. Получение процессорного времени с помощью getrusage() всегда работает для меня.

Функция адаптирована из исходников команд ps и top. Это часть моей программы, которая следит за памятью других процессов.

#ifdef __APPLE__

#include <sys/types.h>
#include <sys/sysctl.h>
#include <sys/vmmeter.h>
#include <mach/mach_init.h>
#include <mach/mach_host.h>
#include <mach/mach_port.h>
#include <mach/mach_traps.h>
#include <mach/task_info.h>
#include <mach/thread_info.h>
#include <mach/thread_act.h>
#include <mach/vm_region.h>
#include <mach/vm_map.h>
#include <mach/task.h>
#include <mach/shared_memory_server.h>

typedef struct vmtotal vmtotal_t;

typedef struct { /* dynamic process information */
    size_t rss, vsize;
    double utime, stime;
} RunProcDyn;

/* On Mac OS X, the only way to get enough information is to become root. Pretty frustrating!*/
int run_get_dynamic_proc_info(pid_t pid, RunProcDyn *rpd)
{
    task_t task;
    kern_return_t error;
    mach_msg_type_number_t count;
    thread_array_t thread_table;
    thread_basic_info_t thi;
    thread_basic_info_data_t thi_data;
    unsigned table_size;
    struct task_basic_info ti;

    error = task_for_pid(mach_task_self(), pid, &task);
    if (error != KERN_SUCCESS) {
        /* fprintf(stderr, "++ Probably you have to set suid or become root.\n"); */
        rpd->rss = rpd->vsize = 0;
        rpd->utime = rpd->stime = 0;
        return 0;
    }
    count = TASK_BASIC_INFO_COUNT;
    error = task_info(task, TASK_BASIC_INFO, (task_info_t)&ti, &count);
    assert(error == KERN_SUCCESS);
    { /* adapted from ps/tasks.c */
        vm_region_basic_info_data_64_t b_info;
        vm_address_t address = GLOBAL_SHARED_TEXT_SEGMENT;
        vm_size_t size;
        mach_port_t object_name;
        count = VM_REGION_BASIC_INFO_COUNT_64;
        error = vm_region_64(task, &address, &size, VM_REGION_BASIC_INFO,
                             (vm_region_info_t)&b_info, &count, &object_name);
        if (error == KERN_SUCCESS) {
            if (b_info.reserved && size == (SHARED_TEXT_REGION_SIZE) &&
                ti.virtual_size > (SHARED_TEXT_REGION_SIZE + SHARED_DATA_REGION_SIZE))
            {
                ti.virtual_size -= (SHARED_TEXT_REGION_SIZE + SHARED_DATA_REGION_SIZE);
            }
        }
        rpd->rss = ti.resident_size;
        rpd->vsize = ti.virtual_size;
    }
    { /* calculate CPU times, adapted from top/libtop.c */
        unsigned i;
        rpd->utime = ti.user_time.seconds + ti.user_time.microseconds * 1e-6;
        rpd->stime = ti.system_time.seconds + ti.system_time.microseconds * 1e-6;
        error = task_threads(task, &thread_table, &table_size);
        assert(error == KERN_SUCCESS);
        thi = &thi_data;
        for (i = 0; i != table_size; ++i) {
            count = THREAD_BASIC_INFO_COUNT;
            error = thread_info(thread_table[i], THREAD_BASIC_INFO, (thread_info_t)thi, &count);
            assert(error == KERN_SUCCESS);
            if ((thi->flags & TH_FLAGS_IDLE) == 0) {
                rpd->utime += thi->user_time.seconds + thi->user_time.microseconds * 1e-6;
                rpd->stime += thi->system_time.seconds + thi->system_time.microseconds * 1e-6;
            }
            if (task != mach_task_self()) {
                error = mach_port_deallocate(mach_task_self(), thread_table[i]);
                assert(error == KERN_SUCCESS);
            }
        }
        error = vm_deallocate(mach_task_self(), (vm_offset_t)thread_table, table_size * sizeof(thread_array_t));
        assert(error == KERN_SUCCESS);
    }
    mach_port_deallocate(mach_task_self(), task);
    return 0;
}

#endif /* __APPLE__ */
person user172818    schedule 09.10.2009
comment
Очень полезный кусок кода. Однако один комментарий: в опубликованном коде оператор mach_port_deallocate(mach_task_self(), thread_table[i]) защищен условием (task != mach_task_self()) и mach_port_deallocate(mach_task_self(), task) (в конце ) выполняется во всех случаях. Я считаю, что это следует переключить (это последний оператор, который должен выполняться условно), после того, как возникли проблемы с вызовом mach_port_deallocate(mach_task_self(), task) с task == mach_task_self(). Это подтверждается документацией, в которой указано, что mach_task_self() возвращает кэшированное значение. - person tcovo; 02.03.2011
comment
Это очень полезно. К сожалению, файл ‹shared_memory_server.h› помечен как устаревший в текущем пакете SDK. - person the_mandrill; 10.05.2012
comment
Макросы GLOBAL_SHARED_TEXT_SEGMENT, SHARED_TEXT_REGION_SIZE и SHARED_DATA_REGION_SIZE определены в ‹mach/shared_memory_server.h›, а не в ‹mach/shared_region.h›. - person martemiev; 30.01.2015
comment
Есть ли у вас справочные страницы для thread_info и т. д. в OS X? Я не вижу ни одного на OS X Yosemite. - person Drux; 19.02.2015
comment
Есть ли способ вычесть общую память из резидентного размера? - person rstackhouse; 13.03.2015
comment
Содержит ли task_info также использование процессора процесса, как это показано в команде /usr/bin/top под заголовком %CPU? Благодарность ! - person ; 23.08.2018

запустите приложение с помощью инструментов. попробуй и оцени результат...

person kent    schedule 09.10.2009
comment
Инструменты, безусловно, очень полезная программа, спасибо, но у меня не всегда есть доступ к приложению. - person David Sykes; 12.10.2009

Вы можете попробовать функцию mallocDebug: http://developer.apple.com/mac/library/DOCUMENTATION/Performance/Conceptual/ManagingMemory/Articles/FindingPatterns.html.

person Patrice Bernassola    schedule 09.10.2009

Следуя совету @user172818, я попробовал getrusage, и у меня это сработало:

#include <sys/time.h>
#include <sys/resource.h>

long getMemoryUsage() 
{
  struct rusage usage;
  if(0 == getrusage(RUSAGE_SELF, &usage))
    return usage.ru_maxrss; // bytes
  else
    return 0;
}

Я использую Mac OS X 10.9.4 с компилятором Apple LLVM version 5.1 (clang-503.0.40) (based on LLVM 3.4svn).

person ChronoTrigger    schedule 28.09.2014
comment
Пробовал это, и, похоже, он не учитывал память, которая была freed, и поэтому возвращал неточные и раздутые цифры. - person mwag; 29.11.2019
comment
Вероятно, это связано с тем, что malloced память не возвращается SO free, процесс повторно использует ее для будущих mallocs. - person ChronoTrigger; 02.12.2019
comment
Какова бы ни была причина, если в самом начале запуска программы цифры были схожими, то к моменту ее завершения цифры, выдаваемые ru_maxrss, были на несколько порядков больше, чем выдаваемые такими инструментами, как Activity Monitor или valgrind. +массив - person mwag; 02.12.2019