16baa0a5aSFrederic Weisbecker #include "../perf.h" 26baa0a5aSFrederic Weisbecker #include <stdlib.h> 36baa0a5aSFrederic Weisbecker #include <stdio.h> 46baa0a5aSFrederic Weisbecker #include <string.h> 5b3165f41SArnaldo Carvalho de Melo #include "session.h" 66baa0a5aSFrederic Weisbecker #include "thread.h" 76baa0a5aSFrederic Weisbecker #include "util.h" 86e086437SFrederic Weisbecker #include "debug.h" 91902efe7SFrederic Weisbecker #include "comm.h" 106baa0a5aSFrederic Weisbecker 1199d725fcSAdrian Hunter struct thread *thread__new(pid_t pid, pid_t tid) 126baa0a5aSFrederic Weisbecker { 131902efe7SFrederic Weisbecker char *comm_str; 141902efe7SFrederic Weisbecker struct comm *comm; 15c824c433SArnaldo Carvalho de Melo struct thread *thread = zalloc(sizeof(*thread)); 166baa0a5aSFrederic Weisbecker 17c824c433SArnaldo Carvalho de Melo if (thread != NULL) { 18*93d5731dSArnaldo Carvalho de Melo thread->mg = map_groups__new(); 19*93d5731dSArnaldo Carvalho de Melo if (thread->mg == NULL) 20*93d5731dSArnaldo Carvalho de Melo goto out_free; 21*93d5731dSArnaldo Carvalho de Melo 22c824c433SArnaldo Carvalho de Melo thread->pid_ = pid; 23c824c433SArnaldo Carvalho de Melo thread->tid = tid; 24c824c433SArnaldo Carvalho de Melo thread->ppid = -1; 251902efe7SFrederic Weisbecker INIT_LIST_HEAD(&thread->comm_list); 261902efe7SFrederic Weisbecker 271902efe7SFrederic Weisbecker comm_str = malloc(32); 281902efe7SFrederic Weisbecker if (!comm_str) 291902efe7SFrederic Weisbecker goto err_thread; 301902efe7SFrederic Weisbecker 311902efe7SFrederic Weisbecker snprintf(comm_str, 32, ":%d", tid); 321902efe7SFrederic Weisbecker comm = comm__new(comm_str, 0); 331902efe7SFrederic Weisbecker free(comm_str); 341902efe7SFrederic Weisbecker if (!comm) 351902efe7SFrederic Weisbecker goto err_thread; 361902efe7SFrederic Weisbecker 371902efe7SFrederic Weisbecker list_add(&comm->list, &thread->comm_list); 386baa0a5aSFrederic Weisbecker } 396baa0a5aSFrederic Weisbecker 40c824c433SArnaldo Carvalho de Melo return thread; 411902efe7SFrederic Weisbecker 421902efe7SFrederic Weisbecker err_thread: 43*93d5731dSArnaldo Carvalho de Melo map_groups__delete(thread->mg); 44*93d5731dSArnaldo Carvalho de Melo out_free: 451902efe7SFrederic Weisbecker free(thread); 461902efe7SFrederic Weisbecker return NULL; 476baa0a5aSFrederic Weisbecker } 486baa0a5aSFrederic Weisbecker 49c824c433SArnaldo Carvalho de Melo void thread__delete(struct thread *thread) 50591765fdSArnaldo Carvalho de Melo { 511902efe7SFrederic Weisbecker struct comm *comm, *tmp; 521902efe7SFrederic Weisbecker 53*93d5731dSArnaldo Carvalho de Melo map_groups__delete(thread->mg); 54*93d5731dSArnaldo Carvalho de Melo thread->mg = NULL; 551902efe7SFrederic Weisbecker list_for_each_entry_safe(comm, tmp, &thread->comm_list, list) { 561902efe7SFrederic Weisbecker list_del(&comm->list); 571902efe7SFrederic Weisbecker comm__free(comm); 581902efe7SFrederic Weisbecker } 591902efe7SFrederic Weisbecker 60c824c433SArnaldo Carvalho de Melo free(thread); 61591765fdSArnaldo Carvalho de Melo } 62591765fdSArnaldo Carvalho de Melo 634dfced35SNamhyung Kim struct comm *thread__comm(const struct thread *thread) 646baa0a5aSFrederic Weisbecker { 651902efe7SFrederic Weisbecker if (list_empty(&thread->comm_list)) 661902efe7SFrederic Weisbecker return NULL; 674385d580SDavid S. Miller 681902efe7SFrederic Weisbecker return list_first_entry(&thread->comm_list, struct comm, list); 694385d580SDavid S. Miller } 701902efe7SFrederic Weisbecker 711902efe7SFrederic Weisbecker /* CHECKME: time should always be 0 if event aren't ordered */ 721902efe7SFrederic Weisbecker int thread__set_comm(struct thread *thread, const char *str, u64 timestamp) 731902efe7SFrederic Weisbecker { 741902efe7SFrederic Weisbecker struct comm *new, *curr = thread__comm(thread); 753178f58bSFrederic Weisbecker int err; 761902efe7SFrederic Weisbecker 771902efe7SFrederic Weisbecker /* Override latest entry if it had no specific time coverage */ 781902efe7SFrederic Weisbecker if (!curr->start) { 793178f58bSFrederic Weisbecker err = comm__override(curr, str, timestamp); 803178f58bSFrederic Weisbecker if (err) 813178f58bSFrederic Weisbecker return err; 82a5285ad9SFrederic Weisbecker } else { 831902efe7SFrederic Weisbecker new = comm__new(str, timestamp); 841902efe7SFrederic Weisbecker if (!new) 851902efe7SFrederic Weisbecker return -ENOMEM; 861902efe7SFrederic Weisbecker list_add(&new->list, &thread->comm_list); 87a5285ad9SFrederic Weisbecker } 88a5285ad9SFrederic Weisbecker 891902efe7SFrederic Weisbecker thread->comm_set = true; 901902efe7SFrederic Weisbecker 911902efe7SFrederic Weisbecker return 0; 926baa0a5aSFrederic Weisbecker } 936baa0a5aSFrederic Weisbecker 94b9c5143aSFrederic Weisbecker const char *thread__comm_str(const struct thread *thread) 95b9c5143aSFrederic Weisbecker { 961902efe7SFrederic Weisbecker const struct comm *comm = thread__comm(thread); 971902efe7SFrederic Weisbecker 981902efe7SFrederic Weisbecker if (!comm) 991902efe7SFrederic Weisbecker return NULL; 1001902efe7SFrederic Weisbecker 1011902efe7SFrederic Weisbecker return comm__str(comm); 102b9c5143aSFrederic Weisbecker } 103b9c5143aSFrederic Weisbecker 1041902efe7SFrederic Weisbecker /* CHECKME: it should probably better return the max comm len from its comm list */ 105c824c433SArnaldo Carvalho de Melo int thread__comm_len(struct thread *thread) 106a4fb581bSFrederic Weisbecker { 107c824c433SArnaldo Carvalho de Melo if (!thread->comm_len) { 1081902efe7SFrederic Weisbecker const char *comm = thread__comm_str(thread); 1091902efe7SFrederic Weisbecker if (!comm) 110a4fb581bSFrederic Weisbecker return 0; 1111902efe7SFrederic Weisbecker thread->comm_len = strlen(comm); 112a4fb581bSFrederic Weisbecker } 113a4fb581bSFrederic Weisbecker 114c824c433SArnaldo Carvalho de Melo return thread->comm_len; 115a4fb581bSFrederic Weisbecker } 116a4fb581bSFrederic Weisbecker 1173f067dcaSArnaldo Carvalho de Melo size_t thread__fprintf(struct thread *thread, FILE *fp) 11895011c60SArnaldo Carvalho de Melo { 119b9c5143aSFrederic Weisbecker return fprintf(fp, "Thread %d %s\n", thread->tid, thread__comm_str(thread)) + 120*93d5731dSArnaldo Carvalho de Melo map_groups__fprintf(thread->mg, verbose, fp); 1216baa0a5aSFrederic Weisbecker } 1226baa0a5aSFrederic Weisbecker 123c824c433SArnaldo Carvalho de Melo void thread__insert_map(struct thread *thread, struct map *map) 1241b46cddfSArnaldo Carvalho de Melo { 125*93d5731dSArnaldo Carvalho de Melo map_groups__fixup_overlappings(thread->mg, map, verbose, stderr); 126*93d5731dSArnaldo Carvalho de Melo map_groups__insert(thread->mg, map); 12795011c60SArnaldo Carvalho de Melo } 12895011c60SArnaldo Carvalho de Melo 1291902efe7SFrederic Weisbecker int thread__fork(struct thread *thread, struct thread *parent, u64 timestamp) 1306baa0a5aSFrederic Weisbecker { 1311902efe7SFrederic Weisbecker int i, err; 1326baa0a5aSFrederic Weisbecker 133faa5c5c3SArnaldo Carvalho de Melo if (parent->comm_set) { 1341902efe7SFrederic Weisbecker const char *comm = thread__comm_str(parent); 1351902efe7SFrederic Weisbecker if (!comm) 1366baa0a5aSFrederic Weisbecker return -ENOMEM; 1371902efe7SFrederic Weisbecker err = thread__set_comm(thread, comm, timestamp); 1388d00be81SDavid Ahern if (err) 1391902efe7SFrederic Weisbecker return err; 140c824c433SArnaldo Carvalho de Melo thread->comm_set = true; 141faa5c5c3SArnaldo Carvalho de Melo } 1426baa0a5aSFrederic Weisbecker 14395011c60SArnaldo Carvalho de Melo for (i = 0; i < MAP__NR_TYPES; ++i) 144*93d5731dSArnaldo Carvalho de Melo if (map_groups__clone(thread->mg, parent->mg, i) < 0) 1456baa0a5aSFrederic Weisbecker return -ENOMEM; 14670c57efbSDavid Ahern 147c824c433SArnaldo Carvalho de Melo thread->ppid = parent->tid; 14870c57efbSDavid Ahern 1496baa0a5aSFrederic Weisbecker return 0; 1506baa0a5aSFrederic Weisbecker } 15152a3cb8cSArnaldo Carvalho de Melo 15252a3cb8cSArnaldo Carvalho de Melo void thread__find_cpumode_addr_location(struct thread *thread, 15352a3cb8cSArnaldo Carvalho de Melo struct machine *machine, 15452a3cb8cSArnaldo Carvalho de Melo enum map_type type, u64 addr, 15552a3cb8cSArnaldo Carvalho de Melo struct addr_location *al) 15652a3cb8cSArnaldo Carvalho de Melo { 15752a3cb8cSArnaldo Carvalho de Melo size_t i; 15852a3cb8cSArnaldo Carvalho de Melo const u8 const cpumodes[] = { 15952a3cb8cSArnaldo Carvalho de Melo PERF_RECORD_MISC_USER, 16052a3cb8cSArnaldo Carvalho de Melo PERF_RECORD_MISC_KERNEL, 16152a3cb8cSArnaldo Carvalho de Melo PERF_RECORD_MISC_GUEST_USER, 16252a3cb8cSArnaldo Carvalho de Melo PERF_RECORD_MISC_GUEST_KERNEL 16352a3cb8cSArnaldo Carvalho de Melo }; 16452a3cb8cSArnaldo Carvalho de Melo 16552a3cb8cSArnaldo Carvalho de Melo for (i = 0; i < ARRAY_SIZE(cpumodes); i++) { 16652a3cb8cSArnaldo Carvalho de Melo thread__find_addr_location(thread, machine, cpumodes[i], type, 16752a3cb8cSArnaldo Carvalho de Melo addr, al); 16852a3cb8cSArnaldo Carvalho de Melo if (al->map) 16952a3cb8cSArnaldo Carvalho de Melo break; 17052a3cb8cSArnaldo Carvalho de Melo } 17152a3cb8cSArnaldo Carvalho de Melo } 172