llvm/clang/test/Analysis/zero-size-non-pod-array.cpp

// RUN: %clang_analyze_cc1 -analyzer-checker=core,debug.ExprInspection -std=c++11 -verify %s
// RUN: %clang_analyze_cc1 -analyzer-checker=core,debug.ExprInspection -std=c++17 -verify %s

void clang_analyzer_eval(bool);

struct S{
    static int CtorInvocationCount;
    static int DtorInvocationCount;

    S(){CtorInvocationCount++;}
    ~S(){DtorInvocationCount++;}
};

int S::CtorInvocationCount = 0;
int S::DtorInvocationCount = 0;

void zeroSizeArrayStack() {
    S::CtorInvocationCount = 0;

    S arr[0];

    clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}}
}

void zeroSizeMultidimensionalArrayStack() {
    S::CtorInvocationCount = 0;
    S::DtorInvocationCount = 0;

    {
        S arr[2][0];
        S arr2[0][2];

        S arr3[0][2][2];
        S arr4[2][2][0];
        S arr5[2][0][2];
    }

    clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}}
    clang_analyzer_eval(S::DtorInvocationCount == 0); //expected-warning{{TRUE}}
}

void zeroSizeArrayStackInLambda() {
    S::CtorInvocationCount = 0;
    S::DtorInvocationCount = 0;

    []{
        S arr[0];
    }();    
    
    clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}}
    clang_analyzer_eval(S::DtorInvocationCount == 0); //expected-warning{{TRUE}}
}

void zeroSizeArrayHeap() {
    S::CtorInvocationCount = 0;
    S::DtorInvocationCount = 0;

    auto *arr = new S[0];
    delete[] arr;
    
    clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}}
    clang_analyzer_eval(S::DtorInvocationCount == 0); //expected-warning{{TRUE}}
}

void zeroSizeMultidimensionalArrayHeap() {
    S::CtorInvocationCount = 0;
    S::DtorInvocationCount = 0;

    auto *arr = new S[2][0];
    delete[] arr;
        
    auto *arr2 = new S[0][2];
    delete[] arr2;

    auto *arr3 = new S[0][2][2];
    delete[] arr3;

    auto *arr4 = new S[2][2][0];
    delete[] arr4;

    auto *arr5 = new S[2][0][2];
    delete[] arr5;

    clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}}
    clang_analyzer_eval(S::DtorInvocationCount == 0); //expected-warning{{TRUE}}
}

#if __cplusplus >= 201703L

void zeroSizeArrayBinding() {
    S::CtorInvocationCount = 0;

    S arr[0];

    // Note: This is an error in gcc but a warning in clang.
    // In MSVC the declaration of 'S arr[0]' is already an error
    // and it doesn't recognize this syntax as a structured binding.
    auto [] = arr; //expected-warning{{ISO C++17 does not allow a decomposition group to be empty}}

    clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}}
}

#endif

void zeroSizeArrayLambdaCapture() {
    S::CtorInvocationCount = 0;
    S::DtorInvocationCount = 0;
        
    S arr[0];

    auto l = [arr]{};
    [arr]{}();    
    
    //FIXME: These should be TRUE. We should avoid calling the destructor 
    // of the temporary that is materialized as the lambda.
    clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}} expected-warning{{FALSE}}
    clang_analyzer_eval(S::DtorInvocationCount == 0); //expected-warning{{TRUE}} expected-warning{{FALSE}}
}

// FIXME: Report a warning if the standard is at least C++17.
#if __cplusplus < 201703L
void zeroSizeArrayLambdaCaptureUndefined1() {
    S arr[0];
    int n;

    auto l = [arr, n]{
        int x = n; //expected-warning{{Assigned value is garbage or undefined}}
        (void) x;
    };

    l();
}
#endif

void zeroSizeArrayLambdaCaptureUndefined2() {
    S arr[0];
    int n;

    [arr, n]{
        int x = n; //expected-warning{{Assigned value is garbage or undefined}}
        (void) x;
    }();
}

struct Wrapper{
    S arr[0];
};

void zeroSizeArrayMember() {
    S::CtorInvocationCount = 0;
    S::DtorInvocationCount = 0;

    {
        Wrapper W;
    }

    clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}}
    clang_analyzer_eval(S::DtorInvocationCount == 0); //expected-warning{{TRUE}}
}

void zeroSizeArrayMemberCopyMove() {
    S::CtorInvocationCount = 0;
    S::DtorInvocationCount = 0;
    
    {
        Wrapper W;
        Wrapper W2 = W;
        Wrapper W3 = (Wrapper&&) W2;
    }

    clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}}
    clang_analyzer_eval(S::DtorInvocationCount == 0); //expected-warning{{TRUE}}
}

struct MultiWrapper{
    S arr[2][0];
};

void zeroSizeMultidimensionalArrayMember() {
    S::CtorInvocationCount = 0;
    S::DtorInvocationCount = 0;

    {
        MultiWrapper MW;
    }

    clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}}
    clang_analyzer_eval(S::DtorInvocationCount == 0); //expected-warning{{TRUE}}
}