update files to correct FSF address
[openocd.git] / src / rtos / linux.c
blob0fc763557c6929e34e20b3ea34e6a880f093d677
1 /***************************************************************************
2 * Copyright (C) 2011 by STEricsson *
3 * Heythem Bouhaja heythem.bouhaja@stericsson.com : creation *
4 * Michel JAOUEN michel.jaouen@stericsson.com : adaptation to rtos *
5 * *
6 * This program is free software; you can redistribute it and/or modify *
7 * it under the terms of the GNU General Public License as published by *
8 * the Free Software Foundation; either version 2 of the License, or *
9 * (at your option) any later version. *
10 * *
11 * This program is distributed in the hope that it will be useful, *
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
14 * GNU General Public License for more details. *
15 * *
16 * You should have received a copy of the GNU General Public License *
17 * along with this program; if not, write to the *
18 * Free Software Foundation, Inc., *
19 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. *
20 ***************************************************************************/
22 #ifdef HAVE_CONFIG_H
23 #include "config.h"
24 #endif
26 #include <helper/time_support.h>
27 #include <jtag/jtag.h>
28 #include "target/target.h"
29 #include "target/target_type.h"
30 #include "helper/log.h"
31 #include "helper/types.h"
32 #include "rtos.h"
33 #include "rtos_standard_stackings.h"
34 #include <target/register.h>
35 #include "server/gdb_server.h"
37 #define LINUX_USER_KERNEL_BORDER 0xc0000000
38 #include "linux_header.h"
39 #define PHYS
40 #define MAX_THREADS 200
41 /* specific task */
42 struct linux_os {
43 char *name;
44 uint32_t init_task_addr;
45 int thread_count;
46 int threadid_count;
47 int preupdtate_threadid_count;
48 int nr_cpus;
49 int threads_lookup;
50 int threads_needs_update;
51 struct current_thread *current_threads;
52 struct threads *thread_list;
53 /* virt2phys parameter */
54 uint32_t phys_mask;
55 uint32_t phys_base;
58 struct current_thread {
59 int64_t threadid;
60 int32_t core_id;
61 #ifdef PID_CHECK
62 uint32_t pid;
63 #endif
64 uint32_t TS;
65 struct current_thread *next;
68 struct threads {
69 char name[17];
70 uint32_t base_addr; /* address to read magic */
71 uint32_t state; /* magic value : filled only at creation */
72 uint32_t pid; /* linux pid : id for identifying a thread */
73 uint32_t oncpu; /* content cpu number in current thread */
74 uint32_t asid; /* filled only at creation */
75 int64_t threadid;
76 int status; /* dead = 1 alive = 2 current = 3 alive and current */
77 /* value that should not change during the live of a thread ? */
78 uint32_t thread_info_addr; /* contain latest thread_info_addr computed */
79 /* retrieve from thread_info */
80 struct cpu_context *context;
81 struct threads *next;
84 struct cpu_context {
85 uint32_t R4;
86 uint32_t R5;
87 uint32_t R6;
88 uint32_t R7;
89 uint32_t R8;
90 uint32_t R9;
91 uint32_t IP;
92 uint32_t FP;
93 uint32_t SP;
94 uint32_t PC;
95 uint32_t preempt_count;
97 struct cpu_context *cpu_context_read(struct target *target, uint32_t base_addr,
98 uint32_t *info_addr);
99 static int insert_into_threadlist(struct target *target, struct threads *t);
101 static int linux_os_create(struct target *target);
103 static int linux_os_dummy_update(struct rtos *rtos)
105 /* update is done only when thread request come
106 * too many thread to do it on each stop */
107 return 0;
110 static int linux_compute_virt2phys(struct target *target, uint32_t address)
112 struct linux_os *linux_os = (struct linux_os *)
113 target->rtos->rtos_specific_params;
114 uint32_t pa = 0;
115 int retval = target->type->virt2phys(target, address, &pa);
116 if (retval != ERROR_OK) {
117 LOG_ERROR("Cannot compute linux virt2phys translation");
118 /* fixes default address */
119 linux_os->phys_base = 0;
120 return ERROR_FAIL;
123 linux_os->init_task_addr = address;
124 address = address & linux_os->phys_mask;
125 linux_os->phys_base = pa - address;
126 return ERROR_OK;
129 static int linux_read_memory(struct target *target,
130 uint32_t address, uint32_t size, uint32_t count,
131 uint8_t *buffer)
133 #ifdef PHYS
134 struct linux_os *linux_os = (struct linux_os *)
135 target->rtos->rtos_specific_params;
136 uint32_t pa = (address & linux_os->phys_mask) + linux_os->phys_base;
137 #endif
138 if (address < 0xc000000) {
139 LOG_ERROR("linux awareness : address in user space");
140 return ERROR_FAIL;
142 #ifdef PHYS
143 target_read_phys_memory(target, pa, size, count, buffer);
144 #endif
145 target_read_memory(target, address, size, count, buffer);
146 return ERROR_OK;
149 static char *reg_converter(char *buffer, void *reg, int size)
151 int i;
153 for (i = 0; i < size; i++)
154 buffer += sprintf(buffer, "%02x", ((uint8_t *) reg)[i]);
156 return buffer;
159 int fill_buffer(struct target *target, uint32_t addr, uint8_t *buffer)
162 if ((addr & 0xfffffffc) != addr)
163 LOG_INFO("unaligned address %x!!", addr);
165 int retval = linux_read_memory(target, addr, 4, 1, buffer);
166 return retval;
170 uint32_t get_buffer(struct target *target, const uint8_t *buffer)
172 uint32_t value = 0;
173 const uint8_t *value_ptr = buffer;
174 value = target_buffer_get_u32(target, value_ptr);
175 return value;
178 static int linux_os_thread_reg_list(struct rtos *rtos,
179 int64_t thread_id, char **hex_reg_list)
181 struct target *target = rtos->target;
182 struct linux_os *linux_os = (struct linux_os *)
183 target->rtos->rtos_specific_params;
184 int i = 0;
185 struct current_thread *tmp = linux_os->current_threads;
186 struct current_thread *next;
187 char *hex_string;
188 int found = 0;
189 int retval;
190 /* check if a current thread is requested */
191 next = tmp;
193 do {
194 if (next->threadid == thread_id)
195 found = 1;
196 else
197 next = next->next;
198 } while ((found == 0) && (next != tmp) && (next != NULL));
200 if (found == 1) {
201 /* search target to perfom the access */
202 struct reg **reg_list;
203 int reg_list_size, reg_packet_size = 0;
204 struct target_list *head;
205 head = target->head;
206 found = 0;
207 do {
208 if (head->target->coreid == next->core_id) {
210 target = head->target;
211 found = 1;
212 } else
213 head = head->next;
215 } while ((head != (struct target_list *)NULL) && (found == 0));
217 if (found == 0) {
218 LOG_ERROR
220 "current thread %" PRIx64 ": no target to perform access of core id %x",
221 thread_id,
222 next->core_id);
223 return ERROR_FAIL;
226 /*LOG_INFO("thread %lx current on core %x",thread_id,
227 * target->coreid);*/
228 retval =
229 target_get_gdb_reg_list(target, &reg_list, &reg_list_size);
231 if (retval != ERROR_OK)
232 return retval;
234 for (i = 0; i < reg_list_size; i++)
235 reg_packet_size += reg_list[i]->size;
237 assert(reg_packet_size > 0);
239 *hex_reg_list = malloc(DIV_ROUND_UP(reg_packet_size, 8) * 2);
241 hex_string = *hex_reg_list;
243 for (i = 0; i < reg_list_size; i++) {
244 if (!reg_list[i]->valid)
245 reg_list[i]->type->get(reg_list[i]);
247 hex_string = reg_converter(hex_string,
248 reg_list[i]->value,
249 (reg_list[i]->size) / 8);
252 free(reg_list);
254 } else {
255 struct threads *temp = linux_os->thread_list;
256 *hex_reg_list = (char *)calloc(1, 500 * sizeof(char));
257 hex_string = *hex_reg_list;
259 for (i = 0; i < 16; i++)
260 hex_string += sprintf(hex_string, "%02x", 0);
262 while ((temp != NULL) &&
263 (temp->threadid != target->rtos->current_threadid))
264 temp = temp->next;
266 if (temp != NULL) {
267 if (temp->context == NULL)
268 temp->context = cpu_context_read(target,
269 temp->
270 base_addr,
271 &temp->
272 thread_info_addr);
274 hex_string =
275 reg_converter(hex_string, &temp->context->R4, 4);
276 hex_string =
277 reg_converter(hex_string, &temp->context->R5, 4);
278 hex_string =
279 reg_converter(hex_string, &temp->context->R6, 4);
280 hex_string =
281 reg_converter(hex_string, &temp->context->R7, 4);
282 hex_string =
283 reg_converter(hex_string, &temp->context->R8, 4);
284 hex_string =
285 reg_converter(hex_string, &temp->context->R9, 4);
287 for (i = 0; i < 4; i++) /*R10 = 0x0 */
288 hex_string += sprintf(hex_string, "%02x", 0);
290 hex_string =
291 reg_converter(hex_string, &temp->context->FP, 4);
292 hex_string =
293 reg_converter(hex_string, &temp->context->IP, 4);
294 hex_string =
295 reg_converter(hex_string, &temp->context->SP, 4);
297 for (i = 0; i < 4; i++)
298 hex_string += sprintf(hex_string, "%02x", 0);
300 hex_string =
301 reg_converter(hex_string, &temp->context->PC, 4);
303 for (i = 0; i < 100; i++) /*100 */
304 hex_string += sprintf(hex_string, "%02x", 0);
306 uint32_t cpsr = 0x00000000;
307 reg_converter(hex_string, &cpsr, 4);
310 return ERROR_OK;
313 static int linux_os_detect(struct target *target)
315 LOG_INFO("should no be called");
316 return 0;
319 static int linux_os_smp_init(struct target *target);
320 static int linux_os_clean(struct target *target);
321 #define INIT_TASK 0
322 static char *linux_symbol_list[] = {
323 "init_task",
324 NULL
327 static int linux_get_symbol_list_to_lookup(symbol_table_elem_t *symbol_list[])
329 unsigned int i;
330 *symbol_list = (symbol_table_elem_t *)
331 malloc(sizeof(symbol_table_elem_t) * ARRAY_SIZE(linux_symbol_list));
333 for (i = 0; i < ARRAY_SIZE(linux_symbol_list); i++)
334 (*symbol_list)[i].symbol_name = linux_symbol_list[i];
336 return 0;
339 static char *linux_ps_command(struct target *target);
341 const struct rtos_type Linux_os = {
342 .name = "linux",
343 .detect_rtos = linux_os_detect,
344 .create = linux_os_create,
345 .smp_init = linux_os_smp_init,
346 .update_threads = linux_os_dummy_update,
347 .get_thread_reg_list = linux_os_thread_reg_list,
348 .get_symbol_list_to_lookup = linux_get_symbol_list_to_lookup,
349 .clean = linux_os_clean,
350 .ps_command = linux_ps_command,
353 static int linux_thread_packet(struct connection *connection, char *packet,
354 int packet_size);
355 static void linux_identify_current_threads(struct target *target);
357 #ifdef PID_CHECK
358 int fill_task_pid(struct target *target, struct threads *t)
360 uint32_t pid_addr = t->base_addr + PID;
361 uint8_t buffer[4];
362 int retval = fill_buffer(target, pid_addr, buffer);
364 if (retval == ERROR_OK) {
365 uint32_t val = get_buffer(target, buffer);
366 t->pid = val;
367 } else
368 LOG_ERROR("fill_task_pid: unable to read memory");
370 return retval;
372 #endif
374 int fill_task(struct target *target, struct threads *t)
376 int retval;
377 uint32_t pid_addr = t->base_addr + PID;
378 uint32_t mem_addr = t->base_addr + MEM;
379 uint32_t on_cpu = t->base_addr + ONCPU;
380 uint8_t *buffer = calloc(1, 4);
381 retval = fill_buffer(target, t->base_addr, buffer);
383 if (retval == ERROR_OK) {
384 uint32_t val = get_buffer(target, buffer);
385 t->state = val;
386 } else
387 LOG_ERROR("fill_task: unable to read memory");
389 retval = fill_buffer(target, pid_addr, buffer);
391 if (retval == ERROR_OK) {
392 uint32_t val = get_buffer(target, buffer);
393 t->pid = val;
394 } else
395 LOG_ERROR("fill task: unable to read memory");
397 retval = fill_buffer(target, on_cpu, buffer);
399 if (retval == ERROR_OK) {
400 uint32_t val = get_buffer(target, buffer);
401 t->oncpu = val;
402 } else
403 LOG_ERROR("fill task: unable to read memory");
405 retval = fill_buffer(target, mem_addr, buffer);
407 if (retval == ERROR_OK) {
408 uint32_t val = get_buffer(target, buffer);
410 if (val != 0) {
411 uint32_t asid_addr = val + MM_CTX;
412 retval = fill_buffer(target, asid_addr, buffer);
414 if (retval == ERROR_OK) {
415 val = get_buffer(target, buffer);
416 t->asid = val;
417 } else
418 LOG_ERROR
419 ("fill task: unable to read memory -- ASID");
420 } else
421 t->asid = 0;
422 } else
423 LOG_ERROR("fill task: unable to read memory");
425 free(buffer);
427 return retval;
430 int get_name(struct target *target, struct threads *t)
432 int retval;
433 uint32_t full_name[4];
434 uint32_t comm = t->base_addr + COMM;
435 int i;
437 for (i = 0; i < 17; i++)
438 t->name[i] = 0;
440 retval = linux_read_memory(target, comm, 4, 4, (uint8_t *) full_name);
442 if (retval != ERROR_OK) {
443 LOG_ERROR("get_name: unable to read memory\n");
444 return ERROR_FAIL;
447 uint32_t raw_name = target_buffer_get_u32(target,
448 (const uint8_t *)
449 &full_name[0]);
450 t->name[3] = raw_name >> 24;
451 t->name[2] = raw_name >> 16;
452 t->name[1] = raw_name >> 8;
453 t->name[0] = raw_name;
454 raw_name =
455 target_buffer_get_u32(target, (const uint8_t *)&full_name[1]);
456 t->name[7] = raw_name >> 24;
457 t->name[6] = raw_name >> 16;
458 t->name[5] = raw_name >> 8;
459 t->name[4] = raw_name;
460 raw_name =
461 target_buffer_get_u32(target, (const uint8_t *)&full_name[2]);
462 t->name[11] = raw_name >> 24;
463 t->name[10] = raw_name >> 16;
464 t->name[9] = raw_name >> 8;
465 t->name[8] = raw_name;
466 raw_name =
467 target_buffer_get_u32(target, (const uint8_t *)&full_name[3]);
468 t->name[15] = raw_name >> 24;
469 t->name[14] = raw_name >> 16;
470 t->name[13] = raw_name >> 8;
471 t->name[12] = raw_name;
472 return ERROR_OK;
476 int get_current(struct target *target, int create)
478 struct target_list *head;
479 head = target->head;
480 uint8_t *buf;
481 uint32_t val;
482 uint32_t ti_addr;
483 uint8_t *buffer = calloc(1, 4);
484 struct linux_os *linux_os = (struct linux_os *)
485 target->rtos->rtos_specific_params;
486 struct current_thread *ctt = linux_os->current_threads;
488 /* invalid current threads content */
489 while (ctt != NULL) {
490 ctt->threadid = -1;
491 ctt->TS = 0xdeadbeef;
492 ctt = ctt->next;
495 while (head != (struct target_list *)NULL) {
496 struct reg **reg_list;
497 int reg_list_size;
498 int retval;
500 if (target_get_gdb_reg_list(head->target, &reg_list,
501 &reg_list_size) != ERROR_OK) {
502 free(buffer);
503 return ERROR_TARGET_FAILURE;
506 if (!reg_list[13]->valid)
507 reg_list[13]->type->get(reg_list[13]);
509 buf = reg_list[13]->value;
510 val = get_buffer(target, buf);
511 ti_addr = (val & 0xffffe000);
512 uint32_t TS_addr = ti_addr + 0xc;
513 retval = fill_buffer(target, TS_addr, buffer);
515 if (retval == ERROR_OK) {
516 uint32_t TS = get_buffer(target, buffer);
517 uint32_t cpu, on_cpu = TS + ONCPU;
518 retval = fill_buffer(target, on_cpu, buffer);
520 if (retval == ERROR_OK) {
521 /*uint32_t cpu = get_buffer(target, buffer);*/
522 struct current_thread *ct =
523 linux_os->current_threads;
524 cpu = head->target->coreid;
526 while ((ct != NULL) && (ct->core_id != (int32_t) cpu))
527 ct = ct->next;
529 if ((ct != NULL) && (ct->TS == 0xdeadbeef))
530 ct->TS = TS;
531 else
532 LOG_ERROR
533 ("error in linux current thread update");
535 if (create) {
536 struct threads *t;
537 t = calloc(1, sizeof(struct threads));
538 t->base_addr = ct->TS;
539 fill_task(target, t);
540 get_name(target, t);
541 t->oncpu = cpu;
542 insert_into_threadlist(target, t);
543 t->status = 3;
544 t->thread_info_addr = 0xdeadbeef;
545 ct->threadid = t->threadid;
546 linux_os->thread_count++;
547 #ifdef PID_CHECK
548 ct->pid = t->pid;
549 #endif
550 /*LOG_INFO("Creation of current thread %s",t->name);*/
555 free(reg_list);
556 head = head->next;
559 free(buffer);
561 return ERROR_OK;
564 struct cpu_context *cpu_context_read(struct target *target, uint32_t base_addr,
565 uint32_t *thread_info_addr_old)
567 struct cpu_context *context = calloc(1, sizeof(struct cpu_context));
568 uint32_t preempt_count_addr = 0;
569 uint32_t registers[10];
570 uint8_t *buffer = calloc(1, 4);
571 uint32_t stack = base_addr + QAT;
572 uint32_t thread_info_addr = 0;
573 uint32_t thread_info_addr_update = 0;
574 int retval = ERROR_FAIL;
575 context->R4 = 0xdeadbeef;
576 context->R5 = 0xdeadbeef;
577 context->R6 = 0xdeadbeef;
578 context->R7 = 0xdeadbeef;
579 context->R8 = 0xdeadbeef;
580 context->R9 = 0xdeadbeef;
581 context->IP = 0xdeadbeef;
582 context->FP = 0xdeadbeef;
583 context->SP = 0xdeadbeef;
584 context->PC = 0xdeadbeef;
585 retry:
587 if (*thread_info_addr_old == 0xdeadbeef) {
588 retval = fill_buffer(target, stack, buffer);
590 if (retval == ERROR_OK)
591 thread_info_addr = get_buffer(target, buffer);
592 else
593 LOG_ERROR("cpu_context: unable to read memory");
595 thread_info_addr_update = thread_info_addr;
596 } else
597 thread_info_addr = *thread_info_addr_old;
599 preempt_count_addr = thread_info_addr + PREEMPT;
600 retval = fill_buffer(target, preempt_count_addr, buffer);
602 if (retval == ERROR_OK)
603 context->preempt_count = get_buffer(target, buffer);
604 else {
605 if (*thread_info_addr_old != 0xdeadbeef) {
606 LOG_ERROR
607 ("cpu_context: cannot read at thread_info_addr");
609 if (*thread_info_addr_old < LINUX_USER_KERNEL_BORDER)
610 LOG_INFO
611 ("cpu_context : thread_info_addr in userspace!!!");
613 *thread_info_addr_old = 0xdeadbeef;
614 goto retry;
617 LOG_ERROR("cpu_context: unable to read memory");
620 thread_info_addr += CPU_CONT;
622 retval = linux_read_memory(target, thread_info_addr, 4, 10,
623 (uint8_t *) registers);
625 if (retval != ERROR_OK) {
626 free(buffer);
627 LOG_ERROR("cpu_context: unable to read memory\n");
628 return context;
631 context->R4 =
632 target_buffer_get_u32(target, (const uint8_t *)&registers[0]);
633 context->R5 =
634 target_buffer_get_u32(target, (const uint8_t *)&registers[1]);
635 context->R6 =
636 target_buffer_get_u32(target, (const uint8_t *)&registers[2]);
637 context->R7 =
638 target_buffer_get_u32(target, (const uint8_t *)&registers[3]);
639 context->R8 =
640 target_buffer_get_u32(target, (const uint8_t *)&registers[4]);
641 context->R9 =
642 target_buffer_get_u32(target, (const uint8_t *)&registers[5]);
643 context->IP =
644 target_buffer_get_u32(target, (const uint8_t *)&registers[6]);
645 context->FP =
646 target_buffer_get_u32(target, (const uint8_t *)&registers[7]);
647 context->SP =
648 target_buffer_get_u32(target, (const uint8_t *)&registers[8]);
649 context->PC =
650 target_buffer_get_u32(target, (const uint8_t *)&registers[9]);
652 if (*thread_info_addr_old == 0xdeadbeef)
653 *thread_info_addr_old = thread_info_addr_update;
655 free(buffer);
657 return context;
660 uint32_t next_task(struct target *target, struct threads *t)
662 uint8_t *buffer = calloc(1, 4);
663 uint32_t next_addr = t->base_addr + NEXT;
664 int retval = fill_buffer(target, next_addr, buffer);
666 if (retval == ERROR_OK) {
667 uint32_t val = get_buffer(target, buffer);
668 val = val - NEXT;
669 free(buffer);
670 return val;
671 } else
672 LOG_ERROR("next task: unable to read memory");
674 free(buffer);
676 return 0;
679 struct current_thread *add_current_thread(struct current_thread *currents,
680 struct current_thread *ct)
682 ct->next = NULL;
684 if (currents == NULL) {
685 currents = ct;
686 return currents;
687 } else {
688 struct current_thread *temp = currents;
690 while (temp->next != NULL)
691 temp = temp->next;
693 temp->next = ct;
694 return currents;
698 struct threads *liste_del_task(struct threads *task_list, struct threads **t,
699 struct threads *prev)
701 LOG_INFO("del task %" PRId64, (*t)->threadid);
702 prev->next = (*t)->next;
704 if (prev == task_list)
705 task_list = prev;
707 /* free content of threads */
708 if ((*t)->context)
709 free((*t)->context);
711 free(*t);
712 *t = prev;
713 return task_list;
716 struct threads *liste_add_task(struct threads *task_list, struct threads *t,
717 struct threads **last)
719 t->next = NULL;
721 if (*last == NULL)
722 if (task_list == NULL) {
723 task_list = t;
724 return task_list;
725 } else {
726 struct threads *temp = task_list;
728 while (temp->next != NULL)
729 temp = temp->next;
731 temp->next = t;
732 *last = t;
733 return task_list;
734 } else {
735 (*last)->next = t;
736 *last = t;
737 return task_list;
741 #ifdef PID_CHECK
742 static int current_pid(struct linux_os *linux_os, uint32_t pid)
743 #else
744 static int current_base_addr(struct linux_os *linux_os, uint32_t base_addr)
745 #endif
747 struct current_thread *ct = linux_os->current_threads;
748 #ifdef PID_CHECK
750 while ((ct != NULL) && (ct->pid != pid))
751 #else
752 while ((ct != NULL) && (ct->TS != base_addr))
753 #endif
754 ct = ct->next;
755 #ifdef PID_CHECK
756 if ((ct != NULL) && (ct->pid == pid))
757 #else
758 if ((ct != NULL) && (ct->TS == base_addr))
759 #endif
760 return 1;
762 return 0;
765 int linux_get_tasks(struct target *target, int context)
767 int loop = 0;
768 int retval = 0;
769 struct linux_os *linux_os = (struct linux_os *)
770 target->rtos->rtos_specific_params;
771 linux_os->thread_list = NULL;
772 linux_os->thread_count = 0;
774 if (linux_os->init_task_addr == 0xdeadbeef) {
775 LOG_INFO("no init symbol\n");
776 return ERROR_FAIL;
779 int64_t start = timeval_ms();
781 struct threads *t = calloc(1, sizeof(struct threads));
782 struct threads *last = NULL;
783 t->base_addr = linux_os->init_task_addr;
784 /* retrieve the thread id , currently running in the different smp core */
785 get_current(target, 1);
787 while (((t->base_addr != linux_os->init_task_addr) &&
788 (t->base_addr != 0)) || (loop == 0)) {
789 loop++;
790 fill_task(target, t);
791 retval = get_name(target, t);
793 if (loop > MAX_THREADS) {
794 free(t);
795 LOG_INFO("more than %d threads !!", MAX_THREADS);
796 return ERROR_FAIL;
799 if (retval != ERROR_OK) {
800 free(t);
801 return ERROR_FAIL;
804 /* check that this thread is not one the current threads already
805 * created */
806 #ifdef PID_CHECK
808 if (!current_pid(linux_os, t->pid)) {
809 #else
810 if (!current_base_addr(linux_os, t->base_addr)) {
811 #endif
812 t->threadid = linux_os->threadid_count;
813 t->status = 1;
814 linux_os->threadid_count++;
816 linux_os->thread_list =
817 liste_add_task(linux_os->thread_list, t, &last);
818 /* no interest to fill the context if it is a current thread. */
819 linux_os->thread_count++;
820 t->thread_info_addr = 0xdeadbeef;
822 if (context)
823 t->context =
824 cpu_context_read(target, t->base_addr,
825 &t->thread_info_addr);
826 } else {
827 /*LOG_INFO("thread %s is a current thread already created",t->name); */
828 free(t);
831 uint32_t base_addr = next_task(target, t);
832 t = calloc(1, sizeof(struct threads));
833 t->base_addr = base_addr;
836 linux_os->threads_lookup = 1;
837 linux_os->threads_needs_update = 0;
838 linux_os->preupdtate_threadid_count = linux_os->threadid_count - 1;
839 /* check that all current threads have been identified */
841 LOG_INFO("complete time %" PRId64 ", thread mean %" PRId64 "\n",
842 (timeval_ms() - start),
843 (timeval_ms() - start) / linux_os->threadid_count);
845 LOG_INFO("threadid count %d", linux_os->threadid_count);
846 free(t);
848 return ERROR_OK;
851 static int clean_threadlist(struct target *target)
853 struct linux_os *linux_os = (struct linux_os *)
854 target->rtos->rtos_specific_params;
855 struct threads *old, *temp = linux_os->thread_list;
857 while (temp != NULL) {
858 old = temp;
860 if (temp->context)
861 free(temp->context);
863 temp = temp->next;
864 free(old);
867 return ERROR_OK;
870 static int linux_os_clean(struct target *target)
872 struct linux_os *os_linux = (struct linux_os *)
873 target->rtos->rtos_specific_params;
874 clean_threadlist(target);
875 os_linux->init_task_addr = 0xdeadbeef;
876 os_linux->name = "linux";
877 os_linux->thread_list = NULL;
878 os_linux->thread_count = 0;
879 os_linux->nr_cpus = 0;
880 os_linux->threads_lookup = 0;
881 os_linux->threads_needs_update = 0;
882 os_linux->threadid_count = 1;
883 return ERROR_OK;
886 static int insert_into_threadlist(struct target *target, struct threads *t)
888 struct linux_os *linux_os = (struct linux_os *)
889 target->rtos->rtos_specific_params;
890 struct threads *temp = linux_os->thread_list;
891 t->threadid = linux_os->threadid_count;
892 linux_os->threadid_count++;
893 t->status = 1;
894 t->next = NULL;
896 if (temp == NULL)
897 linux_os->thread_list = t;
898 else {
899 while (temp->next != NULL)
900 temp = temp->next;
902 t->next = NULL;
903 temp->next = t;
906 return ERROR_OK;
909 static void linux_identify_current_threads(struct target *target)
911 struct linux_os *linux_os = (struct linux_os *)
912 target->rtos->rtos_specific_params;
913 struct threads *thread_list = linux_os->thread_list;
914 struct current_thread *ct = linux_os->current_threads;
915 struct threads *t = NULL;
917 while ((ct != NULL)) {
918 if (ct->threadid == -1) {
920 /* un-identified thread */
921 int found = 0;
922 t = calloc(1, sizeof(struct threads));
923 t->base_addr = ct->TS;
924 #ifdef PID_CHECK
926 if (fill_task_pid(target, t) != ERROR_OK) {
927 error_handling:
928 free(t);
929 LOG_ERROR
930 ("linux identify_current_threads: unable to read pid");
931 return;
933 #endif
935 /* search in the list of threads if pid
936 already present */
937 while ((thread_list != NULL) && (found == 0)) {
938 #ifdef PID_CHECK
939 if (thread_list->pid == t->pid) {
940 #else
941 if (thread_list->base_addr == t->base_addr) {
942 #endif
943 free(t);
944 t = thread_list;
945 found = 1;
947 thread_list = thread_list->next;
950 if (!found) {
951 /* it is a new thread */
952 if (fill_task(target, t) != ERROR_OK)
953 goto error_handling;
955 get_name(target, t);
956 insert_into_threadlist(target, t);
957 t->thread_info_addr = 0xdeadbeef;
960 t->status = 3;
961 ct->threadid = t->threadid;
962 #ifdef PID_CHECK
963 ct->pid = t->pid;
964 #endif
965 linux_os->thread_count++;
966 #if 0
967 if (found == 0)
968 LOG_INFO("current thread core %x identified %s",
969 ct->core_id, t->name);
970 else
971 LOG_INFO("current thread core %x, reused %s",
972 ct->core_id, t->name);
973 #endif
975 #if 0
976 else {
977 struct threads tmp;
978 tmp.base_addr = ct->TS;
979 get_name(target, &tmp);
980 LOG_INFO("current thread core %x , already identified %s !!!",
981 ct->core_id, tmp.name);
983 #endif
984 ct = ct->next;
987 return;
988 #ifndef PID_CHECK
989 error_handling:
990 free(t);
991 LOG_ERROR("unable to read pid");
992 return;
994 #endif
997 static int linux_task_update(struct target *target, int context)
999 struct linux_os *linux_os = (struct linux_os *)
1000 target->rtos->rtos_specific_params;
1001 struct threads *thread_list = linux_os->thread_list;
1002 int retval;
1003 int loop = 0;
1004 linux_os->thread_count = 0;
1006 /*thread_list = thread_list->next; skip init_task*/
1007 while (thread_list != NULL) {
1008 thread_list->status = 0; /*setting all tasks to dead state*/
1010 if (thread_list->context) {
1011 free(thread_list->context);
1012 thread_list->context = NULL;
1015 thread_list = thread_list->next;
1018 int found = 0;
1020 if (linux_os->init_task_addr == 0xdeadbeef) {
1021 LOG_INFO("no init symbol\n");
1022 return ERROR_FAIL;
1024 int64_t start = timeval_ms();
1025 struct threads *t = calloc(1, sizeof(struct threads));
1026 uint32_t previous = 0xdeadbeef;
1027 t->base_addr = linux_os->init_task_addr;
1028 retval = get_current(target, 0);
1029 /*check that all current threads have been identified */
1030 linux_identify_current_threads(target);
1032 while (((t->base_addr != linux_os->init_task_addr) &&
1033 (t->base_addr != previous)) || (loop == 0)) {
1034 /* for avoiding any permanent loop for any reason possibly due to
1035 * target */
1036 loop++;
1037 previous = t->base_addr;
1038 /* read only pid */
1039 #ifdef PID_CHECK
1040 retval = fill_task_pid(target, t);
1041 #endif
1043 if (retval != ERROR_OK) {
1044 free(t);
1045 return ERROR_FAIL;
1048 thread_list = linux_os->thread_list;
1050 while (thread_list != NULL) {
1051 #ifdef PID_CHECK
1052 if (t->pid == thread_list->pid) {
1053 #else
1054 if (t->base_addr == thread_list->base_addr) {
1055 #endif
1056 if (!thread_list->status) {
1057 #ifdef PID_CHECK
1058 if (t->base_addr != thread_list->base_addr)
1059 LOG_INFO("thread base_addr has changed !!");
1060 #endif
1061 /* this is not a current thread */
1062 thread_list->base_addr = t->base_addr;
1063 thread_list->status = 1;
1065 /* we don 't update this field any more */
1067 /*thread_list->state = t->state;
1068 thread_list->oncpu = t->oncpu;
1069 thread_list->asid = t->asid;
1071 if (context)
1072 thread_list->context =
1073 cpu_context_read(target,
1074 thread_list->
1075 base_addr,
1076 &thread_list->
1077 thread_info_addr);
1078 } else {
1079 /* it is a current thread no need to read context */
1082 linux_os->thread_count++;
1083 found = 1;
1084 break;
1085 } else {
1086 found = 0;
1087 thread_list = thread_list->next;
1091 if (found == 0) {
1092 uint32_t base_addr;
1093 fill_task(target, t);
1094 get_name(target, t);
1095 retval = insert_into_threadlist(target, t);
1096 t->thread_info_addr = 0xdeadbeef;
1098 if (context)
1099 t->context =
1100 cpu_context_read(target, t->base_addr,
1101 &t->thread_info_addr);
1103 base_addr = next_task(target, t);
1104 t = calloc(1, sizeof(struct threads));
1105 t->base_addr = base_addr;
1106 linux_os->thread_count++;
1107 } else
1108 t->base_addr = next_task(target, t);
1111 LOG_INFO("update thread done %" PRId64 ", mean%" PRId64 "\n",
1112 (timeval_ms() - start), (timeval_ms() - start) / loop);
1113 free(t);
1114 linux_os->threads_needs_update = 0;
1115 return ERROR_OK;
1118 int linux_gdb_thread_packet(struct target *target,
1119 struct connection *connection, char *packet,
1120 int packet_size)
1122 int retval;
1123 struct linux_os *linux_os =
1124 (struct linux_os *)target->rtos->rtos_specific_params;
1126 if (linux_os->init_task_addr == 0xdeadbeef) {
1127 /* it has not been initialized */
1128 LOG_INFO("received thread request without init task address");
1129 gdb_put_packet(connection, "l", 1);
1130 return ERROR_OK;
1133 retval = linux_get_tasks(target, 1);
1135 if (retval != ERROR_OK)
1136 return ERROR_TARGET_FAILURE;
1138 char *out_str = (char *)calloc(1, 350 * sizeof(int64_t));
1139 char *tmp_str = out_str;
1140 tmp_str += sprintf(tmp_str, "m");
1141 struct threads *temp = linux_os->thread_list;
1142 tmp_str += sprintf(tmp_str, "%016" PRIx64, temp->threadid);
1143 temp = temp->next;
1145 while (temp != NULL) {
1146 tmp_str += sprintf(tmp_str, ",");
1147 tmp_str += sprintf(tmp_str, "%016" PRIx64, temp->threadid);
1148 temp = temp->next;
1151 gdb_put_packet(connection, out_str, strlen(out_str));
1152 return ERROR_OK;
1155 int linux_gdb_thread_update(struct target *target,
1156 struct connection *connection, char *packet,
1157 int packet_size)
1159 int found = 0;
1160 struct linux_os *linux_os = (struct linux_os *)
1161 target->rtos->rtos_specific_params;
1162 struct threads *temp = linux_os->thread_list;
1164 while (temp != NULL) {
1165 if (temp->threadid == linux_os->preupdtate_threadid_count + 1) {
1166 /*LOG_INFO("FOUND");*/
1167 found = 1;
1168 break;
1169 } else
1170 temp = temp->next;
1173 if (found == 1) {
1174 /*LOG_INFO("INTO GDB THREAD UPDATE FOUNDING START TASK");*/
1175 char *out_strr = (char *)calloc(1, 350 * sizeof(int64_t));
1176 char *tmp_strr = out_strr;
1177 tmp_strr += sprintf(tmp_strr, "m");
1178 /*LOG_INFO("CHAR MALLOC & M DONE");*/
1179 tmp_strr += sprintf(tmp_strr, "%016" PRIx64, temp->threadid);
1181 temp = temp->next;
1183 while (temp != NULL) {
1184 /*LOG_INFO("INTO GDB THREAD UPDATE WHILE");*/
1185 tmp_strr += sprintf(tmp_strr, ",");
1186 tmp_strr +=
1187 sprintf(tmp_strr, "%016" PRIx64, temp->threadid);
1188 temp = temp->next;
1191 /*tmp_str[0] = 0;*/
1192 gdb_put_packet(connection, out_strr, strlen(out_strr));
1193 linux_os->preupdtate_threadid_count =
1194 linux_os->threadid_count - 1;
1195 free(out_strr);
1196 } else
1197 gdb_put_packet(connection, "l", 1);
1199 return ERROR_OK;
1202 int linux_thread_extra_info(struct target *target,
1203 struct connection *connection, char *packet,
1204 int packet_size)
1206 int64_t threadid = 0;
1207 struct linux_os *linux_os = (struct linux_os *)
1208 target->rtos->rtos_specific_params;
1209 sscanf(packet, "qThreadExtraInfo,%" SCNx64, &threadid);
1210 /*LOG_INFO("lookup extra info for thread %" SCNx64, threadid);*/
1211 struct threads *temp = linux_os->thread_list;
1213 while (temp != NULL) {
1214 if (temp->threadid == threadid) {
1215 char *pid = " PID: ";
1216 char *pid_current = "*PID: ";
1217 char *name = "NAME: ";
1218 int str_size = strlen(pid) + strlen(name);
1219 char *tmp_str = (char *)calloc(1, str_size + 50);
1220 char *tmp_str_ptr = tmp_str;
1222 /* discriminate current task */
1223 if (temp->status == 3)
1224 tmp_str_ptr += sprintf(tmp_str_ptr, "%s",
1225 pid_current);
1226 else
1227 tmp_str_ptr += sprintf(tmp_str_ptr, "%s", pid);
1229 tmp_str_ptr +=
1230 sprintf(tmp_str_ptr, "%d", (int)temp->pid);
1231 tmp_str_ptr += sprintf(tmp_str_ptr, "%s", " | ");
1232 sprintf(tmp_str_ptr, "%s", name);
1233 sprintf(tmp_str_ptr, "%s", temp->name);
1234 char *hex_str = (char *)calloc(1, strlen(tmp_str) * 2 + 1);
1235 int pkt_len = hexify(hex_str, tmp_str, 0, strlen(tmp_str) * 2 + 1);
1236 gdb_put_packet(connection, hex_str, pkt_len);
1237 free(hex_str);
1238 free(tmp_str);
1239 return ERROR_OK;
1242 temp = temp->next;
1245 LOG_INFO("thread not found");
1246 return ERROR_OK;
1249 int linux_gdb_T_packet(struct connection *connection,
1250 struct target *target, char *packet, int packet_size)
1252 int64_t threadid;
1253 struct linux_os *linux_os = (struct linux_os *)
1254 target->rtos->rtos_specific_params;
1255 int retval = ERROR_OK;
1256 sscanf(packet, "T%" SCNx64, &threadid);
1258 if (linux_os->threads_needs_update == 0) {
1259 struct threads *temp = linux_os->thread_list;
1260 struct threads *prev = linux_os->thread_list;
1262 while (temp != NULL) {
1263 if (temp->threadid == threadid) {
1264 if (temp->status != 0) {
1265 gdb_put_packet(connection, "OK", 2);
1266 return ERROR_OK;
1267 } else {
1268 /* delete item in the list */
1269 linux_os->thread_list =
1270 liste_del_task(linux_os->
1271 thread_list, &temp,
1272 prev);
1273 linux_os->thread_count--;
1274 gdb_put_packet(connection, "E01", 3);
1275 return ERROR_OK;
1279 /* for deletion */
1280 prev = temp;
1281 temp = temp->next;
1284 LOG_INFO("gdb requested status on non existing thread");
1285 gdb_put_packet(connection, "E01", 3);
1286 return ERROR_OK;
1288 } else {
1289 retval = linux_task_update(target, 1);
1290 struct threads *temp = linux_os->thread_list;
1292 while (temp != NULL) {
1293 if (temp->threadid == threadid) {
1294 if (temp->status == 1) {
1295 gdb_put_packet(connection, "OK", 2);
1296 return ERROR_OK;
1297 } else {
1298 gdb_put_packet(connection, "E01", 3);
1299 return ERROR_OK;
1303 temp = temp->next;
1307 return retval;
1310 int linux_gdb_h_packet(struct connection *connection,
1311 struct target *target, char *packet, int packet_size)
1313 struct linux_os *linux_os = (struct linux_os *)
1314 target->rtos->rtos_specific_params;
1315 struct current_thread *ct = linux_os->current_threads;
1317 /* select to display the current thread of the selected target */
1318 while ((ct != NULL) && (ct->core_id != target->coreid))
1319 ct = ct->next;
1321 int64_t current_gdb_thread_rq;
1323 if (linux_os->threads_lookup == 1) {
1324 if ((ct != NULL) && (ct->threadid == -1)) {
1325 ct = linux_os->current_threads;
1327 while ((ct != NULL) && (ct->threadid == -1))
1328 ct = ct->next;
1331 if (ct == NULL) {
1332 /* no current thread can be identified
1333 * any way with smp */
1334 LOG_INFO("no current thread identified");
1335 /* attempt to display the name of the 2 threads identified with
1336 * get_current */
1337 struct threads t;
1338 ct = linux_os->current_threads;
1340 while ((ct != NULL) && (ct->threadid == -1)) {
1341 t.base_addr = ct->TS;
1342 get_name(target, &t);
1343 LOG_INFO("name of unidentified thread %s",
1344 t.name);
1345 ct = ct->next;
1348 gdb_put_packet(connection, "OK", 2);
1349 return ERROR_OK;
1352 if (packet[1] == 'g') {
1353 sscanf(packet, "Hg%16" SCNx64, &current_gdb_thread_rq);
1355 if (current_gdb_thread_rq == 0) {
1356 target->rtos->current_threadid = ct->threadid;
1357 gdb_put_packet(connection, "OK", 2);
1358 } else {
1359 target->rtos->current_threadid =
1360 current_gdb_thread_rq;
1361 gdb_put_packet(connection, "OK", 2);
1363 } else if (packet[1] == 'c') {
1364 sscanf(packet, "Hc%16" SCNx64, &current_gdb_thread_rq);
1366 if ((current_gdb_thread_rq == 0) ||
1367 (current_gdb_thread_rq == ct->threadid)) {
1368 target->rtos->current_threadid = ct->threadid;
1369 gdb_put_packet(connection, "OK", 2);
1370 } else
1371 gdb_put_packet(connection, "E01", 3);
1373 } else
1374 gdb_put_packet(connection, "OK", 2);
1376 return ERROR_OK;
1379 static int linux_thread_packet(struct connection *connection, char *packet,
1380 int packet_size)
1382 int retval = ERROR_OK;
1383 struct current_thread *ct;
1384 struct target *target = get_target_from_connection(connection);
1385 struct linux_os *linux_os = (struct linux_os *)
1386 target->rtos->rtos_specific_params;
1388 switch (packet[0]) {
1389 case 'T': /* Is thread alive?*/
1391 linux_gdb_T_packet(connection, target, packet, packet_size);
1392 break;
1393 case 'H': /* Set current thread */
1394 /* ( 'c' for step and continue, 'g' for all other operations )*/
1395 /*LOG_INFO(" H packet received '%s'", packet);*/
1396 linux_gdb_h_packet(connection, target, packet, packet_size);
1397 break;
1398 case 'q':
1400 if (strncmp(packet, "qSymbol", 7) == 0) {
1401 if (rtos_qsymbol(connection, packet, packet_size) == 1) {
1402 linux_compute_virt2phys(target,
1403 target->rtos->
1404 symbols[INIT_TASK].
1405 address);
1408 break;
1409 } else if (strncmp(packet, "qfThreadInfo", 12) == 0) {
1410 if (linux_os->thread_list == NULL) {
1411 retval = linux_gdb_thread_packet(target,
1412 connection,
1413 packet,
1414 packet_size);
1415 break;
1416 } else {
1417 retval = linux_gdb_thread_update(target,
1418 connection,
1419 packet,
1420 packet_size);
1421 break;
1423 } else if (strncmp(packet, "qsThreadInfo", 12) == 0) {
1424 gdb_put_packet(connection, "l", 1);
1425 break;
1426 } else if (strncmp(packet, "qThreadExtraInfo,", 17) == 0) {
1427 linux_thread_extra_info(target, connection, packet,
1428 packet_size);
1429 break;
1430 } else {
1431 retval = GDB_THREAD_PACKET_NOT_CONSUMED;
1432 break;
1435 case 'Q':
1436 /* previously response was : thread not found
1437 * gdb_put_packet(connection, "E01", 3); */
1438 retval = GDB_THREAD_PACKET_NOT_CONSUMED;
1439 break;
1440 case 'c':
1441 case 's': {
1442 if (linux_os->threads_lookup == 1) {
1443 ct = linux_os->current_threads;
1445 while ((ct != NULL) && (ct->core_id) != target->coreid)
1446 ct = ct->next;
1448 if ((ct != NULL) && (ct->threadid == -1)) {
1449 ct = linux_os->current_threads;
1451 while ((ct != NULL) && (ct->threadid == -1))
1452 ct = ct->next;
1455 if ((ct != NULL) && (ct->threadid !=
1456 target->rtos->
1457 current_threadid)
1458 && (target->rtos->current_threadid != -1))
1459 LOG_WARNING("WARNING! current GDB thread do not match" \
1460 "current thread running." \
1461 "Switch thread in GDB to threadid %d",
1462 (int)ct->threadid);
1464 LOG_INFO("threads_needs_update = 1");
1465 linux_os->threads_needs_update = 1;
1469 /* if a packet handler returned an error, exit input loop */
1470 if (retval != ERROR_OK)
1471 return retval;
1474 return retval;
1477 static int linux_os_smp_init(struct target *target)
1479 struct target_list *head;
1480 /* keep only target->rtos */
1481 struct rtos *rtos = target->rtos;
1482 struct linux_os *os_linux =
1483 (struct linux_os *)rtos->rtos_specific_params;
1484 struct current_thread *ct;
1485 head = target->head;
1487 while (head != (struct target_list *)NULL) {
1488 if (head->target->rtos != rtos) {
1489 struct linux_os *smp_os_linux =
1490 (struct linux_os *)head->target->rtos->
1491 rtos_specific_params;
1492 /* remap smp target on rtos */
1493 free(head->target->rtos);
1494 head->target->rtos = rtos;
1495 /* reuse allocated ct */
1496 ct = smp_os_linux->current_threads;
1497 ct->threadid = -1;
1498 ct->TS = 0xdeadbeef;
1499 ct->core_id = head->target->coreid;
1500 os_linux->current_threads =
1501 add_current_thread(os_linux->current_threads, ct);
1502 os_linux->nr_cpus++;
1503 free(smp_os_linux);
1506 head = head->next;
1509 return ERROR_OK;
1512 static int linux_os_create(struct target *target)
1514 struct linux_os *os_linux = calloc(1, sizeof(struct linux_os));
1515 struct current_thread *ct = calloc(1, sizeof(struct current_thread));
1516 LOG_INFO("linux os creation\n");
1517 os_linux->init_task_addr = 0xdeadbeef;
1518 os_linux->name = "linux";
1519 os_linux->thread_list = NULL;
1520 os_linux->thread_count = 0;
1521 target->rtos->current_threadid = -1;
1522 os_linux->nr_cpus = 1;
1523 os_linux->threads_lookup = 0;
1524 os_linux->threads_needs_update = 0;
1525 os_linux->threadid_count = 1;
1526 os_linux->current_threads = NULL;
1527 target->rtos->rtos_specific_params = (void *)os_linux;
1528 ct->core_id = target->coreid;
1529 ct->threadid = -1;
1530 ct->TS = 0xdeadbeef;
1531 os_linux->current_threads =
1532 add_current_thread(os_linux->current_threads, ct);
1533 /* overload rtos thread default handler */
1534 target->rtos->gdb_thread_packet = linux_thread_packet;
1535 /* initialize a default virt 2 phys translation */
1536 os_linux->phys_mask = ~0xc0000000;
1537 os_linux->phys_base = 0x0;
1538 return JIM_OK;
1541 static char *linux_ps_command(struct target *target)
1543 struct linux_os *linux_os = (struct linux_os *)
1544 target->rtos->rtos_specific_params;
1545 int retval = ERROR_OK;
1546 char *display;
1548 if (linux_os->threads_lookup == 0)
1549 retval = linux_get_tasks(target, 1);
1550 else {
1551 if (linux_os->threads_needs_update != 0)
1552 retval = linux_task_update(target, 0);
1555 if (retval == ERROR_OK) {
1556 struct threads *temp = linux_os->thread_list;
1557 char *tmp;
1558 LOG_INFO("allocation for %d threads line",
1559 linux_os->thread_count);
1560 display = calloc((linux_os->thread_count + 2) * 80, 1);
1562 if (!display)
1563 goto error;
1565 tmp = display;
1566 tmp += sprintf(tmp, "PID\t\tCPU\t\tASID\t\tNAME\n");
1567 tmp += sprintf(tmp, "---\t\t---\t\t----\t\t----\n");
1569 while (temp != NULL) {
1570 if (temp->status) {
1571 if (temp->context)
1572 tmp +=
1573 sprintf(tmp,
1574 "%d\t\t%d\t\t%x\t\t%s\n",
1575 (int)temp->pid, temp->oncpu,
1576 temp->asid, temp->name);
1577 else
1578 tmp +=
1579 sprintf(tmp,
1580 "%d\t\t%d\t\t%x\t\t%s\n",
1581 (int)temp->pid, temp->oncpu,
1582 temp->asid, temp->name);
1585 temp = temp->next;
1588 return display;
1591 error:
1592 display = calloc(40, 1);
1593 sprintf(display, "linux_ps_command failed\n");
1594 return display;