diff options
author | We-unite <3205135446@qq.com> | 2024-07-23 19:32:09 +0800 |
---|---|---|
committer | We-unite <3205135446@qq.com> | 2024-07-25 17:09:45 +0800 |
commit | fc61a4a525846fa31ee2288df4e82f745bb39c95 (patch) | |
tree | e97c7b942c7e843782efbcc48882e6c0854df473 /src/basefunc.go | |
parent | cf5618ff2e2a183c5bdf6444787dccdcbf26ce76 (diff) | |
download | godo-fc61a4a525846fa31ee2288df4e82f745bb39c95.tar.gz godo-fc61a4a525846fa31ee2288df4e82f745bb39c95.zip |
Try ot fix the out-of-order bug, add EXECVE to itthings_left
The Most important work during this time is to find out solution
to the out-of-order bug. Discribe it here in detail: info from
audit may be out of order, which means fork may comes after execve,
even after exit. What an absurd penomenon to see a process not yet
created to work or exit!
To deal with this problem, I've tried several ways:
- in the 2nd coroutine, when EOE msg comes, if it's a fork/clone
event, send it immediately, otherwise wait for some time(such as
100 ms). But after all it delays longer, and has other problems.
- the 2nd coroutine doesn't send directly, but record all the finished
event id in a slice, and another thread checks once every one second,
if there are sth in slice, send corresponding events in the order of
event id. But: event that happens first doesn't always has lower id
or time, for example, 1 forks 2, then 2 execve, the audit in kernel
it self may gets execve before fork(maybe fork makes other settings),
which means execve has earlier timestamp and lower event id. The out-
of-order problem is not completely resolved. If we then add delays
to non-clone event, a more serious problem happens: we must use mutex
to lock the slice recording finished event id to prevent crush between
send thread and wait thread, but the wait thread can't get the mutex
again, because there are to much clone event and frequent send!
- So I use no delay but mongodb, when an execve comes, if pid is not
recorded, just insert it and wait for the fork. It does works, but
some other works is still left to do:
- what should i do if 2 forks 3 comes before 1 forks 2? Now I
suggest it doesn't happen, but what if?
- when execve comes before fork, i recorded it, but if this process
has a parent i don't care, delete, or stays there?
Also, as mentioned above, I've add EXECVE field in process into db,
records all the execve(time, and args) from the same process. Besides,
exit_timestamp and exit_code can be caught now, but too many process
has no exit info. This is also to be fixed.
Now, let's listen to the file changed by process. Don't forget the
to-do works listed above!
Diffstat (limited to '')
-rw-r--r-- | src/basefunc.go | 40 |
1 files changed, 27 insertions, 13 deletions
diff --git a/src/basefunc.go b/src/basefunc.go index 5fff3e8..2f39507 100644 --- a/src/basefunc.go +++ b/src/basefunc.go | |||
@@ -4,32 +4,46 @@ import ( | |||
4 | "bufio" | 4 | "bufio" |
5 | "fmt" | 5 | "fmt" |
6 | "os" | 6 | "os" |
7 | "os/exec" | ||
7 | "path/filepath" | 8 | "path/filepath" |
8 | "regexp" | ||
9 | "strconv" | 9 | "strconv" |
10 | "strings" | 10 | "strings" |
11 | "time" | 11 | "time" |
12 | ) | 12 | ) |
13 | 13 | ||
14 | func figureOutSyscalls() error { | 14 | func figureOutSyscalls() error { |
15 | NRRegex := regexp.MustCompile(`#define __NR_(.*?) (\d+)$`) | 15 | cmd := exec.Command("ausyscall", "--dump") |
16 | file, err := os.Open("/usr/include/asm/unistd_64.h") | 16 | stdout, err := cmd.StdoutPipe() |
17 | if err != nil { | 17 | if err != nil { |
18 | return err | 18 | return err |
19 | } | 19 | } |
20 | defer file.Close() | ||
21 | 20 | ||
22 | scanner := bufio.NewScanner(file) | 21 | if err := cmd.Start(); err != nil { |
23 | for scanner.Scan() { | 22 | return err |
23 | } | ||
24 | |||
25 | scanner := bufio.NewScanner(stdout) | ||
26 | for i := 0; scanner.Scan(); i++ { | ||
27 | if i == 0 { | ||
28 | continue | ||
29 | } | ||
24 | line := scanner.Text() | 30 | line := scanner.Text() |
25 | if NRRegex.MatchString(line) { | 31 | parts := strings.Split(line, "\t") |
26 | match := NRRegex.FindStringSubmatch(line) | 32 | if len(parts) != 2 { |
27 | num, err := strconv.Atoi(match[2]) | 33 | return fmt.Errorf("invalid ausyscall format") |
28 | if err != nil { | ||
29 | return err | ||
30 | } | ||
31 | syscallTable[num] = match[1] | ||
32 | } | 34 | } |
35 | num, err := strconv.Atoi(parts[0]) | ||
36 | if err != nil { | ||
37 | return err | ||
38 | } | ||
39 | syscallTable[num] = parts[1] | ||
40 | } | ||
41 | |||
42 | if err := scanner.Err(); err != nil { | ||
43 | return err | ||
44 | } | ||
45 | if err := cmd.Wait(); err != nil { | ||
46 | return err | ||
33 | } | 47 | } |
34 | return nil | 48 | return nil |
35 | } | 49 | } |