Skip to content

Latest commit

 

History

History
493 lines (352 loc) · 13.8 KB

object_get_info.md

File metadata and controls

493 lines (352 loc) · 13.8 KB

mx_object_get_info

NAME

object_get_info - query information about an object

SYNOPSIS

#include <magenta/syscalls.h>
#include <magenta/syscalls/object.h>

mx_status_t mx_object_get_info(mx_handle_t handle, uint32_t topic,
                               void* buffer, size_t buffer_size,
                               size_t* actual, size_t* avail);

DESCRIPTION

object_get_info() requests information about the provided handle (or the object the handle refers to). The topic parameter indicates what specific information is desired.

buffer is a pointer to a buffer of size buffer_size to return the information.

actual is an optional pointer to return the number of records that were written to buffer.

avail is an optional pointer to return the number of records that are available to read.

If the buffer is insufficiently large, avail will be larger than actual.

[TOC]

TOPICS

MX_INFO_HANDLE_VALID

handle type: Any

buffer type: n/a

Returns MX_OK if handle is valid, a negative status otherwise. No records are returned and buffer may be NULL.

MX_INFO_HANDLE_BASIC

handle type: Any

buffer type: mx_info_handle_basic_t[1]

typedef struct mx_info_handle_basic {
    // The unique id assigned by kernel to the object referenced by the
    // handle.
    mx_koid_t koid;

    // The immutable rights assigned to the handle. Two handles that
    // have the same koid and the same rights are equivalent and
    // interchangeable.
    mx_rights_t rights;

    // The object type: channel, event, socket, etc.
    uint32_t type;                // mx_obj_type_t;

    // The koid of the logical counterpart or parent object of the
    // object referenced by the handle. Otherwise this value is zero.
    mx_koid_t related_koid;

    // Set to MX_OBJ_PROP_WAITABLE if the object referenced by the
    // handle can be waited on; zero otherwise.
    uint32_t props;               // mx_obj_props_t;
} mx_info_handle_basic_t;

MX_INFO_PROCESS

handle type: Process

buffer type: mx_info_process_t[1]

typedef struct mx_info_process {
    // The process's return code; only valid if |exited| is true.
    // Guaranteed to be non-zero if the process was killed by |mx_task_kill|.
    int return_code;

    // True if the process has ever left the initial creation state,
    // even if it has exited as well.
    bool started;

    // If true, the process has exited and |return_code| is valid.
    bool exited;

    // True if a debugger is attached to the process.
    bool debugger_attached;
} mx_info_process_t;

MX_INFO_PROCESS_THREADS

handle type: Process

buffer type: mx_koid_t[n]

Returns an array of mx_koid_t, one for each running thread in the Process at that moment in time.

MX_INFO_RESOURCE_CHILDREN

handle type: Resource

buffer type: mx_rrec_t[n]

Returns an array of mx_rrec_t, one for each child Resource of the provided Resource handle.

MX_INFO_RESOURCE_RECORDS

handle type: Resource

buffer type: mx_rrec_t[n]

Returns an array of mx_rrec_t, one for each Record associated with the provided Resource handle.

MX_INFO_THREAD

handle type: Thread

buffer type: mx_info_thread_t[1]

typedef struct mx_info_thread {
    // One of MX_THREAD_STATE_* values.
    uint32_t state;

    // If nonzero, the thread has gotten an exception and is waiting for
    // the exception to be handled by the specified port.
    // The value is one of MX_EXCEPTION_PORT_TYPE_*.
    // Note: If the thread is waiting for an exception response then |state|
    // will have the value MX_THREAD_STATE_BLOCKED.
    uint32_t wait_exception_port_type;
} mx_info_thread_t;

The values in this struct are mainly for informational and debugging purposes at the moment.

The MX_THREAD_STATE_* values are defined by

#include <magenta/syscalls/object.h>
  • MX_THREAD_STATE_NEW
  • MX_THREAD_STATE_RUNNING
  • MX_THREAD_STATE_SUSPENDED
  • MX_THREAD_STATE_BLOCKED
  • MX_THREAD_STATE_DYING
  • MX_THREAD_STATE_DEAD

The MX_EXCEPTION_PORT_TYPE_* values are defined by

#include <magenta/syscalls/exception.h>
  • MX_EXCEPTION_PORT_TYPE_NONE
  • MX_EXCEPTION_PORT_TYPE_DEBUGGER
  • MX_EXCEPTION_PORT_TYPE_THREAD
  • MX_EXCEPTION_PORT_TYPE_PROCESS
  • MX_EXCEPTION_PORT_TYPE_SYSTEM

MX_INFO_THREAD_EXCEPTION_REPORT

handle type: Thread

