|
@@ -89,18 +89,21 @@
|
|
|
do { memcpy(buffer, string, strlen(string)); \
|
|
|
buffer += strlen(string); } while (0)
|
|
|
|
|
|
-static inline char *task_name(struct task_struct *p, char *buf)
|
|
|
+static inline void task_name(struct seq_file *m, struct task_struct *p)
|
|
|
{
|
|
|
int i;
|
|
|
+ char *buf, *end;
|
|
|
char *name;
|
|
|
char tcomm[sizeof(p->comm)];
|
|
|
|
|
|
get_task_comm(tcomm, p);
|
|
|
|
|
|
- ADDBUF(buf, "Name:\t");
|
|
|
+ seq_printf(m, "Name:\t");
|
|
|
+ end = m->buf + m->size;
|
|
|
+ buf = m->buf + m->count;
|
|
|
name = tcomm;
|
|
|
i = sizeof(tcomm);
|
|
|
- do {
|
|
|
+ while (i && (buf < end)) {
|
|
|
unsigned char c = *name;
|
|
|
name++;
|
|
|
i--;
|
|
@@ -108,20 +111,21 @@ static inline char *task_name(struct task_struct *p, char *buf)
|
|
|
if (!c)
|
|
|
break;
|
|
|
if (c == '\\') {
|
|
|
- buf[1] = c;
|
|
|
- buf += 2;
|
|
|
+ buf++;
|
|
|
+ if (buf < end)
|
|
|
+ *buf++ = c;
|
|
|
continue;
|
|
|
}
|
|
|
if (c == '\n') {
|
|
|
- buf[0] = '\\';
|
|
|
- buf[1] = 'n';
|
|
|
- buf += 2;
|
|
|
+ *buf++ = '\\';
|
|
|
+ if (buf < end)
|
|
|
+ *buf++ = 'n';
|
|
|
continue;
|
|
|
}
|
|
|
buf++;
|
|
|
- } while (i);
|
|
|
- *buf = '\n';
|
|
|
- return buf+1;
|
|
|
+ }
|
|
|
+ m->count = buf - m->buf;
|
|
|
+ seq_printf(m, "\n");
|
|
|
}
|
|
|
|
|
|
/*
|
|
@@ -152,21 +156,20 @@ static inline const char *get_task_state(struct task_struct *tsk)
|
|
|
return *p;
|
|
|
}
|
|
|
|
|
|
-static inline char *task_state(struct task_struct *p, char *buffer)
|
|
|
+static inline void task_state(struct seq_file *m, struct pid_namespace *ns,
|
|
|
+ struct pid *pid, struct task_struct *p)
|
|
|
{
|
|
|
struct group_info *group_info;
|
|
|
int g;
|
|
|
struct fdtable *fdt = NULL;
|
|
|
- struct pid_namespace *ns;
|
|
|
pid_t ppid, tpid;
|
|
|
|
|
|
- ns = current->nsproxy->pid_ns;
|
|
|
rcu_read_lock();
|
|
|
ppid = pid_alive(p) ?
|
|
|
task_tgid_nr_ns(rcu_dereference(p->real_parent), ns) : 0;
|
|
|
tpid = pid_alive(p) && p->ptrace ?
|
|
|
task_pid_nr_ns(rcu_dereference(p->parent), ns) : 0;
|
|
|
- buffer += sprintf(buffer,
|
|
|
+ seq_printf(m,
|
|
|
"State:\t%s\n"
|
|
|
"Tgid:\t%d\n"
|
|
|
"Pid:\t%d\n"
|
|
@@ -176,7 +179,7 @@ static inline char *task_state(struct task_struct *p, char *buffer)
|
|
|
"Gid:\t%d\t%d\t%d\t%d\n",
|
|
|
get_task_state(p),
|
|
|
task_tgid_nr_ns(p, ns),
|
|
|
- task_pid_nr_ns(p, ns),
|
|
|
+ pid_nr_ns(pid, ns),
|
|
|
ppid, tpid,
|
|
|
p->uid, p->euid, p->suid, p->fsuid,
|
|
|
p->gid, p->egid, p->sgid, p->fsgid);
|
|
@@ -184,7 +187,7 @@ static inline char *task_state(struct task_struct *p, char *buffer)
|
|
|
task_lock(p);
|
|
|
if (p->files)
|
|
|
fdt = files_fdtable(p->files);
|
|
|
- buffer += sprintf(buffer,
|
|
|
+ seq_printf(m,
|
|
|
"FDSize:\t%d\n"
|
|
|
"Groups:\t",
|
|
|
fdt ? fdt->max_fds : 0);
|
|
@@ -195,20 +198,18 @@ static inline char *task_state(struct task_struct *p, char *buffer)
|
|
|
task_unlock(p);
|
|
|
|
|
|
for (g = 0; g < min(group_info->ngroups, NGROUPS_SMALL); g++)
|
|
|
- buffer += sprintf(buffer, "%d ", GROUP_AT(group_info, g));
|
|
|
+ seq_printf(m, "%d ", GROUP_AT(group_info, g));
|
|
|
put_group_info(group_info);
|
|
|
|
|
|
- buffer += sprintf(buffer, "\n");
|
|
|
- return buffer;
|
|
|
+ seq_printf(m, "\n");
|
|
|
}
|
|
|
|
|
|
-static char *render_sigset_t(const char *header, sigset_t *set, char *buffer)
|
|
|
+static void render_sigset_t(struct seq_file *m, const char *header,
|
|
|
+ sigset_t *set)
|
|
|
{
|
|
|
- int i, len;
|
|
|
+ int i;
|
|
|
|
|
|
- len = strlen(header);
|
|
|
- memcpy(buffer, header, len);
|
|
|
- buffer += len;
|
|
|
+ seq_printf(m, "%s", header);
|
|
|
|
|
|
i = _NSIG;
|
|
|
do {
|
|
@@ -219,12 +220,10 @@ static char *render_sigset_t(const char *header, sigset_t *set, char *buffer)
|
|
|
if (sigismember(set, i+2)) x |= 2;
|
|
|
if (sigismember(set, i+3)) x |= 4;
|
|
|
if (sigismember(set, i+4)) x |= 8;
|
|
|
- *buffer++ = (x < 10 ? '0' : 'a' - 10) + x;
|
|
|
+ seq_printf(m, "%x", x);
|
|
|
} while (i >= 4);
|
|
|
|
|
|
- *buffer++ = '\n';
|
|
|
- *buffer = 0;
|
|
|
- return buffer;
|
|
|
+ seq_printf(m, "\n");
|
|
|
}
|
|
|
|
|
|
static void collect_sigign_sigcatch(struct task_struct *p, sigset_t *ign,
|
|
@@ -242,7 +241,7 @@ static void collect_sigign_sigcatch(struct task_struct *p, sigset_t *ign,
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-static inline char *task_sig(struct task_struct *p, char *buffer)
|
|
|
+static inline void task_sig(struct seq_file *m, struct task_struct *p)
|
|
|
{
|
|
|
unsigned long flags;
|
|
|
sigset_t pending, shpending, blocked, ignored, caught;
|
|
@@ -269,67 +268,66 @@ static inline char *task_sig(struct task_struct *p, char *buffer)
|
|
|
}
|
|
|
rcu_read_unlock();
|
|
|
|
|
|
- buffer += sprintf(buffer, "Threads:\t%d\n", num_threads);
|
|
|
- buffer += sprintf(buffer, "SigQ:\t%lu/%lu\n", qsize, qlim);
|
|
|
+ seq_printf(m, "Threads:\t%d\n", num_threads);
|
|
|
+ seq_printf(m, "SigQ:\t%lu/%lu\n", qsize, qlim);
|
|
|
|
|
|
/* render them all */
|
|
|
- buffer = render_sigset_t("SigPnd:\t", &pending, buffer);
|
|
|
- buffer = render_sigset_t("ShdPnd:\t", &shpending, buffer);
|
|
|
- buffer = render_sigset_t("SigBlk:\t", &blocked, buffer);
|
|
|
- buffer = render_sigset_t("SigIgn:\t", &ignored, buffer);
|
|
|
- buffer = render_sigset_t("SigCgt:\t", &caught, buffer);
|
|
|
-
|
|
|
- return buffer;
|
|
|
+ render_sigset_t(m, "SigPnd:\t", &pending);
|
|
|
+ render_sigset_t(m, "ShdPnd:\t", &shpending);
|
|
|
+ render_sigset_t(m, "SigBlk:\t", &blocked);
|
|
|
+ render_sigset_t(m, "SigIgn:\t", &ignored);
|
|
|
+ render_sigset_t(m, "SigCgt:\t", &caught);
|
|
|
}
|
|
|
|
|
|
-static char *render_cap_t(const char *header, kernel_cap_t *a, char *buffer)
|
|
|
+static void render_cap_t(struct seq_file *m, const char *header,
|
|
|
+ kernel_cap_t *a)
|
|
|
{
|
|
|
unsigned __capi;
|
|
|
|
|
|
- buffer += sprintf(buffer, "%s", header);
|
|
|
+ seq_printf(m, "%s", header);
|
|
|
CAP_FOR_EACH_U32(__capi) {
|
|
|
- buffer += sprintf(buffer, "%08x",
|
|
|
- a->cap[(_LINUX_CAPABILITY_U32S-1) - __capi]);
|
|
|
+ seq_printf(m, "%08x",
|
|
|
+ a->cap[(_LINUX_CAPABILITY_U32S-1) - __capi]);
|
|
|
}
|
|
|
- return buffer + sprintf(buffer, "\n");
|
|
|
+ seq_printf(m, "\n");
|
|
|
}
|
|
|
|
|
|
-static inline char *task_cap(struct task_struct *p, char *buffer)
|
|
|
+static inline void task_cap(struct seq_file *m, struct task_struct *p)
|
|
|
{
|
|
|
- buffer = render_cap_t("CapInh:\t", &p->cap_inheritable, buffer);
|
|
|
- buffer = render_cap_t("CapPrm:\t", &p->cap_permitted, buffer);
|
|
|
- return render_cap_t("CapEff:\t", &p->cap_effective, buffer);
|
|
|
+ render_cap_t(m, "CapInh:\t", &p->cap_inheritable);
|
|
|
+ render_cap_t(m, "CapPrm:\t", &p->cap_permitted);
|
|
|
+ render_cap_t(m, "CapEff:\t", &p->cap_effective);
|
|
|
}
|
|
|
|
|
|
-static inline char *task_context_switch_counts(struct task_struct *p,
|
|
|
- char *buffer)
|
|
|
+static inline void task_context_switch_counts(struct seq_file *m,
|
|
|
+ struct task_struct *p)
|
|
|
{
|
|
|
- return buffer + sprintf(buffer, "voluntary_ctxt_switches:\t%lu\n"
|
|
|
- "nonvoluntary_ctxt_switches:\t%lu\n",
|
|
|
- p->nvcsw,
|
|
|
- p->nivcsw);
|
|
|
+ seq_printf(m, "voluntary_ctxt_switches:\t%lu\n"
|
|
|
+ "nonvoluntary_ctxt_switches:\t%lu\n",
|
|
|
+ p->nvcsw,
|
|
|
+ p->nivcsw);
|
|
|
}
|
|
|
|
|
|
-int proc_pid_status(struct task_struct *task, char *buffer)
|
|
|
+int proc_pid_status(struct seq_file *m, struct pid_namespace *ns,
|
|
|
+ struct pid *pid, struct task_struct *task)
|
|
|
{
|
|
|
- char *orig = buffer;
|
|
|
struct mm_struct *mm = get_task_mm(task);
|
|
|
|
|
|
- buffer = task_name(task, buffer);
|
|
|
- buffer = task_state(task, buffer);
|
|
|
+ task_name(m, task);
|
|
|
+ task_state(m, ns, pid, task);
|
|
|
|
|
|
if (mm) {
|
|
|
- buffer = task_mem(mm, buffer);
|
|
|
+ task_mem(m, mm);
|
|
|
mmput(mm);
|
|
|
}
|
|
|
- buffer = task_sig(task, buffer);
|
|
|
- buffer = task_cap(task, buffer);
|
|
|
- buffer = cpuset_task_status_allowed(task, buffer);
|
|
|
+ task_sig(m, task);
|
|
|
+ task_cap(m, task);
|
|
|
+ cpuset_task_status_allowed(m, task);
|
|
|
#if defined(CONFIG_S390)
|
|
|
- buffer = task_show_regs(task, buffer);
|
|
|
+ task_show_regs(m, task);
|
|
|
#endif
|
|
|
- buffer = task_context_switch_counts(task, buffer);
|
|
|
- return buffer - orig;
|
|
|
+ task_context_switch_counts(m, task);
|
|
|
+ return 0;
|
|
|
}
|
|
|
|
|
|
/*
|