Window stations and desktops
[reactos.git] / reactos / ntoskrnl / Makefile
1 # $Id: Makefile,v 1.43 2001/06/12 17:50:27 chorns Exp $
2 #
3 # ReactOS Operating System
4 #
5
6 #
7 # Path to the directory containing the root makefile
8 #
9 PATH_TO_TOP := ..
10
11 #
12 # Include details of the kernel configuration
13 #
14
15 include config
16
17 #
18 #
19 #
20 CONFIG :=
21
22 ifeq ($(DBG), 1)
23 CFLAGS_DBG := -g
24 CONFIG += DBG
25 LINKER_SCRIPT := ntoskrnl.lnk
26 STRIP_FLAGS := -Wl,-s
27 else
28 CFLAGS_DBG :=
29 LINKER_SCRIPT := ntoskrnl.lnk
30 STRIP_FLAGS := -Wl,-s
31 endif
32
33 ifeq ($(KDBG), 1)
34 OBJECTS_KDBG := dbg/kdb.o dbg/kdb_keyboard.o dbg/rdebug.o \
35 dbg/i386/kdb_help.o
36 CONFIG += KDBG
37 else
38 OBJECTS_KDBG :=
39 endif
40
41 ifeq ($(MP), 1)
42 CONFIG += MP
43 else
44 CONFIG += UP
45 endif
46
47 ifeq ($(ACPI), 1)
48 CONFIG += ACPI
49 else
50 CONFIG +=
51 endif
52
53 TARGETNAME := ntoskrnl
54
55 OBJECTS_PATH = objects
56
57 ASFLAGS = -Iinclude
58 CFLAGS = -Iinclude -D__NTOSKRNL__ $(CFLAGS_DBG) -Wall -Werror
59
60 include $(PATH_TO_TOP)/rules.mak
61
62 all: $(EXE_PREFIX)depends$(EXE_POSTFIX) include/internal/config.h \
63 $(OBJECTS_PATH) \
64 $(TARGETNAME).nostrip.exe \
65 $(TARGETNAME).exe \
66 $(TARGETNAME).sym
67
68 $(EXE_PREFIX)depends$(EXE_POSTFIX): depends.c
69 $(HOST_CC) -o depends$(EXE_POSTFIX) depends.c
70
71 #
72 # Architecture specific Makefile
73 # Defines $(OBJECTS_ARCH)
74 #
75 include Makefile.$(ARCH)
76
77 # System API (Nt/Zw)
78 OBJECTS_NT = \
79 nt/channel.o \
80 nt/evtpair.o \
81 nt/mutant.o \
82 nt/misc.o \
83 nt/nt.o \
84 nt/ntevent.o \
85 nt/ntsem.o \
86 nt/nttimer.o \
87 nt/plugplay.o \
88 nt/profile.o \
89 nt/zw.o \
90 nt/vdm.o
91
92 # Run-Time Library (Rtl)
93 OBJECTS_RTL = \
94 rtl/atom.o \
95 rtl/bitmap.o \
96 rtl/ctype.o \
97 rtl/error.o \
98 rtl/handle.o \
99 rtl/interlck.o \
100 rtl/largeint.o \
101 rtl/mem.o \
102 rtl/memchr.o \
103 rtl/memcpy.o \
104 rtl/memmove.o \
105 rtl/memset.o \
106 rtl/message.o \
107 rtl/nls.o \
108 rtl/qsort.o \
109 rtl/regio.o \
110 rtl/sprintf.o \
111 rtl/stdlib.o \
112 rtl/string.o \
113 rtl/swprintf.o \
114 rtl/time.o \
115 rtl/timezone.o \
116 rtl/unicode.o \
117 rtl/wstring.o \
118 rtl/bitops.o \
119 rtl/memcmp.o
120
121 # Kernel (Ke)
122 OBJECTS_KE = \
123 ke/apc.o \
124 ke/bug.o \
125 ke/catch.o \
126 ke/critical.o \
127 ke/dpc.o \
128 ke/error.o \
129 ke/event.o \
130 ke/kqueue.o \
131 ke/main.o \
132 ke/mutex.o \
133 ke/process.o \
134 ke/sem.o \
135 ke/spinlock.o \
136 ke/timer.o \
137 ke/wait.o \
138 ke/kthread.o
139
140 # Memory Manager (Mm)
141 OBJECTS_MM = \
142 mm/aspace.o \
143 mm/cont.o \
144 mm/freelist.o \
145 mm/iospace.o \
146 mm/marea.o \
147 mm/mdl.o \
148 mm/mm.o \
149 mm/ncache.o \
150 mm/npool.o \
151 mm/pagfault.o \
152 mm/pagefile.o \
153 mm/paging.o \
154 mm/pool.o \
155 mm/ppool.o \
156 mm/section.o \
157 mm/virtual.o \
158 mm/pager.o \
159 mm/wset.o \
160 mm/mminit.o \
161 mm/kmap.o \
162 mm/mpw.o \
163 mm/pageop.o
164
165 # I/O Subsystem (Io)
166 OBJECTS_IO = \
167 io/adapter.o \
168 io/buildirp.o \
169 io/cancel.o \
170 io/cleanup.o \
171 io/cntrller.o \
172 io/create.o \
173 io/device.o \
174 io/dir.o \
175 io/drvlck.o \
176 io/errlog.o \
177 io/error.o \
178 io/event.o \
179 io/file.o \
180 io/flush.o \
181 io/fs.o \
182 io/iocomp.o \
183 io/ioctrl.o \
184 io/iomgr.o \
185 io/irp.o \
186 io/lock.o \
187 io/mailslot.o \
188 io/mdl.o \
189 io/npipe.o \
190 io/page.o \
191 io/pnpmgr.o \
192 io/pnproot.o \
193 io/process.o \
194 io/queue.o \
195 io/resource.o \
196 io/rw.o \
197 io/share.o \
198 io/shutdown.o \
199 io/symlink.o \
200 io/timer.o \
201 io/vpb.o \
202 io/xhaldisp.o \
203 io/xhaldrv.o
204
205 # Object Manager (Ob)
206 OBJECTS_OB = \
207 ob/dirobj.o \
208 ob/handle.o \
209 ob/namespc.o \
210 ob/ntobj.o \
211 ob/object.o \
212 ob/security.o
213
214 # Process Manager (Ps)
215 OBJECTS_PS = \
216 ps/create.o \
217 ps/idle.o \
218 ps/kill.o \
219 ps/process.o \
220 ps/psmgr.o \
221 ps/thread.o \
222 ps/tinfo.o \
223 ps/debug.o \
224 ps/suspend.o
225
226 # Executive Subsystem (Ex)
227 OBJECTS_EX = \
228 ex/callback.o \
229 ex/fmutex.o \
230 ex/init.o \
231 ex/interlck.o \
232 ex/list.o \
233 ex/locale.o \
234 ex/lookas.o \
235 ex/napi.o \
236 ex/power.o \
237 ex/resource.o \
238 ex/time.o \
239 ex/sysinfo.o \
240 ex/win32k.o \
241 ex/work.o \
242 ex/zone.o
243
244 # Installable File System Run-Time Library (FsRtl)
245 OBJECTS_FS = \
246 fs/dbcsname.o \
247 fs/filelock.o \
248 fs/mcb.o \
249 fs/mdl.o \
250 fs/name.o \
251 fs/notify.o \
252 fs/oplock.o \
253 fs/pool.o \
254 fs/tunnel.o \
255 fs/unc.o \
256 fs/util.o
257
258 # Security Subsystem
259 OBJECTS_SE = \
260 se/access.o \
261 se/acl.o \
262 se/luid.o \
263 se/priv.o \
264 se/sd.o \
265 se/semgr.o \
266 se/sid.o \
267 se/token.o
268
269 # Configuration Manager (Registry)
270 OBJECTS_CM = \
271 cm/registry.o
272
273 # Debugger Support (Dbg)
274 OBJECTS_DBG = \
275 dbg/dbgctrl.o \
276 dbg/errinfo.o \
277 dbg/print.o \
278 dbg/user.o \
279 $(OBJECTS_KDBG)
280
281 # Loader
282 OBJECTS_LDR = \
283 ldr/init.o \
284 ldr/loader.o \
285 ldr/resource.o \
286 ldr/rtl.o \
287 ldr/sysdll.o \
288 ldr/userldr.o
289
290 # Local Procedure Call (Lpc)
291 OBJECTS_LPC = \
292 lpc/close.o \
293 lpc/complete.o \
294 lpc/connect.o \
295 lpc/create.o \
296 lpc/listen.o \
297 lpc/port.o \
298 lpc/query.o \
299 lpc/queue.o \
300 lpc/receive.o \
301 lpc/reply.o \
302 lpc/send.o
303
304 # Nation Language Support Library (Nls)
305 OBJECTS_NLS = nls/nls.o
306
307 # Power Management (Po)
308 OBJECTS_PO = po/power.o
309
310 # Cache Manager (Cc)
311 OBJECTS_CC = \
312 cc/cacheman.o \
313 cc/view.o
314
315 # Kernel Debugger Support (Kd)
316 OBJECTS_KD = \
317 kd/kdebug.o \
318 kd/service.o \
319 kd/dlog.o
320
321 C_OBJECTS := $(OBJECTS_NT) $(OBJECTS_HAL) $(OBJECTS_MM) $(OBJECTS_ARCH) \
322 $(OBJECTS_IO) $(OBJECTS_KE) $(OBJECTS_OB) \
323 $(OBJECTS_PS) $(OBJECTS_EX) $(OBJECTS_CC) $(OBJECTS_FS) $(OBJECTS_SE) \
324 $(OBJECTS_DBG) $(OBJECTS_CM) $(OBJECTS_LDR) $(OBJECTS_LPC) \
325 $(OBJECTS_PO) $(OBJECTS_KD) $(OBJECTS_RTL)
326
327 D_FILES := $(filter-out ex/napi.%, $(filter-out nt/zw.%, $(C_OBJECTS:.o=.d)))
328 D1_FILES := $(join $(dir $(D_FILES)), $(addprefix ., $(notdir $(D_FILES))))
329
330 # Resources
331 OBJECTS_RESOURCE = \
332 $(TARGETNAME).coff
333
334 # IMPORTS/EXPORTS
335 IE_DATA = \
336 $(TARGETNAME).def \
337 $(TARGETNAME).edf
338
339 $(OBJECTS_PATH):
340 mkdir $(OBJECTS_PATH)
341
342 #
343 # Build subsystem intermediate objects
344 #
345 $(OBJECTS_PATH)/hal.o: $(OBJECTS_HAL)
346 $(LD) \
347 -r \
348 -o $(OBJECTS_PATH)/hal.o \
349 $(OBJECTS_HAL)
350
351 $(OBJECTS_PATH)/arch.o: $(OBJECTS_ARCH)
352 $(LD) -r -o $(OBJECTS_PATH)/arch.o $(OBJECTS_ARCH)
353
354 $(OBJECTS_PATH)/io.o: $(OBJECTS_IO)
355 $(LD) \
356 -r \
357 -o $(OBJECTS_PATH)/io.o \
358 $(OBJECTS_IO)
359
360 $(OBJECTS_PATH)/ke.o: $(OBJECTS_KE)
361 $(LD) \
362 -r \
363 -o $(OBJECTS_PATH)/ke.o \
364 $(OBJECTS_KE)
365
366 $(OBJECTS_PATH)/rtl.o: $(OBJECTS_RTL)
367 $(LD) \
368 -r \
369 -o $(OBJECTS_PATH)/rtl.o \
370 $(OBJECTS_RTL)
371
372 $(OBJECTS_PATH)/mm.o: $(OBJECTS_MM)
373 $(LD) \
374 -r \
375 -o $(OBJECTS_PATH)/mm.o \
376 $(OBJECTS_MM)
377
378 $(OBJECTS_PATH)/ob.o: $(OBJECTS_OB)
379 $(LD) \
380 -r \
381 -o $(OBJECTS_PATH)/ob.o \
382 $(OBJECTS_OB)
383
384 $(OBJECTS_PATH)/ps.o: $(OBJECTS_PS)
385 $(LD) \
386 -r \
387 -o $(OBJECTS_PATH)/ps.o \
388 $(OBJECTS_PS)
389
390 $(OBJECTS_PATH)/ex.o: $(OBJECTS_EX)
391 $(LD) \
392 -r \
393 -o $(OBJECTS_PATH)/ex.o \
394 $(OBJECTS_EX)
395
396 $(OBJECTS_PATH)/fs.o: $(OBJECTS_FS)
397 $(LD) \
398 -r \
399 -o $(OBJECTS_PATH)/fs.o \
400 $(OBJECTS_FS)
401
402 $(OBJECTS_PATH)/se.o: $(OBJECTS_SE)
403 $(LD) \
404 -r \
405 -o $(OBJECTS_PATH)/se.o \
406 $(OBJECTS_SE)
407
408 $(OBJECTS_PATH)/cm.o: $(OBJECTS_CM)
409 $(LD) \
410 -r \
411 -o $(OBJECTS_PATH)/cm.o \
412 $(OBJECTS_CM)
413
414 $(OBJECTS_PATH)/dbg.o: $(OBJECTS_DBG)
415 $(LD) \
416 -r \
417 -o $(OBJECTS_PATH)/dbg.o \
418 $(OBJECTS_DBG)
419
420 $(OBJECTS_PATH)/ldr.o: $(OBJECTS_LDR)
421 $(LD) \
422 -r \
423 -o $(OBJECTS_PATH)/ldr.o \
424 $(OBJECTS_LDR)
425
426 $(OBJECTS_PATH)/lpc.o: $(OBJECTS_LPC)
427 $(LD) \
428 -r \
429 -o $(OBJECTS_PATH)/lpc.o \
430 $(OBJECTS_LPC)
431
432 $(OBJECTS_PATH)/nls.o: $(OBJECTS_NLS)
433 $(LD) \
434 -r \
435 -o $(OBJECTS_PATH)/nls.o \
436 $(OBJECTS_NLS)
437
438 $(OBJECTS_PATH)/nt.o: $(OBJECTS_NT)
439 $(LD) \
440 -r \
441 -o $(OBJECTS_PATH)/nt.o \
442 $(OBJECTS_NT)
443
444 $(OBJECTS_PATH)/po.o: $(OBJECTS_PO)
445 $(LD) \
446 -r \
447 -o $(OBJECTS_PATH)/po.o \
448 $(OBJECTS_PO)
449
450 $(OBJECTS_PATH)/cc.o: $(OBJECTS_CC)
451 $(LD) \
452 -r \
453 -o $(OBJECTS_PATH)/cc.o \
454 $(OBJECTS_CC)
455
456 $(OBJECTS_PATH)/kd.o: $(OBJECTS_KD)
457 $(LD) \
458 -r \
459 -o $(OBJECTS_PATH)/kd.o \
460 $(OBJECTS_KD)
461
462 $(TARGETNAME).coff: $(TARGETNAME).rc ../include/reactos/resource.h
463
464
465 # Note: arch.o MUST be the first file!!!
466 OBJECTS := \
467 $(OBJECTS_PATH)/arch.o \
468 $(OBJECTS_PATH)/ke.o \
469 $(OBJECTS_PATH)/hal.o \
470 $(OBJECTS_PATH)/cc.o \
471 $(OBJECTS_PATH)/cm.o \
472 $(OBJECTS_PATH)/dbg.o \
473 $(OBJECTS_PATH)/ex.o \
474 $(OBJECTS_PATH)/lpc.o \
475 $(OBJECTS_PATH)/fs.o \
476 $(OBJECTS_PATH)/io.o \
477 $(OBJECTS_PATH)/kd.o \
478 $(OBJECTS_PATH)/ldr.o \
479 $(OBJECTS_PATH)/mm.o \
480 $(OBJECTS_PATH)/nls.o \
481 $(OBJECTS_PATH)/nt.o \
482 $(OBJECTS_PATH)/ob.o \
483 $(OBJECTS_PATH)/po.o \
484 $(OBJECTS_PATH)/ps.o \
485 $(OBJECTS_PATH)/rtl.o \
486 $(OBJECTS_PATH)/se.o \
487 $(OBJECTS_RESOURCE)
488
489
490 ifeq ($(DOSCLI),yes)
491 CLEAN_FILES = $(OBJECTS_PATH)\*.o cc\*.o cm\*.o dbg\*.o ex\*.o hal\x86\*.o io\*.o \
492 ke\*.o ldr\*.o mm\*.o nt\*.o ob\*.o ps\*.o rtl\*.o se\*.o \
493 ke\i386\*.o mm\i386\*.o fs\*.o po\*.o nls\*.o lpc\*.o \
494 kd\*.o $(TARGETNAME).o $(TARGETNAME).a junk.tmp base.tmp temp.exp \
495 $(TARGETNAME).exe $(TARGETNAME).nostrip.exe $(TARGETNAME).sym $(TARGETNAME).coff $(D1_FILES)
496 else
497 CLEAN_FILES = $(OBJECTS_PATH)/*.o cc/*.o cm/*.o dbg/*.o ex/*.o hal/x86/*.o io/*.o \
498 ke/*.o ldr/*.o mm/*.o nt/*.o ob/*.o ps/*.o rtl/*.o se/*.o \
499 ke/i386/*.o mm/i386/*.o fs/*.o po/*.o nls/*.o lpc/*.o \
500 kd/*.o $(TARGETNAME).o $(TARGETNAME).a junk.tmp base.tmp temp.exp \
501 $(TARGETNAME).exe $(TARGETNAME).nostrip.exe $(TARGETNAME).sym $(TARGETNAME).coff $(D1_FILES)
502 endif
503
504
505 $(TARGETNAME).nostrip.exe: $(TARGETNAME).o $(TARGETNAME).a $(IE_DATA)
506 $(CC) \
507 -Wl,-T,ntoskrnl.lnk \
508 -nostartfiles -nostdlib \
509 -mdll \
510 -o junk.tmp \
511 -Wl,--subsystem,native \
512 -Wl,--image-base,0xc0000000 \
513 -Wl,--file-alignment,0x1000 \
514 -Wl,--section-alignment,0x1000 \
515 -Wl,--base-file,base.tmp \
516 $(TARGETNAME).o -lgcc
517 - $(RM) junk.tmp
518 $(DLLTOOL) \
519 --dllname $(TARGETNAME).exe \
520 --base-file base.tmp \
521 --output-exp temp.exp \
522 --def $(TARGETNAME).edf \
523 --kill-at
524 - $(RM) base.tmp
525 $(CC) \
526 -nostartfiles -nostdlib \
527 -Wl,-T,ntoskrnl.lnk \
528 -mdll \
529 -o $(TARGETNAME).nostrip.exe \
530 -Wl,--subsystem,native \
531 -Wl,--image-base,0xc0000000 \
532 -Wl,--file-alignment,0x1000 \
533 -Wl,--section-alignment,0x1000 \
534 -Wl,temp.exp \
535 $(TARGETNAME).o -lgcc
536 - $(RM) temp.exp
537
538 $(TARGETNAME).exe: $(TARGETNAME).o $(TARGETNAME).a $(LINKER_SCRIPT)
539 $(CC) \
540 -Wl,-T,$(LINKER_SCRIPT) \
541 -nostartfiles -nostdlib \
542 -mdll \
543 -o junk.tmp \
544 -Wl,--subsystem,native \
545 -Wl,--image-base,0xc0000000 \
546 -Wl,--file-alignment,0x1000 \
547 -Wl,--section-alignment,0x1000 \
548 -Wl,--base-file,base.tmp \
549 $(TARGETNAME).o -lgcc
550 - $(RM) junk.tmp
551 $(DLLTOOL) \
552 --dllname $(TARGETNAME).exe \
553 --base-file base.tmp \
554 --output-exp temp.exp \
555 --def $(TARGETNAME).edf \
556 --kill-at
557 - $(RM) base.tmp
558 $(CC) \
559 -Wl,-T,$(LINKER_SCRIPT) $(STRIP_FLAGS)\
560 -nostartfiles -nostdlib \
561 -mdll \
562 -o $(TARGETNAME).exe \
563 -Wl,--subsystem,native \
564 -Wl,--image-base,0xc0000000 \
565 -Wl,--file-alignment,0x1000 \
566 -Wl,--section-alignment,0x1000 \
567 -Wl,temp.exp \
568 $(TARGETNAME).o -lgcc
569 - $(RM) temp.exp
570
571 $(TARGETNAME).dbg.o: $(TARGETNAME).o
572 $(STRIP) --strip-debug -o $(TARGETNAME).dbg.o $(TARGETNAME).o
573
574 $(TARGETNAME).dbg: $(TARGETNAME).dbg.o $(TARGETNAME).a $(TARGETNAME).dbg.lnk
575 $(CC) \
576 -Wl,-T,$(TARGETNAME).dbg.lnk \
577 -nostartfiles -nostdlib \
578 -mdll \
579 -o junk.tmp \
580 -Wl,--subsystem,native \
581 -Wl,--image-base,0xc0000000 \
582 -Wl,--file-alignment,0x1000 \
583 -Wl,--section-alignment,0x1000 \
584 -Wl,--base-file,base.tmp \
585 $(TARGETNAME).dbg.o -lgcc
586 - $(RM) junk.tmp
587 $(DLLTOOL) \
588 --dllname $(TARGETNAME).dbg \
589 --base-file base.tmp \
590 --output-exp temp.exp \
591 --def $(TARGETNAME).edf \
592 --kill-at
593 - $(RM) base.tmp
594 $(CC) \
595 -Wl,-T,$(TARGETNAME).dbg.lnk \
596 -nostartfiles -nostdlib \
597 -mdll \
598 -o $(TARGETNAME).dbg \
599 -Wl,--subsystem,native \
600 -Wl,--image-base,0xc0000000 \
601 -Wl,--file-alignment,0x1000 \
602 -Wl,--section-alignment,0x1000 \
603 -Wl,temp.exp \
604 $(TARGETNAME).dbg.o -lgcc
605 - $(RM) temp.exp
606
607 $(TARGETNAME).sym: $(TARGETNAME).nostrip.exe
608 $(NM) --numeric-sort $(TARGETNAME).nostrip.exe > $(TARGETNAME).sym
609
610 $(TARGETNAME).o: $(OBJECTS)
611 $(LD) \
612 -r \
613 -o $(TARGETNAME).o \
614 $(OBJECTS)
615
616 $(TARGETNAME).a: $(TARGETNAME).def
617 $(DLLTOOL) \
618 --dllname $(TARGETNAME).exe \
619 --def $(TARGETNAME).def \
620 --output-lib $(TARGETNAME).a \
621 --kill-at
622
623 clean: $(CLEAN_FILES:%=%_clean)
624
625 $(CLEAN_FILES:%=%_clean): %_clean:
626 - $(RM) $*
627
628 .PHONY: clean $(CLEAN_FILES:%=%_clean)
629
630 install: $(FLOPPY_DIR)/$(TARGETNAME).exe
631
632 $(FLOPPY_DIR)/$(TARGETNAME).exe: $(TARGETNAME).exe
633 $(CP) $(TARGETNAME).exe $(FLOPPY_DIR)/$(TARGETNAME).exe
634
635 .PHONY: install
636
637 dist: ../$(DIST_DIR)/$(TARGETNAME).exe
638
639 ../$(DIST_DIR)/$(TARGETNAME).exe: $(TARGETNAME).exe
640 $(CP) $(TARGETNAME).exe ../$(DIST_DIR)/$(TARGETNAME).exe
641
642 .PHONY: dist
643
644 ex/napi.o: ex/napi.c ../include/ntdll/napi.h
645
646 ke/main.o: ke/main.c ../include/reactos/buildno.h
647
648 mkconfig$(EXE_SUFFIX): mkconfig.c
649 $(HOST_CC) -g -o mkconfig$(EXE_SUFFIX) mkconfig.c
650
651 config:
652 $(EXE_PREFIX)mkconfig$(EXE_SUFFIX) include/internal/config.h $(CONFIG)
653
654 include/internal/config.h: config mkconfig$(EXE_SUFFIX)
655 $(EXE_PREFIX)mkconfig$(EXE_SUFFIX) include/internal/config.h$(CONFIG)
656
657 ifneq ($(MAKECMDGOALS),clean)
658 include $(D1_FILES)
659 endif
660
661 .%.d: %.c $(EXE_PREFIX)depends$(EXE_POSTFIX)
662 $(CC) $(CFLAGS) -M $< | $(EXE_PREFIX)depends$(EXE_POSTFIX) $(@D) $@
663
664 .%.d: %.s
665 $(CC) $(CFLAGS) -M $< | $(EXE_PREFIX)depends$(EXE_POSTFIX) $(@D) $@
666
667 .%.d: %.S
668 $(CC) $(CFLAGS) -M $< | $(EXE_PREFIX)depends$(EXE_POSTFIX) $(@D) $@
669
670 .%.d: %.asm
671 $(NASM_CMD) $< | $(EXE_PREFIX)depends$(EXE_POSTFIX) $(@D) $@
672
673 # EOF