chapel

error: field "val" used before it is initialized


I was trying to start doing a record for a composite type (vector of ints) but keep getting hung up back on something that must be basic that I'm missing. I've boiled it down to the following which seems like it should be valid.

Thinking like C++ programmer I'm not sure I appreciate the significance of "init"s vs. constructors and assignments. If I do chapel with those, it eventually complains about missing "init=" which takes me down the whole "init" path.

[u2:Chapel] chpl --version
warning: The prototype GPU support implies --no-checks. This may impact debuggability. To suppress this warning, compile with --no-checks explicitly
chpl version 2.3.0
  built with LLVM version 19.1.3
  available LLVM targets: x86-64, x86, nvptx64, nvptx
Copyright 2020-2024 Hewlett Packard Enterprise Development LP
Copyright 2004-2019 Cray Inc.
(See LICENSE file for more details)

[ugpu2:Chapel] cat bug.init.chpl
config const N : int = 4;
type T = uint(32);

record MP {

  var val : [0..#N] T;

  proc init() {
    for i in 0..#N do this.val[i] = 0;
    //this.val = 0;      // does work instead
  }
};

var A : MP;

[u2:Chapel] chpl bug.init.chpl
warning: The prototype GPU support implies --no-checks. This may impact debuggability. To suppress this warning, compile with --no-checks explicitly
bug.init.chpl:9: In initializer:
bug.init.chpl:10: error: field "val" used before it is initialized

Solution

  • @justapony: Chapel's initializers are somewhat akin to C++'s constructors, but because Chapel is designed to promote type safety, its rules are defined to ensure that all fields of a class or record are initialized prior to that field, or the object as a whole, being used. This results in it being somewhat more particular about what appears in the initializer and where. Where C++ uses parenthesized expressions in the lead-in to the constructor's body to initialize fields, Chapel uses the initial statements within the initializer's body to serve this purpose.

    Specifically, Chapel currently requires that the object's fields must be initialized in their declared order using an initialization of the form [this.]field = initExpr;, which is why this.val = 0; (or simply val = 0; or even this.val = for i in 0..#N do 0;) works, yet for i in 0..#N do this.val = 0; doesn't. Effectively, in that loop, the compiler sees a reference to the val field before an (obvious) initialization of it via a [this].val = …; initializing assignment.

    With more work, you could imagine the compiler being taught to understand that your loop is guaranteed to initialize all elements of val, and that a loop like for i in 0..#(N/2) do this.val = 0; would not, but to date, the language has opted for only supporting the simpler model of just looking for whole-field initializations and balking if it sees uses of the field before that.

    Here's an explanation of the difference between initializers (init() procedures), copy initializers (init=() procedures), and assignments (operator =):

    init(...) routines can be thought of as being like C++ constructors and are invoked in the following cases:

    In contrast, the init=() routine can be thought of as being like a copy constructor in C++, and will be invoked in cases like:

    Meanwhile, any = after the initialization of a variable or field is treated as an assignment, and the assignment operator will be used. The reason that it's important to distinguish assignment from initialization, is that in an assignment, the target of the assignment will already store a value that may require cleaning up, whereas in an initialization, it's known that the variable isn't storing anything yet.

    Part of the obvious source for confusion here is that the = symbol can mean assignment or initialization depending on context, so it's not always visually obvious which it is. At times, we've considered adding an explicit init= operator so that a user could write this.val init= initExpr; (say) to be very explicit in their code. But even if we did that, I wouldn't expect Chapel to require it to be used in all initializing contexts since things like var x init= 42; reads so much more cleanly as var x = 42;.

    Hopefully the above answers your questions. I also wanted to point out a few other ways to approach your example, realizing that it's likely a simplified version of what you're really trying to do: