From 1a24d2d223d22009239453d65e882e1ed747e406 Mon Sep 17 00:00:00 2001 From: Paul O'Reilly Date: Thu, 26 Sep 2024 13:28:05 +0100 Subject: [PATCH] SONARJAVA-4287 S3012 has a false positive when using either auto boxing or auto unboxing (#4882) --- .../java/checks/ArrayCopyLoopCheckSample.java | 106 +++++++++++++++++- .../sonar/java/checks/ArrayCopyLoopCheck.java | 5 + 2 files changed, 110 insertions(+), 1 deletion(-) diff --git a/java-checks-test-sources/default/src/main/java/checks/ArrayCopyLoopCheckSample.java b/java-checks-test-sources/default/src/main/java/checks/ArrayCopyLoopCheckSample.java index b7a8678c03a..b02f1dea34c 100644 --- a/java-checks-test-sources/default/src/main/java/checks/ArrayCopyLoopCheckSample.java +++ b/java-checks-test-sources/default/src/main/java/checks/ArrayCopyLoopCheckSample.java @@ -7,6 +7,110 @@ abstract class ArrayCopyLoopCheckSample implements Collection { + // test where src and dst are not primitives but compatible + public void coverageEdgeCase(Long[] arrWE) { + Number[] result = new Number[arrWE.length]; + for (int i = 0; i < arrWE.length; i++) { + result[i] = arrWE[i]; // Noncompliant {{Use "Arrays.copyOf", "Arrays.asList", "Collections.addAll" or "System.arraycopy" instead.}} + } + } + + public Integer[] classToClass(Long[] arrWE) { + Integer[] result = new Integer[arrWE.length]; + for (int i = 0; i < arrWE.length; i++) { + result[i] = /* using auto boxing */ Integer.parseInt(String.valueOf(arrWE[i])); // Compliant + } + return result; + } + + public Long[] classToClassLong(Integer[] arr) { + Long[] result = new Long[arr.length]; + for (int i = 0; i < arr.length; i++) { + result[i] = /* using auto boxing */ Long.parseLong(String.valueOf(arr[i])); // Compliant + } + return result; + } + + public Integer[] boxed(int[] arr) { + Integer[] result = new Integer[arr.length]; + for (int i = 0; i < arr.length; i++) { + result[i] = /* using auto boxing */ arr[i]; // Compliant + } + return result; + } + + public int[] unboxed(Integer[] arr) { + int[] result = new int[arr.length]; + for (int i = 0; i < arr.length; i++) { + result[i] = /* using auto unboxing */ arr[i]; // Compliant + } + return result; + } + + public Long[] boxed(long[] arr) { + Long[] result = new Long[arr.length]; + for (int i = 0; i < arr.length; i++) { + result[i] = /* using auto boxing */ arr[i]; // Compliant + } + return result; + } + + public long[] unboxed(Long[] arr) { + long[] result = new long[arr.length]; + for (int i = 0; i < arr.length; i++) { + result[i] = /* using auto unboxing */ arr[i]; // Compliant + } + return result; + } + + public Float[] boxed(float[] arr) { + Float[] result = new Float[arr.length]; + for (int i = 0; i < arr.length; i++) { + result[i] = /* using auto boxing */ arr[i]; // Compliant + } + return result; + } + + public float[] unboxed(Float[] arr) { + float[] result = new float[arr.length]; + for (int i = 0; i < arr.length; i++) { + result[i] = /* using auto unboxing */ arr[i]; // Compliant + } + return result; + } + + public Double[] boxed(double[] arr) { + Double[] result = new Double[arr.length]; + for (int i = 0; i < arr.length; i++) { + result[i] = /* using auto boxing */ arr[i]; // Compliant + } + return result; + } + + public double[] unboxed(Double[] arr) { + double[] result = new double[arr.length]; + for (int i = 0; i < arr.length; i++) { + result[i] = /* using auto unboxing */ arr[i]; // Compliant + } + return result; + } + + public Character[] boxed(char[] arr) { + Character[] result = new Character[arr.length]; + for (int i = 0; i < arr.length; i++) { + result[i] = /* using auto boxing */ arr[i]; // Compliant + } + return result; + } + + public char[] unboxed(Character[] arr) { + char[] result = new char[arr.length]; + for (int i = 0; i < arr.length; i++) { + result[i] = /* using auto unboxing */ arr[i]; // Compliant + } + return result; + } + int x; void f() throws InterruptedException { @@ -316,7 +420,7 @@ void f() throws InterruptedException { list.add(src[i++]); System.out.println(i); } - + while (c < b) { list.add(src[from]); ++from; diff --git a/java-checks/src/main/java/org/sonar/java/checks/ArrayCopyLoopCheck.java b/java-checks/src/main/java/org/sonar/java/checks/ArrayCopyLoopCheck.java index 6d3170100cf..46ba46f5fe4 100644 --- a/java-checks/src/main/java/org/sonar/java/checks/ArrayCopyLoopCheck.java +++ b/java-checks/src/main/java/org/sonar/java/checks/ArrayCopyLoopCheck.java @@ -234,6 +234,11 @@ private static boolean isArrayToArrayCopy(ExpressionTree expression, Symbol coun if (lhs.is(Kind.ARRAY_ACCESS_EXPRESSION) && rhs.is(Kind.ARRAY_ACCESS_EXPRESSION)) { ArrayAccessExpressionTree src = (ArrayAccessExpressionTree) rhs; ArrayAccessExpressionTree dst = (ArrayAccessExpressionTree) lhs; + // check they are not boxing or unboxing; this can happen if we have different types and one side is primitive + if (!src.symbolType().equals(dst.symbolType()) && (src.symbolType().isPrimitive() || dst.symbolType().isPrimitive())) { + return false; + } + // otherwise determine the expressions return isCounter(src.dimension().expression(), counter) && isCounter(dst.dimension().expression(), counter); } }