-
Notifications
You must be signed in to change notification settings - Fork 71
/
Copy pathsr.h
847 lines (764 loc) · 29.2 KB
/
sr.h
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
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
/* SR.H (c)Copyright Greg Smith, 2004-2012 */
/* Suspend/Resume a Hercules session */
/* */
/* Released under "The Q Public License Version 1" */
/* (http://www.hercules-390.org/herclic.html) as modifications to */
/* Hercules. */
/*
* The suspend/resume functions allow a hercules instance to be
* captured to a file and later resumed. Note that the suspend
* function also terminates the hercules instance.
*
* In order for an instance to be resumed, hercules must be started
* with a config file describing the configuration at suspend time.
* For example, mainsize and xpndsize must match. Also, all devices
* present at suspend time must be present at resume time.
*
* Disk devices must be at the same state as they were at suspend
* time. They can, however, be a different file type. That is,
* a disk could be a cckd disk at suspend time. Then a ckd disk
* could be created using dasdcopy and hercules resumed using the
* ckd disk instead.
*
* Also, hercules must be configured similarly as at suspend time.
* For example, if 4 emulated CPUs were active at suspend time
* then the session can not be resumed on a hercules with a
* maximum of two CPUs. Another example, you will not be able
* to resume a session in z900 architecture mode for a hercules
* that was built without z900 architecture.
*
* Device state
*
* Currently, device state is only fully saved for CKD disks.
* Each device class (eg TAPE, RDR, PUN, CTC) will need code
* to save and restore their state. Some states may not be
* possible to restore (eg active tcp/ip connections at the
* time of suspend).
*
* Further limitations
*
* Currently the vector facility state is not saved.
* Also, the ecpsvm state is not currently saved.
*
* File Structure
*
* The suspend/resume file (.srf) contains some number of `text
* units'. A text unit has an 8 byte header followed by zero or
* more bytes of data.
*
* The file is designed to be hercules release independent and
* to be host architecture independent. For example, I should be
* able to take an srf file created on hercules 3.02 on an intel
* machine and resume on a Sun machine running hercules 3.04.
*
* The header contains a 4 byte key and a 4 byte length. Both
* the key and the length are stored in big-endian byte order.
*
* There are 3 types of data: STRING, BUF and VALUE.
*
* A string is a null terminated sequence of bytes. The string
* includes the null terminator byte. The total length of a
* string cannot exceed SR_MAX_STRING_LENGTH (4096). The
* length is checked by SR_READ_STRING.
*
* A buf is a sequence of bytes whose length must be provided.
* The length should be checked before issuing SR_READ_BUF.
*
* A value is an arithmetic number. It's length can be
* 0, 1, 2, 4 or 8 bytes. Values are stored in big-endian
* byte order. A zero length indicates the value is 0.
*
* Text Units
*
* There are 4 categories (so far) of text units:
* HDR ... fields that describe the file
* SYS ... fields from SYSBLK
* CPU ... fields from REGS
* DEV ... fields from DEVBLK
*
* The format of a text unit key value is
* ace c t xxx
*
* ace -- all keys start with 0xace
* c -- category (0 - HDR, 1 - SYS, 2 - CPU, 3 - DEV)
* t -- for DEV keys, identifies the subtype for the
* device type (0 - general, 1 - CKD, ...)
* xxx -- field identifier
*
* Note that there is no array data type. When an array
* needs to be used, the elements should have ascending
* text unit key values. For example:
*
* #define SR_CPU_GR 0xace20020
* #define SR_CPU_GR_0 0xace20020
* #define SR_CPU_GR_1 0xace20021
* #define SR_CPU_GR_2 0xace20022
* #define SR_CPU_GR_3 0xace20023
* . . . . . .
*
* The array can be written during suspend as follows:
*
* for (i = 0; i < 16; i++)
* SR_WRITE_VALUE(fd, SR_CPU_GR+i, regs->gr[i], sizeof(regs->gr[0]));
*
* The array can be processed during resume as follows
*
* case SR_CPU_GR_0:
* case SR_CPU_GR_1:
* case SR_CPU_GR_2:
* case SR_CPU_GR_3:
* . . . . . .
* i = key - SR_CPU_GR;
* SR_READ_VALUE(fd, len, ®s->gr[i], sizeof(regs->gr[0]));
* break;
*
* The format of the .srf file is deliberately unstructured to
* allow for flexibility in future enhancements. However there
* are a few restrictions.
*
* o Key SR_SYS_ARCHNAME shoud be specified before any
* SR_CPU keys. The corresponding CPU is configured
* when the SR_CPU key is read, so sysblk.arch_mode must
* already be correctly set.
* o SR_CPU_ keys must follow the corresponding SR_CPU key.
* o Likewise SR_DEV_ keys must follow the corresponding SR_DEV key
*
* There may be other instances where the processing of one
* key requires that another key has been previously processed.
*
*/
#ifndef _HERCULES_SR_H
#define _HERCULES_SR_H
#include "opcode.h"
#define SR_ID "Hercules suspend/resume file"
#define SR_MAX_STRING_LENGTH 4096
#define SR_SKIP_CHUNKSIZE 256
#define SR_BUF_CHUNKSIZE (256*1024*1024)
#define SR_KEY_ID_MASK 0xfff00000
#define SR_KEY_ID 0xace00000
#define SR_HDR_ID 0xace00000
#define SR_HDR_VERSION 0xace00001
#define SR_HDR_DATE 0xace00002
#define SR_SYS_MASK 0xfffff000
#define SR_SYS_STARTED_MASK 0xace10000
#define SR_SYS_INTS_STATE 0xace10001
#define SR_SYS_ARCH_NAME 0xace10002
#define SR_SYS_MAINSIZE 0xace10007
#define SR_SYS_MAINSTOR 0xace10008
#define SR_SYS_SKEYSIZE 0xace10009
#define SR_SYS_STORKEYS 0xace1000a
#define SR_SYS_XPNDSIZE 0xace1000b
#define SR_SYS_XPNDSTOR 0xace1000c
#define SR_SYS_CPUID 0xace1000d
#define SR_SYS_IPLDEV 0xace1000e
#define SR_SYS_IPLCPU 0xace1000f
#define SR_SYS_MBO 0xace10010
#define SR_SYS_MBK 0xace10011
#define SR_SYS_MBM 0xace10012
#define SR_SYS_MBD 0xace10013
#define SR_SYS_IOINTQ 0xace10020
#define SR_SYS_IOPENDING 0xace10021
#define SR_SYS_PCIPENDING 0xace10022
#define SR_SYS_ATTNPENDING 0xace10023
#define SR_SYS_IPLED 0xace10030
#define SR_SYS_CRWCOUNT 0xace10031
#define SR_SYS_CRWARRAY 0xace10032
#define SR_SYS_CRWINDEX 0xace10033
#define SR_SYS_CRWENDIAN 0xace10034
#define SR_SYS_SERVPARM 0xace10040
#define SR_SYS_SIGINTREQ 0xace10041
#define SR_SYS_VMACTIVE 0xace10042
#define SR_SYS_MSCHDELAY 0xace10043
#define SR_SYS_LOADPARM 0xace10044
/*
* Following 3 tags added for Multiple
* Logical Channel Subsystem support
*/
#define SR_SYS_IOPENDING_LCSS 0xace10045
#define SR_SYS_PCIPENDING_LCSS 0xace10046
#define SR_SYS_ATTNPENDING_LCSS 0xace10047
#define SR_SYS_CPUMODEL 0xace10048
#define SR_SYS_CPUVERSION 0xace10049
#define SR_SYS_CPUSERIAL 0xace10050
#define SR_SYS_LPARMODE 0xace10051
#define SR_SYS_LPARNUM 0xace10052
#define SR_SYS_CPUIDFMT 0xace10053
#define SR_SYS_OPERATION_MODE 0xace10054
#define SR_SYS_SERVC 0xace11000
#define SR_SYS_CLOCK 0xace12000
#define SR_CPU 0xace20000
#define SR_CPU_ARCHMODE 0xace20001
#define SR_CPU_PX 0xace20002
#define SR_CPU_PSW 0xace20003
#define SR_CPU_GR 0xace20020
#define SR_CPU_GR_0 0xace20020
#define SR_CPU_GR_1 0xace20021
#define SR_CPU_GR_2 0xace20022
#define SR_CPU_GR_3 0xace20023
#define SR_CPU_GR_4 0xace20024
#define SR_CPU_GR_5 0xace20025
#define SR_CPU_GR_6 0xace20026
#define SR_CPU_GR_7 0xace20027
#define SR_CPU_GR_8 0xace20028
#define SR_CPU_GR_9 0xace20029
#define SR_CPU_GR_10 0xace2002a
#define SR_CPU_GR_11 0xace2002b
#define SR_CPU_GR_12 0xace2002c
#define SR_CPU_GR_13 0xace2002d
#define SR_CPU_GR_14 0xace2002e
#define SR_CPU_GR_15 0xace2002f
#define SR_CPU_CR 0xace20040
#define SR_CPU_CR_0 0xace20040
#define SR_CPU_CR_1 0xace20041
#define SR_CPU_CR_2 0xace20042
#define SR_CPU_CR_3 0xace20043
#define SR_CPU_CR_4 0xace20044
#define SR_CPU_CR_5 0xace20045
#define SR_CPU_CR_6 0xace20046
#define SR_CPU_CR_7 0xace20047
#define SR_CPU_CR_8 0xace20048
#define SR_CPU_CR_9 0xace20049
#define SR_CPU_CR_10 0xace2004a
#define SR_CPU_CR_11 0xace2004b
#define SR_CPU_CR_12 0xace2004c
#define SR_CPU_CR_13 0xace2004d
#define SR_CPU_CR_14 0xace2004e
#define SR_CPU_CR_15 0xace2004f
#define SR_CPU_AR 0xace20060
#define SR_CPU_AR_0 0xace20060
#define SR_CPU_AR_1 0xace20061
#define SR_CPU_AR_2 0xace20062
#define SR_CPU_AR_3 0xace20063
#define SR_CPU_AR_4 0xace20064
#define SR_CPU_AR_5 0xace20065
#define SR_CPU_AR_6 0xace20066
#define SR_CPU_AR_7 0xace20067
#define SR_CPU_AR_8 0xace20068
#define SR_CPU_AR_9 0xace20069
#define SR_CPU_AR_10 0xace2006a
#define SR_CPU_AR_11 0xace2006b
#define SR_CPU_AR_12 0xace2006c
#define SR_CPU_AR_13 0xace2006d
#define SR_CPU_AR_14 0xace2006e
#define SR_CPU_AR_15 0xace2006f
#define SR_CPU_FPR 0xace20080
#define SR_CPU_FPR_0 0xace20080
#define SR_CPU_FPR_1 0xace20081
#define SR_CPU_FPR_2 0xace20082
#define SR_CPU_FPR_3 0xace20083
#define SR_CPU_FPR_4 0xace20084
#define SR_CPU_FPR_5 0xace20085
#define SR_CPU_FPR_6 0xace20086
#define SR_CPU_FPR_7 0xace20087
#define SR_CPU_FPR_8 0xace20088
#define SR_CPU_FPR_9 0xace20089
#define SR_CPU_FPR_10 0xace2008a
#define SR_CPU_FPR_11 0xace2008b
#define SR_CPU_FPR_12 0xace2008c
#define SR_CPU_FPR_13 0xace2008d
#define SR_CPU_FPR_14 0xace2008e
#define SR_CPU_FPR_15 0xace2008f
#define SR_CPU_FPR_16 0xace20090
#define SR_CPU_FPR_17 0xace20091
#define SR_CPU_FPR_18 0xace20092
#define SR_CPU_FPR_19 0xace20093
#define SR_CPU_FPR_20 0xace20094
#define SR_CPU_FPR_21 0xace20095
#define SR_CPU_FPR_22 0xace20096
#define SR_CPU_FPR_23 0xace20097
#define SR_CPU_FPR_24 0xace20098
#define SR_CPU_FPR_25 0xace20099
#define SR_CPU_FPR_26 0xace2009a
#define SR_CPU_FPR_27 0xace2009b
#define SR_CPU_FPR_28 0xace2009c
#define SR_CPU_FPR_29 0xace2009d
#define SR_CPU_FPR_30 0xace2009e
#define SR_CPU_FPR_31 0xace2009f
#define SR_CPU_FPC 0xace20100
#define SR_CPU_DXC 0xace20101
#define SR_CPU_MC 0xace20102
#define SR_CPU_EA 0xace20103
#define SR_CPU_PTIMER 0xace20104
#define SR_CPU_CLKC 0xace20105
#define SR_CPU_CHANSET 0xace20106
#define SR_CPU_TODPR 0xace20107
#define SR_CPU_MONCLASS 0xace20108
#define SR_CPU_EXCARID 0xace20109
#define SR_CPU_INTS_STATE 0xace2010a
#define SR_CPU_INTS_MASK 0xace2010b
#define SR_CPU_EXTCCPU 0xace2010c
#define SR_CPU_BEAR 0xace2010d
#define SR_CPU_OPNDRID 0xace20110
#define SR_CPU_CHECKSTOP 0xace20111
#define SR_CPU_HOSTINT 0xace20112
#define SR_CPU_EXECFLAG 0xace20113
#define SR_CPU_INSTVALID 0xace20114
#define SR_CPU_PERMODE 0xace20115
#define SR_CPU_LOADSTATE 0xace20116
#define SR_CPU_INVALIDATE 0xace20117
#define SR_CPU_RESET_OPCTAB 0xace20118
#define SR_CPU_SIGPRESET 0xace20119
#define SR_CPU_SIGPIRESET 0xace2011a
#define SR_CPU_VTIMERINT 0xace2011b
#define SR_CPU_RTIMERINT 0xace2011c
#define SR_CPU_MALFCPU 0xace20120
#define SR_CPU_MALFCPU_0 0xace20120
#define SR_CPU_MALFCPU_1 0xace20121
#define SR_CPU_MALFCPU_2 0xace20122
#define SR_CPU_MALFCPU_3 0xace20123
#define SR_CPU_MALFCPU_4 0xace20124
#define SR_CPU_MALFCPU_5 0xace20125
#define SR_CPU_MALFCPU_6 0xace20126
#define SR_CPU_MALFCPU_7 0xace20127
#define SR_CPU_MALFCPU_8 0xace20128
#define SR_CPU_MALFCPU_9 0xace20129
#define SR_CPU_MALFCPU_10 0xace2012a
#define SR_CPU_MALFCPU_11 0xace2012b
#define SR_CPU_MALFCPU_12 0xace2012c
#define SR_CPU_MALFCPU_13 0xace2012d
#define SR_CPU_MALFCPU_14 0xace2012e
#define SR_CPU_MALFCPU_15 0xace2012f
#define SR_CPU_MALFCPU_16 0xace20130
#define SR_CPU_MALFCPU_17 0xace20131
#define SR_CPU_MALFCPU_18 0xace20132
#define SR_CPU_MALFCPU_19 0xace20133
#define SR_CPU_MALFCPU_20 0xace20134
#define SR_CPU_MALFCPU_21 0xace20135
#define SR_CPU_MALFCPU_22 0xace20136
#define SR_CPU_MALFCPU_23 0xace20137
#define SR_CPU_MALFCPU_24 0xace20138
#define SR_CPU_MALFCPU_25 0xace20139
#define SR_CPU_MALFCPU_26 0xace2013a
#define SR_CPU_MALFCPU_27 0xace2013b
#define SR_CPU_MALFCPU_28 0xace2013c
#define SR_CPU_MALFCPU_29 0xace2013d
#define SR_CPU_MALFCPU_30 0xace2013e
#define SR_CPU_MALFCPU_31 0xace2013f
#define SR_CPU_EMERCPU 0xace20140
#define SR_CPU_EMERCPU_0 0xace20140
#define SR_CPU_EMERCPU_1 0xace20141
#define SR_CPU_EMERCPU_2 0xace20142
#define SR_CPU_EMERCPU_3 0xace20143
#define SR_CPU_EMERCPU_4 0xace20144
#define SR_CPU_EMERCPU_5 0xace20145
#define SR_CPU_EMERCPU_6 0xace20146
#define SR_CPU_EMERCPU_7 0xace20147
#define SR_CPU_EMERCPU_8 0xace20148
#define SR_CPU_EMERCPU_9 0xace20149
#define SR_CPU_EMERCPU_10 0xace2014a
#define SR_CPU_EMERCPU_11 0xace2014b
#define SR_CPU_EMERCPU_12 0xace2014c
#define SR_CPU_EMERCPU_13 0xace2014d
#define SR_CPU_EMERCPU_14 0xace2014e
#define SR_CPU_EMERCPU_15 0xace2014f
#define SR_CPU_EMERCPU_16 0xace20150
#define SR_CPU_EMERCPU_17 0xace20151
#define SR_CPU_EMERCPU_18 0xace20152
#define SR_CPU_EMERCPU_19 0xace20153
#define SR_CPU_EMERCPU_20 0xace20154
#define SR_CPU_EMERCPU_21 0xace20155
#define SR_CPU_EMERCPU_22 0xace20156
#define SR_CPU_EMERCPU_23 0xace20157
#define SR_CPU_EMERCPU_24 0xace20158
#define SR_CPU_EMERCPU_25 0xace20159
#define SR_CPU_EMERCPU_26 0xace2015a
#define SR_CPU_EMERCPU_27 0xace2015b
#define SR_CPU_EMERCPU_28 0xace2015c
#define SR_CPU_EMERCPU_29 0xace2015d
#define SR_CPU_EMERCPU_30 0xace2015e
#define SR_CPU_EMERCPU_31 0xace2015f
#define SR_DEV 0xace30000
#define SR_DEV_DEVTYPE 0xace30001
#define SR_DEV_ARGC 0xace30002
#define SR_DEV_ARGV 0xace30003
#define SR_DEV_TYPNAME 0xace30004
/*
* Following tag added for multiple Logical
* Channel subsystem support
*/
#define SR_DEV_LCSS 0xace30005
#define SR_DEV_ORB 0xace30010
#define SR_DEV_PMCW 0xace30011
#define SR_DEV_SCSW 0xace30012
#define SR_DEV_PCISCSW 0xace30013
#define SR_DEV_ATTNSCSW 0xace30014
/* available 0xace30015 */
/* available 0xace30016 */
/* available 0xace30017 */
#define SR_DEV_ESW 0xace30018
#define SR_DEV_ECW 0xace30019
#define SR_DEV_SENSE 0xace3001a
#define SR_DEV_PGSTAT 0xace3001b
#define SR_DEV_PGID 0xace3001c
/* By Adrian - SR_DEV_DRVPWD */
#define SR_DEV_DRVPWD 0xace3001d
/* By Ian - SR_DEV_NUMCONFDEV */
#define SR_DEV_NUMCONFDEV 0xace3001e
#define SR_DEV_BUSY 0xace30020
#define SR_DEV_RESERVED 0xace30021
#define SR_DEV_SUSPENDED 0xace30022
#define SR_DEV_PENDING 0xace30023
#define SR_DEV_PCIPENDING 0xace30024
#define SR_DEV_ATTNPENDING 0xace30025
#define SR_DEV_STARTPENDING 0xace30026
// (available) 0xace30027
#define SR_DEV_CCWADDR 0xace30028
#define SR_DEV_IDAPMASK 0xace30029
#define SR_DEV_IDAWFMT 0xace3002a
#define SR_DEV_CCWFMT 0xace3002b
#define SR_DEV_CCWKEY 0xace3002c
#define SR_DEV_MASK 0xfffff000
#define SR_DEV_CKD 0xace31000
#define SR_DEV_FBA 0xace32000
#define SR_DEV_TTY 0xace33000
#define SR_DEV_3270 0xace34000
#define SR_DEV_RDR 0xace35000
#define SR_DEV_PUN 0xace36000
#define SR_DEV_PRT 0xace37000
#define SR_DEV_TAPE 0xace38000
#define SR_DEV_COMM 0xace39000
#define SR_DEV_CTC 0xace3a000
#define SR_DEV_CTCI 0xace3b000
#define SR_DEV_CTCT 0xace3c000
#define SR_DEV_VMNET 0xace3d000
#define SR_DEV_LCS 0xace3e000
#define SR_DEV_CTCE 0xace3f000
#define SR_DELIMITER 0xaceffffe
#define SR_EOF 0xacefffff
#if defined (_HERCULES_SR_C)
#define SR_WRITE_ERROR goto sr_write_error
#define SR_READ_ERROR goto sr_read_error
#define SR_SEEK_ERROR goto sr_seek_error
#define SR_VALUE_ERROR goto sr_value_error
#define SR_STRING_ERROR goto sr_string_error
#else
#define SR_WRITE_ERROR do { sr_write_error_(); return -1; } while (0)
#define SR_READ_ERROR do { sr_read_error_(); return -1; } while (0)
#define SR_SEEK_ERROR do { sr_seek_error_(); return -1; } while (0)
#define SR_VALUE_ERROR do { sr_value_error_(); return -1; } while (0)
#define SR_STRING_ERROR do { sr_string_error_(); return -1; } while (0)
#endif
#ifdef HAVE_LIBZ
#define SR_DEFAULT_FILENAME "hercules.srf.gz"
#define SR_FILE gzFile
#define SR_OPEN(_path, _mode) \
gzopen((_path), (_mode))
#define SR_READ(_ptr, _size, _nmemb, _stream) \
gzread((gzFile)(_stream), (_ptr), (unsigned int)((_size) * (_nmemb)))
#define SR_WRITE(_ptr, _size, _nmemb, _stream) \
gzwrite((gzFile)(_stream), (_ptr), (unsigned int)((_size) * (_nmemb)))
#define SR_SEEK(_stream, _offset, _whence) \
gzseek((gzFile)(_stream), (_offset), (_whence))
#define SR_CLOSE(_stream) \
gzclose((gzFile)(_stream))
#else
#define SR_DEFAULT_FILENAME "hercules.srf"
#define SR_FILE FILE *
#define SR_OPEN(_path, _mode) \
fopen((_path), (_mode))
#define SR_READ(_ptr, _size, _nmemb, _stream) \
fread((_ptr), (_size), (_nmemb), (_stream))
#define SR_WRITE(_ptr, _size, _nmemb, _stream) \
fwrite((_ptr), (_size), (_nmemb), (_stream))
#define SR_SEEK(_stream, _offset, _whence) \
fseek((_stream), (_offset), (_whence))
#define SR_CLOSE(_stream) \
fclose((_stream))
#endif
static INLINE int sr_write_hdr (FILE* file, U32 key, U32 len);
static INLINE int sr_write_value (FILE* file, U32 key, U64 val, U32 len);
static INLINE int sr_write_buf (FILE* file, U32 key, void *buf, U64 len);
static INLINE int sr_write_string (FILE* file, U32 key, void *str );
static INLINE int sr_read_hdr (FILE* file, U32 *key, U32 *len);
static INLINE int sr_read_value (FILE* file, U32 origlen, void *p, U32 len);
static INLINE int sr_read_buf (FILE* file, void *p, U64 len);
static INLINE int sr_read_string (FILE* file, void *p, U32 len);
static INLINE int sr_read_skip (FILE* file, U32 len);
static INLINE void sr_write_error_();
static INLINE void sr_read_error_();
static INLINE void sr_seek_error_();
static INLINE void sr_value_error_();
static INLINE void sr_string_error_();
#define SR_WRITE_HDR( _file, _key, _len) \
do {if (sr_write_hdr((FILE*)(_file), (U32)(_key), (U32)(_len)) != 0) return -1; } while (0)
#define SR_WRITE_STRING( _file, _key, _str) \
do {if (sr_write_string((FILE*)(_file), (U32)(_key), (void*)(_str)) != 0) return -1; } while (0)
#define SR_WRITE_BUF( _file, _key, _buf, _len) \
do {if (sr_write_buf((FILE*)(_file), (U32)(_key), (void*)(_buf), (U64)(_len)) != 0) return -1; } while (0)
#define SR_WRITE_VALUE( _file, _key, _val, _len) \
do {if (sr_write_value((FILE*)(_file), (U32)(_key), (U64)(_val), (U32)(_len)) != 0) return -1; } while (0)
#define SR_READ_HDR( _file, _key, _len) \
do { \
U32 k, l; \
if (sr_read_hdr((FILE*)(_file), &k, &l) != 0) \
return -1; \
(_key) = k; \
(_len) = l; \
} while (0)
#define SR_READ_SKIP( _file, _len) \
do {if (sr_read_skip((FILE*)(_file), (U32)(_len)) != 0) return -1; } while (0)
#define SR_READ_STRING( _file, _p, _len) \
do {if (sr_read_string((FILE*)(_file), (void*)(_p), (U32)(_len)) != 0) return -1; } while (0)
#define SR_READ_BUF( _file, _p, _len) \
do {if (sr_read_buf((FILE*)(_file), (void*)(_p), (U64)(_len)) != 0) return -1; } while (0)
#define SR_READ_VALUE( _file, _suslen, _p, _reslen) \
do {if (sr_read_value((FILE*)(_file), (U32)(_suslen), (void*)(_p), (U32)(_reslen)) != 0) return -1; } while (0)
#define SR_SKIP_NULL_DEV(_dev, _file, _len) \
if ((_dev) == NULL) { \
SR_READ_SKIP((_file),(_len)); \
break; \
}
/*********************************************************************/
/* sr_write_hdr */
/*********************************************************************/
static INLINE int sr_write_hdr (FILE* file, U32 key, U32 len)
{
BYTE buf[8];
TRACE("SR: sr_write_hdr: key=0x%8.8x, len=0x%8.8x\n", key, len);
store_fw (buf, key);
store_fw (buf+4, len);
if (SR_WRITE(buf, 1, 8, file) != 8)
{
sr_write_error_();
return -1;
}
return 0;
}
/*********************************************************************/
/* sr_write_string */
/*********************************************************************/
static INLINE int sr_write_string (FILE* file, U32 key, void* str)
{
size_t len = strlen(str) + 1;
TRACE("SR: sr_write_string: key=0x%8.8x\n", key);
if (len > SR_MAX_STRING_LENGTH)
{
sr_string_error_();
return -1;
}
if (sr_write_hdr(file, key, (U32)len) != 0)
return -1;
if ((size_t)SR_WRITE(str, 1, len, file) != len)
{
sr_write_error_();
return -1;
}
return 0;
}
/*********************************************************************/
/* sr_write_buf */
/*********************************************************************/
static INLINE int sr_write_buf (FILE* file, U32 key, void* p, U64 len)
{
U32 siz;
U64 tot = len;
BYTE* buf = p;
if (sr_write_hdr(file, key, len) != 0)
return -1;
TRACE("SR: sr_write_buf: key=0x%8.8x, len=0x%16.16"PRIx64"\n", key, len);
while (tot)
{
siz = tot < SR_BUF_CHUNKSIZE ? tot : SR_BUF_CHUNKSIZE;
if ((U32)SR_WRITE(buf, 1, siz, file) != siz)
{
sr_write_error_();
return -1;
}
tot -= siz;
buf += siz;
}
return 0;
}
/*********************************************************************/
/* sr_write_value */
/*********************************************************************/
static INLINE int sr_write_value (FILE* file, U32 key, U64 val, U32 len)
{
BYTE buf[8];
TRACE("SR: sr_write_value: key=0x%8.8x, len=0x%8.8x, val=0x%16.16"PRIx64"\n", key, len, val);
if (len != 1 && len != 2 && len != 4 && len != 8)
{
sr_value_error_();
return -1;
}
if (sr_write_hdr(file, key, len) != 0)
return -1;
switch (len)
{
case 1: buf[0] = (BYTE)val; break;
case 2: store_hw (buf, (U16)val); break;
case 4: store_fw (buf, (U32)val); break;
case 8: store_dw (buf, (U64)val); break;
}
if ((U32)SR_WRITE(buf, 1, len, file) != len)
{
sr_write_error_();
return -1;
}
return 0;
}
/*********************************************************************/
/* sr_read_hdr */
/*********************************************************************/
static INLINE int sr_read_hdr (FILE* file, U32* key, U32* len)
{
BYTE buf[8];
if (SR_READ(buf, 1, 8, file) != 8)
{
sr_read_error_();
return -1;
}
*key = fetch_fw (buf);
*len = fetch_fw (buf+4);
TRACE("SR: sr_read_hdr: key=0x%8.8x, len=0x%8.8x\n", *key, *len);
return 0;
}
/*********************************************************************/
/* sr_read_skip */
/*********************************************************************/
static INLINE int sr_read_skip (FILE* file, U32 len)
{
/* FIXME: Workaround for problem involving gzseek
and large files. Just read the data. */
BYTE buf[SR_SKIP_CHUNKSIZE];
size_t siz;
size_t tot;
TRACE("SR: sr_read_skip: len=0x%8.8x\n", len);
tot = len;
while (tot)
{
siz = tot < SR_SKIP_CHUNKSIZE ? tot : SR_SKIP_CHUNKSIZE;
if ((size_t)SR_READ(buf, 1, siz, file) != siz)
{
sr_read_error_();
return -1;
}
tot -= siz;
}
return 0;
}
/*********************************************************************/
/* sr_read_string */
/*********************************************************************/
static INLINE int sr_read_string (FILE* file, void* p, U32 len)
{
TRACE("SR: sr_read_string: len=0x%8.8x\n", len);
if (len > SR_MAX_STRING_LENGTH)
{
sr_string_error_();
return -1;
}
if ((U32)SR_READ(p, 1, len, file) != len)
{
sr_read_error_();
return -1;
}
return 0;
}
/*********************************************************************/
/* sr_read_buf */
/*********************************************************************/
static INLINE int sr_read_buf (FILE* file, void* p, U64 len)
{
U32 siz;
U64 tot = len;
BYTE* buf = p;
TRACE("SR: sr_read_buf: len=0x%16.16"PRIx64"\n", len);
while (tot)
{
siz = tot < SR_BUF_CHUNKSIZE ? tot : SR_BUF_CHUNKSIZE;
if ((U32)SR_READ(buf, 1, siz, file) != siz)
{
sr_read_error_();
return -1;
}
tot -= siz;
buf += siz;
}
return 0;
}
/*********************************************************************/
/* sr_read_value */
/*********************************************************************/
static INLINE int sr_read_value (FILE* file, U32 suslen, void* p, U32 reslen)
{
BYTE buf[8];
U64 value;
TRACE("SR: sr_read_value: suslen=0x%8.8x, reslen=0x%8.8x,\n", suslen, reslen);
if (suslen != 1 && suslen != 2 && suslen != 4 && suslen != 8)
{
sr_value_error_();
return -1;
}
if ((U32)SR_READ(buf, 1, suslen, file) != suslen)
{
sr_read_error_();
return -1;
}
switch (suslen)
{
case 1: value = buf[0]; break;
case 2: value = fetch_hw (buf); break;
case 4: value = fetch_fw (buf); break;
case 8: value = fetch_dw (buf); break;
default: value = 0; break; /* To ward off gcc -Wall */
}
TRACE(" val=0x%16.16"PRIx64"\n", value);
switch (reslen)
{
case 1:
{
BYTE* ptr = p;
*ptr = (BYTE)(value & 0xFF);
break;
}
case 2:
{
U16* ptr = p;
*ptr = (U16)(value & 0xFFFF);
break;
}
case 4:
{
U32* ptr = p;
*ptr = (U32)(value & 0xFFFFFFFF);
break;
}
case 8:
{
U64* ptr = p;
*ptr = (U64)(value & 0xFFFFFFFFFFFFFFFFULL);
break;
}
}
return 0;
}
/*********************************************************************/
/* sr_xxxxx_error_ */
/*********************************************************************/
static INLINE void sr_write_error_()
{
// "SR: error in function '%s': '%s'"
WRMSG(HHC02001, "E", "write()", strerror(errno));
}
static INLINE void sr_read_error_()
{
// "SR: error in function '%s': '%s'"
WRMSG(HHC02001, "E", "read()", strerror(errno));
}
static INLINE void sr_seek_error_()
{
// "SR: error in function '%s': '%s'"
WRMSG(HHC02001, "E", "lseek()", strerror(errno));
}
static INLINE void sr_value_error_()
{
// "SR: value error, incorrect length"
WRMSG(HHC02020, "E");
}
static INLINE void sr_string_error_()
{
// "SR: string error, incorrect length"
WRMSG(HHC02021, "E");
}
#endif /* !defined(_HERCULES_SR_H) */