1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
|
;;; test-org-drill-determine-next-interval-sm2.el --- Tests for SM2 algorithm
;;; Commentary:
;; Unit tests for org-drill-determine-next-interval-sm2, the SuperMemo 2 (SM2)
;; spaced repetition algorithm implementation.
;;
;; The SM2 algorithm calculates the next review interval based on:
;; - Last interval (days since last review)
;; - Number of successful repeats (n)
;; - Easiness factor (EF) - modified by recall quality
;; - Quality of recall (0-5 scale)
;; - Failure count and statistics
;;
;; Function signature:
;; (test-scheduler--call-sm2 last-interval n ef quality
;; failures meanq total-repeats)
;;
;; Returns: (INTERVAL REPEATS EF FAILURES MEAN TOTAL-REPEATS OFMATRIX)
;;; Code:
(require 'ert)
(require 'assess)
(require 'org-drill)
(add-to-list 'load-path
(file-name-directory (or load-file-name buffer-file-name)))
(require 'testutil-scheduler)
;;; Test Data and Constants
;; Default SM2 parameters
(defconst test-sm2-default-ef 2.5
"Default easiness factor for new items.")
(defconst test-sm2-min-ef 1.3
"Minimum easiness factor (SM2 floor).")
;;; Normal Cases - Successful Reviews
(ert-deftest test-org-drill-determine-next-interval-sm2-normal-first-review-quality-4 ()
"Test first successful review with quality 4.
First review (n=1) should return interval of 1 day."
(let* ((result (test-scheduler--call-sm2 0 1 nil 4 0 nil 0))
(interval (test-scheduler--extract-interval result))
(repeats (test-scheduler--extract-repeats result))
(ef (test-scheduler--extract-ef result)))
(should (= interval 1))
(should (= repeats 2))
(should ef))) ; EF should be calculated
(ert-deftest test-org-drill-determine-next-interval-sm2-normal-second-review-quality-4 ()
"Test second successful review with quality 4.
Second review (n=2) should return interval of 6 days (no random noise)."
(let ((org-drill-add-random-noise-to-intervals-p nil))
(let* ((result (test-scheduler--call-sm2 1 2 2.5 4 0 nil 0))
(interval (test-scheduler--extract-interval result))
(repeats (test-scheduler--extract-repeats result)))
(should (= interval 6))
(should (= repeats 3)))))
(ert-deftest test-org-drill-determine-next-interval-sm2-normal-third-review-quality-4 ()
"Test third successful review with quality 4.
Third review (n=3) uses formula: last-interval * EF."
(let ((org-drill-add-random-noise-to-intervals-p nil))
(let* ((ef 2.5)
(last-interval 6)
(result (test-scheduler--call-sm2 last-interval 3 ef 4 0 nil 0))
(interval (test-scheduler--extract-interval result))
(new-ef (test-scheduler--extract-ef result)))
;; Interval should be approximately last-interval * new-ef
(should (> interval last-interval))
(should (= (test-scheduler--extract-repeats result) 4)))))
(ert-deftest test-org-drill-determine-next-interval-sm2-normal-quality-5-perfect-recall ()
"Test review with perfect recall (quality 5).
Quality 5 should increase EF and result in longer intervals."
(let ((org-drill-add-random-noise-to-intervals-p nil))
(let* ((result (test-scheduler--call-sm2 10 3 2.5 5 0 nil 0))
(ef (test-scheduler--extract-ef result)))
;; Quality 5 should maintain or increase EF
(should (>= ef 2.5)))))
(ert-deftest test-org-drill-determine-next-interval-sm2-normal-quality-3-adequate-recall ()
"Test review with adequate recall (quality 3).
Quality 3 should maintain relatively stable EF."
(let ((org-drill-add-random-noise-to-intervals-p nil))
(let* ((result (test-scheduler--call-sm2 10 3 2.5 3 0 nil 0))
(ef (test-scheduler--extract-ef result)))
;; Quality 3 should result in moderate EF
(should (> ef test-sm2-min-ef))
(should (< ef 2.5))))) ; EF likely decreases slightly
;;; Normal Cases - Failed Reviews
(ert-deftest test-org-drill-determine-next-interval-sm2-normal-failure-quality-0 ()
"Test failed review with quality 0.
Quality 0 (<= org-drill-failure-quality) should reset interval to -1."
(let* ((result (test-scheduler--call-sm2 10 3 2.5 0 0 nil 0))
(interval (test-scheduler--extract-interval result))
(repeats (test-scheduler--extract-repeats result))
(ef (test-scheduler--extract-ef result))
(failures (test-scheduler--extract-failures result)))
(should (= interval -1))
(should (= repeats 1)) ; Reset to 1
(should (= ef 2.5)) ; EF unchanged on failure
(should (= failures 1)))) ; Failure count incremented
(ert-deftest test-org-drill-determine-next-interval-sm2-normal-failure-quality-1 ()
"Test failed review with quality 1.
Quality 1 (<= org-drill-failure-quality) should reset interval."
(let* ((result (test-scheduler--call-sm2 15 5 2.6 1 2 3.5 10))
(interval (test-scheduler--extract-interval result))
(ef (test-scheduler--extract-ef result))
(failures (test-scheduler--extract-failures result)))
(should (= interval -1))
(should (= ef 2.6)) ; EF unchanged
(should (= failures 3)))) ; Previous failures + 1
(ert-deftest test-org-drill-determine-next-interval-sm2-normal-failure-quality-2 ()
"Test failed review with quality 2.
Quality 2 (= org-drill-failure-quality default) should reset interval."
(let* ((result (test-scheduler--call-sm2 10 3 2.5 2 0 nil 0))
(interval (test-scheduler--extract-interval result)))
(should (= interval -1))))
;;; Boundary Cases - Default Values
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-nil-ef-uses-default ()
"Test that nil EF defaults to 2.5."
(let* ((result (test-scheduler--call-sm2 0 1 nil 4 0 nil 0))
(ef (test-scheduler--extract-ef result)))
(should ef) ; EF should be set (modified from default 2.5)
(should (> ef 0))))
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-zero-n-becomes-one ()
"Test that n=0 is treated as n=1."
(let* ((result (test-scheduler--call-sm2 0 0 2.5 4 0 nil 0))
(interval (test-scheduler--extract-interval result))
(repeats (test-scheduler--extract-repeats result)))
(should (= interval 1)) ; First review interval
(should (= repeats 2)))) ; n incremented from 1 to 2
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-nil-meanq-uses-quality ()
"Test that nil meanq initializes to current quality."
(let* ((quality 4)
(result (test-scheduler--call-sm2 0 1 2.5 quality 0 nil 0))
(meanq (test-scheduler--extract-meanq result)))
(should (= meanq quality))))
;;; Boundary Cases - Quality Extremes
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-quality-5-maximum ()
"Test maximum quality (5) - perfect recall."
(let ((org-drill-add-random-noise-to-intervals-p nil))
(let* ((result (test-scheduler--call-sm2 1 2 2.5 5 0 nil 0))
(interval (test-scheduler--extract-interval result)))
(should (= interval 6))))) ; Second review with quality 5
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-quality-0-minimum ()
"Test minimum quality (0) - complete failure."
(let* ((result (test-scheduler--call-sm2 10 3 2.5 0 0 nil 0))
(interval (test-scheduler--extract-interval result)))
(should (= interval -1)))) ; Failed review
;;; Boundary Cases - Repeat Count
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-n-equals-1 ()
"Test first successful review (n=1) returns interval of 1."
(let* ((result (test-scheduler--call-sm2 0 1 2.5 4 0 nil 0))
(interval (test-scheduler--extract-interval result)))
(should (= interval 1))))
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-n-equals-2-no-noise ()
"Test second review (n=2) with no random noise.
Should return interval of 6 days regardless of quality (if passing)."
(let ((org-drill-add-random-noise-to-intervals-p nil))
(let* ((result (test-scheduler--call-sm2 1 2 2.5 4 0 nil 0))
(interval (test-scheduler--extract-interval result)))
(should (= interval 6)))))
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-n-equals-2-with-noise ()
"Test second review (n=2) with random noise enabled.
Interval should vary by quality, with quality 5 having highest base interval (6)."
(let ((org-drill-add-random-noise-to-intervals-p t))
(let* ((result-q5 (test-scheduler--call-sm2 1 2 2.5 5 0 nil 0))
(result-q4 (test-scheduler--call-sm2 1 2 2.5 4 0 nil 0))
(result-q3 (test-scheduler--call-sm2 1 2 2.5 3 0 nil 0))
(interval-q5 (test-scheduler--extract-interval result-q5))
(interval-q4 (test-scheduler--extract-interval result-q4))
(interval-q3 (test-scheduler--extract-interval result-q3)))
;; Base intervals before noise are: Q5=6, Q4=4, Q3=3
;; After random noise, verify intervals are positive and roughly in expected ranges
(should (> interval-q5 0))
(should (> interval-q4 0))
(should (> interval-q3 0))
;; Quality 5 base interval (6) should generally be higher than quality 3 base (3)
;; even with noise applied
(should (> interval-q5 2)))))
;;; Boundary Cases - Mean Quality Calculation
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-meanq-weighted-average ()
"Test that meanq is correctly calculated as weighted average.
meanq = (quality + meanq * total-repeats) / (total-repeats + 1)"
(let* ((quality 4)
(meanq 3.0)
(total-repeats 10)
(result (test-scheduler--call-sm2 10 3 2.5 quality 0 meanq total-repeats))
(new-meanq (test-scheduler--extract-meanq result))
(expected-meanq (/ (+ quality (* meanq total-repeats 1.0))
(1+ total-repeats))))
(should (< (abs (- new-meanq expected-meanq)) 0.0001)))) ; Floating point comparison
;;; Boundary Cases - EF Modification
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-ef-minimum-floor ()
"Test that EF floor of 1.3 is applied to input EF when it's below 1.3.
If input EF < 1.3, org-drill-modify-e-factor returns 1.3 exactly."
(let* ((result (test-scheduler--call-sm2 10 3 1.0 4 0 nil 0))
(ef (test-scheduler--extract-ef result)))
;; Input EF was 1.0 < 1.3, so it should be raised to 1.3 first,
;; then modified based on quality 4 (which increases EF)
(should (>= ef 1.3))))
;;; Boundary Cases - Total Repeats
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-total-repeats-increments ()
"Test that total-repeats is always incremented by 1."
(let* ((total-repeats 42)
(result (test-scheduler--call-sm2 10 3 2.5 4 0 3.5 total-repeats))
(new-total (test-scheduler--extract-total-repeats result)))
(should (= new-total (1+ total-repeats)))))
;;; Return Value Structure
(ert-deftest test-org-drill-determine-next-interval-sm2-normal-return-value-structure ()
"Test that return value has correct structure.
Should return 7-element list: (INTERVAL REPEATS EF FAILURES MEAN TOTAL-REPEATS OFMATRIX)"
(let ((result (test-scheduler--call-sm2 10 3 2.5 4 0 3.5 10)))
(should (listp result))
(should (= (length result) 7))
;; Verify first 6 elements are numbers
(should (numberp (nth 0 result))) ; interval
(should (numberp (nth 1 result))) ; repeats
(should (numberp (nth 2 result))) ; ef
(should (numberp (nth 3 result))) ; failures
(should (numberp (nth 4 result))) ; meanq
(should (numberp (nth 5 result))) ; total-repeats
;; 7th element (index 6) is OFMATRIX - can be nil or a matrix
;; Just verify list has 7 elements (already checked above)
))
;;; Algorithm Verification
(ert-deftest test-org-drill-determine-next-interval-sm2-algorithm-ef-increases-with-quality ()
"Test that higher quality results in higher EF (or maintains it).
Quality 5 should result in EF >= initial EF."
(let* ((initial-ef 2.5)
(result (test-scheduler--call-sm2 10 3 initial-ef 5 0 nil 0))
(new-ef (test-scheduler--extract-ef result)))
(should (>= new-ef initial-ef))))
(ert-deftest test-org-drill-determine-next-interval-sm2-algorithm-ef-decreases-with-low-quality ()
"Test that low quality results in decreased EF.
Quality 3 should result in EF < initial EF."
(let* ((initial-ef 2.5)
(result (test-scheduler--call-sm2 10 3 initial-ef 3 0 nil 0))
(new-ef (test-scheduler--extract-ef result)))
(should (< new-ef initial-ef))))
(ert-deftest test-org-drill-determine-next-interval-sm2-algorithm-interval-grows-exponentially ()
"Test that intervals grow approximately exponentially for consistent quality.
Third review interval should be significantly larger than second."
(let ((org-drill-add-random-noise-to-intervals-p nil))
(let* ((ef 2.5)
(result-2nd (test-scheduler--call-sm2 1 2 ef 4 0 nil 0))
(interval-2nd (test-scheduler--extract-interval result-2nd))
(ef-2nd (test-scheduler--extract-ef result-2nd))
(result-3rd (test-scheduler--call-sm2 interval-2nd 3 ef-2nd 4 0 nil 1))
(interval-3rd (test-scheduler--extract-interval result-3rd)))
(should (> interval-3rd (* interval-2nd 1.5)))))) ; Significant growth
;;; Aggressive Boundary Cases
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-very-high-repeat-count ()
"Test SM2 with very high repeat count (1000 reviews).
Algorithm should remain stable with extreme n values."
(let ((org-drill-add-random-noise-to-intervals-p nil))
(let* ((result (test-scheduler--call-sm2 365 1000 2.5 4 0 4.0 1000))
(interval (test-scheduler--extract-interval result))
(repeats (test-scheduler--extract-repeats result))
(ef (test-scheduler--extract-ef result)))
(should (numberp interval))
(should (> interval 0))
(should (= repeats 1001))
(should (>= ef test-sm2-min-ef)))))
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-very-long-interval ()
"Test SM2 with very long last interval (10 years = 3650 days).
Algorithm should handle extreme intervals without overflow."
(let ((org-drill-add-random-noise-to-intervals-p nil))
(let* ((result (test-scheduler--call-sm2 3650 100 2.5 4 0 4.0 100))
(interval (test-scheduler--extract-interval result)))
(should (numberp interval))
(should (> interval 0)))))
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-very-high-failure-count ()
"Test SM2 with very high failure count (100+ failures).
Failure count should be tracked correctly even at extreme values."
(let* ((failures 100)
(result (test-scheduler--call-sm2 10 3 2.5 0 failures 3.0 150))
(new-failures (test-scheduler--extract-failures result)))
(should (= new-failures (1+ failures)))))
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-very-high-total-repeats ()
"Test SM2 with very high total-repeats (10000+).
Total repeats should increment correctly even at extreme values."
(let* ((total-repeats 10000)
(result (test-scheduler--call-sm2 10 3 2.5 4 0 4.0 total-repeats))
(new-total (test-scheduler--extract-total-repeats result)))
(should (= new-total (1+ total-repeats)))))
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-ef-just-above-floor ()
"Test SM2 with EF just above minimum floor (1.31).
EF should be maintained or modified normally, not clamped."
(let* ((result (test-scheduler--call-sm2 10 3 1.31 4 0 4.0 10))
(ef (test-scheduler--extract-ef result)))
(should (>= ef test-sm2-min-ef))))
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-ef-at-floor ()
"Test SM2 with EF exactly at minimum floor (1.3).
EF at floor with quality 4 should increase slightly."
(let* ((result (test-scheduler--call-sm2 10 3 1.3 4 0 4.0 10))
(ef (test-scheduler--extract-ef result)))
(should (>= ef test-sm2-min-ef))))
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-perfect-recall-after-many-failures ()
"Test perfect recall (quality 5) after long failure streak.
Perfect recall should not reset failures, but should calculate normal interval."
(let ((org-drill-add-random-noise-to-intervals-p nil))
(let* ((failures 50)
(result (test-scheduler--call-sm2 10 3 2.5 5 failures 3.0 100))
(interval (test-scheduler--extract-interval result))
(new-failures (test-scheduler--extract-failures result)))
(should (> interval 0))
(should (= new-failures failures))))) ; Failures unchanged on success
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-failure-after-long-streak ()
"Test complete failure (quality 0) after long successful streak.
Failure should reset interval to -1 regardless of previous success."
(let* ((result (test-scheduler--call-sm2 365 100 2.8 0 0 4.5 100))
(interval (test-scheduler--extract-interval result))
(repeats (test-scheduler--extract-repeats result))
(failures (test-scheduler--extract-failures result)))
(should (= interval -1))
(should (= repeats 1)) ; Reset to 1
(should (= failures 1))))
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-meanq-at-maximum ()
"Test SM2 with meanq at maximum (5.0).
High quality review with already-perfect meanq."
(let* ((result (test-scheduler--call-sm2 10 3 2.5 5 0 5.0 100))
(meanq (test-scheduler--extract-meanq result)))
(should (<= meanq 5.0))
(should (>= meanq 4.5)))) ; Should remain very high
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-meanq-at-minimum ()
"Test SM2 with meanq at minimum (0.0).
Low quality history with poor current recall."
(let* ((result (test-scheduler--call-sm2 10 3 2.5 3 5 0.0 100))
(meanq (test-scheduler--extract-meanq result)))
(should (>= meanq 0.0))
(should (<= meanq 1.0)))) ; Should remain low with quality 3
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-high-total-repeats-low-meanq ()
"Test SM2 with very high total-repeats but low meanq.
Indicates item reviewed many times but poorly recalled."
(let* ((total-repeats 1000)
(meanq 2.0)
(result (test-scheduler--call-sm2 5 10 1.5 3 20 meanq total-repeats))
(new-meanq (test-scheduler--extract-meanq result)))
(should (numberp new-meanq))
;; meanq should remain low given poor history
(should (< new-meanq 3.0))))
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-zero-last-interval ()
"Test SM2 with zero last-interval.
Should handle as first review (n=1 case)."
(let* ((result (test-scheduler--call-sm2 0 1 2.5 4 0 4.0 0))
(interval (test-scheduler--extract-interval result)))
(should (= interval 1))))
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-float-vs-int-intervals ()
"Test SM2 with float last-interval vs integer.
Algorithm should handle both correctly."
(let ((org-drill-add-random-noise-to-intervals-p nil))
(let* ((result-float (test-scheduler--call-sm2 10.5 3 2.5 4 0 4.0 10))
(result-int (test-scheduler--call-sm2 10 3 2.5 4 0 4.0 10))
(interval-float (test-scheduler--extract-interval result-float))
(interval-int (test-scheduler--extract-interval result-int)))
(should (numberp interval-float))
(should (numberp interval-int))
;; Both should produce reasonable intervals
(should (> interval-float 10))
(should (> interval-int 10)))))
(ert-deftest test-org-drill-determine-next-interval-sm2-boundary-alternating-quality-pattern ()
"Test SM2 behavior with alternating quality (5, 0, 5, 0 pattern).
Simulates inconsistent learning."
(let* ((result-1 (test-scheduler--call-sm2 1 2 2.5 5 0 5.0 1))
(ef-1 (test-scheduler--extract-ef result-1))
(result-2 (test-scheduler--call-sm2 6 3 ef-1 0 0 4.5 2))
(interval-2 (test-scheduler--extract-interval result-2))
(failures-2 (test-scheduler--extract-failures result-2)))
;; After perfect then fail, should reset interval
(should (= interval-2 -1))
(should (= failures-2 1))))
;;; Error Cases - cl-assert invariant violations
;; The function asserts (> n 0) and (and (>= quality 0) (<= quality 5)).
;; n is normalized from 0 to 1 first, so only a negative n trips its assert.
(defmacro test-scheduler--should-cl-assert (&rest body)
"Assert BODY signals a cl-assertion-failed.
Mirrors the sm5 and simple8 files' helper — see their commentary
for the ERT-29 signal-hook quirk that forces a `skip-unless' on
Emacs <30. Under undercover's edebug instrumentation the same
quirk drops into a blocking debugger in batch mode, so a bare
`should-error' here would hang the coverage job rather than fail."
`(progn
(skip-unless (>= emacs-major-version 30))
(let ((caught
(condition-case _err
(progn ,@body 'no-error)
(cl-assertion-failed 'caught))))
(unless (eq caught 'caught)
(ert-fail "expected cl-assertion-failed signal, got none")))))
(ert-deftest test-org-drill-determine-next-interval-sm2-error-quality-above-max ()
"Quality above the 0-5 range trips the quality assertion."
(test-scheduler--should-cl-assert
(test-scheduler--call-sm2 1 2 2.5 6 0 4.0 1)))
(ert-deftest test-org-drill-determine-next-interval-sm2-error-quality-below-min ()
"Quality below the 0-5 range trips the quality assertion."
(test-scheduler--should-cl-assert
(test-scheduler--call-sm2 1 2 2.5 -1 0 4.0 1)))
(ert-deftest test-org-drill-determine-next-interval-sm2-error-negative-n ()
"A negative repeat count trips the (> n 0) assertion. Note n=0 is
normalized to 1 before the assert, so zero is not an error case."
(test-scheduler--should-cl-assert
(test-scheduler--call-sm2 1 -1 2.5 4 0 4.0 1)))
(provide 'test-org-drill-determine-next-interval-sm2)
;;; test-org-drill-determine-next-interval-sm2.el ends here
|