-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathpli-get.py
722 lines (601 loc) · 23.4 KB
/
pli-get.py
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
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
"""
pli acquisition
RT, 27 june 2023, 10h53
RT, VS, 14 july 2023, 21h29
RT, 24 july 2023, 22h31
"""
import pathlib
import os
import asyncio
import argparse
import time
import numpy as np
from pypylon import pylon
from skimage import io
import serial
import serial.tools.list_ports
import matplotlib.pyplot as plt
from matplotlib.widgets import TextBox
## PLI machine
class DummySerial:
'''Dummy serial port for testing.'''
in_waiting = False
def __init__(self):
pass
def write(self, string):
print(f"DummySerial > {string.strip()}")
def readline(self):
return "ready"
class DummyCamera:
'''Dummy camera for testing.'''
def __init__(self):
pass
def Open(self):
pass
def Close(self):
pass
def StartGrabbing(self, grab_strategy):
pass
def RetrieveResult(self, timeout, timeout_handling):
# return DummyGrabResult()
return None
def StopGrabbing(self):
pass
class PLI:
'''PLI machine class.'''
pli_serial = None # serial connection to the PLI machine
camera = None # Basler camera
xy_stage_serial = None # XY stage
images = [] # array to store all images
channel = 1 # green, default image channel to capture
steps = 0 # current number of steps in the PLI machine
status = "idle"
motor_busy = [False, False, False]
debug = False
n_polarisers = None
n_stepper_steps = None
n_large_gear_teeth = None
n_small_gear_teeth = None
n_angles = None
interactive_mode = False
# PLI machine
def process_pli_machine_message(self, message):
'''Process a message received from the PLI machine.'''
print("PLI >", message)
if message == "ready":
self.status = "ready"
arr = message.split(": ")
if arr[0] == "done":
motor_number = int(arr[1])
self.motor_busy[motor_number] = False
if self.motor_busy[1] is False and self.motor_busy[2] is False:
self.status = "ready"
print("motors ready")
async def listen_to_pli_machine_messages(self, ser):
'''Listen to messages from the PLI machine.'''
while self.status != "done":
if ser.in_waiting:
message = ser.readline().decode().strip()
self.process_pli_machine_message(message)
await asyncio.sleep(0.1)
async def wait_for_ready(self):
'''Wait for the PLI machine to be ready.'''
while self.status != "ready":
await asyncio.sleep(0.1)
def new_pli(self):
'''Create a new serial connection to the PLI machine.'''
list_of_known_arduino_ports = [
"AB0LS12X", # Paris
"ttyUSB0" # Alicante
]
ports = serial.tools.list_ports.comports()
arduino_port = None
for port, desc, hwid in sorted(ports):
if self.debug:
print(f"{port}: {desc} [{hwid}]")
# if "usbmodem" in port or "usbserial" in port:
if max([p in port for p in list_of_known_arduino_ports]) is True:
arduino_port = port
if self.debug:
print(f"arduino_port: {arduino_port}")
ser = None
if arduino_port is None:
print("Arduino not found: returning dummy serial port.")
ser = DummySerial()
else:
ser = serial.Serial(arduino_port, 9600)
return ser
def pli_write(self, string, sleep=0.01):
'''Send raw commands to the PLI machine.
Commands:
delay n: delay between steps of the machine in ms.
From 1 (fast) to 30 (slow) are reasonable values
i{+, -}n: i is the index of the motor (1 or 2),
+ and - indicate the direction of the motion, and
n indicates the number of steps to move.
For example, "1+100" moves motor 1, ccw, 100 steps.
wait n: pauses the execution of commands for n seconds.
This command is not sent to the PLI machine but
executed locally.
'''
bin_cmd = bytes(string, "ascii")
if string.startswith("delay"):
bin_cmd += b"\r\n"
self.pli_serial.write(bin_cmd)
if sleep:
time.sleep(sleep)
def move(self, motor_number, delta):
'''move one motor by delta steps.'''
direction = "+" if delta > 0 else "-"
string = f"{motor_number}{direction}{abs(delta)}\n"
self.pli_write(string)
if motor_number == 2:
self.steps -= delta
self.status = "moving"
self.motor_busy[motor_number] = True
def move2(self, delta):
'''move two motors by delta steps.'''
self.move(1, delta)
self.move(2, -delta)
def move_to(self, pos):
'''move to a given position.'''
delta = pos - self.steps
if self.debug:
print(f"move_to delta: {delta}")
if delta == 0:
return
self.move(1, delta)
self.move(2, -delta)
def move_to_home(self):
'''move to home position.'''
if self.debug:
print(f"move_to_home from {self.steps} to 0")
self.move_to(0)
def reset_steps(self):
'''reset the number of steps.'''
self.steps = 0
# XY stage
async def listen_to_xy_stage_messages(self, ser):
'''Listen for messages from the XY stage.'''
print("listen for stage messages")
while True:
if ser.in_waiting:
message = ser.readline().decode().strip()
print(message)
await asyncio.sleep(0.1)
def new_xy_stage(self):
ports = serial.tools.list_ports.comports()
xy_stage_port = None
for port, desc, hwid in sorted(ports):
if self.debug:
print(f"{port}: {desc} [{hwid}]")
if "140" in port:
xy_stage_port = port
if self.debug:
print(f"xy_stage_port: {xy_stage_port}")
ser = None
if xy_stage_port is None:
print("XY stage not found.")
else:
ser = serial.Serial(xy_stage_port, 115200)
for _ in range(2):
output = ser.readline()
print(output)
ser.write(b"$G\n")
for _ in range(2):
output = ser.readline()
print(output)
return ser
def xy_stage_write(self, string, sleep=0.01):
'''Send raw commands to the xy stage, semi-colon separated.
The xy stage listens to g-code. For example, G21G91G1X120F2000
will move the x motor 120 steps with a feed rate of 2000.
In addition, the command "wait n" will pause the execution of
commands for n seconds. This command is not sent to the XY
stage, but executed locally.
'''
bin_cmd = string.encode() # bytes(string, "ascii")
# add carriage return at the end
bin_cmd += b"\r\n"
print(bin_cmd)
self.xy_stage_serial.write(bin_cmd)
for _ in range(1):
output = self.xy_stage_serial.readline()
print(output)
if sleep:
time.sleep(sleep)
## Basler camera
def new_camera(self):
'''Create a new camera instance.'''
camera = None
try:
camera = pylon.InstantCamera(pylon.TlFactory.GetInstance().CreateFirstDevice())
print("Got camera:", camera.GetDeviceInfo().GetModelName())
except:
print("Camera not found: returning empty camera.")
return camera
def camera_open(self):
'''Open the camera.'''
if self.camera is None:
print("Empty camera: Open")
else:
self.camera.Open()
def camera_close(self):
'''Close the camera.'''
if self.camera is None:
print("Empty camera: Close")
else:
self.camera.Close()
## Settings
def set_n_angles(self, n_angles):
'''Set the number of angles.'''
self.n_angles = n_angles
def set_n_polarisers(self, n_polarisers):
'''Set the number of polarisers.'''
if n_polarisers != 1 and n_polarisers != 2:
raise ValueError("n_polarisers must be 1 or 2")
self.n_polarisers = n_polarisers
def set_n_stepper_steps(self, n_stepper_steps):
'''Set the number of steps in a whole turn of the stepper motors.'''
self.n_stepper_steps = n_stepper_steps
def set_n_large_gear_teeth(self, n_large_gear_teeth):
'''Set the number of teeth in the large gear.'''
self.n_large_gear_teeth = n_large_gear_teeth
def set_n_small_gear_teeth(self, n_small_gear_teeth):
'''Set the number of teeth in the small gear.'''
self.n_small_gear_teeth = n_small_gear_teeth
def set_color_mode(self, color_mode):
'''Set the color mode of the camera.'''
if self.camera is None:
print(f"Empty camera: {color_mode}")
return
self.camera.Open()
if color_mode == "RGB8":
self.camera.PixelFormat.SetValue("RGB8")
elif color_mode == "Mono8":
self.camera.PixelFormat.SetValue("Mono8")
elif color_mode == "Mono12":
self.camera.PixelFormat.SetValue("Mono12")
else:
raise ValueError("color_mode must be RGB8 or Mono8 or Mono12")
self.camera.Close()
def set_gain(self, gain):
'''Set the gain of the camera.'''
if self.camera is None:
print(f"Empty camera: {gain}")
return
self.camera.Open()
self.camera.GainAuto.SetValue("Off")
self.camera.Gain.SetValue(gain)
self.camera.Close()
def set_exposure(self, exposure):
'''Set the exposure of the camera.'''
if self.camera is None:
print(f"Empty camera: {exposure}")
return
self.camera.Open()
self.camera.ExposureAuto.SetValue("Off")
self.camera.ExposureTime.SetValue(exposure)
self.camera.Close()
def set_gamma(self, gamma):
'''Set the gamma of the camera.'''
if self.camera is None:
print(f"Empty camera: {gamma}")
return
self.camera.Open()
self.camera.Gamma.SetValue(gamma)
## Acquisition functions
# grab one image from the camera
def grab_image(self, channel=1):
'''Grab one image from the camera.
Returns:
img: numpy array of the image
'''
img = None
if self.camera is None:
print("Empty camera: Image")
img = np.zeros((100, 100, 3), dtype=np.uint8)
else:
self.camera.StartGrabbing(pylon.GrabStrategy_LatestImageOnly)
grab_result = self.camera.RetrieveResult(500, pylon.TimeoutHandling_ThrowException)
if grab_result.GrabSucceeded():
img = grab_result.Array[:, :]
grab_result.Release()
self.camera.StopGrabbing()
return img
def grab(self, image_name = "test.png"):
'''Grab one image from the camera. Appends the image in Image_Array.
Returns:
img: numpy array of the image
'''
img = self.grab_image()
self.images.append([img, image_name])
# print(f"Image shape: {img.shape[0]} x {img.shape[1]}")
return img
def save_all_images(self):
'''Save all images in Image_Array.'''
print("saving images...")
_, computed_image_name = self.images[0]
pathlib.Path(
os.path.dirname(f"{computed_image_name}")
).mkdir(
parents=True,
exist_ok=True
)
for img, computed_image_name in self.images:
io.imsave(f"{computed_image_name}", img)
print("all images saved.")
## Autocalibration
def mean_value_image(self, img=None):
'''Get the mean value of an image for calibration. If no image is passed, acquire a new one.'''
if img is None:
img = self.grab_image()[::10, ::10]
else:
img = img[::10, ::10]
res = np.mean(img.ravel())
return res
async def calibrate(self, initial_step):
'''Calibrate the background of the PLI machine.
Parameters:
initial_step: initial step size
Returns:
vals: array of mean values
pos: position of the PLI machine
'''
if self.pli_serial is None or self.camera is None:
print("Empty PLI and camera: Calibrate")
return [0], 0
time.sleep(1)
self.pli_write("delay 5")
self.set_gamma(1)
pos = 0
dx = initial_step # pylint: disable=C0103
val0 = self.mean_value_image()
print("initial value:", val0)
vals = [val0]
while dx != 0:
self.move(1, dx)
await self.wait_for_ready()
pos += dx
val1 = self.mean_value_image()
vals.append(val1)
dy = val1 - val0 # pylint: disable=C0103
if dy > 0:
dx = -int(dx/2) # pylint: disable=C0103
val0 = val1
return vals, pos
async def interactive(self):
'''Interactive mode for configuring exposure, gain and gamma.'''
# if self.pli_serial is None or self.camera is None:
if self.camera is None:
print("Empty PLI and camera: Interactive mode")
return
time.sleep(1)
self.pli_write("delay 5")
self.interactive_mode = True
plt.rcParams['toolbar'] = 'None'
fig, ax_dict = plt.subplot_mosaic("AAB;AAB;AAB;AAB;AAC;AAD;AAE", figsize=(10, 5))
def on_close(event):
print(f"--exposure {self.camera.ExposureTime.Value} --gain {self.camera.Gain.Value} --gamma {self.camera.Gamma.Value}")
self.interactive_mode = False
fig.canvas.mpl_connect('close_event', on_close)
fig.subplots_adjust(left=0.05, bottom=0.05, right=0.95, top=0.95, wspace=0.05, hspace=0.05)
self.camera.Close()
self.camera.Open()
self.camera.UserSetSelector = "Default"
self.camera.UserSetLoad.Execute()
self.camera.PixelFormat.SetValue("Mono12")
print(self.camera.PixelFormat.Value)
# self.camera.Height = 2000
# self.camera.Width = 2000
# self.camera.AcquisitionFrameRateEnable.SetValue(True)
# self.camera.AcquisitionFrameRate.SetValue(1.0)
# self.camera.StreamGrabber.MaxBufferSize = 20286016
# self.camera.StreamGrabber.MaxTransferSize = 262144
input_box1 = TextBox(ax_dict["C"], 'Exposure', initial=self.camera.ExposureTime.Value)
def exposure(val):
self.camera.ExposureAuto.SetValue("Off")
self.camera.ExposureTime.SetValue(float(val))
input_box1.on_submit(exposure)
input_box2 = TextBox(ax_dict["D"], 'Gain', initial=self.camera.Gain.Value)
def gain(val):
self.camera.GainAuto.SetValue("Off")
self.camera.Gain.SetValue(float(val))
input_box2.on_submit(gain)
input_box3 = TextBox(ax_dict["E"], 'Gamma', initial=self.camera.Gamma.Value)
def gamma(val):
self.camera.Gamma.SetValue(float(val))
input_box3.on_submit(gamma)
self.camera.StartGrabbing(pylon.GrabStrategy_LatestImageOnly)
while self.interactive_mode is True:
with self.camera.RetrieveResult(5000, pylon.TimeoutHandling_ThrowException) as grab_result:
img = grab_result.Array[::10, ::10]
ax_dict["A"].clear()
ax_dict["A"].imshow(img, cmap="gray")
ax_dict["B"].clear()
ax_dict["B"].hist(
img.flatten(),
bins=64,
range=(0, self.camera.PixelDynamicRangeMax.Value),
density=True)
ax_dict["B"].get_yaxis().set_visible(False)
plt.pause(0.1)
plt.close(fig)
self.camera.StopGrabbing()
self.camera.Close()
## Constructor
def __init__(self):
self.camera = self.new_camera()
self.xy_stage_serial = self.new_xy_stage()
self.pli_serial = self.new_pli()
async def calibrate_task(pli):
'''Calibrate the PLI machine.'''
vals, pos = await pli.calibrate(20)
print(vals)
print(pos)
pli.status = "done" # this stops the listener task and ends the script
async def interactive_task(pli):
'''Interactive mode.'''
await pli.interactive()
pli.status = "done" # this stops the listener task and ends the script
async def acquire_task(pli, base_path):
'''Acquire one slice of the sample.'''
print("""
@todo: the core of this function should be moved to a function called acquire within the PLI class, like for calibrate
""")
steps_whole_turn = pli.n_large_gear_teeth/pli.n_small_gear_teeth * pli.n_stepper_steps
print("steps for a whole turn:", steps_whole_turn)
# angles
angle_step = steps_whole_turn/pli.n_angles
print("setps in one angular displacement:", angle_step)
pli.images = []
dark_array = []
await pli.wait_for_ready()
pli.pli_write("delay 5")
print("moving home")
pli.move_to_home()
await pli.wait_for_ready()
print("home reached")
# print("calibrating")
# vals, pos = pli.calibrate(20)
# await pli.wait_for_ready()
# print(pos, vals[-1])
for iteration in range(pli.n_angles):
print(f"angle: {iteration}")
pli.move_to(int(iteration * angle_step))
await pli.wait_for_ready()
img = pli.grab(f"{base_path}/{iteration}.tif")
dark_array.append(pli.mean_value_image(img))
pli.move_to_home()
await pli.wait_for_ready()
print(f"mean dark value: {np.mean(dark_array)}")
print(f"dark value range: {np.min(dark_array)} - {np.max(dark_array)}")
# plt.plot(dark_array)
pli.save_all_images()
pli.status = "done" # this stops the listener task and ends the script
def main(args):
'''Main function.'''
# create a new PLI machine instance,
# and configure it
pli = PLI()
loop = asyncio.get_event_loop()
listener_task = loop.create_task(
pli.listen_to_pli_machine_messages(pli.pli_serial))
# xy_stage_listener_task = loop.create_task(
# pli.listen_to_xy_stage_messages(pli.xy_stage_serial))
if args.calibrate is True:
commands_task = loop.create_task(calibrate_task(pli))
tasks = [listener_task, commands_task]
loop.run_until_complete(asyncio.wait(tasks))
print("done calibrating")
elif args.pli_commands:
commands = args.pli_commands.split(";")
for command in commands:
command = command.strip()
print(f"command: [{command}]")
if command.startswith("wait"):
secs = float(command.split(" ")[1])
print(f"waiting for {secs} seconds")
time.sleep(secs)
else:
pli.pli_write(command)
time.sleep(1)
elif args.xy_commands:
commands = args.xy_commands.split(";")
for command in commands:
command = command.strip()
print(f"command: [{command}]")
if command.startswith("wait"):
secs = float(command.split(" ")[1])
print(f"waiting for {secs} seconds")
time.sleep(secs)
else:
pli.xy_stage_write(command)
time.sleep(1)
elif args.interactive is True:
commands_task = loop.create_task(interactive_task(pli))
tasks = [listener_task, commands_task]
loop.run_until_complete(asyncio.wait(tasks))
print("done interactive mode")
elif args.acquire is True:
if args.base_path is None:
raise ValueError("base_path is required")
if args.n_angles is None:
raise ValueError("n_angles is required")
if args.n_polarisers is None:
raise ValueError("n_polarisers is required")
if args.n_stepper_steps is None:
raise ValueError("n_stepper_steps is required")
if args.n_large_gear_teeth is None:
raise ValueError("n_large_gear_teeth is required")
if args.n_small_gear_teeth is None:
raise ValueError("n_small_gear_teeth is required")
if args.color_mode is None:
raise ValueError("color_mode is required")
if args.gain is None:
raise ValueError("gain is required")
if args.exposure is None:
raise ValueError("exposure is required")
if args.gamma is None:
raise ValueError("gamma is required")
base_path = args.base_path
n_angles = args.n_angles
n_polarisers = args.n_polarisers
n_stepper_steps = args.n_stepper_steps
n_large_gear_teeth = args.n_large_gear_teeth
n_small_gear_teeth = args.n_small_gear_teeth
color_mode = args.color_mode
gain = args.gain
exposure = args.exposure
gamma = args.gamma
pli.pli_write("delay 5", sleep=0)
pli.reset_steps()
pli.set_n_angles(n_angles)
pli.set_n_polarisers(n_polarisers)
pli.set_n_stepper_steps(n_stepper_steps)
pli.set_n_large_gear_teeth(n_large_gear_teeth)
pli.set_n_small_gear_teeth(n_small_gear_teeth)
pli.set_color_mode(color_mode)
pli.set_gain(gain)
pli.set_exposure(exposure)
pli.set_gamma(gamma)
# start the acquisition
commands_task = loop.create_task(acquire_task(pli, base_path))
tasks = [listener_task, commands_task]
loop.run_until_complete(asyncio.wait(tasks))
if __name__ == "__main__":
parser = argparse.ArgumentParser(description='Acquire PLI data.')
parser.add_argument('--calibrate', action='store_true', help='auto-calibrate the PLI machine')
parser.add_argument('--interactive', action='store_true', help='start interactive mode for camera configuration')
parser.add_argument('--acquire', action='store_true', help='acquire an image')
parser.add_argument('--base_path', type=str, help='base path for the images')
parser.add_argument('--n_angles', type=int, help='number of angles')
parser.add_argument('--n_polarisers', type=int, help='machine number of polarisers')
parser.add_argument('--n_stepper_steps', type=int, help='number of steps in a whole turn of the stepper motors')
parser.add_argument('--n_large_gear_teeth', type=int, help='machine number of teeth in the large gear')
parser.add_argument('--n_small_gear_teeth', type=int, help='machine number of teeth in the small gear')
parser.add_argument('--color_mode', type=str, help='camera color mode')
parser.add_argument('--gain', type=float, help='camera gain')
parser.add_argument('--exposure', type=float, help='camera exposure time')
parser.add_argument('--gamma', type=float, help='camera gamma')
parser.add_argument('--pli_commands', type=str, help='raw commands to send to the PLI machine, separated by a semi-colon')
parser.add_argument('--xy_commands', type=str, help='raw commands to send to the XY stage, separated by a semi-colon')
main(parser.parse_args())
# my_dict = {
# "calibrate": False,
# "pli_commands": False,
# "xy_commands": False,
# "acquire": True,
# "base_path": "~/Desktop/test",
# "n_angles": 36,
# "n_polarisers": 1,
# "n_stepper_steps": 800,
# "n_large_gear_teeth": 112,
# "n_small_gear_teeth": 42,
# "color_mode": "Mono8",
# "gain": 9.555,
# "exposure": 11000,
# "gamma": 3.999
# }
# args = argparse.Namespace(**my_dict)
# main(args)