buffer type: mx_exception_report_t[1]

#include <magenta/syscalls/exception.h>

If the thread is currently in an exception and is waiting for an exception response, then this returns the exception report as a single mx_exception_report_t, with status MX_OK.

Returns MX_ERR_BAD_STATE if the thread is not in an exception and waiting for an exception response.

MX_INFO_VMAR

handle type: VM Address Region

buffer type: mx_info_vmar_t[1]

typedef struct mx_info_vmar {
    // Base address of the region.
    uintptr_t base;

    // Length of the region, in bytes.
    size_t len;
} mx_info_vmar_t;

MX_INFO_JOB_CHILDREN

handle type: Job

buffer type: mx_koid_t[n]

Returns an array of mx_koid_t, one for each direct child Job of the provided Job handle.

MX_INFO_JOB_PROCESSES

handle type: Job

buffer type: mx_koid_t[n]

Returns an array of mx_koid_t, one for each direct child Process of the provided Job handle.

MX_INFO_TASK_STATS

handle type: Process

buffer type: mx_info_task_stats_t[1]

// Statistics about resources (e.g., memory) used by a task. Can be relatively
// expensive to gather.
typedef struct mx_info_task_stats {
    // The total size of mapped memory ranges in the task.
    // Not all will be backed by physical memory.
    size_t mem_mapped_bytes;

    // For the fields below, a byte is considered committed if it's backed by
    // physical memory. Some of the memory may be double-mapped, and thus
    // double-counted.

    // Committed memory that is only mapped into this task.
    size_t mem_private_bytes;

    // Committed memory that is mapped into this and at least one other task.
    size_t mem_shared_bytes;

    // A number that estimates the fraction of mem_shared_bytes that this
    // task is responsible for keeping alive.
    //
    // An estimate of:
    //   For each shared, committed byte:
    //   mem_scaled_shared_bytes += 1 / (number of tasks mapping this byte)
    //
    // This number is strictly smaller than mem_shared_bytes.
    size_t mem_scaled_shared_bytes;
} mx_info_task_stats_t;

Additional errors:

  • MX_ERR_BAD_STATE: If the target process is not currently running.

MX_INFO_PROCESS_MAPS

handle type: Process other than your own, with MX_RIGHT_READ

buffer type: mx_info_maps_t[n]

The mx_info_maps_t array is a depth-first pre-order walk of the target process's Aspace/VMAR/Mapping tree.

typedef struct mx_info_maps {
    // Name if available; empty string otherwise.
    char name[MX_MAX_NAME_LEN];
    // Base address.
    mx_vaddr_t base;
    // Size in bytes.
    size_t size;

    // The depth of this node in the tree.
    // Can be used for indentation, or to rebuild the tree from an array
    // of mx_info_maps_t entries, which will be in depth-first pre-order.
    size_t depth;
    // The type of this entry; indicates which union entry is valid.
    uint32_t type; // mx_info_maps_type_t
    union {
        mx_info_maps_mapping_t mapping;
        // No additional fields for other types.
    } u;
} mx_info_maps_t;

The depth field of each entry describes its relationship to the nodes that come before it. Depth 0 is the root Aspace, depth 1 is the root VMAR, and all other entries have depth 2 or greater.

To get a full picture of how a process uses its VMOs and how a VMO is used by various processes, you may need to combine this information with MX_INFO_PROCESS_VMOS.

See the vmaps command-line tool for an example user of this topic, and to dump the maps of arbitrary processes by koid.

Additional errors:

  • MX_ERR_ACCESS_DENIED: If the appropriate rights are missing, or if a process attempts to call this on a handle to itself. It's not safe to examine yourself: buffer will live inside the Aspace being examined, and the kernel can't safely fault in pages of the buffer while walking the Aspace.
  • MX_ERR_BAD_STATE: If the target process is not currently running, or if its address space has been destroyed.

MX_INFO_PROCESS_VMOS

handle type: Process other than your own, with MX_RIGHT_READ

buffer type: mx_info_vmos_t[n]

The mx_info_vmos_t array is list of all VMOs pointed to by the target process. Some VMOs are mapped, some are pointed to by handles, and some are both.

Note: The same VMO may appear multiple times due to multiple mappings/handles. Also, because VMOs can change as the target process runs, the same VMO may have different values each time it appears. It is the caller's job to resolve any duplicates.

To get a full picture of how a process uses its VMOs and how a VMO is used by various processes, you may need to combine this information with MX_INFO_PROCESS_MAPS.

