Skip to main content
aboutsummaryrefslogtreecommitdiffstats
blob: 39a8b16461ba65b31307510348046a6fba5670f4 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
/*******************************************************************************
 * Copyright (c) 2007, 2010 Wind River Systems, Inc. and others.
 * All rights reserved. This program and the accompanying materials
 * are made available under the terms of the Eclipse Public License v1.0
 * and Eclipse Distribution License v1.0 which accompany this distribution.
 * The Eclipse Public License is available at
 * http://www.eclipse.org/legal/epl-v10.html
 * and the Eclipse Distribution License is available at
 * http://www.eclipse.org/org/documents/edl-v10.php.
 *
 * Contributors:
 *     Wind River Systems - initial API and implementation
 *******************************************************************************/

/*
 * This is utility module that contains context PID hash table implementation.
 * System specific debug context code can use this module to implement context lookup.
 */

/* TODO: consider splitting the code into two files: some clients might want to reuse PID hash table for context_find_from_pid(),
 * but use some other means to implement id2ctx() */

#define CONTEXT_PID_HASH_SIZE 1024
#define CONTEXT_PID_HASH(PID) ((unsigned)(PID) % CONTEXT_PID_HASH_SIZE)

static LINK context_pid_hash[CONTEXT_PID_HASH_SIZE];
static size_t pid_hash_link_offset = 0;

#define ctx2pidlink(ctx) ((LINK *)((char *)(ctx) + pid_hash_link_offset))
#define pidlink2ctx(lnk) ((Context *)((char *)(lnk) - pid_hash_link_offset))

static void link_context(Context * ctx) {
    LINK * h = context_pid_hash + CONTEXT_PID_HASH(EXT(ctx)->pid);

    assert(ctx->mem != NULL);
    assert(EXT(ctx)->pid != 0);
    assert(context_find_from_pid(EXT(ctx)->pid, ctx->parent != NULL) == NULL);
    list_add_first(&ctx->ctxl, &context_root);
    list_add_first(ctx2pidlink(ctx), h);
    ctx->ref_count++;
}

Context * context_find_from_pid(pid_t pid, int thread) {
    LINK * h = context_pid_hash + CONTEXT_PID_HASH(pid);
    LINK * l = h->next;

    assert(is_dispatch_thread());
    if (l == NULL) return NULL;
    while (l != h) {
        Context * ctx = pidlink2ctx(l);
        if (EXT(ctx)->pid == pid &&
            (ctx->parent != NULL) == (thread != 0)) return ctx;
        l = l->next;
    }
    return NULL;
}

Context * id2ctx(const char * id) {
    pid_t parent = 0;
    pid_t pid = id2pid(id, &parent);
    if (pid == 0) return NULL;
    return context_find_from_pid(pid, parent != 0);
}

static void pid_hash_context_exited(Context * ctx, void * args) {
    list_remove(ctx2pidlink(ctx));
}

static void ini_context_pid_hash(void) {
    int i;
    static ContextEventListener l = { NULL, pid_hash_context_exited };
    for (i = 0; i < CONTEXT_PID_HASH_SIZE; i++) list_init(context_pid_hash + i);
    pid_hash_link_offset = context_extension(sizeof(LINK));
    add_context_event_listener(&l, NULL);
}

Back to the top