#!/usr/bin/env groovy // Copyright (c) 2017 - 2021 Advanced Micro Devices, Inc. All Rights Reserved. // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. // Generated from snippet generator 'properties; set job properties' properties([buildDiscarder(logRotator( artifactDaysToKeepStr: '', artifactNumToKeepStr: '', daysToKeepStr: '', numToKeepStr: '10')), disableConcurrentBuilds(), parameters([booleanParam( name: 'push_image_to_docker_hub', defaultValue: false, description: 'Push hip & hcc image to rocm docker-hub' )]), [$class: 'CopyArtifactPermissionProperty', projectNames: '*'] ]) //////////////////////////////////////////////////////////////////////// // -- AUXILLARY HELPER FUNCTIONS //////////////////////////////////////////////////////////////////////// // Return build number of upstream job @NonCPS int get_upstream_build_num( ) { def upstream_cause = currentBuild.rawBuild.getCause( hudson.model.Cause$UpstreamCause ) if( upstream_cause == null) return 0 return upstream_cause.getUpstreamBuild() } //////////////////////////////////////////////////////////////////////// // Return project name of upstream job @NonCPS String get_upstream_build_project( ) { def upstream_cause = currentBuild.rawBuild.getCause( hudson.model.Cause$UpstreamCause ) if( upstream_cause == null) return null return upstream_cause.getUpstreamProject() } //////////////////////////////////////////////////////////////////////// // Construct the docker build image name String docker_build_image_name( ) { return "build-ubuntu-16.04" } //////////////////////////////////////////////////////////////////////// // Construct the relative path of the build directory String build_directory_rel( String build_config ) { if( build_config.equalsIgnoreCase( 'release' ) ) { return "build/release" } else { return "build/debug" } } //////////////////////////////////////////////////////////////////////// // Lots of images are created above; no apparent way to delete images:tags with docker global variable def docker_clean_images( String org, String image_name ) { // Check if any images exist first grepping for image names int docker_images = sh( script: "docker images | grep \"${org}/${image_name}\"", returnStatus: true ) // The script returns a 0 for success (images were found ) if( docker_images == 0 ) { // Deleting images can fail, if other projects have built on top of that image and are now dependent on it. // This should not be treated as a hip build failure. This requires cleanup at a later time, possibly through // another job try { // Best attempt to run bash script to clean images // deleting images based on hash seems to be more stable than through name:tag values because of tags sh "docker images | grep \"${org}/${image_name}\" | awk '{print \$1 \":\" \$2}' | xargs docker rmi" } catch( err ) { println 'Failed to cleanup a few images; probably the images are used as a base for other images' currentBuild.result = 'SUCCESS' } } } //////////////////////////////////////////////////////////////////////// // -- BUILD RELATED FUNCTIONS //////////////////////////////////////////////////////////////////////// // Checkout source code, source dependencies and update version number numbers // Returns a relative path to the directory where the source exists in the workspace String checkout_and_version( String platform ) { String source_dir_rel = "src" String source_hip_rel = "${source_dir_rel}/hip" stage("${platform} clone") { dir( "${source_hip_rel}" ) { // checkout hip checkout([ $class: 'GitSCM', branches: scm.branches, doGenerateSubmoduleConfigurations: scm.doGenerateSubmoduleConfigurations, extensions: scm.extensions + [[$class: 'CleanCheckout']], userRemoteConfigs: scm.userRemoteConfigs ]) } } return source_hip_rel } //////////////////////////////////////////////////////////////////////// // This creates the docker image that we use to build the project in // The docker images contains all dependencies, including OS platform, to build def docker_build_image( String platform, String org, String optional_build_parm, String source_hip_rel, String from_image ) { String build_image_name = docker_build_image_name( ) String dockerfile_name = "dockerfile-build-ubuntu-16.04" def build_image = null stage("${platform} build image") { dir("${source_hip_rel}") { def user_uid = sh( script: 'id -u', returnStdout: true ).trim() // Docker 17.05 introduced the ability to use ARG values in FROM statements // Docker inspect failing on FROM statements with ARG https://issues.jenkins-ci.org/browse/JENKINS-44836 // build_image = docker.build( "${org}/${build_image_name}:latest", "--pull -f docker/${dockerfile_name} --build-arg user_uid=${user_uid} --build-arg base_image=${from_image} ." ) // JENKINS-44836 workaround by using a bash script instead of docker.build() sh "docker build -t ${org}/${build_image_name}:latest -f docker/${dockerfile_name} ${optional_build_parm} --build-arg user_uid=${user_uid} --build-arg base_image=${from_image} ." build_image = docker.image( "${org}/${build_image_name}:latest" ) } } return build_image } //////////////////////////////////////////////////////////////////////// // This encapsulates the cmake configure, build and package commands // Leverages docker containers to encapsulate the build in a fixed environment def docker_build_inside_image( def build_image, String inside_args, String platform, String optional_configure, String build_config, String source_hip_rel, String build_dir_rel ) { String source_hip_abs = pwd() + "/" + source_hip_rel build_image.inside( inside_args ) { stage("${platform} make ${build_config}") { // The rm command needs to run as sudo because the test steps below create files owned by root sh """#!/usr/bin/env bash set -x rm -rf ${build_dir_rel} mkdir -p ${build_dir_rel} cd ${build_dir_rel} cmake -DCMAKE_BUILD_TYPE=${build_config} -DCMAKE_INSTALL_PREFIX=staging ${optional_configure} ${source_hip_abs} make -j\$(nproc) """ } // Cap the maximum amount of testing, in case of hangs // Excluding hipMultiThreadDevice-pyramid & hipMemoryAllocateCoherentDriver tests from automation; due to its flakiness which requires some investigation timeout(time: 1, unit: 'HOURS') { stage("${platform} unit testing") { sh """#!/usr/bin/env bash set -x cd ${build_dir_rel} make install -j\$(nproc) make build_tests -i -j\$(nproc) ctest --output-on-failure -E "(hipMultiThreadDevice-pyramid|hipMemoryAllocateCoherentDriver)" """ // If unit tests output a junit or xunit file in the future, jenkins can parse that file // to display test results on the dashboard // junit "${build_dir_rel}/*.xml" } } // Only create packages from hcc based builds if( platform.toLowerCase( ).startsWith( 'rocm-' ) ) { stage("${platform} packaging") { sh """#!/usr/bin/env bash set -x cd ${build_dir_rel} make package """ // No matter the base platform, all packages have the same name // Only upload 1 set of packages, so we don't have a race condition uploading packages if( platform.toLowerCase( ).startsWith( 'rocm-head' ) ) { archiveArtifacts artifacts: "${build_dir_rel}/*.deb", fingerprint: true archiveArtifacts artifacts: "${build_dir_rel}/*.rpm", fingerprint: true } } } } return void } //////////////////////////////////////////////////////////////////////// // This builds a fresh docker image FROM a clean base image, with no build dependencies included // Uploads the new docker image to internal artifactory String docker_upload_artifactory( String hcc_ver, String artifactory_org, String from_image, String source_hip_rel, String build_dir_rel ) { def hip_install_image = null String image_name = "hip-${hcc_ver}-ubuntu-16.04" stage( 'artifactory' ) { println "artifactory_org: ${artifactory_org}" // We copy the docker files into the bin directory where the .deb lives so that it's a clean build everytime sh "cp -r ${source_hip_rel}/docker/* ${build_dir_rel}" // Docker 17.05 introduced the ability to use ARG values in FROM statements // Docker inspect failing on FROM statements with ARG https://issues.jenkins-ci.org/browse/JENKINS-44836 // hip_install_image = docker.build( "${artifactory_org}/${image_name}:${env.BUILD_NUMBER}", "--pull -f ${build_dir_rel}/dockerfile-hip-ubuntu-16.04 --build-arg base_image=${from_image} ${build_dir_rel}" ) // JENKINS-44836 workaround by using a bash script instead of docker.build() sh "docker build -t ${artifactory_org}/${image_name} --pull -f ${build_dir_rel}/dockerfile-hip-ubuntu-16.04 --build-arg base_image=${from_image} ${build_dir_rel}" hip_install_image = docker.image( "${artifactory_org}/${image_name}" ) // The connection to artifactory can fail sometimes, but this should not be treated as a build fail try { // Don't push pull requests to artifactory, these tend to accumulate over time if( env.BRANCH_NAME.toLowerCase( ).startsWith( 'pr-' ) ) { println 'Pull Request (PR-xxx) detected; NOT pushing to artifactory' } else { docker.withRegistry('http://compute-artifactory:5001', 'artifactory-cred' ) { hip_install_image.push( "${env.BUILD_NUMBER}" ) hip_install_image.push( 'latest' ) } } } catch( err ) { currentBuild.result = 'SUCCESS' } } return image_name } //////////////////////////////////////////////////////////////////////// // Uploads the new docker image to the public docker-hub def docker_upload_dockerhub( String local_org, String image_name, String remote_org ) { stage( 'docker-hub' ) { // Do not treat failures to push to docker-hub as a build fail try { sh """#!/usr/bin/env bash set -x echo inside sh docker tag ${local_org}/${image_name} ${remote_org}/${image_name} """ docker_hub_image = docker.image( "${remote_org}/${image_name}" ) docker.withRegistry('https://registry.hub.docker.com', 'docker-hub-cred' ) { docker_hub_image.push( "${env.BUILD_NUMBER}" ) docker_hub_image.push( 'latest' ) } } catch( err ) { currentBuild.result = 'SUCCESS' } } } //////////////////////////////////////////////////////////////////////// // -- MAIN // Following this line is the start of MAIN of this Jenkinsfile String build_config = 'Release' String job_name = env.JOB_NAME.toLowerCase( ) // The following launches 3 builds in parallel: rocm-head, rocm-3.3.x and cuda-10.x parallel rocm_3_3: { node('hip-rocm') { String hcc_ver = 'rocm-3.3.x' String from_image = 'ci_test_nodes/rocm-3.3.x/ubuntu-16.04:latest' String inside_args = '--device=/dev/kfd --device=/dev/dri --group-add=video' // Checkout source code, dependencies and version files String source_hip_rel = checkout_and_version( hcc_ver ) // Create/reuse a docker image that represents the hip build environment def hip_build_image = docker_build_image( hcc_ver, 'hip', '', source_hip_rel, from_image ) // Print system information for the log hip_build_image.inside( inside_args ) { sh """#!/usr/bin/env bash set -x /opt/rocm/bin/rocm_agent_enumerator -t ALL /opt/rocm/bin/hcc --version """ } // Conctruct a binary directory path based on build config String build_hip_rel = build_directory_rel( build_config ); // Build hip inside of the build environment docker_build_inside_image( hip_build_image, inside_args, hcc_ver, '', build_config, source_hip_rel, build_hip_rel ) // Clean docker build image docker_clean_images( 'hip', docker_build_image_name( ) ) // After a successful build, upload a docker image of the results /* String hip_image_name = docker_upload_artifactory( hcc_ver, job_name, from_image, source_hip_rel, build_hip_rel ) if( params.push_image_to_docker_hub ) { docker_upload_dockerhub( job_name, hip_image_name, 'rocm' ) docker_clean_images( 'rocm', hip_image_name ) } docker_clean_images( job_name, hip_image_name ) */ } }, rocm_head: { node('hip-rocm') { String hcc_ver = 'rocm-head' String from_image = 'ci_test_nodes/rocm-head/ubuntu-16.04:latest' String inside_args = '--device=/dev/kfd --device=/dev/dri --group-add=video' // Checkout source code, dependencies and version files String source_hip_rel = checkout_and_version( hcc_ver ) // Create/reuse a docker image that represents the hip build environment def hip_build_image = docker_build_image( hcc_ver, 'hip', '', source_hip_rel, from_image ) // Print system information for the log hip_build_image.inside( inside_args ) { sh """#!/usr/bin/env bash set -x /opt/rocm/bin/rocm_agent_enumerator -t ALL /opt/rocm/bin/hcc --version """ } // Conctruct a binary directory path based on build config String build_hip_rel = build_directory_rel( build_config ); // Build hip inside of the build environment docker_build_inside_image( hip_build_image, inside_args, hcc_ver, '', build_config, source_hip_rel, build_hip_rel ) // Clean docker image docker_clean_images( 'hip', docker_build_image_name( ) ) // After a successful build, upload a docker image of the results /* String hip_image_name = docker_upload_artifactory( hcc_ver, job_name, from_image, source_hip_rel, build_hip_rel ) if( params.push_image_to_docker_hub ) { docker_upload_dockerhub( job_name, hip_image_name, 'rocm' ) docker_clean_images( 'rocm', hip_image_name ) } docker_clean_images( job_name, hip_image_name ) */ } }, cuda_10_x: { node('hip-cuda') { //////////////////////////////////////////////////////////////////////// // Block of string constants customizing behavior for cuda String nvcc_ver = 'cuda-10.x' String from_image = 'ci_test_nodes/cuda-10.x/ubuntu-16.04:latest' String inside_args = '--gpus all'; // Checkout source code, dependencies and version files String source_hip_rel = checkout_and_version( nvcc_ver ) // Create/reuse a docker image that represents the hip build environment def hip_build_image = docker_build_image( nvcc_ver, 'hip', '', source_hip_rel, from_image ) // Print system information for the log hip_build_image.inside( inside_args ) { sh """#!/usr/bin/env bash set -x nvidia-smi nvcc --version """ } // Conctruct a binary directory path based on build config String build_hip_rel = build_directory_rel( build_config ); // Build hip inside of the build environment docker_build_inside_image( hip_build_image, inside_args, nvcc_ver, "-DHIP_NVCC_FLAGS=--Wno-deprecated-gpu-targets", build_config, source_hip_rel, build_hip_rel ) // Clean docker image docker_clean_images( 'hip', docker_build_image_name( ) ) } }