// Describes a VMO.
typedef struct mx_info_vmo {
    // The koid of this VMO.
    mx_koid_t koid;

    // The name of this VMO.
    char name[MX_MAX_NAME_LEN];

    // The size of this VMO; i.e., the amount of virtual address space it
    // would consume if mapped.
    uint64_t size_bytes;

    // If this VMO is a clone, the koid of its parent. Otherwise, zero.
    // See |flags| for the type of clone.
    mx_koid_t parent_koid;

    // The number of clones of this VMO, if any.
    size_t num_children;

    // The number of times this VMO is currently mapped into VMARs.
    // Note that the same process will often map the same VMO twice,
    // and both mappings will be counted here. (I.e., this is not a count
    // of the number of processes that map this VMO; see share_count.)
    size_t num_mappings;

    // An estimate of the number of unique address spaces that
    // this VMO is mapped into. Every process has its own address space,
    // and so does the kernel.
    size_t share_count;

    // Bitwise OR of MX_INFO_VMO_* values.
    uint32_t flags;

    // If |MX_INFO_VMO_TYPE(flags) == MX_INFO_VMO_TYPE_PAGED|, the amount of
    // memory currently allocated to this VMO; i.e., the amount of physical
    // memory it consumes. Undefined otherwise.
    uint64_t committed_bytes;

    // If |flags & MX_INFO_VMO_VIA_HANDLE|, the handle rights.
    // Undefined otherwise.
    mx_rights_t handle_rights;
} mx_info_vmo_t;

See the vmos command-line tool for an example user of this topic, and to dump the VMOs of arbitrary processes by koid.

MX_INFO_KMEM_STATS

handle type: Resource (Specifically, the root resource)

buffer type: mx_info_kmem_stats_t[1]

// Information about kernel memory usage.
// Can be expensive to gather.
typedef struct mx_info_kmem_stats {
    // The total amount of physical memory available to the system.
    size_t total_bytes;

    // The amount of unallocated memory.
    size_t free_bytes;

    // The amount of memory reserved by and mapped into the kernel for reasons
    // not covered by other fields in this struct. Typically for readonly data
    // like the ram disk and kernel image, and for early-boot dynamic memory.
    size_t wired_bytes;

    // The amount of memory allocated to the kernel heap.
    size_t total_heap_bytes;

    // The portion of |total_heap_bytes| that is not in use.
    size_t free_heap_bytes;

    // The amount of memory committed to VMOs, both kernel and user.
    // A superset of all userspace memory.
    // Does not include certain VMOs that fall under |wired_bytes|.
    //
    // TODO(dbort): Break this into at least two pieces: userspace VMOs that
    // have koids, and kernel VMOs that don't. Or maybe look at VMOs
    // mapped into the kernel aspace vs. everything else.
    size_t vmo_bytes;

    // The amount of memory used for architecture-specific MMU metadata
    // like page tables.
    size_t mmu_overhead_bytes;

    // Non-free memory that isn't accounted for in any other field.
    size_t other_bytes;
} mx_info_kmem_stats_t;

RETURN VALUE

mx_object_get_info() returns MX_OK on success. In the event of failure, a negative error value is returned.

ERRORS

MX_ERR_BAD_HANDLE handle is not a valid handle.

MX_ERR_WRONG_TYPE handle is not an appropriate type for topic

MX_ERR_ACCESS_DENIED: If handle does not have the necessary rights for the operation.

MX_ERR_INVALID_ARGS buffer, actual, or avail are invalid pointers.

MX_ERR_NO_MEMORY Temporary out of memory failure.

MX_ERR_BUFFER_TOO_SMALL The topic returns a fixed number of records, but the provided buffer is not large enough for these records.

MX_ERR_NOT_SUPPORTED topic does not exist.

EXAMPLES

bool is_handle_valid(mx_handle_t handle) {
    return mx_object_get_info(
        handle, MX_INFO_HANDLE_VALID, NULL, 0, NULL, NULL) == MX_OK;
}

mx_koid_t get_object_koid(mx_handle_t handle) {
    mx_info_handle_basic_t info;
    if (mx_object_get_info(handle, MX_INFO_HANDLE_BASIC,
                           &info, sizeof(info), NULL, NULL) != MX_OK) {
        return 0;
    }
    return info.koid;
}

void examine_threads(mx_handle_t proc) {
    mx_koid_t threads[128];
    size_t count, avail;

    if (mx_object_get_info(proc, MX_INFO_PROCESS_THREADS, threads,
                           sizeof(threads), &count, &avail) != MX_OK) {
        // Error!
    } else {
        if (avail > count) {
            // More threads than space in array;
            // could call again with larger array.
        }
        for (size_t n = 0; n < count; n++) {
            do_something(thread[n]);
        }
    }
}

SEE ALSO

handle_close, handle_duplicate, handle_replace, object_get_child.