#!/bin/bash -e READLINK=readlink OS=$(uname -s) USER=$(whoami) if [ $OS = "Darwin" ];then READLINK=greadlink else echo "ERR: only run at macos env" exit -1 fi function err_env() { echo "check_env failed: pls refs ${SRC_DIR}/scripts/whl/BUILD_PYTHON_WHL_README.md" echo "try call ./scripts/whl/macos/macos_whl_env_prepare.sh to init env" exit -1 } function append_path_env_and_check() { export PATH=/usr/local/opt/findutils/libexec/gnubin:$PATH export PATH=/usr/local/opt/binutils/bin:$PATH export PATH=/usr/local/opt/llvm/bin:$PATH export PATH=/Users/${USER}/megengine_use_cmake/install/bin:$PATH if [ ! -f "/usr/local/opt/llvm/bin/llvm-strip" ]; then err_env fi which cmake if [ ! -f "/Users/${USER}/megengine_use_cmake/install/bin/cmake" ]; then err_env fi } append_path_env_and_check SRC_DIR=$($READLINK -f "`dirname $0`/../../../") source ${SRC_DIR}/scripts/whl/utils/utils.sh ALL_PYTHON=${ALL_PYTHON} FULL_PYTHON_VER="3.5.9 3.6.10 3.7.7 3.8.3" if [[ -z ${ALL_PYTHON} ]] then ALL_PYTHON=${FULL_PYTHON_VER} else check_python_version_is_valid "${ALL_PYTHON}" "${FULL_PYTHON_VER}" fi PYTHON_DIR= PYTHON_LIBRARY= PYTHON_INCLUDE_DIR= MACOS_WHL_HOME=${SRC_DIR}/scripts/whl/macos/macos_whl_home if [ -e "${MACOS_WHL_HOME}" ]; then echo "remove old macos whl file" rm -rf ${MACOS_WHL_HOME} fi mkdir -p ${MACOS_WHL_HOME} function config_python_env() { PYTHON_DIR=/Users/${USER}/.pyenv/versions/$1/ PYTHON_BIN=/Users/${USER}/.pyenv/versions/$1/bin if [ ! -f "$PYTHON_BIN/python3" ]; then echo "ERR: can not find $PYTHON_BIN , Invalid python package" echo "now support list: ${FULL_PYTHON_VER}" err_env else echo "put python3 to env..." export PATH=${PYTHON_BIN}:$PATH which python3 fi echo ${ver} if [ "$1" = "3.5.9" ]; then PYTHON_INCLUDE_DIR=${PYTHON_DIR}include/python3.5m PYTHON_LIBRARY=${PYTHON_DIR}/lib/libpython3.5m.dylib elif [ "$1" = "3.6.10" ]; then PYTHON_INCLUDE_DIR=${PYTHON_DIR}include/python3.6m PYTHON_LIBRARY=${PYTHON_DIR}/lib/libpython3.6m.dylib elif [ "$1" = "3.7.7" ]; then PYTHON_INCLUDE_DIR=${PYTHON_DIR}include/python3.7m PYTHON_LIBRARY=${PYTHON_DIR}/lib/libpython3.7m.dylib elif [ "$1" = "3.8.3" ]; then PYTHON_INCLUDE_DIR=${PYTHON_DIR}include/python3.8 PYTHON_LIBRARY=${PYTHON_DIR}/lib/libpython3.8.dylib else echo "ERR: DO NOT SUPPORT PYTHON VERSION" echo "now support list: ${FULL_PYTHON_VER}" exit -1 fi } MEGENGINE_LIB="${SRC_DIR}/build_dir/host/MGE_WITH_CUDA_OFF/MGE_INFERENCE_ONLY_OFF/Release/build/src/libmegengine_export.dylib" function depend_real_copy() { REAL_DST=$1 echo "real copy lib to $1" cp "${MEGENGINE_LIB}" ${REAL_DST} } BUILD_DIR=${SRC_DIR}/build_dir/host/MGE_WITH_CUDA_OFF/MGE_INFERENCE_ONLY_OFF/Release/build/ # here we just treat dnn/src/common/conv_bias.cpp should not in the increment build file list INCREMENT_KEY_WORDS="conv_bias.cpp.o is dirty" IS_IN_FIRST_LOOP=TRUE ORG_EXTRA_CMAKE_FLAG=${EXTRA_CMAKE_FLAG} function do_build() { for ver in ${ALL_PYTHON} do # we want run a full clean build at the first loop if [ ${IS_IN_FIRST_LOOP} = "TRUE" ]; then # TODO: may all cmake issue can be resolved after rm CMakeCache? # if YES, remove this to use old cache and speed up CI echo "warning: remove old build_dir for the first loop" rm -rf ${BUILD_DIR} fi #config config_python_env ${ver} #check env if [ ! -f "$PYTHON_LIBRARY" ]; then echo "ERR: can not find $PYTHON_LIBRARY , Invalid python package" err_env fi if [ ! -d "$PYTHON_INCLUDE_DIR" ]; then echo "ERR: can not find $PYTHON_INCLUDE_DIR , Invalid python package" err_env fi echo "PYTHON_LIBRARY: ${PYTHON_LIBRARY}" echo "PYTHON_INCLUDE_DIR: ${PYTHON_INCLUDE_DIR}" #config build type to RelWithDebInfo to enable MGB_ENABLE_DEBUG_UTIL etc export EXTRA_CMAKE_ARGS="${ORG_EXTRA_CMAKE_FLAG} -DCMAKE_BUILD_TYPE=RelWithDebInfo" #append cmake args for config python export EXTRA_CMAKE_ARGS="${EXTRA_CMAKE_ARGS} -DPYTHON_EXECUTABLE=${PYTHON_DIR}/bin/python3" export EXTRA_CMAKE_ARGS="${EXTRA_CMAKE_ARGS} -DPYTHON_LIBRARY=${PYTHON_LIBRARY}" export EXTRA_CMAKE_ARGS="${EXTRA_CMAKE_ARGS} -DPYTHON_INCLUDE_DIR=${PYTHON_INCLUDE_DIR}" #we use std::visit in src, so set osx version minimum to 10.14, but 10.14 have objdump #issue, so we now config to 10.15, whl name to 10.14 #TODO: can set to 10.12 after remove use std::visit export EXTRA_CMAKE_ARGS="${EXTRA_CMAKE_ARGS} -DCMAKE_OSX_DEPLOYMENT_TARGET=10.15 " if [ -d "${BUILD_DIR}" ]; then # insure rm have args touch ${BUILD_DIR}/empty.so touch ${BUILD_DIR}/CMakeCache.txt find ${BUILD_DIR} -name "*.so" | xargs rm # as we now use increment build mode when switch python # But I do not known any more issue at CMakeLists.txt or not # so Force remove CMakeCache.txt find ${BUILD_DIR} -name CMakeCache.txt | xargs rm fi HOST_BUILD_ARGS="-t -s" # call ninja dry run and check increment is invalid or not if [ ${IS_IN_FIRST_LOOP} = "FALSE" ]; then ninja_dry_run_and_check_increment "${SRC_DIR}/scripts/cmake-build/host_build.sh" "${HOST_BUILD_ARGS}" "${INCREMENT_KEY_WORDS}" fi # call real build echo "host_build.sh HOST_BUILD_ARGS: ${HOST_BUILD_ARGS}" ${SRC_DIR}/scripts/cmake-build/host_build.sh ${HOST_BUILD_ARGS} # check python api call setup.py cd ${BUILD_DIR} check_build_ninja_python_api ${ver} rm -rf staging mkdir -p staging cp -a imperative/python/{megengine,setup.py,requires.txt,requires-style.txt,requires-test.txt} staging/ cd ${BUILD_DIR}/staging/megengine/core rt_file=`ls _imperative_rt.*.so` echo "rt file is: ${rt_file}" if [[ -z ${rt_file} ]] then echo "ERR: can not find valid rt file" exit -1 fi llvm-strip -s ${rt_file} mv ${rt_file} _imperative_rt.so echo "check so valid or not..." otool_out=`otool -L _imperative_rt.so` if [[ "${otool_out}" =~ "ython" ]]; then echo "ERR: invalid _imperative_rt.so which depend on python lib, detail: log" echo ${otool_out} exit -1 else echo "valid..." fi #handle dlopen path install_name_tool -change @rpath/libmegengine_export.dylib @loader_path/lib/libmegengine_export.dylib _imperative_rt.so #copy megbrain_export lib DEPEND_LIB=${BUILD_DIR}/staging/megengine/core/lib/ rm -rf ${DEPEND_LIB} mkdir ${DEPEND_LIB} depend_real_copy ${DEPEND_LIB} # handle megenginelite cd ${BUILD_DIR} mkdir -p staging/megenginelite cp ${SRC_DIR}/lite/pylite/megenginelite/* staging/megenginelite/ mkdir -p ${BUILD_DIR}/staging/megenginelite/libs LITE_LIB=${BUILD_DIR}/staging/megenginelite/libs/liblite_shared_whl.so cp ${SRC_DIR}/build_dir/host/MGE_WITH_CUDA_OFF/MGE_INFERENCE_ONLY_OFF/Release/build/lite/liblite_shared_whl.dylib ${LITE_LIB} llvm-strip -s ${LITE_LIB} #handle dlopen path install_name_tool -change @rpath/libmegengine_export.dylib @loader_path/../../megengine/core/lib/libmegengine_export.dylib ${LITE_LIB} cd ${BUILD_DIR}/staging ${PYTHON_DIR}/bin/python3 setup.py bdist_wheel cd ${BUILD_DIR}/staging/dist/ org_whl_name=`ls Meg*.whl` index=`awk -v a="${org_whl_name}" -v b="-macosx" 'BEGIN{print index(a,b)}'` compat_whl_name=`echo ${org_whl_name} |cut -b -$index`macosx_10_14_x86_64.whl echo "org whl name: ${org_whl_name}" echo "comapt whl name: ${compat_whl_name}" cp ${BUILD_DIR}/staging/dist/Meg*.whl ${MACOS_WHL_HOME}/${compat_whl_name} cd ${SRC_DIR} echo "" echo "##############################################################################################" echo "macos whl package location: ${MACOS_WHL_HOME}" ls ${MACOS_WHL_HOME} echo "##############################################################################################" IS_IN_FIRST_LOOP=FALSE done } function third_party_prepare() { echo "init third_party..." ${SRC_DIR}/third_party/prepare.sh if [[ -z ${ALREADY_INSTALL_MKL} ]] then echo "init third_party..." ${SRC_DIR}/third_party/install-mkl.sh else echo "skip init mkl internal" fi } ###################### third_party_prepare do_build