• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2007 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.android.dexgen.dex.code;
18 
19 import com.android.dexgen.rop.code.LocalItem;
20 import com.android.dexgen.rop.code.RegisterSpec;
21 import com.android.dexgen.rop.code.RegisterSpecList;
22 import com.android.dexgen.rop.code.RegisterSpecSet;
23 import com.android.dexgen.rop.code.SourcePosition;
24 import com.android.dexgen.rop.cst.Constant;
25 import com.android.dexgen.rop.cst.CstMemberRef;
26 import com.android.dexgen.rop.cst.CstType;
27 import com.android.dexgen.rop.cst.CstUtf8;
28 import com.android.dexgen.rop.type.Type;
29 
30 import java.util.ArrayList;
31 import java.util.HashSet;
32 
33 /**
34  * Processor for instruction lists, which takes a "first cut" of
35  * instruction selection as a basis and produces a "final cut" in the
36  * form of a {@link DalvInsnList} instance.
37  */
38 public final class OutputFinisher {
39     /**
40      * {@code >= 0;} register count for the method, not including any extra
41      * "reserved" registers needed to translate "difficult" instructions
42      */
43     private final int unreservedRegCount;
44 
45     /** {@code non-null;} the list of instructions, per se */
46     private ArrayList<DalvInsn> insns;
47 
48     /** whether any instruction has position info */
49     private boolean hasAnyPositionInfo;
50 
51     /** whether any instruction has local variable info */
52     private boolean hasAnyLocalInfo;
53 
54     /**
55      * {@code >= 0;} the count of reserved registers (low-numbered
56      * registers used when expanding instructions that can't be
57      * represented simply); becomes valid after a call to {@link
58      * #massageInstructions}
59      */
60     private int reservedCount;
61 
62     /**
63      * Constructs an instance. It initially contains no instructions.
64      *
65      * @param regCount {@code >= 0;} register count for the method
66      * @param initialCapacity {@code >= 0;} initial capacity of the instructions
67      * list
68      */
OutputFinisher(int initialCapacity, int regCount)69     public OutputFinisher(int initialCapacity, int regCount) {
70         this.unreservedRegCount = regCount;
71         this.insns = new ArrayList<DalvInsn>(initialCapacity);
72         this.reservedCount = -1;
73         this.hasAnyPositionInfo = false;
74         this.hasAnyLocalInfo = false;
75     }
76 
77     /**
78      * Returns whether any of the instructions added to this instance
79      * come with position info.
80      *
81      * @return whether any of the instructions added to this instance
82      * come with position info
83      */
hasAnyPositionInfo()84     public boolean hasAnyPositionInfo() {
85         return hasAnyPositionInfo;
86     }
87 
88     /**
89      * Returns whether this instance has any local variable information.
90      *
91      * @return whether this instance has any local variable information
92      */
hasAnyLocalInfo()93     public boolean hasAnyLocalInfo() {
94         return hasAnyLocalInfo;
95     }
96 
97     /**
98      * Helper for {@link #add} which scrutinizes a single
99      * instruction for local variable information.
100      *
101      * @param insn {@code non-null;} instruction to scrutinize
102      * @return {@code true} iff the instruction refers to any
103      * named locals
104      */
hasLocalInfo(DalvInsn insn)105     private static boolean hasLocalInfo(DalvInsn insn) {
106         if (insn instanceof LocalSnapshot) {
107             RegisterSpecSet specs = ((LocalSnapshot) insn).getLocals();
108             int size = specs.size();
109             for (int i = 0; i < size; i++) {
110                 if (hasLocalInfo(specs.get(i))) {
111                     return true;
112                 }
113             }
114         } else if (insn instanceof LocalStart) {
115             RegisterSpec spec = ((LocalStart) insn).getLocal();
116             if (hasLocalInfo(spec)) {
117                 return true;
118             }
119         }
120 
121         return false;
122     }
123 
124     /**
125      * Helper for {@link #hasAnyLocalInfo} which scrutinizes a single
126      * register spec.
127      *
128      * @param spec {@code non-null;} spec to scrutinize
129      * @return {@code true} iff the spec refers to any
130      * named locals
131      */
hasLocalInfo(RegisterSpec spec)132     private static boolean hasLocalInfo(RegisterSpec spec) {
133         return (spec != null)
134             && (spec.getLocalItem().getName() != null);
135     }
136 
137     /**
138      * Returns the set of all constants referred to by instructions added
139      * to this instance.
140      *
141      * @return {@code non-null;} the set of constants
142      */
getAllConstants()143     public HashSet<Constant> getAllConstants() {
144         HashSet<Constant> result = new HashSet<Constant>(20);
145 
146         for (DalvInsn insn : insns) {
147             addConstants(result, insn);
148         }
149 
150         return result;
151     }
152 
153     /**
154      * Helper for {@link #getAllConstants} which adds all the info for
155      * a single instruction.
156      *
157      * @param result {@code non-null;} result set to add to
158      * @param insn {@code non-null;} instruction to scrutinize
159      */
addConstants(HashSet<Constant> result, DalvInsn insn)160     private static void addConstants(HashSet<Constant> result,
161             DalvInsn insn) {
162         if (insn instanceof CstInsn) {
163             Constant cst = ((CstInsn) insn).getConstant();
164             result.add(cst);
165         } else if (insn instanceof LocalSnapshot) {
166             RegisterSpecSet specs = ((LocalSnapshot) insn).getLocals();
167             int size = specs.size();
168             for (int i = 0; i < size; i++) {
169                 addConstants(result, specs.get(i));
170             }
171         } else if (insn instanceof LocalStart) {
172             RegisterSpec spec = ((LocalStart) insn).getLocal();
173             addConstants(result, spec);
174         }
175     }
176 
177     /**
178      * Helper for {@link #getAllConstants} which adds all the info for
179      * a single {@code RegisterSpec}.
180      *
181      * @param result {@code non-null;} result set to add to
182      * @param spec {@code null-ok;} register spec to add
183      */
addConstants(HashSet<Constant> result, RegisterSpec spec)184     private static void addConstants(HashSet<Constant> result,
185             RegisterSpec spec) {
186         if (spec == null) {
187             return;
188         }
189 
190         LocalItem local = spec.getLocalItem();
191         CstUtf8 name = local.getName();
192         CstUtf8 signature = local.getSignature();
193         Type type = spec.getType();
194 
195         if (type != Type.KNOWN_NULL) {
196             result.add(CstType.intern(type));
197         }
198 
199         if (name != null) {
200             result.add(name);
201         }
202 
203         if (signature != null) {
204             result.add(signature);
205         }
206     }
207 
208     /**
209      * Adds an instruction to the output.
210      *
211      * @param insn {@code non-null;} the instruction to add
212      */
add(DalvInsn insn)213     public void add(DalvInsn insn) {
214         insns.add(insn);
215         updateInfo(insn);
216     }
217 
218     /**
219      * Inserts an instruction in the output at the given offset.
220      *
221      * @param at {@code >= 0;} what index to insert at
222      * @param insn {@code non-null;} the instruction to insert
223      */
insert(int at, DalvInsn insn)224     public void insert(int at, DalvInsn insn) {
225         insns.add(at, insn);
226         updateInfo(insn);
227     }
228 
229     /**
230      * Helper for {@link #add} and {@link #insert},
231      * which updates the position and local info flags.
232      *
233      * @param insn {@code non-null;} an instruction that was just introduced
234      */
updateInfo(DalvInsn insn)235     private void updateInfo(DalvInsn insn) {
236         if (! hasAnyPositionInfo) {
237             SourcePosition pos = insn.getPosition();
238             if (pos.getLine() >= 0) {
239                 hasAnyPositionInfo = true;
240             }
241         }
242 
243         if (! hasAnyLocalInfo) {
244             if (hasLocalInfo(insn)) {
245                 hasAnyLocalInfo = true;
246             }
247         }
248     }
249 
250     /**
251      * Reverses a branch which is buried a given number of instructions
252      * backward in the output. It is illegal to call this unless the
253      * indicated instruction really is a reversible branch.
254      *
255      * @param which how many instructions back to find the branch;
256      * {@code 0} is the most recently added instruction,
257      * {@code 1} is the instruction before that, etc.
258      * @param newTarget {@code non-null;} the new target for the reversed branch
259      */
reverseBranch(int which, CodeAddress newTarget)260     public void reverseBranch(int which, CodeAddress newTarget) {
261         int size = insns.size();
262         int index = size - which - 1;
263         TargetInsn targetInsn;
264 
265         try {
266             targetInsn = (TargetInsn) insns.get(index);
267         } catch (IndexOutOfBoundsException ex) {
268             // Translate the exception.
269             throw new IllegalArgumentException("too few instructions");
270         } catch (ClassCastException ex) {
271             // Translate the exception.
272             throw new IllegalArgumentException("non-reversible instruction");
273         }
274 
275         /*
276          * No need to call this.set(), since the format and other info
277          * are the same.
278          */
279         insns.set(index, targetInsn.withNewTargetAndReversed(newTarget));
280     }
281 
282     /**
283      * Assigns indices in all instructions that need them, using the
284      * given callback to perform lookups. This should be called before
285      * calling {@link #finishProcessingAndGetList}.
286      *
287      * @param callback {@code non-null;} callback object
288      */
assignIndices(DalvCode.AssignIndicesCallback callback)289     public void assignIndices(DalvCode.AssignIndicesCallback callback) {
290         for (DalvInsn insn : insns) {
291             if (insn instanceof CstInsn) {
292                 assignIndices((CstInsn) insn, callback);
293             }
294         }
295     }
296 
297     /**
298      * Helper for {@link #assignIndices} which does assignment for one
299      * instruction.
300      *
301      * @param insn {@code non-null;} the instruction
302      * @param callback {@code non-null;} the callback
303      */
assignIndices(CstInsn insn, DalvCode.AssignIndicesCallback callback)304     private static void assignIndices(CstInsn insn,
305             DalvCode.AssignIndicesCallback callback) {
306         Constant cst = insn.getConstant();
307         int index = callback.getIndex(cst);
308 
309         if (index >= 0) {
310             insn.setIndex(index);
311         }
312 
313         if (cst instanceof CstMemberRef) {
314             CstMemberRef member = (CstMemberRef) cst;
315             CstType definer = member.getDefiningClass();
316             index = callback.getIndex(definer);
317             if (index >= 0) {
318                 insn.setClassIndex(index);
319             }
320         }
321     }
322 
323     /**
324      * Does final processing on this instance and gets the output as
325      * a {@link DalvInsnList}. Final processing consists of:
326      *
327      * <ul>
328      *   <li>optionally renumbering registers (to make room as needed for
329      *   expanded instructions)</li>
330      *   <li>picking a final opcode for each instruction</li>
331      *   <li>rewriting instructions, because of register number,
332      *   constant pool index, or branch target size issues</li>
333      *   <li>assigning final addresses</li>
334      * </ul>
335      *
336      * <p><b>Note:</b> This method may only be called once per instance
337      * of this class.</p>
338      *
339      * @return {@code non-null;} the output list
340      * @throws UnsupportedOperationException if this method has
341      * already been called
342      */
finishProcessingAndGetList()343     public DalvInsnList finishProcessingAndGetList() {
344         if (reservedCount >= 0) {
345             throw new UnsupportedOperationException("already processed");
346         }
347 
348         InsnFormat[] formats = makeFormatsArray();
349         reserveRegisters(formats);
350         massageInstructions(formats);
351         assignAddressesAndFixBranches();
352 
353         return DalvInsnList.makeImmutable(insns,
354                 reservedCount + unreservedRegCount);
355     }
356 
357     /**
358      * Helper for {@link #finishProcessingAndGetList}, which extracts
359      * the format out of each instruction into a separate array, to be
360      * further manipulated as things progress.
361      *
362      * @return {@code non-null;} the array of formats
363      */
makeFormatsArray()364     private InsnFormat[] makeFormatsArray() {
365         int size = insns.size();
366         InsnFormat[] result = new InsnFormat[size];
367 
368         for (int i = 0; i < size; i++) {
369             result[i] = insns.get(i).getOpcode().getFormat();
370         }
371 
372         return result;
373     }
374 
375     /**
376      * Helper for {@link #finishProcessingAndGetList}, which figures
377      * out how many reserved registers are required and then reserving
378      * them. It also updates the given {@code formats} array so
379      * as to avoid extra work when constructing the massaged
380      * instruction list.
381      *
382      * @param formats {@code non-null;} array of per-instruction format selections
383      */
reserveRegisters(InsnFormat[] formats)384     private void reserveRegisters(InsnFormat[] formats) {
385         int oldReservedCount = (reservedCount < 0) ? 0 : reservedCount;
386 
387         /*
388          * Call calculateReservedCount() and then perform register
389          * reservation, repeatedly until no new reservations happen.
390          */
391         for (;;) {
392             int newReservedCount = calculateReservedCount(formats);
393             if (oldReservedCount >= newReservedCount) {
394                 break;
395             }
396 
397             int reservedDifference = newReservedCount - oldReservedCount;
398             int size = insns.size();
399 
400             for (int i = 0; i < size; i++) {
401                 /*
402                  * CodeAddress instance identity is used to link
403                  * TargetInsns to their targets, so it is
404                  * inappropriate to make replacements, and they don't
405                  * have registers in any case. Hence, the instanceof
406                  * test below.
407                  */
408                 DalvInsn insn = insns.get(i);
409                 if (!(insn instanceof CodeAddress)) {
410                     /*
411                      * No need to call this.set() since the format and
412                      * other info are the same.
413                      */
414                     insns.set(i, insn.withRegisterOffset(reservedDifference));
415                 }
416             }
417 
418             oldReservedCount = newReservedCount;
419         }
420 
421         reservedCount = oldReservedCount;
422     }
423 
424     /**
425      * Helper for {@link #reserveRegisters}, which does one
426      * pass over the instructions, calculating the number of
427      * registers that need to be reserved. It also updates the
428      * {@code formats} list to help avoid extra work in future
429      * register reservation passes.
430      *
431      * @param formats {@code non-null;} array of per-instruction format selections
432      * @return {@code >= 0;} the count of reserved registers
433      */
calculateReservedCount(InsnFormat[] formats)434     private int calculateReservedCount(InsnFormat[] formats) {
435         int size = insns.size();
436 
437         /*
438          * Potential new value of reservedCount, which gets updated in the
439          * following loop. It starts out with the existing reservedCount
440          * and gets increased if it turns out that additional registers
441          * need to be reserved.
442          */
443         int newReservedCount = reservedCount;
444 
445         for (int i = 0; i < size; i++) {
446             DalvInsn insn = insns.get(i);
447             InsnFormat originalFormat = formats[i];
448             InsnFormat newFormat = findFormatForInsn(insn, originalFormat);
449 
450             if (originalFormat == newFormat) {
451                 continue;
452             }
453 
454             if (newFormat == null) {
455                 /*
456                  * The instruction will need to be expanded, so reserve
457                  * registers for it.
458                  */
459                 int reserve = insn.getMinimumRegisterRequirement();
460                 if (reserve > newReservedCount) {
461                     newReservedCount = reserve;
462                 }
463             }
464 
465             formats[i] = newFormat;
466         }
467 
468         return newReservedCount;
469     }
470 
471     /**
472      * Attempts to fit the given instruction into a format, returning
473      * either a format that the instruction fits into or {@code null}
474      * to indicate that the instruction will need to be expanded. This
475      * fitting process starts with the given format as a first "best
476      * guess" and then pessimizes from there if necessary.
477      *
478      * @param insn {@code non-null;} the instruction in question
479      * @param format {@code null-ok;} the current guess as to the best instruction
480      * format to use; {@code null} means that no simple format fits
481      * @return {@code null-ok;} a possibly-different format, which is either a
482      * good fit or {@code null} to indicate that no simple format
483      * fits
484      */
findFormatForInsn(DalvInsn insn, InsnFormat format)485     private InsnFormat findFormatForInsn(DalvInsn insn, InsnFormat format) {
486         if (format == null) {
487             // The instruction is already known not to fit any simple format.
488             return format;
489         }
490 
491         if (format.isCompatible(insn)) {
492             // The instruction already fits in the current best-known format.
493             return format;
494         }
495 
496         Dop dop = insn.getOpcode();
497         int family = dop.getFamily();
498 
499         for (;;) {
500             format = format.nextUp();
501             if ((format == null) ||
502                     (format.isCompatible(insn) &&
503                      (Dops.getOrNull(family, format) != null))) {
504                 break;
505             }
506         }
507 
508         return format;
509     }
510 
511     /**
512      * Helper for {@link #finishProcessingAndGetList}, which goes
513      * through each instruction in the output, making sure its opcode
514      * can accomodate its arguments. In cases where the opcode is
515      * unable to do so, this replaces the instruction with a larger
516      * instruction with identical semantics that <i>will</i> work.
517      *
518      * <p>This method may also reserve a number of low-numbered
519      * registers, renumbering the instructions' original registers, in
520      * order to have register space available in which to move
521      * very-high registers when expanding instructions into
522      * multi-instruction sequences. This expansion is done when no
523      * simple instruction format can be found for a given instruction that
524      * is able to accomodate that instruction's registers.</p>
525      *
526      * <p>This method ignores issues of branch target size, since
527      * final addresses aren't known at the point that this method is
528      * called.</p>
529      *
530      * @param formats {@code non-null;} array of per-instruction format selections
531      */
massageInstructions(InsnFormat[] formats)532     private void massageInstructions(InsnFormat[] formats) {
533         if (reservedCount == 0) {
534             /*
535              * The easy common case: No registers were reserved, so we
536              * merely need to replace any instructions whose format changed
537              * during the reservation pass, but all instructions will stay
538              * at their original indices, and the instruction list doesn't
539              * grow.
540              */
541             int size = insns.size();
542 
543             for (int i = 0; i < size; i++) {
544                 DalvInsn insn = insns.get(i);
545                 Dop dop = insn.getOpcode();
546                 InsnFormat format = formats[i];
547 
548                 if (format != dop.getFormat()) {
549                     dop = Dops.getOrNull(dop.getFamily(), format);
550                     insns.set(i, insn.withOpcode(dop));
551                 }
552             }
553         } else {
554             /*
555              * The difficult uncommon case: Some instructions have to be
556              * expanded to deal with high registers.
557              */
558             insns = performExpansion(formats);
559         }
560     }
561 
562     /**
563      * Helper for {@link #massageInstructions}, which constructs a
564      * replacement list, where each {link DalvInsn} instance that
565      * couldn't be represented simply (due to register representation
566      * problems) is expanded into a series of instances that together
567      * perform the proper function.
568      *
569      * @param formats {@code non-null;} array of per-instruction format selections
570      * @return {@code non-null;} the replacement list
571      */
performExpansion(InsnFormat[] formats)572     private ArrayList<DalvInsn> performExpansion(InsnFormat[] formats) {
573         int size = insns.size();
574         ArrayList<DalvInsn> result = new ArrayList<DalvInsn>(size * 2);
575 
576         for (int i = 0; i < size; i++) {
577             DalvInsn insn = insns.get(i);
578             Dop dop = insn.getOpcode();
579             InsnFormat originalFormat = dop.getFormat();
580             InsnFormat currentFormat = formats[i];
581             DalvInsn prefix;
582             DalvInsn suffix;
583 
584             if (currentFormat != null) {
585                 // No expansion is necessary.
586                 prefix = null;
587                 suffix = null;
588             } else {
589                 // Expansion is required.
590                 prefix = insn.hrPrefix();
591                 suffix = insn.hrSuffix();
592 
593                 /*
594                  * Get the initial guess as to the hr version, but then
595                  * let findFormatForInsn() pick a better format, if any.
596                  */
597                 insn = insn.hrVersion();
598                 originalFormat = insn.getOpcode().getFormat();
599                 currentFormat = findFormatForInsn(insn, originalFormat);
600             }
601 
602             if (prefix != null) {
603                 result.add(prefix);
604             }
605 
606             if (currentFormat != originalFormat) {
607                 dop = Dops.getOrNull(dop.getFamily(), currentFormat);
608                 insn = insn.withOpcode(dop);
609             }
610             result.add(insn);
611 
612             if (suffix != null) {
613                 result.add(suffix);
614             }
615         }
616 
617         return result;
618     }
619 
620     /**
621      * Helper for {@link #finishProcessingAndGetList}, which assigns
622      * addresses to each instruction, possibly rewriting branches to
623      * fix ones that wouldn't otherwise be able to reach their
624      * targets.
625      */
assignAddressesAndFixBranches()626     private void assignAddressesAndFixBranches() {
627         for (;;) {
628             assignAddresses();
629             if (!fixBranches()) {
630                 break;
631             }
632         }
633     }
634 
635     /**
636      * Helper for {@link #assignAddressesAndFixBranches}, which
637      * assigns an address to each instruction, in order.
638      */
assignAddresses()639     private void assignAddresses() {
640         int address = 0;
641         int size = insns.size();
642 
643         for (int i = 0; i < size; i++) {
644             DalvInsn insn = insns.get(i);
645             insn.setAddress(address);
646             address += insn.codeSize();
647         }
648     }
649 
650     /**
651      * Helper for {@link #assignAddressesAndFixBranches}, which checks
652      * the branch target size requirement of each branch instruction
653      * to make sure it fits. For instructions that don't fit, this
654      * rewrites them to use a {@code goto} of some sort. In the
655      * case of a conditional branch that doesn't fit, the sense of the
656      * test is reversed in order to branch around a {@code goto}
657      * to the original target.
658      *
659      * @return whether any branches had to be fixed
660      */
fixBranches()661     private boolean fixBranches() {
662         int size = insns.size();
663         boolean anyFixed = false;
664 
665         for (int i = 0; i < size; i++) {
666             DalvInsn insn = insns.get(i);
667             if (!(insn instanceof TargetInsn)) {
668                 // This loop only needs to inspect TargetInsns.
669                 continue;
670             }
671 
672             Dop dop = insn.getOpcode();
673             InsnFormat format = dop.getFormat();
674             TargetInsn target = (TargetInsn) insn;
675 
676             if (format.branchFits(target)) {
677                 continue;
678             }
679 
680             if (dop.getFamily() == DalvOps.GOTO) {
681                 // It is a goto; widen it if possible.
682                 InsnFormat newFormat = findFormatForInsn(insn, format);
683                 if (newFormat == null) {
684                     /*
685                      * The branch is already maximally large. This should
686                      * only be possible if a method somehow manages to have
687                      * more than 2^31 code units.
688                      */
689                     throw new UnsupportedOperationException("method too long");
690                 }
691                 dop = Dops.getOrNull(dop.getFamily(), newFormat);
692                 insn = insn.withOpcode(dop);
693                 insns.set(i, insn);
694             } else {
695                 /*
696                  * It is a conditional: Reverse its sense, and arrange for
697                  * it to branch around an absolute goto to the original
698                  * branch target.
699                  *
700                  * Note: An invariant of the list being processed is
701                  * that every TargetInsn is followed by a CodeAddress.
702                  * Hence, it is always safe to get the next element
703                  * after a TargetInsn and cast it to CodeAddress, as
704                  * is happening a few lines down.
705                  *
706                  * Also note: Size gets incremented by one here, as we
707                  * have -- in the net -- added one additional element
708                  * to the list, so we increment i to match. The added
709                  * and changed elements will be inspected by a repeat
710                  * call to this method after this invocation returns.
711                  */
712                 CodeAddress newTarget;
713                 try {
714                     newTarget = (CodeAddress) insns.get(i + 1);
715                 } catch (IndexOutOfBoundsException ex) {
716                     // The TargetInsn / CodeAddress invariant was violated.
717                     throw new IllegalStateException(
718                             "unpaired TargetInsn (dangling)");
719                 } catch (ClassCastException ex) {
720                     // The TargetInsn / CodeAddress invariant was violated.
721                     throw new IllegalStateException("unpaired TargetInsn");
722                 }
723                 TargetInsn gotoInsn =
724                     new TargetInsn(Dops.GOTO, target.getPosition(),
725                             RegisterSpecList.EMPTY, target.getTarget());
726                 insns.set(i, gotoInsn);
727                 insns.add(i, target.withNewTargetAndReversed(newTarget));
728                 size++;
729                 i++;
730             }
731 
732             anyFixed = true;
733         }
734 
735         return anyFixed;
736     }
737 }
738