forked from sinojelly/mockcpp
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathUnixCodeModifier.cpp
More file actions
65 lines (50 loc) · 2.51 KB
/
UnixCodeModifier.cpp
File metadata and controls
65 lines (50 loc) · 2.51 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
/***
mockcpp is a C/C++ mock framework.
Copyright [2008] [Darwin Yuan <[email protected]>]
[Chen Guodong <[email protected]>]
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
***/
#include <string.h>
#include <inttypes.h>
#include <sys/mman.h>
#include <unistd.h>
#include <mockcpp/CodeModifier.h>
//////////////////////////////////////////////////////////////////
#define ALIGN_TO_PAGE_BOUNDARY(addr, page_size) (void*) (((uintptr_t)addr) & (~(page_size - 1)))
//////////////////////////////////////////////////////////////////
MOCKCPP_NS_START
bool CodeModifier::modify(void *dest, const void *src, size_t size)
{
int page_size = getpagesize();
if(::mprotect(ALIGN_TO_PAGE_BOUNDARY(dest, page_size), page_size * 2, PROT_EXEC | PROT_WRITE | PROT_READ ) != 0)
{
return false;
}
::memcpy(dest, src, size);
#if 0
#if BUILD_FOR_X86
//(void)memcpy(dest, src, size); // something wrong on linux: after memcpy(or 5 single byte copy), the 4 bytes following jmp, src is 0x07c951b0, but dest is 0x07b851b0. so use unsigned int *, it works ok.
*((unsigned char *)dest) = *((unsigned char *)src);
*((unsigned long *)((unsigned long)dest + 1)) = *((unsigned long *)((unsigned long)src + 1));
#else
*((unsigned char *)dest) = *((unsigned char *)src);
*((unsigned char *)((unsigned long)dest + 1)) = *((unsigned char *)((unsigned long)src + 1));
// after this line, dest+2 is 0x00c90000, not 0, so change it.
*((unsigned char *)((unsigned long)dest + 2)) = *((unsigned char *)((unsigned long)src + 2));
*((unsigned char *)((unsigned long)dest + 3)) = *((unsigned char *)((unsigned long)src + 3));
*((unsigned char *)((unsigned long)dest + 4)) = *((unsigned char *)((unsigned long)src + 4));
*((unsigned char *)((unsigned long)dest + 5)) = *((unsigned char *)((unsigned long)src + 5));
*((unsigned long *)((unsigned long)dest + 6)) = *((unsigned long *)((unsigned long)src + 6));
#endif
#endif
return true;
}
MOCKCPP_NS_END