nas:build_a_linux_nas:tune_the_system
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
nas:build_a_linux_nas:tune_the_system [2021/09/25 23:17] – peter | nas:build_a_linux_nas:tune_the_system [2021/09/26 12:39] (current) – peter | ||
---|---|---|---|
Line 72: | Line 72: | ||
* Test with stripe_cache_size=32768 | * Test with stripe_cache_size=32768 | ||
- | At each test, check the Read and Write performance, | + | At each test, check the **Write** performance, |
dd if=testfile of=/ | dd if=testfile of=/ | ||
361680+0 records in | 361680+0 records in | ||
Line 78: | Line 78: | ||
2962882560 bytes (3.0 GB) copied, 0.92471 s, 3.2 GB/s | 2962882560 bytes (3.0 GB) copied, 0.92471 s, 3.2 GB/s | ||
</ | </ | ||
+ | |||
Read speed test: <code bash> | Read speed test: <code bash> | ||
Line 116: | Line 117: | ||
* Test with Read Ahead @ 524288 | * Test with Read Ahead @ 524288 | ||
- | At each test, check the Read performance, | + | At each test, check the **Read** performance, |
dd if=testfile of=/ | dd if=testfile of=/ | ||
361680+0 records in | 361680+0 records in | ||
Line 137: | Line 138: | ||
</ | </ | ||
+ | |||
+ | ---- | ||
+ | |||
+ | ===== Script to auto-determine best tuning changes ===== | ||
+ | |||
+ | <code bash> | ||
+ | #!/bin/bash | ||
+ | # | ||
+ | # Please note this test requires 30 GB of free space | ||
+ | # in your RAID md device | ||
+ | # | ||
+ | # The aim of this script is to find the best settings for performance | ||
+ | # of your RAID by testing each setting separately. | ||
+ | # This script does make some system modification, | ||
+ | # make these changes permanent (e.g. write them in / | ||
+ | # At the next boot all the changes will be lost, | ||
+ | # so fill free to play with it! | ||
+ | # | ||
+ | # developed by alfonso / Jan 2012 | ||
+ | # | ||
+ | # | ||
+ | # this is your mount point for the RAID | ||
+ | # PLEASE NOTE the script will REMOVE any file called testfile*.out in this folder!! | ||
+ | MNT=/ | ||
+ | # this is device from which to get input | ||
+ | # no need to change this | ||
+ | INPUT=/ | ||
+ | |||
+ | # test for privileges | ||
+ | if [ " | ||
+ | then | ||
+ | echo Need to be root! | ||
+ | echo ABORT | ||
+ | exit 1 | ||
+ | fi | ||
+ | |||
+ | if ! [ -d $MNT/ | ||
+ | then | ||
+ | echo | ||
+ | echo "$MNT is not a file system! Something went wrong?" | ||
+ | echo ABORT | ||
+ | exit 1 | ||
+ | fi | ||
+ | |||
+ | # find out which one is your md | ||
+ | # note that the script only works for one md. If you have more than one | ||
+ | # just uncomment the line below and type something like MDDEV=md0 | ||
+ | MDDEV=" | ||
+ | # MDDEV=md0 | ||
+ | |||
+ | if [ -z " | ||
+ | then | ||
+ | echo | ||
+ | echo "I can\'t find any md" | ||
+ | echo ABORT | ||
+ | exit 1 | ||
+ | fi | ||
+ | |||
+ | # | ||
+ | # get the letter of all devices from cat / | ||
+ | # | ||
+ | # this expression takes the output of / | ||
+ | # then takes the line of our md | ||
+ | # then changes spaces into new lines | ||
+ | # then takes only lines starting with sd | ||
+ | # then take the 3rd character (a for sda1, etc) | ||
+ | # and then remove new lines to make a single string | ||
+ | DEVS=" | ||
+ | echo "These are devices found in $MDDEV: $DEVS" | ||
+ | |||
+ | function test_write() | ||
+ | { | ||
+ | # writing tests: | ||
+ | |||
+ | echo -n . | ||
+ | hdparm -f / | ||
+ | WRTE1=`dd if=$INPUT of=$MNT/ | ||
+ | WRUN1=`echo $WRTE1 | awk ' { print ( $(NF) ) }'` | ||
+ | WRSP1=`echo $WRTE1 | awk ' { print ( $(NF-1) ) }'` | ||
+ | if [ $WRUN1 != " | ||
+ | then | ||
+ | echo | ||
+ | echo "This script was created for all speeds measured in MB/s" | ||
+ | echo ABORT | ||
+ | exit 1 | ||
+ | fi | ||
+ | | ||
+ | echo -n . | ||
+ | hdparm -f / | ||
+ | WRTE2=`dd if=$INPUT of=$MNT/ | ||
+ | WRUN2=`echo $WRTE2 | awk ' { print ( $(NF) ) }'` | ||
+ | WRSP2=`echo $WRTE2 | awk ' { print ( $(NF-1) ) }'` | ||
+ | if [ $WRUN2 != " | ||
+ | then | ||
+ | echo | ||
+ | echo "This script was created for all speeds measured in MB/s" | ||
+ | echo ABORT | ||
+ | exit 1 | ||
+ | fi | ||
+ | | ||
+ | echo -n . | ||
+ | hdparm -f / | ||
+ | WRTE3=`dd if=$INPUT of=$MNT/ | ||
+ | WRUN3=`echo $WRTE3 | awk ' { print ( $(NF) ) }'` | ||
+ | WRSP3=`echo $WRTE3 | awk ' { print ( $(NF-1) ) }'` | ||
+ | if [ $WRUN3 != " | ||
+ | then | ||
+ | echo | ||
+ | echo "This script was created for all speeds measured in MB/s" | ||
+ | echo ABORT | ||
+ | exit 1 | ||
+ | fi | ||
+ | | ||
+ | AVG_WRITE=`echo " | ||
+ | | ||
+ | #echo " Average write is $AVG_WRITE MB/s NOTE: there should be a dot before the last 2 digits" | ||
+ | echo " average write is `echo " | ||
+ | | ||
+ | # echo $WRTE1 | ||
+ | # echo $WRTE2 | ||
+ | # echo $WRTE3 | ||
+ | } | ||
+ | |||
+ | function test_read() | ||
+ | { | ||
+ | |||
+ | # reading tests: | ||
+ | |||
+ | echo -n . | ||
+ | hdparm -f / | ||
+ | READ1=`dd if=$MNT/ | ||
+ | RDUN1=`echo $READ1 | awk ' { print ( $(NF) ) }'` | ||
+ | RDSP1=`echo $READ1 | awk ' { print ( $(NF-1) ) }'` | ||
+ | if [ $RDUN1 != " | ||
+ | then | ||
+ | echo | ||
+ | echo "This script was created for all speeds measured in MB/s" | ||
+ | echo ABORT | ||
+ | exit 1 | ||
+ | fi | ||
+ | | ||
+ | echo -n . | ||
+ | hdparm -f / | ||
+ | READ2=`dd if=$MNT/ | ||
+ | RDUN2=`echo $READ2 | awk ' { print ( $(NF) ) }'` | ||
+ | RDSP2=`echo $READ2 | awk ' { print ( $(NF-1) ) }'` | ||
+ | if [ $RDUN2 != " | ||
+ | then | ||
+ | echo | ||
+ | echo "This script was created for all speeds measured in MB/s" | ||
+ | echo ABORT | ||
+ | exit 1 | ||
+ | fi | ||
+ | | ||
+ | echo -n . | ||
+ | hdparm -f / | ||
+ | READ3=`dd if=$MNT/ | ||
+ | RDUN3=`echo $READ3 | awk ' { print ( $(NF) ) }'` | ||
+ | RDSP3=`echo $READ3 | awk ' { print ( $(NF-1) ) }'` | ||
+ | if [ $RDUN3 != " | ||
+ | then | ||
+ | echo | ||
+ | echo "This script was created for all speeds measured in MB/s" | ||
+ | echo ABORT | ||
+ | exit 1 | ||
+ | fi | ||
+ | | ||
+ | AVG_READ=`echo " | ||
+ | | ||
+ | #echo " Average read is $AVG_READ MB/s NOTE: there should be a dot before the last 2 digits" | ||
+ | echo " average read is `echo " | ||
+ | | ||
+ | #echo $READ1 | ||
+ | #echo $READ2 | ||
+ | #echo $READ3 | ||
+ | } | ||
+ | |||
+ | echo | ||
+ | echo CURRENT SYSTEM SETTINGS | ||
+ | echo your current value of / | ||
+ | echo your current value of disk readahead is `blockdev --getra / | ||
+ | echo your current value of md readahead is `blockdev --getra / | ||
+ | DEVINDEX=0 | ||
+ | NUMDEVS=${# | ||
+ | until [ $DEVINDEX -ge $NUMDEVS ] | ||
+ | do | ||
+ | DEVLETTER=${DEVS: | ||
+ | echo your current value of / | ||
+ | DEVINDEX=$[$DEVINDEX+1] | ||
+ | done | ||
+ | echo | ||
+ | |||
+ | for i in 1 2 3 4 | ||
+ | #for i in 1 | ||
+ | # 1 when testing / | ||
+ | # 2 when testing disk readahead | ||
+ | # 3 when testing md readahead | ||
+ | # 4 when testing / | ||
+ | do | ||
+ | BEST_WRITE=0 | ||
+ | BEST_WRITE_ID=0 | ||
+ | WORST_WRITE=0 | ||
+ | WORST_WRITE_ID=0 | ||
+ | BEST_READ=0 | ||
+ | BEST_READ_ID=0 | ||
+ | WORST_READ=0 | ||
+ | WORST_READ_ID=0 | ||
+ | for j in 64 128 256 512 1024 2048 4096 8192 16384 | ||
+ | # for j in 64 16384 | ||
+ | do | ||
+ | #echo | ||
+ | #echo SYSTEM SETTINGS | ||
+ | #echo your current value of / | ||
+ | #echo your current value of disk readahead is `blockdev --getra / | ||
+ | #echo your current value of md readahead is `blockdev --getra / | ||
+ | #DEVINDEX=0 | ||
+ | # | ||
+ | #until [ $DEVINDEX -ge $NUMDEVS ] | ||
+ | #do | ||
+ | # DEVLETTER=${DEVS: | ||
+ | # echo your current value of / | ||
+ | # DEVINDEX=$[$DEVINDEX+1] | ||
+ | #done | ||
+ | #echo | ||
+ | case " | ||
+ | 1) echo "We are testing md stripe_cache_size" | ||
+ | echo $j > / | ||
+ | echo "step 1/4: NOW your current value of / | ||
+ | ;; | ||
+ | 2) echo "We are testing disks readahead" | ||
+ | blockdev --setra $j / | ||
+ | echo "step 2/4: NOW your current value of disk readahead is `blockdev --getra / | ||
+ | ;; | ||
+ | 3) echo "We are testing md readahead" | ||
+ | blockdev --setra $j /dev/$MDDEV | ||
+ | echo "step 3/4 NOW your current value of md readahead is `blockdev --getra / | ||
+ | ;; | ||
+ | 4) echo "We are testing disks max_sectors_kb" | ||
+ | DEVINDEX=0 | ||
+ | NUMDEVS=${# | ||
+ | until [ $DEVINDEX -ge $NUMDEVS ] | ||
+ | do | ||
+ | DEVLETTER=${DEVS: | ||
+ | echo $j > / | ||
+ | echo "step 4/4 NOW your current value of / | ||
+ | DEVINDEX=$[$DEVINDEX+1] | ||
+ | done | ||
+ | ;; | ||
+ | *) echo "This text should never appear" | ||
+ | echo ABORT | ||
+ | exit 1 | ||
+ | ;; | ||
+ | esac | ||
+ | rm $MNT/ | ||
+ | test_write | ||
+ | if [ " | ||
+ | then | ||
+ | #echo 1st test BEST_WRITE | ||
+ | BEST_WRITE=$AVG_WRITE | ||
+ | BEST_WRITE_ID=$j | ||
+ | fi | ||
+ | if [ " | ||
+ | then | ||
+ | #echo 1st test WORST_WRITE | ||
+ | WORST_WRITE=$AVG_WRITE | ||
+ | WORST_WRITE_ID=$j | ||
+ | fi | ||
+ | if [ " | ||
+ | then | ||
+ | echo "found new best write - old: `echo " | ||
+ | #echo "old: $BEST_WRITE new: $AVG_WRITE" | ||
+ | BEST_WRITE=$AVG_WRITE | ||
+ | BEST_WRITE_ID=$j | ||
+ | fi | ||
+ | if [ " | ||
+ | then | ||
+ | echo "found new worst write - old: `echo " | ||
+ | #echo old: $WORST_WRITE new: $AVG_WRITE | ||
+ | WORST_WRITE=$AVG_WRITE | ||
+ | WORST_WRITE_ID=$j | ||
+ | fi | ||
+ | test_read | ||
+ | if [ " | ||
+ | then | ||
+ | #echo 1st test BEST_READ | ||
+ | BEST_READ=$AVG_READ | ||
+ | BEST_READ_ID=$j | ||
+ | fi | ||
+ | if [ " | ||
+ | then | ||
+ | #echo 1st test WORST_READ | ||
+ | WORST_READ=$AVG_READ | ||
+ | WORST_READ_ID=$j | ||
+ | fi | ||
+ | if [ " | ||
+ | then | ||
+ | echo "found new best read - old: `echo " | ||
+ | #echo old: $BEST_READ new: $AVG_READ | ||
+ | BEST_READ=$AVG_READ | ||
+ | BEST_READ_ID=$j | ||
+ | fi | ||
+ | if [ " | ||
+ | then | ||
+ | echo "found new worst read - old: `echo " | ||
+ | #echo old: $WORST_READ new: $AVG_READ | ||
+ | WORST_READ=$AVG_READ | ||
+ | WORST_READ_ID=$j | ||
+ | fi | ||
+ | rm $MNT/ | ||
+ | rm $MNT/ | ||
+ | rm $MNT/ | ||
+ | |||
+ | done | ||
+ | echo BEST_WRITE is $BEST_WRITE | ||
+ | echo BEST_WRITE_ID is $BEST_WRITE_ID | ||
+ | echo WORST_WRITE is $WORST_WRITE | ||
+ | echo WORST_WRITE_ID is $WORST_WRITE_ID | ||
+ | echo BEST_READ is $BEST_READ | ||
+ | echo BEST_READ_ID is $BEST_READ_ID | ||
+ | echo WORST_READ is $WORST_READ | ||
+ | echo WORST_READ_ID is $WORST_READ_ID | ||
+ | # now we want to understand if this test affected more READ or WRITE performances | ||
+ | DIFF_WRITE=$[ BEST_WRITE - WORST_WRITE ] | ||
+ | DIFF_READ=$[ BEST_READ - WORST_READ ] | ||
+ | if [ " | ||
+ | then | ||
+ | echo this test affected more READ than WRITE | ||
+ | BEST_OVERALL_ID=$BEST_READ_ID | ||
+ | WORST_OVERALL_ID=$WORST_READ_ID | ||
+ | else | ||
+ | echo this test affected more WRITE than READ | ||
+ | BEST_OVERALL_ID=$BEST_WRITE_ID | ||
+ | WORST_OVERALL_ID=$WORST_WRITE_ID | ||
+ | fi | ||
+ | case " | ||
+ | 1) echo " | ||
+ | BEST_1_ID=$BEST_OVERALL_ID | ||
+ | echo $BEST_OVERALL_ID > / | ||
+ | ;; | ||
+ | 2) echo " | ||
+ | BEST_2_ID=$BEST_OVERALL_ID | ||
+ | blockdev --setra $BEST_OVERALL_ID / | ||
+ | ;; | ||
+ | 3) echo " | ||
+ | BEST_3_ID=$BEST_OVERALL_ID | ||
+ | blockdev --setra $BEST_OVERALL_ID /dev/$MDDEV | ||
+ | ;; | ||
+ | 4) echo " | ||
+ | BEST_4_ID=$BEST_OVERALL_ID | ||
+ | DEVINDEX=0 | ||
+ | NUMDEVS=${# | ||
+ | until [ $DEVINDEX -ge $NUMDEVS ] | ||
+ | do | ||
+ | DEVLETTER=${DEVS: | ||
+ | echo $BEST_OVERALL_ID > / | ||
+ | DEVINDEX=$[$DEVINDEX+1] | ||
+ | done | ||
+ | ;; | ||
+ | *) echo "This text should never appear" | ||
+ | echo ABORT | ||
+ | exit 1 | ||
+ | ;; | ||
+ | esac | ||
+ | done | ||
+ | |||
+ | echo the best for md stripe_cache_size is $BEST_1_ID | ||
+ | echo the best for disks readahead is $BEST_2_ID | ||
+ | echo the best for md readahead is $BEST_3_ID | ||
+ | echo the best for max_sectors_kb is $BEST_4_ID | ||
+ | echo | ||
+ | echo "Add the following lines to your / | ||
+ | echo | ||
+ | echo "echo $BEST_1_ID > / | ||
+ | echo " | ||
+ | echo " | ||
+ | DEVINDEX=0 | ||
+ | NUMDEVS=${# | ||
+ | until [ $DEVINDEX -ge $NUMDEVS ] | ||
+ | do | ||
+ | DEVLETTER=${DEVS: | ||
+ | echo "echo $BEST_4_ID > / | ||
+ | DEVINDEX=$[$DEVINDEX+1] | ||
+ | done | ||
+ | |||
+ | exit 0 | ||
+ | </ | ||
+ | |||
---- | ---- | ||
Line 149: | Line 537: | ||
echo *number* | sudo tee / | echo *number* | sudo tee / | ||
blockdev --setra *number* /dev/md0 | blockdev --setra *number* /dev/md0 | ||
+ | |||
+ | For example: | ||
+ | |||
+ | echo 8192 > / | ||
+ | echo 256 > / | ||
+ | echo 256 > / | ||
+ | echo 256 > / | ||
+ | echo 256 > / | ||
+ | blockdev --setra 64 / | ||
+ | blockdev --setra 16384 /dev/md0 | ||
</ | </ | ||
<WRAP info> | <WRAP info> | ||
- | **NOTE: | + | **NOTE: |
</ | </ | ||
Line 167: | Line 565: | ||
https:// | https:// | ||
+ |
nas/build_a_linux_nas/tune_the_system.1632611870.txt.gz · Last modified: 2021/09/25 23:17 by peter