On Mon, 2008-05-05 at 21:10 -0700, Neil Brown wrote: > On Monday May 5, dan.j.williams@xxxxxxxxx wrote: > > The STRIPE_OP_* flags record the state of stripe operations which are > > performed outside the stripe lock. Their use in indicating which > > operations need to be run is straightforward; however, interpolating what > > the next state of the stripe should be based on a given combination of > > these flags is not straightforward, and has led to bugs. An easier to read > > implementation with minimal degrees of freedom is needed. > > Thanks a lot for looking at this. > > Yes, fewer degrees of freedom does seem like a good direction. > > I cannot quite decide if I like what you have done or not. That fact > that you found a bug speaks highly in favour of the approach you > took. However it doesn't seem to be removing any flag bits, which > bothers me. I had hoped that at least one and possibly two of > ops.{pending,ack,complete} could go. They are removed by this approach. They still exist in this patch in order to test the new code in the presence of the old code. ops.{ack,complete} are removed from the check path, and ops.pending is replaced with stripe_head_state.ops_run. In this way requesting an operation is never globally visible, which should be more robust (or at least less code). The only globally visible state is whether or not an operation is quiesced. Other things on the hit list: * kill ops.count: requesting service from raid5_run_ops is no longer globally visible so all we need is sh->count to tell when handle_stripe needs to be called again. * kill test_and_ack_op: no need for it after ops.{pending, ack, complete} are gone. * reduce prototype of raid5_run_ops to void raid5_run_ops(struct stripe_head *sh, struct stripe_head_state *s) > You have made the state-machine aspect of the code more explicit by > introducing labels for the high-level states. > When you have a multi-threads machine as we do, I think it is > sometimes easier to encode just the low-level state of the data (which > is harder to get wrong due to races) and each time make a decision > based on that. > This is what we were previously doing, but it seems to be just a > little to complex. Maybe that is telling us the for sufficiently > complex machines, focussing on the high-level states is the best way > to solve the complexity... but I'm not 100% convinced yet and am keen > to explore. > > One aspect of your states that seems a bit odd is that you have both > "idle" and "done" states. As states, these seem to me to mean much > the same thing - nothing is happening. You do some "finish up" > tasking in the "done" state. Yes, the '_done' state really is a misnomer, how about '_process_result'? Any event where parity is touched appears to have a _process_result state. Reads (biofill operations) seem to be the only case where we can go straight from '_run' to '_idle'. > Maybe these should be done on the > transition from "run" back to "idle".... but maybe locking > requirements make that awkward... You mean add the finish up tasks to the ops_complete_* routines? We need an up-to-date stripe_head_state to direct some completion actions. So yes, moving these tasks outside of handle_stripe seems awkward. > As I said, I would like to see only one bitfield in place of the > current pending,ack,complete. > The way I imagine this is as follows: > When the machine enters some state (xxx_run) there are some number > of tasks that need to be accomplished. > We set "state = xxx_run" and ops.pending = set.of.bits. > > Then outside the lock we doing: > if (test_and_clean(bit, &ops.pending)) > atomic_inc(&sh->count); > start 'bit' operation > for each bit. > When the operation completes, sh->count is decremented. > > So when ops.pending is 0, and sh->count is zero, xxx_run must have > completed. > Possibly we inc sh->count when we set the bit in pending rather than > when we clear it. That makes it universal that if sh->count is 0, > then it is time to assess the current state and move on. > I remains to be seen if the various states, particularly of parity > check and repair, can be adequately deduced from the general state of > the stripe, or whether we need an explicit state variable like you > introduced. > Would you be interested in trying to remove .ack and .complete as > described above and see where that leave us? I think this is done for check_states... I'll take a look at implementing reconstruct_states to see if anything falls out. > Thanks, > NeilBrown Thanks, Dan -- To unsubscribe from this list: send the line "unsubscribe linux-raid" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html