From: Caspar Zhang <czhang@redhat.com>
To: Garrett Cooper <yanegomi@gmail.com>
Cc: ltp-list@lists.sourceforge.net
Subject: [LTP] Final version [Re: [PATCH v4] make pid_list dynamically sized with memory]
Date: Fri, 07 Jan 2011 02:42:12 +0800 [thread overview]
Message-ID: <4D260D04.3080102@redhat.com> (raw)
In-Reply-To: <1294339211-19449-1-git-send-email-czhang@redhat.com>
Hi Garrett, this is the *final* version, please review.
On 01/07/2011 02:40 AM, czhang@redhat.com wrote:
> From: Caspar Zhang <czhang@redhat.com>
>
> We get segfaults during testing mtest01 on a 5TB memory machine, the
> problem was traced to the array pid_list[] went to overflow and
> corrupted memory. This fix makes pid_list dynamically sized with
> correct memory size to avoid overflow.
>
> v2: not split into different bits when allocating pid_list.
> v3: 1) fix optargs: -b and -p should not be used at the same time.
> 2) pre_mem maybe not initialized before using if -p option not used.
> fix it by moving it outside ``if(maxpercent)'' block.
> v4: fix code format.
>
> Signed-off-by: Caspar Zhang <czhang@redhat.com>
> ---
> testcases/kernel/mem/mtest01/mtest01.c | 409 +++++++++++++++++---------------
> 1 files changed, 215 insertions(+), 194 deletions(-)
>
> diff --git a/testcases/kernel/mem/mtest01/mtest01.c b/testcases/kernel/mem/mtest01/mtest01.c
> index aee0d51..94b429a 100644
> --- a/testcases/kernel/mem/mtest01/mtest01.c
> +++ b/testcases/kernel/mem/mtest01/mtest01.c
> @@ -44,223 +44,244 @@
>
> #include "test.h"
>
> +#define FIVE_HUNDRED_KB (unsigned long long)(500*1024*1024)
> +#define ONE_MEGABYTE (unsigned long long)(1024*1024*1024)
> +#define THREE_MEGABYTES (unsigned long long)(3*ONE_MEGABYTE)
> +
> char *TCID = "mtest01";
> int TST_TOTAL = 1;
> -
> int pid_count = 0;
>
> void handler(int signo)
> {
> - pid_count++;
> + pid_count++;
> }
>
> -int main(int argc, char* argv[]) {
> - char* mem;
> - float percent;
> - unsigned int maxpercent=0, dowrite=0, verbose=0, j, c;
> - unsigned long bytecount, alloc_bytes;
> - unsigned long long original_maxbytes,maxbytes=0;
> - unsigned long long pre_mem, post_mem;
> - extern char* optarg;
> - int chunksize = 1024*1024; /* one meg at a time by default */
> - struct sysinfo sstats;
> - int i,pid_cntr;
> - pid_t pid,pid_list[1000];
> - struct sigaction act;
> -
> - act.sa_handler = handler;
> - act.sa_flags = 0;
> - sigemptyset(&act.sa_mask);
> - sigaction(SIGRTMIN, &act, 0);
> -
> - for (i=0;i<1000;i++)
> - pid_list[i]=(pid_t)0;
> -
> - while ((c=getopt(argc, argv, "c:b:p:wvh")) != EOF) {
> - switch((char)c) {
> - case 'c':
> - chunksize = atoi(optarg);
> - break;
> - case 'b':
> - maxbytes = atoll(optarg);
> - break;
> - case 'p':
> - maxpercent = atoi(optarg);
> - if (maxpercent <= 0) {
> - tst_resm(TFAIL, "ERROR: -p option requires number greater than 0");
> - exit(1);}
> - if (maxpercent > 99) {
> - tst_resm(TFAIL, "ERROR: -p option cannot be greater than 99");
> - exit(1);}
> - break;
> - case 'w':
> - dowrite = 1;
> - break;
> - case 'v':
> - verbose = 1;
> - break;
> - case 'h':
> - default:
> - printf("Usage: %s [-c <bytes>] [-b <bytes>|-p <percent>] [-v]\n", argv[0]);
> - printf("\t-c <num>\tsize of chunk in bytes to malloc on each pass\n");
> - printf("\t-b <bytes>\tmaximum number of bytes to allocate before stopping\n");
> - printf("\t-p <bytes>\tpercent of total memory used at which the program stops\n");
> - printf("\t-w\t\twrite to the memory after allocating\n");
> - printf("\t-v\t\tverbose\n");
> - printf("\t-h\t\tdisplay usage\n");
> - exit(-1);
> - }
> - }
> -
> - sysinfo(&sstats);
> - if (maxpercent) {
> +int main(int argc, char* argv[])
> +{
> + char* mem;
> + float percent;
> + unsigned int maxpercent = 0, dowrite = 0, verbose=0, j, c;
> + unsigned long bytecount, alloc_bytes, max_pids;
> + unsigned long long original_maxbytes, maxbytes = 0;
> + unsigned long long pre_mem, post_mem;
> unsigned long long total_ram, total_free, D, C;
> - percent=(float)maxpercent/100.00;
> -
> - total_ram=sstats.totalram;
> - total_ram=total_ram+sstats.totalswap;
> -
> - total_free=sstats.freeram;
> - total_free=total_free+sstats.freeswap;
> -
> - /* Total memory used needed to reach maxpercent */
> - D = percent*(sstats.mem_unit*total_ram);
> - tst_resm(TINFO, "Total memory used needed to reach maxpercent = %llu kbytes", D/1024);
> -
> - /* Total memory already used */
> - C = sstats.mem_unit*(total_ram-total_free);
> - tst_resm(TINFO, "Total memory already used on system = %llu kbytes", C/1024);
> + extern char* optarg;
> + int chunksize = 1024*1024; /* one meg at a time by default */
> + struct sysinfo sstats;
> + int i, pid_cntr;
> + pid_t pid, *pid_list;
> + struct sigaction act;
>
> + act.sa_handler = handler;
> + act.sa_flags = 0;
> + sigemptyset(&act.sa_mask);
> + sigaction(SIGRTMIN, &act, 0);
> +
> + while ((c = getopt(argc, argv, "c:b:p:wvh")) != EOF)
> + {
> + switch((char)c)
> + {
> + case 'c':
> + chunksize = atoi(optarg);
> + break;
> + case 'b':
> + if (maxpercent != 0) {
> + tst_resm(TFAIL, "ERROR: -b option cannot be used with -p option at the same time");
> + exit(1);
> + }
> + maxbytes = atoll(optarg);
> + break;
> + case 'p':
> + if (maxbytes != 0) {
> + tst_resm(TFAIL, "ERROR: -p option cannot be used with -b option at the same time");
> + exit(1);
> + }
> + maxpercent = atoi(optarg);
> + if (maxpercent <= 0) {
> + tst_resm(TFAIL, "ERROR: -p option requires number greater than 0");
> + exit(1);
> + }
> + if (maxpercent > 99) {
> + tst_resm(TFAIL, "ERROR: -p option cannot be greater than 99");
> + exit(1);
> + }
> + break;
> + case 'w':
> + dowrite = 1;
> + break;
> + case 'v':
> + verbose = 1;
> + break;
> + case 'h':
> + default:
> + printf("Usage: %s [-c <bytes>] [-b <bytes>|-p <percent>] [-v]\n", argv[0]);
> + printf("\t-c <num>\tsize of chunk in bytes to malloc on each pass\n");
> + printf("\t-b <bytes>\tmaximum number of bytes to allocate before stopping\n");
> + printf("\t-p <bytes>\tpercent of total memory used at which the program stops\n");
> + printf("\t-w\t\twrite to the memory after allocating\n");
> + printf("\t-v\t\tverbose\n");
> + printf("\t-h\t\tdisplay usage\n");
> + exit(-1);
> + }
> + }
> +
> + sysinfo(&sstats);
> + total_ram = sstats.totalram + sstats.totalswap;
> + total_free = sstats.freeram + sstats.freeswap;
> /* Total Free Pre-Test RAM */
> - pre_mem = sstats.mem_unit*total_free;
> -
> - /* Are we already using more than maxpercent? */
> - if (C>D) {
> - tst_resm(TFAIL, "More memory than the maximum amount you specified is already being used");
> - exit(1);
> + pre_mem = sstats.mem_unit * total_free;
> + max_pids = total_ram / (unsigned long)FIVE_HUNDRED_KB + 1;
> +
> + if ((pid_list = malloc(max_pids * sizeof(pid_t))) == NULL)
> + {
> + tst_resm(TBROK|TERRNO, "malloc failed.");
> + exit(1);
> }
> - else
> - pre_mem = sstats.mem_unit*total_free;
> + memset(pid_list, 0, max_pids * sizeof(pid_t));
> +
> + /* Currently used memory */
> + C = sstats.mem_unit * (total_ram - total_free);
> + tst_resm(TINFO, "Total memory already used on system = %llu kbytes", C/1024);
> +
> + if (maxpercent)
> + {
> + percent = (float)maxpercent / 100.00;
>
> - /* set maxbytes to the extra amount we want to allocate */
> - maxbytes = D-C;
> - tst_resm(TINFO, "Filling up %d%% of ram which is %llu kbytes", maxpercent, maxbytes/1024);
> - }
> - original_maxbytes=maxbytes;
> - i=0;
> - pid_cntr=0;
> - pid=fork();
> - if (pid != 0)
> - pid_cntr++;
> - pid_list[i]=pid;
> + /* Desired memory needed to reach maxpercent */
> + D = percent * (sstats.mem_unit * total_ram);
> + tst_resm(TINFO, "Total memory used needed to reach maxpercent = %llu kbytes", D/1024);
> +
> + /* Are we already using more than maxpercent? */
> + if (C > D)
> + {
> + tst_resm(TFAIL, "More memory than the maximum amount you specified is already being used");
> + free(pid_list);
> + exit(1);
> + }
> +
> + /* set maxbytes to the extra amount we want to allocate */
> + maxbytes = D - C;
> + tst_resm(TINFO, "Filling up %d%% of ram which is %llu kbytes", maxpercent, maxbytes/1024);
> + }
> + original_maxbytes = maxbytes;
> + i = 0;
> + pid_cntr = 0;
> + pid = fork();
> + if (pid != 0)
> + pid_cntr++;
> + pid_list[i] = pid;
>
> #if defined (_s390_) /* s390's 31bit addressing requires smaller chunks */
> -#define FIVE_HUNDRED_KB (500*1024*1024)
> -#define ONE_MEGABYTE (1024*1024*1024)
> -#define THREE_MEGABYTES (3*ONE_MEGABYTE)
> - while (pid != 0 && maxbytes > FIVE_HUNDRED_KB)
> - {
> - i++;
> - maxbytes -= FIVE_HUNDRED_KB;
> - pid = fork();
> - if (pid != 0) {
> - pid_cntr++;
> - pid_list[i] = pid;
> + while (pid != 0 && maxbytes > FIVE_HUNDRED_KB)
> + {
> + i++;
> + maxbytes -= FIVE_HUNDRED_KB;
> + pid = fork();
> + if (pid != 0)
> + {
> + pid_cntr++;
> + pid_list[i] = pid;
> + }
> }
> - }
> - if (maxbytes > FIVE_HUNDRED_KB)
> - alloc_bytes FIVE_HUNDRED_KB;
> - else
> - alloc_bytes = (unsigned long) maxbytes;
> -
> + if (maxbytes > FIVE_HUNDRED_KB)
> + alloc_bytes = FIVE_HUNDRED_KB;
> + else
> + alloc_bytes = (unsigned long) maxbytes;
> +
> #elif __WORDSIZE==32
> - while (pid != 0 && maxbytes > ONE_MEGABYTE)
> - {
> - i++;
> - maxbytes -= ONE_MEGABYTE;
> - pid = fork();
> - if (pid != 0) {
> - pid_cntr++;
> - pid_list[i]=pid;
> + while (pid != 0 && maxbytes > ONE_MEGABYTE)
> + {
> + i++;
> + maxbytes -= ONE_MEGABYTE;
> + pid = fork();
> + if (pid != 0)
> + {
> + pid_cntr++;
> + pid_list[i]=pid;
> + }
> }
> - }
> - if (maxbytes > ONE_MEGABYTE)
> - alloc_bytes = ONE_MEGABYTE;
> - else
> - alloc_bytes = (unsigned long)maxbytes;
> -
> + if (maxbytes > ONE_MEGABYTE)
> + alloc_bytes = ONE_MEGABYTE;
> + else
> + alloc_bytes = (unsigned long)maxbytes;
> +
> #elif __WORDSIZE==64
> - while (pid!=0 && maxbytes > THREE_MEGABYTES)
> - {
> - i++;
> - maxbytes -= THREE_MEGABYTES;
> - pid=fork();
> - if (pid != 0) {
> - pid_cntr++;
> - pid_list[i] = pid;
> - }
> - }
> - if (maxbytes > THREE_MEGABYTES)
> - alloc_bytes = THREE_MEGABYTES;
> - else
> - alloc_bytes = maxbytes;
> -#endif
> -
> - if (pid == 0) /** CHILD **/
> - {
> - bytecount=chunksize;
> - while (1) {
> - if ((mem = (char*)malloc(chunksize)) == NULL) {
> - tst_resm(TINFO, "stopped at %lu bytes", bytecount);
> - exit(1);
> - }
> - if (dowrite)
> - for (j=0; j<chunksize; j++)
> - *(mem+j)='a';
> - if (verbose)
> - tst_resm(TINFO, "allocated %lu bytes chunksize is %d", bytecount, chunksize);
> - bytecount+=chunksize;
> - if (alloc_bytes && (bytecount >= alloc_bytes))
> - break;
> + while (pid!=0 && maxbytes > THREE_MEGABYTES)
> + {
> + i++;
> + maxbytes -= THREE_MEGABYTES;
> + pid = fork();
> + if (pid != 0)
> + {
> + pid_cntr++;
> + pid_list[i] = pid;
> + }
> }
> - if (dowrite)
> - tst_resm(TINFO, "... %lu bytes allocated and used.", bytecount);
> + if (maxbytes > THREE_MEGABYTES)
> + alloc_bytes = THREE_MEGABYTES;
> else
> - tst_resm(TINFO, "... %lu bytes allocated only.", bytecount);
> - kill(getppid(),SIGRTMIN);
> - while (1)
> - sleep(1);
> - }
> - else /** PARENT **/
> - {
> -
> - i=0;
> - sysinfo(&sstats);
> + alloc_bytes = maxbytes;
> +#endif
>
> - if (dowrite)
> + if (pid == 0) /** CHILD **/
> {
> - /* Total Free Post-Test RAM */
> - post_mem = (unsigned long long)sstats.mem_unit*sstats.freeram;
> - post_mem = post_mem+((unsigned long long)sstats.mem_unit*sstats.freeswap);
> -
> - while ((((unsigned long long)pre_mem - post_mem) < (unsigned long long)original_maxbytes) &&
> - (pid_count < pid_cntr) )
> - {
> - sleep(1);
> - sysinfo(&sstats);
> - post_mem = (unsigned long long)sstats.mem_unit*sstats.freeram;
> - post_mem = post_mem+((unsigned long long)sstats.mem_unit*sstats.freeswap);
> - }
> + bytecount = chunksize;
> + while (1)
> + {
> + if ((mem = malloc(chunksize)) == NULL)
> + {
> + tst_resm(TBROK|TERRNO, "stopped at %lu bytes", bytecount);
> + free(pid_list);
> + exit(1);
> + }
> + if (dowrite)
> + for (j = 0; j < chunksize; j++)
> + *(mem+j) = 'a';
> + if (verbose)
> + tst_resm(TINFO, "allocated %lu bytes chunksize is %d", bytecount, chunksize);
> + bytecount += chunksize;
> + if (alloc_bytes && bytecount >= alloc_bytes)
> + break;
> + }
> + if (dowrite)
> + tst_resm(TINFO, "... %lu bytes allocated and used.", bytecount);
> + else
> + tst_resm(TINFO, "... %lu bytes allocated only.", bytecount);
> + kill(getppid(), SIGRTMIN);
> + while (1)
> + sleep(1);
> }
> - while (pid_list[i]!=0)
> + else /** PARENT **/
> {
> - kill(pid_list[i],SIGKILL);
> - i++;
> + i = 0;
> + sysinfo(&sstats);
> +
> + if (dowrite)
> + {
> + /* Total Free Post-Test RAM */
> + post_mem = (unsigned long long)sstats.mem_unit * sstats.freeram;
> + post_mem = post_mem + (unsigned long long)sstats.mem_unit * sstats.freeswap;
> +
> + while ((((unsigned long long)pre_mem - post_mem) < (unsigned long long)original_maxbytes) &&
> + (pid_count < pid_cntr))
> + {
> + sleep(1);
> + sysinfo(&sstats);
> + post_mem = (unsigned long long)sstats.mem_unit * sstats.freeram;
> + post_mem = post_mem + (unsigned long long)sstats.mem_unit * sstats.freeswap;
> + }
> + }
> + while (pid_list[i] != 0)
> + {
> + kill(pid_list[i], SIGKILL);
> + i++;
> + }
> + if (dowrite)
> + tst_resm(TPASS, "%llu kbytes allocated and used.", original_maxbytes/1024);
> + else
> + tst_resm(TPASS, "%llu kbytes allocated only.", original_maxbytes/1024);
> }
> - if (dowrite)
> - tst_resm(TPASS, "%llu kbytes allocated and used.", original_maxbytes/1024);
> - else
> - tst_resm(TPASS, "%llu kbytes allocated only.", original_maxbytes/1024);
> - }
> - exit(0);
> -}
> \ No newline at end of file
> + free(pid_list);
> + exit(0);
> +}
--
Quality Assurance Associate (Kernel) in
Red Hat Software (Beijing) Co., R&D Branch
TEL: +86-10-62608150
WEB: http://www.redhat.com/
--
C-A-S-P-A-R, Caspar /kæspɑ:/ ;-)
------------------------------------------------------------------------------
Learn how Oracle Real Application Clusters (RAC) One Node allows customers
to consolidate database storage, standardize their database environment, and,
should the need arise, upgrade to a full multi-node Oracle RAC database
without downtime or disruption
http://p.sf.net/sfu/oracle-sfdevnl
_______________________________________________
Ltp-list mailing list
Ltp-list@lists.sourceforge.net
https://lists.sourceforge.net/lists/listinfo/ltp-list
next prev parent reply other threads:[~2011-01-06 18:42 UTC|newest]
Thread overview: 8+ messages / expand[flat|nested] mbox.gz Atom feed top
2011-01-06 18:40 [LTP] [PATCH v4] make pid_list dynamically sized with memory czhang
2011-01-06 18:42 ` Caspar Zhang [this message]
2011-01-11 9:26 ` [LTP] Final version [Re: [PATCH v4] make pid_list dynamically sized with memory] Caspar Zhang
2011-01-16 11:05 ` [LTP] [PATCH v4] make pid_list dynamically sized with memory Caspar Zhang
2011-01-18 8:29 ` Caspar Zhang
2011-01-18 8:33 ` Garrett Cooper
2011-01-18 8:45 ` Caspar Zhang
2011-01-18 9:14 ` [LTP] [PATCH v5] " Caspar Zhang
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=4D260D04.3080102@redhat.com \
--to=czhang@redhat.com \
--cc=ltp-list@lists.sourceforge.net \
--cc=yanegomi@gmail.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox