Skip to content

Latest commit

 

History

History
184 lines (146 loc) · 6.64 KB

README.md

File metadata and controls

184 lines (146 loc) · 6.64 KB

countbits

exploring the speed of varying languages at counting "on" bits in integers, and other fun comparisons

Runs ec2 instances for each benchmark test and outputs results to s3

This is based on an interview question I encountered where the interviewer asked me how I would count "on" bits in an integer, given repeated unpredictable requests (i.e. a service) and I suggested a lookup table, and he scoffed that this would be even remotely feasible. Told me that even if it could fit (uhh, 32 bits is 4 gb?) it would take too long to generate. (He wanted me to propose a caching scheme. I did not pursue the job.)

So, let's see how long it does take to generate ;)

Stats

Countbits, average per 10mm:

  • Go 1.15.2 ~0.5s
  • C ~1.0s
  • C# dotnet core 3.1.402 ~1.7s
  • Javascript / Node 14 (Naive write) OOM ~5.4s
  • Javascript / Node 14 (Readable -> pipe) ~7.7s
  • Java 11 (AdoptOpenJDK 11.0.8+10) (Classes) ~18s
  • Java 11 (AdoptOpenJDK 11.0.8+10) (Optimized primitives) ~18s
  • JRuby 9.2.13.0 ~23s
  • PHP 7.4.3 ~27s
  • Ruby 2.7.1 ~28s
  • Python 3.8.2 ~101s
  • Rust 1.46.0 ~?s (needs entire implementation)

Fibonacci 42:

  • C ~1.2s
  • Rust 1.46.0 ~1.7s
  • Java 11 (AdoptOpenJDK 11.0.8+10) ~1.8s
  • C# dotnet core 3.1.402 ~2.7s
  • Javascript / Node 14 ~4.5s
  • Go 1.15.2 ~4.9s
  • JRuby 9.2.13.0 ~22s
  • PHP 7.4.3 ~23s
  • Ruby 2.7.1 ~36s
  • Python 3.8.2 ~109s

Significant changes relative to the past test:

  • Go got much slower on both countbits and fibonacci -- until some improvements came in for countbits to speed it up a lot!
  • (the go improvements make me wonder how much other languages could be trivially optimized as well)
  • Python got faster (slightly) -- but still a half dead slug
  • C# got 5x faster on fibonacci (previous result was very different vs other langues compared to countbits relative perf)

Running

After one time setup of your ec2 account as per instructions at the bottom (role, policy, bucket, security group, keypair, etc)

First set your PROFILE and BUCKET in run.sh and clean.sh ala:

AWS_PROFILE=yourAWSprofile
AWS_BUCKET=yourAWSs3bucket
sed -i "s|PROFILE=...|PROFILE=$AWS_PROFILE|" run.sh
sed -i "s|BUCKET=...|BUCKET=$AWS_BUCKET|" run.sh
sed -i "s|PROFILE=...|PROFILE=$AWS_PROFILE|" clean.sh
sed -i "s|BUCKET=...|BUCKET=$AWS_BUCKET|" clean.sh

Run the perf benchmarks with . run.sh countbits or . run.sh fibonacci 42 (you can run more than one at a time).

Then download the results with . clean.sh (which also removes the results from your s3 bucket -- note that it does not stop running ec2 instances)

Debugging

Some debugging and lookup tools you might want afterwards:

# http://docs.aws.amazon.com/AWSEC2/latest/UserGuide/AccessingInstancesLinux.html
ssh -i ~/.ssh/$KEYPAIR.pem ubuntu@$DOMAIN

tail -f /var/log/cloud-init-output.log

sudo su
cat /root/countbits/fibonacci/csharp/fibonacci_*_out.txt


aws s3 ls --profile $PROFILE $BUCKET/$TYPE/
aws s3 cp --profile $PROFILE s3://$BUCKET/$TYPE/output_DATETIME.json ./

# https://stedolan.github.io/jq/tutorial/
sudo apt-get install jq
cat output_DATETIME.json | jq


aws ec2 describe-instances --profile $PROFILE --region $REGION --query 'Reservations[*].Instances[*].[LaunchTime, InstanceId, State.Name, Tags[0].Value]' --output text

aws ec2 terminate-instances --profile $PROFILE --region $REGION --instance-ids $INSTANCEID

Initial configuration

Get all the necessary AWS stuff setup, permissions, etc:

# http://docs.aws.amazon.com/AWSEC2/latest/UserGuide/iam-roles-for-amazon-ec2.html
cat << EOF > ec2-role-trust-policy.json
{
  "Version": "2012-10-17",
  "Statement": [
    {
      "Effect": "Allow",
      "Principal": { "Service": "ec2.amazonaws.com"},
      "Action": "sts:AssumeRole"
    }
  ]
}
EOF

aws iam create-role \
    --profile $PROFILE \
    --region $REGION \
    --role-name s3-put-role \
    --description "Write access to S3 $BUCKET" \
    --assume-role-policy-document file://ec2-role-trust-policy.json

# https://aws.amazon.com/blogs/security/a-safer-way-to-distribute-aws-credentials-to-ec2/
# http://docs.aws.amazon.com/IAM/latest/UserGuide/reference_policies_examples_s3_deny-except-bucket.html
# http://docs.aws.amazon.com/IAM/latest/UserGuide/reference_policies_examples_s3_rw-bucket-console.html
cat << EOF > s3-put-policy.json
{
  "Version": "2012-10-17",
  "Statement": [
    {
      "Effect": "Allow",
      "Action": [
        "s3:PutObject"
      ],
      "Resource": [
        "arn:aws:s3:::$BUCKET/*"
      ]
    }
  ]
}
EOF

aws iam put-role-policy \
    --profile $PROFILE \
    --region $REGION \
    --role-name s3-put-role \
    --policy-name s3-put-policy \
    --policy-document file://s3-put-policy.json

aws iam create-instance-profile --profile $PROFILE --region $REGION --instance-profile-name s3-put-profile

aws iam add-role-to-instance-profile --profile $PROFILE --region $REGION --instance-profile-name s3-put-profile --role-name s3-put-role


# http://docs.aws.amazon.com/cli/latest/reference/s3api/create-bucket.html
aws s3api list-buckets --profile $PROFILE --region $REGION
aws s3api create-bucket --profile $PROFILE --region $REGION --bucket $BUCKET --acl private

# http://docs.aws.amazon.com/cli/latest/userguide/cli-ec2-keypairs.html
# http://docs.aws.amazon.com/AWSEC2/latest/UserGuide/ec2-key-pairs.html#how-to-generate-your-own-key-and-import-it-to-aws
KEYPAIR=ec2-keypair
REGION=us-east-1
aws ec2 delete-key-pair --key-name $KEYPAIR --region $REGION
aws ec2 import-key-pair --key-name $KEYPAIR --public-key-material fileb://~/.ssh/id_rsa.pub --region $REGION
aws ec2 describe-key-pairs --profile $PROFILE --region $REGION
#aws ec2 create-key-pair --profile $PROFILE --key-name $KEYPAIR --query 'KeyMaterial' --output text > ~/.ssh/$KEYPAIR.pem
#chmod 400 ~/.ssh/$KEYPAIR.pem

# http://skillslane.com/aws-tutorial-vpc-launch-instance-cli/
# http://docs.aws.amazon.com/AmazonVPC/latest/UserGuide/vpc-subnets-commands-example.html
# https://console.aws.amazon.com/vpc/home?region=$REGION
VPCID=`aws ec2 describe-vpcs --profile $PROFILE --region $REGION --query 'Vpcs[*].VpcId' --output text`
echo $VPCID

# http://docs.aws.amazon.com/cli/latest/userguide/cli-ec2-sg.html
# http://docs.aws.amazon.com/AWSEC2/latest/UserGuide/using-network-security.html
# http://docs.aws.amazon.com/AmazonVPC/latest/UserGuide/VPC_SecurityGroups.html#VPC_Security_Group_Differences
aws ec2 describe-security-groups --profile $PROFILE --region $REGION
SGID=$(aws ec2 create-security-group \
    --profile $PROFILE \
    --region $REGION \
    --vpc-id $VPCID \
    --group-name ssh-anywhere \
    --description "SSH from anywhere"
    --query 'GroupId'
    --output text)
aws ec2 authorize-security-group-ingress --profile $PROFILE --region $REGION --group-id $SGID --protocol tcp --port 22 --cidr 0.0.0.0/0