Skip to content

Instantly share code, notes, and snippets.

@gjedeer
Created January 11, 2017 09:11
Show Gist options
  • Select an option

  • Save gjedeer/5295786a1f2e0e94b40e29664d4281dd to your computer and use it in GitHub Desktop.

Select an option

Save gjedeer/5295786a1f2e0e94b40e29664d4281dd to your computer and use it in GitHub Desktop.
/*
* test process event connector - cn_proc.c
*
* cc $(shell pkg-config --cflags glib-2.0) \
* $(shell pkg-config --libs glib-2.0) \
* -o cn_proc cn_proc.c
*
* Listens for process events (fork, exec, change uid/gid/..., and exit)
* received through a kernel connector and prints them.
*
* Copyright (C) 2010 Canonical, Ltd.
* Improved by Kees Cook <kees@ubuntu.com>
* - Updated for current kernel structures
* - added event filtering
* - added more extensive process command line reporting
*
* Copyright (C) Matt Helsley, IBM Corp. 2005
* Derived from fcctl.c by Guillaume Thouvenin
* Original copyright notice follows:
*
* Copyright (C) 2005 BULL SA.
* Written by Guillaume Thouvenin <guillaume.thouvenin@bull.net>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <linux/connector.h>
#include <linux/netlink.h>
#include <linux/cn_proc.h>
#include <signal.h>
#include <setjmp.h>
#include <glib.h>
#include <stdint.h>
#include <stddef.h>
#include <inttypes.h>
#define SEND_MESSAGE_LEN (NLMSG_LENGTH(sizeof(struct cn_msg) + \
sizeof(enum proc_cn_mcast_op)))
#define RECV_MESSAGE_LEN (NLMSG_LENGTH(sizeof(struct cn_msg) + \
sizeof(struct proc_event)))
#define SEND_MESSAGE_SIZE (NLMSG_SPACE(SEND_MESSAGE_LEN))
#define RECV_MESSAGE_SIZE (NLMSG_SPACE(RECV_MESSAGE_LEN))
#define max(x,y) ((y)<(x)?(x):(y))
#define min(x,y) ((y)>(x)?(x):(y))
#define BUFF_SIZE (max(max(SEND_MESSAGE_SIZE, RECV_MESSAGE_SIZE), 1024))
#define MIN_RECV_SIZE (min(SEND_MESSAGE_SIZE, RECV_MESSAGE_SIZE))
#define PROC_CN_MCAST_LISTEN (1)
#define PROC_CN_MCAST_IGNORE (2)
/*
* SIGINT causes the program to exit gracefully
* this could happen any time after the LISTEN message has
* been sent
*/
#define INTR_SIG SIGINT
sigjmp_buf g_jmp;
void handle_intr (int signum)
{
siglongjmp(g_jmp, signum);
}
unsigned int show_event = PROC_EVENT_EXEC;
int show_seq = 0;
int show_cpu = 0;
int show_security_context = 1;
gchar *attr_result;
char *
attr_current(int pid)
{
char *filename;
gsize length;
GError *error = NULL;
if (attr_result) {
g_free(attr_result);
attr_result = NULL;
}
if (!asprintf(&filename,"/proc/%d/attr/current", pid)) {
perror("asprintf");
exit(1);
}
if (g_file_get_contents(filename, &attr_result, &length, &error)) {
if (length>1 && attr_result[length-1] == '\n') attr_result[length-1] = 0;
}
else {
goto failed;
}
free(filename);
if (error) g_error_free(error);
return attr_result;
failed:
free(filename);
if (error) g_error_free(error);
return "";
}
GString * cmdline_string;
char *cmdline_result;
char *
cmdline(int pid)
{
gchar *filename;
gchar *contents;
gsize i, length;
GError *error = NULL;
if (!cmdline_string) {
cmdline_string = g_string_new("");
}
if (cmdline_result) {
g_free(cmdline_result);
cmdline_result = NULL;
}
if (!asprintf(&filename,"/proc/%d/cmdline", pid)) {
perror("asprintf");
exit(1);
}
if (!g_file_get_contents(filename, &cmdline_result, &length, &error)) {
perror(error->message);
goto failed;
}
for (i=0;i<length;i++) {
if (i != length-1 && cmdline_result[i]=='\0') {
cmdline_result[i]=' ';
}
}
/* is this a kernel task? */
if (strlen(cmdline_result)==0) {
char *start, *end;
sprintf(filename,"/proc/%d/status",pid);
if (!g_file_get_contents(filename, &cmdline_result, &length, &error)) {
perror(error->message);
exit(1);
}
start = cmdline_result;
if (strncmp("Name:", start, 5)) {
fprintf(stderr,"'Name' not on first line of /proc status file:%s\n",cmdline_result);
goto failed;
}
start += 5;
while (*start++ == ' ') ;
end = start;
while (*end != '\n') end++;
*end = '\0';
g_string_printf(cmdline_string, "[%s]", start);
}
else {
g_string_assign(cmdline_string, cmdline_result);
if (show_security_context) {
g_string_append_printf(cmdline_string, " (%s)", attr_current(pid));
}
}
free(filename);
if (error) {
g_error_free(error);
error = NULL;
}
return cmdline_string->str;
failed:
free(filename);
if (error) {
g_error_free(error);
error = NULL;
}
return "";
}
void handle_msg (struct cn_msg *cn_hdr)
{
struct proc_event *ev;
/* print the message */
ev = (struct proc_event*)cn_hdr->data;
/* uninteresting event, bail out */
if ((show_event & ev->what) == 0) return;
if (show_seq || show_cpu) printf("\n");
if (show_seq) {
printf("cn seq: %d\ncn ack: %d\n",
cn_hdr->seq, cn_hdr->ack);
}
if (show_cpu) {
printf("ev cpu: %d\n", ev->cpu);
}
switch(ev->what){
case PROC_EVENT_NONE:
printf("event: none\n");
break;
case PROC_EVENT_FORK:
printf("event: fork: parent: %d %d (%s) child: %d %d\n",
ev->event_data.fork.parent_pid,
ev->event_data.fork.parent_tgid,
cmdline(ev->event_data.fork.parent_pid),
ev->event_data.fork.child_pid,
ev->event_data.fork.child_tgid);
break;
case PROC_EVENT_EXEC:
printf("event: exec %d %d: %s\n",
ev->event_data.exec.process_pid,
ev->event_data.exec.process_tgid,
cmdline(ev->event_data.exec.process_pid));
break;
case PROC_EVENT_UID:
printf("event: uid %d %d (%s): -> %d %d\n",
ev->event_data.id.process_pid,
ev->event_data.id.process_tgid,
cmdline(ev->event_data.fork.parent_pid),
ev->event_data.id.r.ruid,
ev->event_data.id.e.euid);
break;
case PROC_EVENT_GID:
printf("event: gid %d %d (%s): -> %d %d\n",
ev->event_data.id.process_pid,
ev->event_data.id.process_tgid,
cmdline(ev->event_data.fork.parent_pid),
ev->event_data.id.r.rgid,
ev->event_data.id.e.egid);
break;
#ifdef PROC_EVENT_SID
case PROC_EVENT_SID:
printf("event: sid %d %d (%s)\n",
ev->event_data.id.process_pid,
ev->event_data.id.process_tgid,
cmdline(ev->event_data.fork.parent_pid));
break;
#endif
case PROC_EVENT_EXIT:
printf("event: exit %d %d: exit code: %d\n",
ev->event_data.exit.process_pid,
ev->event_data.exit.process_tgid,
ev->event_data.exit.exit_code);
break;
default:
printf("event unknown: %d\n", ev->what);
break;
}
}
int main(int argc, char **argv)
{
int sk_nl;
int err;
struct sockaddr_nl my_nla, kern_nla, from_nla;
socklen_t from_nla_len;
char buff[BUFF_SIZE];
int rc = -1;
struct nlmsghdr *nl_hdr;
struct cn_msg *cn_hdr;
enum proc_cn_mcast_op *mcop_msg;
size_t recv_len = 0;
/* FIXME: actually test for CAP_NET_ADMIN */
if (getuid() != 0) {
fprintf(stderr,"Only users with CAP_NET_ADMIN can start/stop the fork connector\n");
return 0;
}
if (argc != 1)
return 0;
/*
* Create an endpoint for communication. Use the kernel user
* interface device (PF_NETLINK) which is a datagram oriented
* service (SOCK_DGRAM). The protocol used is the connector
* protocol (NETLINK_CONNECTOR)
*/
sk_nl = socket(PF_NETLINK, SOCK_DGRAM, NETLINK_CONNECTOR);
if (sk_nl == -1) {
fprintf(stderr,"socket sk_nl error");
return rc;
}
my_nla.nl_family = AF_NETLINK;
my_nla.nl_groups = CN_IDX_PROC;
my_nla.nl_pid = getpid();
kern_nla.nl_family = AF_NETLINK;
kern_nla.nl_groups = CN_IDX_PROC;
kern_nla.nl_pid = 1;
err = bind(sk_nl, (struct sockaddr *)&my_nla, sizeof(my_nla));
if (err == -1) {
perror("binding sk_nl error");
goto close_and_exit;
}
nl_hdr = (struct nlmsghdr *)buff;
cn_hdr = (struct cn_msg *)NLMSG_DATA(nl_hdr);
mcop_msg = (enum proc_cn_mcast_op*)&cn_hdr->data[0];
if (sigsetjmp(g_jmp, INTR_SIG) != 0) {
fprintf(stderr,"sending proc connector: PROC_CN_MCAST_IGNORE... ");
memset(buff, 0, sizeof(buff));
*mcop_msg = PROC_CN_MCAST_IGNORE;
} else {
fprintf(stderr,"sending proc connector: PROC_CN_MCAST_LISTEN... ");
memset(buff, 0, sizeof(buff));
*mcop_msg = PROC_CN_MCAST_LISTEN;
signal(INTR_SIG, handle_intr);
}
/* fill the netlink header */
nl_hdr->nlmsg_len = SEND_MESSAGE_LEN;
nl_hdr->nlmsg_type = NLMSG_DONE;
nl_hdr->nlmsg_flags = 0;
nl_hdr->nlmsg_seq = 0;
nl_hdr->nlmsg_pid = getpid();
/* fill the connector header */
cn_hdr->id.idx = CN_IDX_PROC;
cn_hdr->id.val = CN_VAL_PROC;
cn_hdr->seq = 0;
cn_hdr->ack = 0;
cn_hdr->len = sizeof(enum proc_cn_mcast_op);
if (send(sk_nl, nl_hdr, nl_hdr->nlmsg_len, 0) != nl_hdr->nlmsg_len) {
fprintf(stderr,"failed to send proc connector mcast ctl op!\n");
goto close_and_exit;
}
fprintf(stderr,"sent\n");
if (*mcop_msg == PROC_CN_MCAST_IGNORE) {
rc = 0;
goto close_and_exit;
}
fprintf(stderr,"Reading process events from proc connector.\n"
"Hit Ctrl-C to exit\n");
for(memset(buff, 0, sizeof(buff)), from_nla_len = sizeof(from_nla);
; memset(buff, 0, sizeof(buff)), from_nla_len = sizeof(from_nla)) {
struct nlmsghdr *nlh = (struct nlmsghdr*)buff;
memcpy(&from_nla, &kern_nla, sizeof(from_nla));
recv_len = recvfrom(sk_nl, buff, BUFF_SIZE, 0,
(struct sockaddr*)&from_nla, &from_nla_len);
if (recv_len < 1)
continue;
while (NLMSG_OK(nlh, recv_len)) {
cn_hdr = NLMSG_DATA(nlh);
if (nlh->nlmsg_type == NLMSG_NOOP)
continue;
if ((nlh->nlmsg_type == NLMSG_ERROR) ||
(nlh->nlmsg_type == NLMSG_OVERRUN))
break;
handle_msg(cn_hdr);
if (nlh->nlmsg_type == NLMSG_DONE)
break;
nlh = NLMSG_NEXT(nlh, recv_len);
}
}
close_and_exit:
close(sk_nl);
exit(rc);
return 0;
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment