diff --git a/.hgtags-top-repo b/.hgtags-top-repo index c0108b217b9..16ebe0c9938 100644 --- a/.hgtags-top-repo +++ b/.hgtags-top-repo @@ -126,3 +126,4 @@ f42e3d9394b40a423d345b8da22687b5462e5f25 jdk8-b01 69f592185747226a9c765a9fe139c1d34d616f9c jdk8-b02 587bb549dff83131b65f40aa51864f69562f34a7 jdk8-b03 0b66a233bfb9ba2ebda1e5cdfdb0373d6c1e3c69 jdk8-b04 +b910aac18c772b823b1f7da03e2c6528725cc6de jdk8-b05 diff --git a/corba/.hgtags b/corba/.hgtags index f8e0d5abe74..fc40ae67186 100644 --- a/corba/.hgtags +++ b/corba/.hgtags @@ -126,3 +126,4 @@ a2f340a048c88d10cbedc0504f5cf03d39925a40 jdk7-b142 ed8d94519a87b4adac270c3eec9134ff1f62bff5 jdk8-b02 cd0da00694fbce642db9be936d3e4909a71d911d jdk8-b03 60a68d688e24473cf84dedd1e60901a61ab82555 jdk8-b04 +cc1b599b986a37cb57de4584c5e58169766ca535 jdk8-b05 diff --git a/hotspot/.hgtags b/hotspot/.hgtags index a0f3ddd463c..04fd5117f83 100644 --- a/hotspot/.hgtags +++ b/hotspot/.hgtags @@ -178,3 +178,4 @@ c149193c768b8b7233da4c3a3fdc0756b975848e jdk7-b143 31e253c1da429124bb87570ab095d9bc89850d0a jdk8-b02 3a2fb61165dfc72e398179a2796d740c8da5b8c0 jdk8-b03 0fa3ace511fe98fe948e751531f3e2b7c60c8376 jdk8-b04 +dce7d24674f4d0bed00de24f00119057fdce7cfb jdk8-b05 diff --git a/jaxp/.hgtags b/jaxp/.hgtags index 0b2e3c4326a..466478d345d 100644 --- a/jaxp/.hgtags +++ b/jaxp/.hgtags @@ -126,3 +126,4 @@ fc268cd1dd5d2e903ccd4b0275e1f9c2461ed30c jdk7-b147 ca4d6ad55a660f0469882e85b4dacf7822d50abf jdk8-b02 7a74371ce0c64108b857c497ae130dfe9514532c jdk8-b03 acbcadef0b21582abf406f07f1b74d2b8f80dc01 jdk8-b04 +ff0a3d78e7a22743eabbaa71e9d17b2f094ddf62 jdk8-b05 diff --git a/jdk/.hgtags b/jdk/.hgtags index 75feff6b3b4..23e8fa2a74e 100644 --- a/jdk/.hgtags +++ b/jdk/.hgtags @@ -126,3 +126,4 @@ f097ca2434b1412b12ab4a5c2397ce271bf681e7 jdk7-b147 13e70aa1398eb959c54bc68b783ca0eef1286ad2 jdk8-b02 7989ee9fe673a87f4db3917fa6a005732a6a9b85 jdk8-b03 fc569517f3cf242f90ce3503b743eb5553938946 jdk8-b04 +0b32369b83d81c226a2e79e730f3a8c0d2595e92 jdk8-b05 diff --git a/jdk/make/com/sun/servicetag/Makefile b/jdk/make/com/sun/servicetag/Makefile index 6a8bf6da52c..a31b9aed177 100644 --- a/jdk/make/com/sun/servicetag/Makefile +++ b/jdk/make/com/sun/servicetag/Makefile @@ -47,7 +47,7 @@ FILES_copy = $(SERVICETAG_RESOURCES_DIR)/product_registration.xsd \ # Add all properties files to the FILES_copy list SWORDFISH_properties := $(shell \ $(CD) $(SHARE_SRC)/classes/com/sun/servicetag/resources; \ - $(FIND) . -name 'javase_*_swordfish.properties' -print ; \ + $(FIND) . -name 'javase_*.properties' -print ; \ ) FILES_copy += $(shell \ for f in $(SWORDFISH_properties) ; do \ diff --git a/jdk/make/jprt.properties b/jdk/make/jprt.properties index 0759ecd8ef8..e07910e18b9 100644 --- a/jdk/make/jprt.properties +++ b/jdk/make/jprt.properties @@ -258,6 +258,15 @@ jprt.make.rule.all.test.targets= \ windows_i586_5.1-product-c1-jdk_security3, \ windows_x64_5.2-product-c2-jdk_security3, \ \ + solaris_sparc_5.10-product-c1-jdk_sound, \ + solaris_sparcv9_5.10-product-c2-jdk_sound, \ + solaris_i586_5.10-product-c1-jdk_sound, \ + solaris_x64_5.10-product-c2-jdk_sound, \ + linux_i586_2.6-product-{c1|c2}-jdk_sound, \ + linux_x64_2.6-product-c2-jdk_sound, \ + windows_i586_5.1-product-c1-jdk_sound, \ + windows_x64_5.2-product-c2-jdk_sound, \ + \ solaris_sparc_5.10-product-c1-jdk_swing, \ solaris_sparcv9_5.10-product-c2-jdk_swing, \ solaris_i586_5.10-product-c1-jdk_swing, \ diff --git a/jdk/src/share/classes/com/sun/imageio/plugins/jpeg/JFIFMarkerSegment.java b/jdk/src/share/classes/com/sun/imageio/plugins/jpeg/JFIFMarkerSegment.java index 3adcddce5e9..64d09af7b8c 100644 --- a/jdk/src/share/classes/com/sun/imageio/plugins/jpeg/JFIFMarkerSegment.java +++ b/jdk/src/share/classes/com/sun/imageio/plugins/jpeg/JFIFMarkerSegment.java @@ -763,7 +763,7 @@ class JFIFMarkerSegment extends MarkerSegment { } } catch (IllegalThumbException e) { // Should never happen - throw new InternalError("Illegal thumb in setThumbnail!"); + throw new InternalError("Illegal thumb in setThumbnail!", e); } } diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/MotifFileChooserUI.java b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/MotifFileChooserUI.java index ada499264f8..048e76647c4 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/MotifFileChooserUI.java +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/MotifFileChooserUI.java @@ -529,21 +529,25 @@ public class MotifFileChooserUI extends BasicFileChooserUI { Locale l = fc.getLocale(); enterFolderNameLabelText = UIManager.getString("FileChooser.enterFolderNameLabelText",l); - enterFolderNameLabelMnemonic = UIManager.getInt("FileChooser.enterFolderNameLabelMnemonic"); + enterFolderNameLabelMnemonic = getMnemonic("FileChooser.enterFolderNameLabelMnemonic", l); enterFileNameLabelText = UIManager.getString("FileChooser.enterFileNameLabelText",l); - enterFileNameLabelMnemonic = UIManager.getInt("FileChooser.enterFileNameLabelMnemonic"); + enterFileNameLabelMnemonic = getMnemonic("FileChooser.enterFileNameLabelMnemonic", l); filesLabelText = UIManager.getString("FileChooser.filesLabelText",l); - filesLabelMnemonic = UIManager.getInt("FileChooser.filesLabelMnemonic"); + filesLabelMnemonic = getMnemonic("FileChooser.filesLabelMnemonic", l); foldersLabelText = UIManager.getString("FileChooser.foldersLabelText",l); - foldersLabelMnemonic = UIManager.getInt("FileChooser.foldersLabelMnemonic"); + foldersLabelMnemonic = getMnemonic("FileChooser.foldersLabelMnemonic", l); pathLabelText = UIManager.getString("FileChooser.pathLabelText",l); - pathLabelMnemonic = UIManager.getInt("FileChooser.pathLabelMnemonic"); + pathLabelMnemonic = getMnemonic("FileChooser.pathLabelMnemonic", l); filterLabelText = UIManager.getString("FileChooser.filterLabelText",l); - filterLabelMnemonic = UIManager.getInt("FileChooser.filterLabelMnemonic"); + filterLabelMnemonic = getMnemonic("FileChooser.filterLabelMnemonic", l); + } + + private Integer getMnemonic(String key, Locale l) { + return SwingUtilities2.getUIDefaultsInt(key, l); } protected void installIcons(JFileChooser fc) { diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/MotifLookAndFeel.java b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/MotifLookAndFeel.java index 60ab1ac5926..923cd9d18b9 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/MotifLookAndFeel.java +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/MotifLookAndFeel.java @@ -1215,11 +1215,6 @@ public class MotifLookAndFeel extends BasicLookAndFeel "EditorPane.focusInputMap", multilineInputMap, - "FileChooser.pathLabelMnemonic", new Integer(KeyEvent.VK_P), // 'p' - "FileChooser.filterLabelMnemonic", new Integer (KeyEvent.VK_R), // 'r' - "FileChooser.foldersLabelMnemonic", new Integer (KeyEvent.VK_L), // 'l' - "FileChooser.filesLabelMnemonic", new Integer (KeyEvent.VK_I), // 'i' - "FileChooser.enterFileNameLabelMnemonic", new Integer (KeyEvent.VK_N), // 'n' "FileChooser.ancestorInputMap", new UIDefaults.LazyInputMap(new Object[] { "ESCAPE", "cancelSelection" diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif.properties index eb32a6d04a7..d3fee304abe 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif.properties @@ -27,10 +27,15 @@ FileChooser.openDialogTitleText=Open FileChooser.updateButtonText=Update FileChooser.helpButtonText=Help FileChooser.pathLabelText=Enter path or folder name: +FileChooser.pathLabelMnemonic=80 FileChooser.filterLabelText=Filter +FileChooser.filterLabelMnemonic=82 FileChooser.foldersLabelText=Folders +FileChooser.foldersLabelMnemonic=76 FileChooser.filesLabelText=Files +FileChooser.filesLabelMnemonic=73 FileChooser.enterFileNameLabelText=Enter file name: +FileChooser.enterFileNameLabelMnemonic=78 FileChooser.enterFolderNameLabelText=Enter folder name: FileChooser.cancelButtonToolTipText=Abort file chooser dialog. diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_de.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_de.properties index f1ee7ed76b1..e7723574a6d 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_de.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_de.properties @@ -27,10 +27,15 @@ FileChooser.openDialogTitleText=\u00D6ffnen FileChooser.updateButtonText=Aktualisieren FileChooser.helpButtonText=Hilfe FileChooser.pathLabelText=Pfad- oder Ordnernamen eingeben: +FileChooser.pathLabelMnemonic=80 FileChooser.filterLabelText=Filter +FileChooser.filterLabelMnemonic=82 FileChooser.foldersLabelText=Ordner +FileChooser.foldersLabelMnemonic=76 FileChooser.filesLabelText=Dateien +FileChooser.filesLabelMnemonic=73 FileChooser.enterFileNameLabelText=Dateinamen eingeben: +FileChooser.enterFileNameLabelMnemonic=78 FileChooser.enterFolderNameLabelText=Ordnernamen eingeben: FileChooser.cancelButtonToolTipText=Dialogfeld f\u00FCr Dateiauswahl schlie\u00DFen. diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_es.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_es.properties index c4e9108ff0e..6d9aa8f16ba 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_es.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_es.properties @@ -27,10 +27,15 @@ FileChooser.openDialogTitleText=Abrir FileChooser.updateButtonText=Actualizar FileChooser.helpButtonText=Ayuda FileChooser.pathLabelText=Introducir nombre de la ruta de acceso o carpeta: +FileChooser.pathLabelMnemonic=80 FileChooser.filterLabelText=Filtro +FileChooser.filterLabelMnemonic=82 FileChooser.foldersLabelText=Carpetas +FileChooser.foldersLabelMnemonic=76 FileChooser.filesLabelText=Archivos +FileChooser.filesLabelMnemonic=73 FileChooser.enterFileNameLabelText=Introducir nombre de archivo: +FileChooser.enterFileNameLabelMnemonic=78 FileChooser.enterFolderNameLabelText=Introducir nombre de carpeta: FileChooser.cancelButtonToolTipText=Abortar cuadro de di\u00E1logo del selector de archivos. diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_fr.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_fr.properties index 8644dea7930..392ecee5bd2 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_fr.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_fr.properties @@ -27,10 +27,15 @@ FileChooser.openDialogTitleText=Ouvrir FileChooser.updateButtonText=Mettre \u00E0 jour FileChooser.helpButtonText=Aide FileChooser.pathLabelText=Entrez le chemin ou le nom du dossier : +FileChooser.pathLabelMnemonic=80 FileChooser.filterLabelText=Filtre +FileChooser.filterLabelMnemonic=82 FileChooser.foldersLabelText=Dossiers +FileChooser.foldersLabelMnemonic=76 FileChooser.filesLabelText=Fichiers +FileChooser.filesLabelMnemonic=73 FileChooser.enterFileNameLabelText=Entrez le nom du fichier : +FileChooser.enterFileNameLabelMnemonic=78 FileChooser.enterFolderNameLabelText=Entrez le nom du dossier : FileChooser.cancelButtonToolTipText=Ferme la bo\u00EEte de dialogue du s\u00E9lecteur de fichiers. diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_it.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_it.properties index e637d6b5e8e..9763045aee2 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_it.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_it.properties @@ -27,10 +27,15 @@ FileChooser.openDialogTitleText=Apri FileChooser.updateButtonText=Aggiorna FileChooser.helpButtonText=? FileChooser.pathLabelText=Percorso o nome cartella: +FileChooser.pathLabelMnemonic=80 FileChooser.filterLabelText=Filtro +FileChooser.filterLabelMnemonic=82 FileChooser.foldersLabelText=Cartelle +FileChooser.foldersLabelMnemonic=76 FileChooser.filesLabelText=File +FileChooser.filesLabelMnemonic=73 FileChooser.enterFileNameLabelText=Nome file: +FileChooser.enterFileNameLabelMnemonic=78 FileChooser.enterFolderNameLabelText=Nome cartella: FileChooser.cancelButtonToolTipText=Chiude la finestra di dialogo di selezione file. diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_ja.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_ja.properties index 68342cc8544..97b505b5741 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_ja.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_ja.properties @@ -27,10 +27,15 @@ FileChooser.openDialogTitleText=\u958B\u304F FileChooser.updateButtonText=\u66F4\u65B0 FileChooser.helpButtonText=\u30D8\u30EB\u30D7 FileChooser.pathLabelText=\u30D1\u30B9\u307E\u305F\u306F\u30D5\u30A9\u30EB\u30C0\u540D\u3092\u5165\u529B: +FileChooser.pathLabelMnemonic=80 FileChooser.filterLabelText=\u30D5\u30A3\u30EB\u30BF +FileChooser.filterLabelMnemonic=82 FileChooser.foldersLabelText=\u30D5\u30A9\u30EB\u30C0 +FileChooser.foldersLabelMnemonic=76 FileChooser.filesLabelText=\u30D5\u30A1\u30A4\u30EB +FileChooser.filesLabelMnemonic=73 FileChooser.enterFileNameLabelText=\u30D5\u30A1\u30A4\u30EB\u540D\u3092\u5165\u529B: +FileChooser.enterFileNameLabelMnemonic=78 FileChooser.enterFolderNameLabelText=\u30D5\u30A9\u30EB\u30C0\u540D\u3092\u5165\u529B: FileChooser.cancelButtonToolTipText=\u30D5\u30A1\u30A4\u30EB\u30FB\u30C1\u30E5\u30FC\u30B6\u30FB\u30C0\u30A4\u30A2\u30ED\u30B0\u3092\u7D42\u4E86\u3057\u307E\u3059\u3002 diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_ko.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_ko.properties index f522359b907..61c06c6949d 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_ko.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_ko.properties @@ -27,10 +27,15 @@ FileChooser.openDialogTitleText=\uC5F4\uAE30 FileChooser.updateButtonText=\uAC31\uC2E0 FileChooser.helpButtonText=\uB3C4\uC6C0\uB9D0 FileChooser.pathLabelText=\uACBD\uB85C \uB610\uB294 \uD3F4\uB354 \uC774\uB984 \uC785\uB825: +FileChooser.pathLabelMnemonic=80 FileChooser.filterLabelText=\uD544\uD130 +FileChooser.filterLabelMnemonic=82 FileChooser.foldersLabelText=\uD3F4\uB354 +FileChooser.foldersLabelMnemonic=76 FileChooser.filesLabelText=\uD30C\uC77C +FileChooser.filesLabelMnemonic=73 FileChooser.enterFileNameLabelText=\uD30C\uC77C \uC774\uB984 \uC785\uB825: +FileChooser.enterFileNameLabelMnemonic=78 FileChooser.enterFolderNameLabelText=\uD3F4\uB354 \uC774\uB984 \uC785\uB825: FileChooser.cancelButtonToolTipText=\uD30C\uC77C \uC120\uD0DD\uAE30 \uB300\uD654\uC0C1\uC790\uB97C \uC911\uB2E8\uD569\uB2C8\uB2E4. diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_pt_BR.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_pt_BR.properties index 63b5d9d0c2f..6e40ef51c06 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_pt_BR.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_pt_BR.properties @@ -27,10 +27,15 @@ FileChooser.openDialogTitleText=Abrir FileChooser.updateButtonText=Atualizar FileChooser.helpButtonText=Ajuda FileChooser.pathLabelText=Informar caminho ou nome da pasta: +FileChooser.pathLabelMnemonic=80 FileChooser.filterLabelText=Filtro +FileChooser.filterLabelMnemonic=82 FileChooser.foldersLabelText=Pastas +FileChooser.foldersLabelMnemonic=76 FileChooser.filesLabelText=Arquivos +FileChooser.filesLabelMnemonic=73 FileChooser.enterFileNameLabelText=Informar nome do arquivo: +FileChooser.enterFileNameLabelMnemonic=78 FileChooser.enterFolderNameLabelText=Informar nome da pasta: FileChooser.cancelButtonToolTipText=Abortar caixa de di\u00E1logo do seletor de arquivos. diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_sv.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_sv.properties index 67a6c078768..4cd736baa9d 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_sv.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_sv.properties @@ -27,10 +27,15 @@ FileChooser.openDialogTitleText=\u00D6ppna FileChooser.updateButtonText=Uppdatera FileChooser.helpButtonText=Hj\u00E4lp FileChooser.pathLabelText=Ange s\u00F6kv\u00E4g eller mappnamn: +FileChooser.pathLabelMnemonic=80 FileChooser.filterLabelText=Filter +FileChooser.filterLabelMnemonic=82 FileChooser.foldersLabelText=Mappar +FileChooser.foldersLabelMnemonic=76 FileChooser.filesLabelText=Filer +FileChooser.filesLabelMnemonic=73 FileChooser.enterFileNameLabelText=Ange filnamn: +FileChooser.enterFileNameLabelMnemonic=78 FileChooser.enterFolderNameLabelText=Ange ett mappnamn: FileChooser.cancelButtonToolTipText=Avbryt dialogrutan Filv\u00E4ljare. diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_zh_CN.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_zh_CN.properties index a1576618dca..121fe60a8ff 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_zh_CN.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_zh_CN.properties @@ -27,10 +27,15 @@ FileChooser.openDialogTitleText=\u6253\u5F00 FileChooser.updateButtonText=\u66F4\u65B0 FileChooser.helpButtonText=\u5E2E\u52A9 FileChooser.pathLabelText=\u952E\u5165\u8DEF\u5F84\u6216\u6587\u4EF6\u5939\u540D: +FileChooser.pathLabelMnemonic=80 FileChooser.filterLabelText=\u7B5B\u9009\u5668 +FileChooser.filterLabelMnemonic=82 FileChooser.foldersLabelText=\u6587\u4EF6\u5939 +FileChooser.foldersLabelMnemonic=76 FileChooser.filesLabelText=\u6587\u4EF6 +FileChooser.filesLabelMnemonic=73 FileChooser.enterFileNameLabelText=\u952E\u5165\u6587\u4EF6\u540D: +FileChooser.enterFileNameLabelMnemonic=78 FileChooser.enterFolderNameLabelText=\u8F93\u5165\u6587\u4EF6\u5939\u540D: FileChooser.cancelButtonToolTipText=\u4E2D\u6B62\u6587\u4EF6\u9009\u62E9\u5668\u5BF9\u8BDD\u6846\u3002 diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_zh_TW.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_zh_TW.properties index f2985492b1a..1b2d9d76c4b 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_zh_TW.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/motif/resources/motif_zh_TW.properties @@ -27,10 +27,15 @@ FileChooser.openDialogTitleText=\u958B\u555F FileChooser.updateButtonText=\u66F4\u65B0 FileChooser.helpButtonText=\u8AAA\u660E FileChooser.pathLabelText=\u8F38\u5165\u8DEF\u5F91\u6216\u8CC7\u6599\u593E\u540D\u7A31: +FileChooser.pathLabelMnemonic=80 FileChooser.filterLabelText=\u7BE9\u9078 +FileChooser.filterLabelMnemonic=82 FileChooser.foldersLabelText=\u8CC7\u6599\u593E +FileChooser.foldersLabelMnemonic=76 FileChooser.filesLabelText=\u6A94\u6848 +FileChooser.filesLabelMnemonic=73 FileChooser.enterFileNameLabelText=\u8F38\u5165\u6A94\u6848\u540D\u7A31: +FileChooser.enterFileNameLabelMnemonic=78 FileChooser.enterFolderNameLabelText=\u8F38\u5165\u8CC7\u6599\u593E\u540D\u7A31: FileChooser.cancelButtonToolTipText=\u4E2D\u6B62\u6A94\u6848\u9078\u64C7\u5668\u5C0D\u8A71\u65B9\u584A\u3002 diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/WindowsFileChooserUI.java b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/WindowsFileChooserUI.java index dec7d1b2c4a..2e1168199c9 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/WindowsFileChooserUI.java +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/WindowsFileChooserUI.java @@ -528,16 +528,16 @@ public class WindowsFileChooserUI extends BasicFileChooserUI { Locale l = fc.getLocale(); - lookInLabelMnemonic = UIManager.getInt("FileChooser.lookInLabelMnemonic"); + lookInLabelMnemonic = getMnemonic("FileChooser.lookInLabelMnemonic", l); lookInLabelText = UIManager.getString("FileChooser.lookInLabelText",l); saveInLabelText = UIManager.getString("FileChooser.saveInLabelText",l); - fileNameLabelMnemonic = UIManager.getInt("FileChooser.fileNameLabelMnemonic"); + fileNameLabelMnemonic = getMnemonic("FileChooser.fileNameLabelMnemonic", l); fileNameLabelText = UIManager.getString("FileChooser.fileNameLabelText",l); - folderNameLabelMnemonic = UIManager.getInt("FileChooser.folderNameLabelMnemonic"); + folderNameLabelMnemonic = getMnemonic("FileChooser.folderNameLabelMnemonic", l); folderNameLabelText = UIManager.getString("FileChooser.folderNameLabelText",l); - filesOfTypeLabelMnemonic = UIManager.getInt("FileChooser.filesOfTypeLabelMnemonic"); + filesOfTypeLabelMnemonic = getMnemonic("FileChooser.filesOfTypeLabelMnemonic", l); filesOfTypeLabelText = UIManager.getString("FileChooser.filesOfTypeLabelText",l); upFolderToolTipText = UIManager.getString("FileChooser.upFolderToolTipText",l); @@ -550,6 +550,10 @@ public class WindowsFileChooserUI extends BasicFileChooserUI { viewMenuButtonAccessibleName = UIManager.getString("FileChooser.viewMenuButtonAccessibleName",l); } + private Integer getMnemonic(String key, Locale l) { + return SwingUtilities2.getUIDefaultsInt(key, l); + } + protected void installListeners(JFileChooser fc) { super.installListeners(fc); ActionMap actionMap = getActionMap(); diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/WindowsLookAndFeel.java b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/WindowsLookAndFeel.java index 18c76d780a7..b8723035a32 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/WindowsLookAndFeel.java +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/WindowsLookAndFeel.java @@ -770,9 +770,6 @@ public class WindowsLookAndFeel extends BasicLookAndFeel "icons/NewFolder.gif"), "FileChooser.useSystemExtensionHiding", Boolean.TRUE, - "FileChooser.lookInLabelMnemonic", Integer.valueOf(KeyEvent.VK_I), - "FileChooser.fileNameLabelMnemonic", Integer.valueOf(KeyEvent.VK_N), - "FileChooser.filesOfTypeLabelMnemonic", Integer.valueOf(KeyEvent.VK_T), "FileChooser.usesSingleFilePane", Boolean.TRUE, "FileChooser.noPlacesBar", new DesktopProperty("win.comdlg.noPlacesBar", Boolean.FALSE), diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows.properties index 0900aa3eb69..0f27244f37a 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=Look in: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=Save in: FileChooser.fileNameLabelText=File name: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=Folder name: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=Files of type: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=Up One Level FileChooser.upFolderAccessibleName=Up FileChooser.homeFolderToolTipText=Home diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_de.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_de.properties index 993b26445f3..b9d8c341f47 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_de.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_de.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=Suchen in: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=Speichern in: FileChooser.fileNameLabelText=Dateiname: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=Ordnername: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=Dateityp: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=Eine Ebene h\u00F6her FileChooser.upFolderAccessibleName=Nach oben FileChooser.homeFolderToolTipText=Home diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_es.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_es.properties index adf964b0013..d7b610e746d 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_es.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_es.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=Buscar en: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=Guardar en: FileChooser.fileNameLabelText=Nombre de Archivo: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=Nombre de la Carpeta: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=Archivos de Tipo: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=Subir un Nivel FileChooser.upFolderAccessibleName=Arriba FileChooser.homeFolderToolTipText=Inicio diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_fr.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_fr.properties index d4561daaadb..39736484777 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_fr.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_fr.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=Rechercher dans : +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=Enregistrer dans : FileChooser.fileNameLabelText=Nom du fichier : +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=Nom du dossier : +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=Fichiers de type : +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=Remonte d'un niveau. FileChooser.upFolderAccessibleName=Monter FileChooser.homeFolderToolTipText=R\u00E9pertoire d'origine diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_it.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_it.properties index 4edf194a173..5e3e6f12634 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_it.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_it.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=Cerca in: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=Salva in: FileChooser.fileNameLabelText=Nome file: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=Nome della cartella: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=Tipo file: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=Cartella superiore FileChooser.upFolderAccessibleName=Superiore FileChooser.homeFolderToolTipText=Home diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_ja.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_ja.properties index 7c6c3053372..adc5d930d69 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_ja.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_ja.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=\u53C2\u7167: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=\u4FDD\u5B58: FileChooser.fileNameLabelText=\u30D5\u30A1\u30A4\u30EB\u540D: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=\u30D5\u30A9\u30EB\u30C0\u540D: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=\u30D5\u30A1\u30A4\u30EB\u306E\u30BF\u30A4\u30D7: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=1\u30EC\u30D9\u30EB\u4E0A\u3078 FileChooser.upFolderAccessibleName=\u4E0A\u3078 FileChooser.homeFolderToolTipText=\u30DB\u30FC\u30E0 diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_ko.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_ko.properties index 53dedb4d672..6922556c77e 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_ko.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_ko.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=\uAC80\uC0C9 \uC704\uCE58: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=\uC800\uC7A5 \uC704\uCE58: FileChooser.fileNameLabelText=\uD30C\uC77C \uC774\uB984: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=\uD3F4\uB354 \uC774\uB984: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=\uD30C\uC77C \uC720\uD615: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=\uD55C \uB808\uBCA8 \uC704\uB85C FileChooser.upFolderAccessibleName=\uC704\uB85C FileChooser.homeFolderToolTipText=\uD648 diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_pt_BR.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_pt_BR.properties index 526a0f9e5f1..7faf8c19c80 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_pt_BR.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_pt_BR.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=Consultar em: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=Salvar em: FileChooser.fileNameLabelText=Nome do arquivo: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=Nome da pasta: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=Arquivos do tipo: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=Um N\u00EDvel Acima FileChooser.upFolderAccessibleName=Acima FileChooser.homeFolderToolTipText=In\u00EDcio diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_sv.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_sv.properties index fc8358df46d..4e83498093f 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_sv.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_sv.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=Leta i: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=Spara i: FileChooser.fileNameLabelText=Filnamn: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=Mapp: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=Filformat: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=Upp en niv\u00E5 FileChooser.upFolderAccessibleName=Upp FileChooser.homeFolderToolTipText=Hem diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_zh_CN.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_zh_CN.properties index ae6fe8033cb..3a4c0cd6942 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_zh_CN.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_zh_CN.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=\u67E5\u770B: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=\u4FDD\u5B58: FileChooser.fileNameLabelText=\u6587\u4EF6\u540D: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=\u6587\u4EF6\u5939\u540D: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=\u6587\u4EF6\u7C7B\u578B: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=\u5411\u4E0A\u4E00\u7EA7 FileChooser.upFolderAccessibleName=\u5411\u4E0A FileChooser.homeFolderToolTipText=\u4E3B\u76EE\u5F55 diff --git a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_zh_TW.properties b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_zh_TW.properties index fe18bdc2fa2..e4c8ad0c081 100644 --- a/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_zh_TW.properties +++ b/jdk/src/share/classes/com/sun/java/swing/plaf/windows/resources/windows_zh_TW.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=\u67E5\u8A62: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=\u5132\u5B58\u65BC: FileChooser.fileNameLabelText=\u6A94\u6848\u540D\u7A31: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=\u8CC7\u6599\u593E\u540D\u7A31: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=\u6A94\u6848\u985E\u578B: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=\u5F80\u4E0A\u4E00\u5C64 FileChooser.upFolderAccessibleName=\u5F80\u4E0A FileChooser.homeFolderToolTipText=\u4E3B\u76EE\u9304 diff --git a/jdk/src/share/classes/com/sun/jmx/snmp/SnmpCounter64.java b/jdk/src/share/classes/com/sun/jmx/snmp/SnmpCounter64.java index 128903628d1..ed23942eeb6 100644 --- a/jdk/src/share/classes/com/sun/jmx/snmp/SnmpCounter64.java +++ b/jdk/src/share/classes/com/sun/jmx/snmp/SnmpCounter64.java @@ -186,7 +186,7 @@ public class SnmpCounter64 extends SnmpValue { newclone = (SnmpCounter64) super.clone() ; newclone.value = value ; } catch (CloneNotSupportedException e) { - throw new InternalError() ; // vm bug. + throw new InternalError(e) ; // vm bug. } return newclone ; } diff --git a/jdk/src/share/classes/com/sun/jmx/snmp/SnmpInt.java b/jdk/src/share/classes/com/sun/jmx/snmp/SnmpInt.java index a5faf5c4260..ca1cf52f564 100644 --- a/jdk/src/share/classes/com/sun/jmx/snmp/SnmpInt.java +++ b/jdk/src/share/classes/com/sun/jmx/snmp/SnmpInt.java @@ -232,7 +232,7 @@ public class SnmpInt extends SnmpValue { newclone = (SnmpInt) super.clone() ; newclone.value = value ; } catch (CloneNotSupportedException e) { - throw new InternalError() ; // vm bug. + throw new InternalError(e) ; // vm bug. } return newclone ; } diff --git a/jdk/src/share/classes/com/sun/jmx/snmp/SnmpNull.java b/jdk/src/share/classes/com/sun/jmx/snmp/SnmpNull.java index 18fe8b72a27..f6274b406c9 100644 --- a/jdk/src/share/classes/com/sun/jmx/snmp/SnmpNull.java +++ b/jdk/src/share/classes/com/sun/jmx/snmp/SnmpNull.java @@ -129,7 +129,7 @@ public class SnmpNull extends SnmpValue { newclone = (SnmpNull) super.clone() ; newclone.tag = tag ; } catch (CloneNotSupportedException e) { - throw new InternalError() ; // vm bug. + throw new InternalError(e) ; // vm bug. } return newclone ; } diff --git a/jdk/src/share/classes/com/sun/jmx/snmp/SnmpString.java b/jdk/src/share/classes/com/sun/jmx/snmp/SnmpString.java index 47cfbdf6ae4..1cf86adf86e 100644 --- a/jdk/src/share/classes/com/sun/jmx/snmp/SnmpString.java +++ b/jdk/src/share/classes/com/sun/jmx/snmp/SnmpString.java @@ -250,7 +250,7 @@ public class SnmpString extends SnmpValue { newclone.value = new byte[value.length] ; System.arraycopy(value, 0, newclone.value, 0, value.length) ; } catch (CloneNotSupportedException e) { - throw new InternalError() ; // vm bug. + throw new InternalError(e) ; // vm bug. } return newclone ; } diff --git a/jdk/src/share/classes/com/sun/jmx/snmp/daemon/SnmpRequestHandler.java b/jdk/src/share/classes/com/sun/jmx/snmp/daemon/SnmpRequestHandler.java index 53b9f0bd7df..dc32e6b35ad 100644 --- a/jdk/src/share/classes/com/sun/jmx/snmp/daemon/SnmpRequestHandler.java +++ b/jdk/src/share/classes/com/sun/jmx/snmp/daemon/SnmpRequestHandler.java @@ -921,7 +921,7 @@ class SnmpRequestHandler extends ClientHandler implements SnmpDefinitions { SNMP_ADAPTOR_LOGGER.logp(Level.FINEST, dbgTag, "newTooBigMessage", "Internal error", x); } - throw new InternalError() ; + throw new InternalError(x) ; } return result ; diff --git a/jdk/src/share/classes/com/sun/security/auth/module/NTSystem.java b/jdk/src/share/classes/com/sun/security/auth/module/NTSystem.java index 5ed6c357df1..51f9a2c688c 100644 --- a/jdk/src/share/classes/com/sun/security/auth/module/NTSystem.java +++ b/jdk/src/share/classes/com/sun/security/auth/module/NTSystem.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2000, 2010, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2000, 2011, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -33,6 +33,7 @@ package com.sun.security.auth.module; public class NTSystem { private native void getCurrent(boolean debug); + private native long getImpersonationToken0(); private String userName; private String domain; @@ -132,10 +133,14 @@ public class NTSystem { * * @return an impersonation token for the current NT user. */ - public long getImpersonationToken() { + public synchronized long getImpersonationToken() { + if (impersonationToken == 0) { + impersonationToken = getImpersonationToken0(); + } return impersonationToken; } + private void loadNative() { System.loadLibrary("jaas_nt"); } diff --git a/jdk/src/share/classes/com/sun/servicetag/BrowserSupport.java b/jdk/src/share/classes/com/sun/servicetag/BrowserSupport.java index 442267bf36c..86d9204863e 100644 --- a/jdk/src/share/classes/com/sun/servicetag/BrowserSupport.java +++ b/jdk/src/share/classes/com/sun/servicetag/BrowserSupport.java @@ -77,9 +77,7 @@ class BrowserSupport { result = (Boolean) isDesktopSupportedMethod.invoke(null); } catch (IllegalAccessException e) { // should never reach here - InternalError x = - new InternalError("Desktop.getDesktop() method not found"); - x.initCause(e); + throw new InternalError("Desktop.getDesktop() method not found", e); } catch (InvocationTargetException e) { // browser not supported if (Util.isVerbose()) { @@ -101,28 +99,10 @@ class BrowserSupport { result = (Boolean) isSupportedMethod.invoke(desktopObj, browseField.get(null)); supported = result.booleanValue(); } - } catch (ClassNotFoundException e) { - // browser not supported - if (Util.isVerbose()) { - e.printStackTrace(); - } - } catch (NoSuchMethodException e) { - // browser not supported - if (Util.isVerbose()) { - e.printStackTrace(); - } - } catch (NoSuchFieldException e) { - // browser not supported - if (Util.isVerbose()) { - e.printStackTrace(); - } } catch (IllegalAccessException e) { // should never reach here - InternalError x = - new InternalError("Desktop.getDesktop() method not found"); - x.initCause(e); - throw x; - } catch (InvocationTargetException e) { + throw new InternalError("Desktop.getDesktop() method not found", e); + } catch (ReflectiveOperationException e) { // browser not supported if (Util.isVerbose()) { e.printStackTrace(); @@ -175,10 +155,7 @@ class BrowserSupport { browseMethod.invoke(desktop, uri); } catch (IllegalAccessException e) { // should never reach here - InternalError x = - new InternalError("Desktop.getDesktop() method not found"); - x.initCause(e); - throw x; + throw new InternalError("Desktop.getDesktop() method not found", e); } catch (InvocationTargetException e) { Throwable x = e.getCause(); if (x != null) { diff --git a/jdk/src/share/classes/com/sun/servicetag/Installer.java b/jdk/src/share/classes/com/sun/servicetag/Installer.java index ae5ed4bc4fb..40ae99ed84d 100644 --- a/jdk/src/share/classes/com/sun/servicetag/Installer.java +++ b/jdk/src/share/classes/com/sun/servicetag/Installer.java @@ -61,8 +61,8 @@ public class Installer { private static RegistrationData registration; private static boolean supportRegistration; private static String registerHtmlParent; - private static Set supportedLocales = new HashSet(); - private static Properties swordfishProps = null; + private static Set supportedLocales = new HashSet<>(); + private static Properties svcTagProps = null; private static String[] jreArchs = null; static { String dir = System.getProperty(SVCTAG_DIR_PATH); @@ -94,7 +94,7 @@ public class Installer { boolean cleanup = false; try { // Check if we have the swordfish entries for this JRE version - if (loadSwordfishEntries() == null) { + if (loadServiceTagProps() == null) { return null; } @@ -144,18 +144,14 @@ public class Installer { return registration; } if (regXmlFile.exists()) { - BufferedInputStream in = null; - try { - in = new BufferedInputStream(new FileInputStream(regXmlFile)); + try (BufferedInputStream in = + new BufferedInputStream(new FileInputStream(regXmlFile))) + { registration = RegistrationData.loadFromXML(in); } catch (IllegalArgumentException ex) { System.err.println("Error: Bad registration data \"" + regXmlFile + "\":" + ex.getMessage()); throw ex; - } finally { - if (in != null) { - in.close(); - } } } else { registration = new RegistrationData(); @@ -186,18 +182,14 @@ public class Installer { deleteRegistrationHtmlPage(); getRegistrationHtmlPage(); - BufferedOutputStream out = null; - try { - out = new BufferedOutputStream(new FileOutputStream(regXmlFile)); + try (BufferedOutputStream out = + new BufferedOutputStream(new FileOutputStream(regXmlFile))) + { getRegistrationData().storeToXML(out); } catch (IllegalArgumentException ex) { System.err.println("Error: Bad registration data \"" + regXmlFile + "\":" + ex.getMessage()); throw ex; - } finally { - if (out != null) { - out.close(); - } } } @@ -206,11 +198,9 @@ public class Installer { * or empty set if file not exists. */ private static Set getInstalledURNs() throws IOException { - Set urnSet = new HashSet(); + Set urnSet = new HashSet<>(); if (serviceTagFile.exists()) { - BufferedReader in = null; - try { - in = new BufferedReader(new FileReader(serviceTagFile)); + try (BufferedReader in = new BufferedReader(new FileReader(serviceTagFile))) { String urn; while ((urn = in.readLine()) != null) { urn = urn.trim(); @@ -218,10 +208,6 @@ public class Installer { urnSet.add(urn); } } - } finally { - if (in != null) { - in.close(); - } } } return urnSet; @@ -237,9 +223,9 @@ public class Installer { private static ServiceTag[] getJavaServiceTagArray() throws IOException { RegistrationData regData = getRegistrationData(); Set svcTags = regData.getServiceTags(); - Set result = new HashSet(); + Set result = new HashSet<>(); - Properties props = loadSwordfishEntries(); + Properties props = loadServiceTagProps(); String jdkUrn = props.getProperty("servicetag.jdk.urn"); String jreUrn = props.getProperty("servicetag.jre.urn"); for (ServiceTag st : svcTags) { @@ -343,8 +329,7 @@ public class Installer { } private static ServiceTag newServiceTag(String svcTagSource) throws IOException { - // Load the swoRDFish information for the service tag creation - Properties props = loadSwordfishEntries(); + Properties props = loadServiceTagProps(); // Determine the product URN and name String productURN; @@ -442,52 +427,35 @@ public class Installer { return; } - PrintWriter out = null; - try { - out = new PrintWriter(serviceTagFile); - + try (PrintWriter out = new PrintWriter(serviceTagFile)) { ServiceTag[] javaSvcTags = getJavaServiceTagArray(); for (ServiceTag st : javaSvcTags) { // Write the instance_run to the servicetag file String instanceURN = st.getInstanceURN(); out.println(instanceURN); } - } finally { - if (out != null) { - out.close(); - } } } /** - * Load the values associated with the swoRDFish metadata entries - * for Java SE. The swoRDFish metadata entries are different for - * different release. + * Load the properties for generating Java SE service tags. * * @param version Version of Java SE */ - private static synchronized Properties loadSwordfishEntries() throws IOException { - if (swordfishProps != null) { - return swordfishProps; + private static synchronized Properties loadServiceTagProps() throws IOException { + if (svcTagProps != null) { + return svcTagProps; } - // The version string for Java SE 6 is 1.6.0 - // We just need the minor number in the version string - int version = Util.getJdkVersion(); - - String filename = "/com/sun/servicetag/resources/javase_" + - version + "_swordfish.properties"; - InputStream in = Installer.class.getResourceAsStream(filename); - if (in == null) { - return null; + // For Java SE 8 and later releases, JDK and JRE both use + // the same product number. The sworRDFish metadata were + // for legacy Sun part number. + String filename = "/com/sun/servicetag/resources/javase_servicetag.properties"; + try (InputStream in = Installer.class.getResourceAsStream(filename)) { + svcTagProps = new Properties(); + svcTagProps.load(in); } - swordfishProps = new Properties(); - try { - swordfishProps.load(in); - } finally { - in.close(); - } - return swordfishProps; + return svcTagProps; } /** @@ -546,7 +514,7 @@ public class Installer { return jreArchs; } - Set archs = new HashSet(); + Set archs = new HashSet<>(); String os = System.getProperty("os.name"); if (os.equals("SunOS") || os.equals("Linux")) { @@ -681,16 +649,16 @@ public class Installer { String country = locale.getCountry(); String variant = locale.getVariant(); - List locales = new ArrayList(3); + List locales = new ArrayList<>(3); if (variant.length() > 0) { locales.add(locale); } if (country.length() > 0) { - locales.add((locales.size() == 0) ? + locales.add((locales.isEmpty()) ? locale : new Locale(language, country, "")); } if (language.length() > 0) { - locales.add((locales.size() == 0) ? + locales.add((locales.isEmpty()) ? locale : new Locale(language, "", "")); } return locales; @@ -788,14 +756,11 @@ public class Installer { // Format the registration data in one single line StringBuilder payload = new StringBuilder(); String xml = regData.toString().replaceAll("\"", "%22"); - BufferedReader reader = new BufferedReader(new StringReader(xml)); - try { + try (BufferedReader reader = new BufferedReader(new StringReader(xml))) { String line = null; while ((line = reader.readLine()) != null) { payload.append(line.trim()); } - } finally { - reader.close(); } String resourceFilename = "/com/sun/servicetag/resources/register"; diff --git a/jdk/src/share/classes/com/sun/servicetag/RegistrationDocument.java b/jdk/src/share/classes/com/sun/servicetag/RegistrationDocument.java index 876446bc864..eb33570972e 100644 --- a/jdk/src/share/classes/com/sun/servicetag/RegistrationDocument.java +++ b/jdk/src/share/classes/com/sun/servicetag/RegistrationDocument.java @@ -150,9 +150,7 @@ class RegistrationDocument { } catch (ParserConfigurationException pce) { // Parser with specific options can't be built // should not reach here - InternalError x = new InternalError("Error in creating the new document"); - x.initCause(pce); - throw x; + throw new InternalError("Error in creating the new document", pce); } } @@ -172,9 +170,7 @@ class RegistrationDocument { } catch (ParserConfigurationException pce) { // Parser with specified options can't be built // should not reach here - InternalError x = new InternalError("Error in creating the new document"); - x.initCause(pce); - throw x; + throw new InternalError("Error in creating the new document", pce); } } @@ -195,20 +191,14 @@ class RegistrationDocument { new StreamResult(new BufferedWriter(new OutputStreamWriter(os, "UTF-8")))); } catch (UnsupportedEncodingException ue) { // Should not reach here - InternalError x = new InternalError("Error generated during transformation"); - x.initCause(ue); - throw x; + throw new InternalError("Error generated during transformation", ue); } catch (TransformerConfigurationException tce) { // Error generated by the parser // Should not reach here - InternalError x = new InternalError("Error in creating the new document"); - x.initCause(tce); - throw x; + throw new InternalError("Error in creating the new document", tce); } catch (TransformerException te) { // Error generated by the transformer - InternalError x = new InternalError("Error generated during transformation"); - x.initCause(te); - throw x; + throw new InternalError("Error generated during transformation", te); } } diff --git a/jdk/src/share/classes/com/sun/servicetag/resources/javase_servicetag.properties b/jdk/src/share/classes/com/sun/servicetag/resources/javase_servicetag.properties new file mode 100644 index 00000000000..7ff81b27827 --- /dev/null +++ b/jdk/src/share/classes/com/sun/servicetag/resources/javase_servicetag.properties @@ -0,0 +1,29 @@ +# Copyright (c) 2011, Oracle and/or its affiliates. All rights reserved. +# DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. +# +# This code is free software; you can redistribute it and/or modify it +# under the terms of the GNU General Public License version 2 only, as +# published by the Free Software Foundation. Oracle designates this +# particular file as subject to the "Classpath" exception as provided +# by Oracle in the LICENSE file that accompanied this code. +# +# This code is distributed in the hope that it will be useful, but WITHOUT +# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or +# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +# version 2 for more details (a copy is included in the LICENSE file that +# accompanied this code). +# +# You should have received a copy of the GNU General Public License version +# 2 along with this work; if not, write to the Free Software Foundation, +# Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. +# +# Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA +# or visit www.oracle.com if you need additional information or have any +# questions. + +servicetag.jdk.urn = Q8549 +servicetag.jdk.name = Java Development Kit +servicetag.jre.urn = Q8549 +servicetag.jre.name = Java Runtime Environment +servicetag.parent.urn = Q8549 +servicetag.parent.name = Java Platform, Standard Edition diff --git a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal.properties b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal.properties index f7fd7fca4d8..3ae82055bd9 100644 --- a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal.properties +++ b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=Look In: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=Save In: FileChooser.fileNameLabelText=File Name: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=Folder name: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=Files of Type: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=Up One Level FileChooser.upFolderAccessibleName=Up FileChooser.homeFolderToolTipText=Home diff --git a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_de.properties b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_de.properties index 14c676a0e08..124449e236a 100644 --- a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_de.properties +++ b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_de.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=Suchen in: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=Speichern in: FileChooser.fileNameLabelText=Dateiname: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=Ordnername: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=Dateityp: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=Eine Ebene h\u00F6her FileChooser.upFolderAccessibleName=Nach oben FileChooser.homeFolderToolTipText=Home diff --git a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_es.properties b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_es.properties index 6cd153b4e5c..4bfeaaf0e61 100644 --- a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_es.properties +++ b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_es.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=Buscar en: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=Guardar en: FileChooser.fileNameLabelText=Nombre de Archivo: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=Nombre de la Carpeta: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=Archivos de Tipo: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=Subir un Nivel FileChooser.upFolderAccessibleName=Arriba FileChooser.homeFolderToolTipText=Inicio diff --git a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_fr.properties b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_fr.properties index ae099682341..2f91a500365 100644 --- a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_fr.properties +++ b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_fr.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=Rechercher dans : +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=Enregistrer dans : FileChooser.fileNameLabelText=Nom du fichier : +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=Nom du dossier : +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=Fichiers de type : +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=Remonte d'un niveau. FileChooser.upFolderAccessibleName=Monter FileChooser.homeFolderToolTipText=R\u00E9pertoire d'origine diff --git a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_it.properties b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_it.properties index ff581c782be..b0bd9f03070 100644 --- a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_it.properties +++ b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_it.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=Cerca in: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=Salva in: FileChooser.fileNameLabelText=Nome file: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=Nome della cartella: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=Tipo file: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=Cartella superiore FileChooser.upFolderAccessibleName=Superiore FileChooser.homeFolderToolTipText=Home diff --git a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_ja.properties b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_ja.properties index d669fd5059e..4e467cec4db 100644 --- a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_ja.properties +++ b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_ja.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=\u53C2\u7167: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=\u4FDD\u5B58: FileChooser.fileNameLabelText=\u30D5\u30A1\u30A4\u30EB\u540D: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=\u30D5\u30A9\u30EB\u30C0\u540D: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=\u30D5\u30A1\u30A4\u30EB\u306E\u30BF\u30A4\u30D7: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=1\u30EC\u30D9\u30EB\u4E0A\u3078 FileChooser.upFolderAccessibleName=\u4E0A\u3078 FileChooser.homeFolderToolTipText=\u30DB\u30FC\u30E0 diff --git a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_ko.properties b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_ko.properties index 7e8c3b483cb..3cd5e20da12 100644 --- a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_ko.properties +++ b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_ko.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=\uAC80\uC0C9 \uC704\uCE58: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=\uC800\uC7A5 \uC704\uCE58: FileChooser.fileNameLabelText=\uD30C\uC77C \uC774\uB984: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=\uD3F4\uB354 \uC774\uB984: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=\uD30C\uC77C \uC720\uD615: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=\uD55C \uB808\uBCA8 \uC704\uB85C FileChooser.upFolderAccessibleName=\uC704\uB85C FileChooser.homeFolderToolTipText=\uD648 diff --git a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_pt_BR.properties b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_pt_BR.properties index 4e0fb3e9340..903f1d1bae0 100644 --- a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_pt_BR.properties +++ b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_pt_BR.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=Consultar Em: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=Salvar Em: FileChooser.fileNameLabelText=Nome do Arquivo: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=Nome da pasta: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=Arquivos do Tipo: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=Um N\u00EDvel Acima FileChooser.upFolderAccessibleName=Acima FileChooser.homeFolderToolTipText=In\u00EDcio diff --git a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_sv.properties b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_sv.properties index df1a453ae14..0f76c987f12 100644 --- a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_sv.properties +++ b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_sv.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=Leta i: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=Spara i: FileChooser.fileNameLabelText=Filnamn: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=Mapp: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=Filformat: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=Upp en niv\u00E5 FileChooser.upFolderAccessibleName=Upp FileChooser.homeFolderToolTipText=Hem diff --git a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_zh_CN.properties b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_zh_CN.properties index e3eca13ca02..73405a09398 100644 --- a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_zh_CN.properties +++ b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_zh_CN.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=\u67E5\u770B: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=\u4FDD\u5B58: FileChooser.fileNameLabelText=\u6587\u4EF6\u540D: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=\u6587\u4EF6\u5939\u540D: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=\u6587\u4EF6\u7C7B\u578B: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=\u5411\u4E0A\u4E00\u7EA7 FileChooser.upFolderAccessibleName=\u5411\u4E0A FileChooser.homeFolderToolTipText=\u4E3B\u76EE\u5F55 diff --git a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_zh_TW.properties b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_zh_TW.properties index a24f3e9edca..6f70ad739dd 100644 --- a/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_zh_TW.properties +++ b/jdk/src/share/classes/com/sun/swing/internal/plaf/metal/resources/metal_zh_TW.properties @@ -19,10 +19,14 @@ ############ FILE CHOOSER STRINGS ############# FileChooser.lookInLabelText=\u67E5\u8A62: +FileChooser.lookInLabelMnemonic=73 FileChooser.saveInLabelText=\u5132\u5B58\u65BC: FileChooser.fileNameLabelText=\u6A94\u6848\u540D\u7A31: +FileChooser.fileNameLabelMnemonic=78 FileChooser.folderNameLabelText=\u8CC7\u6599\u593E\u540D\u7A31: +FileChooser.folderNameLabelMnemonic=78 FileChooser.filesOfTypeLabelText=\u6A94\u6848\u985E\u578B: +FileChooser.filesOfTypeLabelMnemonic=84 FileChooser.upFolderToolTipText=\u5F80\u4E0A\u4E00\u5C64 FileChooser.upFolderAccessibleName=\u5F80\u4E0A FileChooser.homeFolderToolTipText=\u4E3B\u76EE\u9304 diff --git a/jdk/src/share/classes/java/awt/BufferCapabilities.java b/jdk/src/share/classes/java/awt/BufferCapabilities.java index a4b860bacbc..61ce15eeeee 100644 --- a/jdk/src/share/classes/java/awt/BufferCapabilities.java +++ b/jdk/src/share/classes/java/awt/BufferCapabilities.java @@ -137,7 +137,7 @@ public class BufferCapabilities implements Cloneable { return super.clone(); } catch (CloneNotSupportedException e) { // Since we implement Cloneable, this should never happen - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/awt/Component.java b/jdk/src/share/classes/java/awt/Component.java index 2225633c0db..09d745f6d7a 100644 --- a/jdk/src/share/classes/java/awt/Component.java +++ b/jdk/src/share/classes/java/awt/Component.java @@ -3776,11 +3776,10 @@ public abstract class Component implements ImageObserver, MenuContainer, createBufferStrategy(numBuffers, bufferCaps); return; // Success } catch (AWTException e) { - // Failed + // Code should never reach here (an unaccelerated blitting + // strategy should always work) + throw new InternalError("Could not create a buffer strategy", e); } - // Code should never reach here (an unaccelerated blitting - // strategy should always work) - throw new InternalError("Could not create a buffer strategy"); } /** @@ -7910,7 +7909,7 @@ public abstract class Component implements ImageObserver, MenuContainer, res = toFocus.requestFocusInWindow(CausedFocusEvent.Cause.TRAVERSAL_BACKWARD); } } - if (!res) { + if (clearOnFailure && !res) { if (focusLog.isLoggable(PlatformLogger.FINER)) { focusLog.finer("clear global focus owner"); } diff --git a/jdk/src/share/classes/java/awt/GridBagConstraints.java b/jdk/src/share/classes/java/awt/GridBagConstraints.java index 9fb3093f1e5..c524efd0c63 100644 --- a/jdk/src/share/classes/java/awt/GridBagConstraints.java +++ b/jdk/src/share/classes/java/awt/GridBagConstraints.java @@ -653,7 +653,7 @@ public class GridBagConstraints implements Cloneable, java.io.Serializable { return c; } catch (CloneNotSupportedException e) { // this shouldn't happen, since we are Cloneable - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/awt/ImageCapabilities.java b/jdk/src/share/classes/java/awt/ImageCapabilities.java index fb6ddc5db19..d0ad474a944 100644 --- a/jdk/src/share/classes/java/awt/ImageCapabilities.java +++ b/jdk/src/share/classes/java/awt/ImageCapabilities.java @@ -74,7 +74,7 @@ public class ImageCapabilities implements Cloneable { return super.clone(); } catch (CloneNotSupportedException e) { // Since we implement Cloneable, this should never happen - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/awt/Insets.java b/jdk/src/share/classes/java/awt/Insets.java index c9e38059047..f049a9e8bf6 100644 --- a/jdk/src/share/classes/java/awt/Insets.java +++ b/jdk/src/share/classes/java/awt/Insets.java @@ -177,7 +177,7 @@ public class Insets implements Cloneable, java.io.Serializable { return super.clone(); } catch (CloneNotSupportedException e) { // this shouldn't happen, since we are Cloneable - throw new InternalError(); + throw new InternalError(e); } } /** diff --git a/jdk/src/share/classes/java/awt/JobAttributes.java b/jdk/src/share/classes/java/awt/JobAttributes.java index c144e1b7439..205b360ac65 100644 --- a/jdk/src/share/classes/java/awt/JobAttributes.java +++ b/jdk/src/share/classes/java/awt/JobAttributes.java @@ -361,7 +361,7 @@ public final class JobAttributes implements Cloneable { return super.clone(); } catch (CloneNotSupportedException e) { // Since we implement Cloneable, this should never happen - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/awt/PageAttributes.java b/jdk/src/share/classes/java/awt/PageAttributes.java index f5c883369f2..85f52ffbe14 100644 --- a/jdk/src/share/classes/java/awt/PageAttributes.java +++ b/jdk/src/share/classes/java/awt/PageAttributes.java @@ -969,7 +969,7 @@ public final class PageAttributes implements Cloneable { return super.clone(); } catch (CloneNotSupportedException e) { // Since we implement Cloneable, this should never happen - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/awt/RenderingHints.java b/jdk/src/share/classes/java/awt/RenderingHints.java index 7d33dbe3f5a..5b8daa9e519 100644 --- a/jdk/src/share/classes/java/awt/RenderingHints.java +++ b/jdk/src/share/classes/java/awt/RenderingHints.java @@ -1276,7 +1276,7 @@ public class RenderingHints } } catch (CloneNotSupportedException e) { // this shouldn't happen, since we are Cloneable - throw new InternalError(); + throw new InternalError(e); } return rh; diff --git a/jdk/src/share/classes/java/awt/font/TextLayout.java b/jdk/src/share/classes/java/awt/font/TextLayout.java index c5e204afeac..51b0dc64e80 100644 --- a/jdk/src/share/classes/java/awt/font/TextLayout.java +++ b/jdk/src/share/classes/java/awt/font/TextLayout.java @@ -753,7 +753,7 @@ public final class TextLayout implements Cloneable { return super.clone(); } catch (CloneNotSupportedException e) { - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/awt/geom/AffineTransform.java b/jdk/src/share/classes/java/awt/geom/AffineTransform.java index a3b8134fe1f..2fa1dc2137a 100644 --- a/jdk/src/share/classes/java/awt/geom/AffineTransform.java +++ b/jdk/src/share/classes/java/awt/geom/AffineTransform.java @@ -3856,7 +3856,7 @@ public class AffineTransform implements Cloneable, java.io.Serializable { return super.clone(); } catch (CloneNotSupportedException e) { // this shouldn't happen, since we are Cloneable - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/awt/geom/CubicCurve2D.java b/jdk/src/share/classes/java/awt/geom/CubicCurve2D.java index 3f6bc9de4dd..5e1fc770f2f 100644 --- a/jdk/src/share/classes/java/awt/geom/CubicCurve2D.java +++ b/jdk/src/share/classes/java/awt/geom/CubicCurve2D.java @@ -1569,7 +1569,7 @@ public abstract class CubicCurve2D implements Shape, Cloneable { return super.clone(); } catch (CloneNotSupportedException e) { // this shouldn't happen, since we are Cloneable - throw new InternalError(); + throw new InternalError(e); } } } diff --git a/jdk/src/share/classes/java/awt/geom/Dimension2D.java b/jdk/src/share/classes/java/awt/geom/Dimension2D.java index 437c827d6a9..35725397ce9 100644 --- a/jdk/src/share/classes/java/awt/geom/Dimension2D.java +++ b/jdk/src/share/classes/java/awt/geom/Dimension2D.java @@ -108,7 +108,7 @@ public abstract class Dimension2D implements Cloneable { return super.clone(); } catch (CloneNotSupportedException e) { // this shouldn't happen, since we are Cloneable - throw new InternalError(); + throw new InternalError(e); } } } diff --git a/jdk/src/share/classes/java/awt/geom/Line2D.java b/jdk/src/share/classes/java/awt/geom/Line2D.java index 312a0e0e2db..2eccbd0c5ad 100644 --- a/jdk/src/share/classes/java/awt/geom/Line2D.java +++ b/jdk/src/share/classes/java/awt/geom/Line2D.java @@ -1122,7 +1122,7 @@ public abstract class Line2D implements Shape, Cloneable { return super.clone(); } catch (CloneNotSupportedException e) { // this shouldn't happen, since we are Cloneable - throw new InternalError(); + throw new InternalError(e); } } } diff --git a/jdk/src/share/classes/java/awt/geom/Point2D.java b/jdk/src/share/classes/java/awt/geom/Point2D.java index d5455055c71..2320ea8f241 100644 --- a/jdk/src/share/classes/java/awt/geom/Point2D.java +++ b/jdk/src/share/classes/java/awt/geom/Point2D.java @@ -393,7 +393,7 @@ public abstract class Point2D implements Cloneable { return super.clone(); } catch (CloneNotSupportedException e) { // this shouldn't happen, since we are Cloneable - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/awt/geom/QuadCurve2D.java b/jdk/src/share/classes/java/awt/geom/QuadCurve2D.java index d2351bc0d82..4df791e40b2 100644 --- a/jdk/src/share/classes/java/awt/geom/QuadCurve2D.java +++ b/jdk/src/share/classes/java/awt/geom/QuadCurve2D.java @@ -1395,7 +1395,7 @@ public abstract class QuadCurve2D implements Shape, Cloneable { return super.clone(); } catch (CloneNotSupportedException e) { // this shouldn't happen, since we are Cloneable - throw new InternalError(); + throw new InternalError(e); } } } diff --git a/jdk/src/share/classes/java/awt/geom/RectangularShape.java b/jdk/src/share/classes/java/awt/geom/RectangularShape.java index 3c04832301f..1a468dd0109 100644 --- a/jdk/src/share/classes/java/awt/geom/RectangularShape.java +++ b/jdk/src/share/classes/java/awt/geom/RectangularShape.java @@ -391,7 +391,7 @@ public abstract class RectangularShape implements Shape, Cloneable { return super.clone(); } catch (CloneNotSupportedException e) { // this shouldn't happen, since we are Cloneable - throw new InternalError(); + throw new InternalError(e); } } } diff --git a/jdk/src/share/classes/java/awt/image/ImageFilter.java b/jdk/src/share/classes/java/awt/image/ImageFilter.java index 2cd0da2880f..7b72807844b 100644 --- a/jdk/src/share/classes/java/awt/image/ImageFilter.java +++ b/jdk/src/share/classes/java/awt/image/ImageFilter.java @@ -252,7 +252,7 @@ public class ImageFilter implements ImageConsumer, Cloneable { return super.clone(); } catch (CloneNotSupportedException e) { // this shouldn't happen, since we are Cloneable - throw new InternalError(); + throw new InternalError(e); } } } diff --git a/jdk/src/share/classes/java/awt/image/Kernel.java b/jdk/src/share/classes/java/awt/image/Kernel.java index e5da78c2bb7..a0f3ea64c84 100644 --- a/jdk/src/share/classes/java/awt/image/Kernel.java +++ b/jdk/src/share/classes/java/awt/image/Kernel.java @@ -147,7 +147,7 @@ public class Kernel implements Cloneable { return super.clone(); } catch (CloneNotSupportedException e) { // this shouldn't happen, since we are Cloneable - throw new InternalError(); + throw new InternalError(e); } } } diff --git a/jdk/src/share/classes/java/io/ObjectStreamClass.java b/jdk/src/share/classes/java/io/ObjectStreamClass.java index ec13d4533f8..f1c0d5b782f 100644 --- a/jdk/src/share/classes/java/io/ObjectStreamClass.java +++ b/jdk/src/share/classes/java/io/ObjectStreamClass.java @@ -478,7 +478,7 @@ public class ObjectStreamClass implements Serializable { fieldRefl = getReflector(fields, this); } catch (InvalidClassException ex) { // field mismatches impossible when matching local fields vs. self - throw new InternalError(); + throw new InternalError(ex); } if (deserializeEx == null) { @@ -941,7 +941,7 @@ public class ObjectStreamClass implements Serializable { return cons.newInstance(); } catch (IllegalAccessException ex) { // should not occur, as access checks have been suppressed - throw new InternalError(); + throw new InternalError(ex); } } else { throw new UnsupportedOperationException(); @@ -969,7 +969,7 @@ public class ObjectStreamClass implements Serializable { } } catch (IllegalAccessException ex) { // should not occur, as access checks have been suppressed - throw new InternalError(); + throw new InternalError(ex); } } else { throw new UnsupportedOperationException(); @@ -1000,7 +1000,7 @@ public class ObjectStreamClass implements Serializable { } } catch (IllegalAccessException ex) { // should not occur, as access checks have been suppressed - throw new InternalError(); + throw new InternalError(ex); } } else { throw new UnsupportedOperationException(); @@ -1028,7 +1028,7 @@ public class ObjectStreamClass implements Serializable { } } catch (IllegalAccessException ex) { // should not occur, as access checks have been suppressed - throw new InternalError(); + throw new InternalError(ex); } } else { throw new UnsupportedOperationException(); @@ -1053,11 +1053,11 @@ public class ObjectStreamClass implements Serializable { throw (ObjectStreamException) th; } else { throwMiscException(th); - throw new InternalError(); // never reached + throw new InternalError(th); // never reached } } catch (IllegalAccessException ex) { // should not occur, as access checks have been suppressed - throw new InternalError(); + throw new InternalError(ex); } } else { throw new UnsupportedOperationException(); @@ -1082,11 +1082,11 @@ public class ObjectStreamClass implements Serializable { throw (ObjectStreamException) th; } else { throwMiscException(th); - throw new InternalError(); // never reached + throw new InternalError(th); // never reached } } catch (IllegalAccessException ex) { // should not occur, as access checks have been suppressed - throw new InternalError(); + throw new InternalError(ex); } } else { throw new UnsupportedOperationException(); @@ -1774,7 +1774,7 @@ public class ObjectStreamClass implements Serializable { } return hash; } catch (IOException ex) { - throw new InternalError(); + throw new InternalError(ex); } catch (NoSuchAlgorithmException ex) { throw new SecurityException(ex.getMessage()); } diff --git a/jdk/src/share/classes/java/lang/CharacterName.java b/jdk/src/share/classes/java/lang/CharacterName.java index fa0b2d6dae3..a2d84635151 100644 --- a/jdk/src/share/classes/java/lang/CharacterName.java +++ b/jdk/src/share/classes/java/lang/CharacterName.java @@ -83,7 +83,7 @@ class CharacterName { dis.readFully(strPool); refStrPool = new SoftReference<>(strPool); } catch (Exception x) { - throw new InternalError(x.getMessage()); + throw new InternalError(x.getMessage(), x); } finally { try { if (dis != null) diff --git a/jdk/src/share/classes/java/lang/Class.java b/jdk/src/share/classes/java/lang/Class.java index 6f74179a28b..01e21b9a030 100644 --- a/jdk/src/share/classes/java/lang/Class.java +++ b/jdk/src/share/classes/java/lang/Class.java @@ -974,8 +974,7 @@ public final descriptor = (String) enclosingInfo[2]; assert((name != null && descriptor != null) || name == descriptor); } catch (ClassCastException cce) { - throw (InternalError) - new InternalError("Invalid type in enclosing method information").initCause(cce); + throw new InternalError("Invalid type in enclosing method information", cce); } } @@ -1241,8 +1240,7 @@ public final try { return getName().substring(enclosingClass.getName().length()); } catch (IndexOutOfBoundsException ex) { - throw (InternalError) - new InternalError("Malformed class name").initCause(ex); + throw new InternalError("Malformed class name", ex); } } diff --git a/jdk/src/share/classes/java/lang/invoke/CallSite.java b/jdk/src/share/classes/java/lang/invoke/CallSite.java index 276931ac1c9..465cd779be6 100644 --- a/jdk/src/share/classes/java/lang/invoke/CallSite.java +++ b/jdk/src/share/classes/java/lang/invoke/CallSite.java @@ -27,7 +27,6 @@ package java.lang.invoke; import sun.invoke.empty.Empty; import sun.misc.Unsafe; -import static java.lang.invoke.MethodHandleStatics.*; import static java.lang.invoke.MethodHandles.Lookup.IMPL_LOOKUP; /** @@ -244,8 +243,8 @@ public class CallSite { try { GET_TARGET = IMPL_LOOKUP. findVirtual(CallSite.class, "getTarget", MethodType.methodType(MethodHandle.class)); - } catch (ReflectiveOperationException ignore) { - throw new InternalError(); + } catch (ReflectiveOperationException e) { + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/lang/invoke/Invokers.java b/jdk/src/share/classes/java/lang/invoke/Invokers.java index 24b7d3ebc34..a1ae3743174 100644 --- a/jdk/src/share/classes/java/lang/invoke/Invokers.java +++ b/jdk/src/share/classes/java/lang/invoke/Invokers.java @@ -88,7 +88,7 @@ class Invokers { try { invoker = IMPL_LOOKUP.findVirtual(MethodHandle.class, name, targetType); } catch (ReflectiveOperationException ex) { - throw new InternalError("JVM cannot find invoker for "+targetType); + throw new InternalError("JVM cannot find invoker for "+targetType, ex); } assert(invokerType(targetType) == invoker.type()); assert(!invoker.isVarargsCollector()); diff --git a/jdk/src/share/classes/java/lang/invoke/MemberName.java b/jdk/src/share/classes/java/lang/invoke/MemberName.java index 088c5682fe6..bf72c5bce2d 100644 --- a/jdk/src/share/classes/java/lang/invoke/MemberName.java +++ b/jdk/src/share/classes/java/lang/invoke/MemberName.java @@ -382,7 +382,7 @@ import static java.lang.invoke.MethodHandleStatics.*; try { return (MemberName) super.clone(); } catch (CloneNotSupportedException ex) { - throw new InternalError(); + throw new InternalError(ex); } } diff --git a/jdk/src/share/classes/java/lang/invoke/MethodHandleStatics.java b/jdk/src/share/classes/java/lang/invoke/MethodHandleStatics.java index e812d245bbc..3cad363c7a1 100644 --- a/jdk/src/share/classes/java/lang/invoke/MethodHandleStatics.java +++ b/jdk/src/share/classes/java/lang/invoke/MethodHandleStatics.java @@ -108,9 +108,7 @@ import java.security.PrivilegedAction; return new IllegalArgumentException(message(message, obj, obj2)); } /*non-public*/ static Error uncaughtException(Exception ex) { - Error err = new InternalError("uncaught exception"); - err.initCause(ex); - return err; + throw new InternalError("uncaught exception", ex); } private static String message(String message, Object obj) { if (obj != null) message = message + ": " + obj; diff --git a/jdk/src/share/classes/java/lang/invoke/MethodTypeForm.java b/jdk/src/share/classes/java/lang/invoke/MethodTypeForm.java index 768d1ab3782..75b0538c2be 100644 --- a/jdk/src/share/classes/java/lang/invoke/MethodTypeForm.java +++ b/jdk/src/share/classes/java/lang/invoke/MethodTypeForm.java @@ -461,7 +461,7 @@ class MethodTypeForm { // Trigger adapter creation. genericInvoker = InvokeGeneric.generalInvokerOf(erasedType); } catch (Exception ex) { - Error err = new InternalError("Exception while resolving inexact invoke"); + Error err = new InternalError("Exception while resolving inexact invoke", ex); err.initCause(ex); throw err; } diff --git a/jdk/src/share/classes/java/lang/reflect/Proxy.java b/jdk/src/share/classes/java/lang/reflect/Proxy.java index e83ab4085e0..c2420915031 100644 --- a/jdk/src/share/classes/java/lang/reflect/Proxy.java +++ b/jdk/src/share/classes/java/lang/reflect/Proxy.java @@ -610,7 +610,7 @@ public class Proxy implements java.io.Serializable { IllegalAccessException | InstantiationException | InvocationTargetException e) { - throw new InternalError(e.toString()); + throw new InternalError(e.toString(), e); } } diff --git a/jdk/src/share/classes/java/lang/reflect/TypeVariable.java b/jdk/src/share/classes/java/lang/reflect/TypeVariable.java index 4c5d3deaf23..b55a99f013a 100644 --- a/jdk/src/share/classes/java/lang/reflect/TypeVariable.java +++ b/jdk/src/share/classes/java/lang/reflect/TypeVariable.java @@ -48,7 +48,7 @@ package java.lang.reflect; * * @since 1.5 */ -public interface TypeVariable extends Type { +public interface TypeVariable extends Type, AnnotatedElement { /** * Returns an array of {@code Type} objects representing the * upper bound(s) of this type variable. Note that if no upper bound is diff --git a/jdk/src/share/classes/java/math/BigDecimal.java b/jdk/src/share/classes/java/math/BigDecimal.java index 0549890f125..d95b1cb27b3 100644 --- a/jdk/src/share/classes/java/math/BigDecimal.java +++ b/jdk/src/share/classes/java/math/BigDecimal.java @@ -215,6 +215,7 @@ import static java.math.BigInteger.LONG_MASK; * @author Josh Bloch * @author Mike Cowlishaw * @author Joseph D. Darcy + * @author Sergey V. Kuksenko */ public class BigDecimal extends Number implements Comparable { /** @@ -224,7 +225,7 @@ public class BigDecimal extends Number implements Comparable { * @serial * @see #unscaledValue */ - private volatile BigInteger intVal; + private final BigInteger intVal; /** * The scale of this BigDecimal, as returned by {@link #scale}. @@ -232,8 +233,9 @@ public class BigDecimal extends Number implements Comparable { * @serial * @see #scale */ - private int scale; // Note: this may have any value, so - // calculations must be done in longs + private final int scale; // Note: this may have any value, so + // calculations must be done in longs + /** * The number of decimal digits in this BigDecimal, or 0 if the * number of digits are not known (lookaside information). If @@ -256,19 +258,19 @@ public class BigDecimal extends Number implements Comparable { */ static final long INFLATED = Long.MIN_VALUE; + private static final BigInteger INFLATED_BIGINT = BigInteger.valueOf(INFLATED); + /** * If the absolute value of the significand of this BigDecimal is * less than or equal to {@code Long.MAX_VALUE}, the value can be * compactly stored in this field and used in computations. */ - private transient long intCompact; + private final transient long intCompact; // All 18-digit base ten strings fit into a long; not all 19-digit // strings will private static final int MAX_COMPACT_DIGITS = 18; - private static final int MAX_BIGINT_BITS = 62; - /* Appease the serialization gods */ private static final long serialVersionUID = 6108874887143696463L; @@ -282,17 +284,17 @@ public class BigDecimal extends Number implements Comparable { // Cache of common small BigDecimal values. private static final BigDecimal zeroThroughTen[] = { - new BigDecimal(BigInteger.ZERO, 0, 0, 1), - new BigDecimal(BigInteger.ONE, 1, 0, 1), - new BigDecimal(BigInteger.valueOf(2), 2, 0, 1), - new BigDecimal(BigInteger.valueOf(3), 3, 0, 1), - new BigDecimal(BigInteger.valueOf(4), 4, 0, 1), - new BigDecimal(BigInteger.valueOf(5), 5, 0, 1), - new BigDecimal(BigInteger.valueOf(6), 6, 0, 1), - new BigDecimal(BigInteger.valueOf(7), 7, 0, 1), - new BigDecimal(BigInteger.valueOf(8), 8, 0, 1), - new BigDecimal(BigInteger.valueOf(9), 9, 0, 1), - new BigDecimal(BigInteger.TEN, 10, 0, 2), + new BigDecimal(BigInteger.ZERO, 0, 0, 1), + new BigDecimal(BigInteger.ONE, 1, 0, 1), + new BigDecimal(BigInteger.valueOf(2), 2, 0, 1), + new BigDecimal(BigInteger.valueOf(3), 3, 0, 1), + new BigDecimal(BigInteger.valueOf(4), 4, 0, 1), + new BigDecimal(BigInteger.valueOf(5), 5, 0, 1), + new BigDecimal(BigInteger.valueOf(6), 6, 0, 1), + new BigDecimal(BigInteger.valueOf(7), 7, 0, 1), + new BigDecimal(BigInteger.valueOf(8), 8, 0, 1), + new BigDecimal(BigInteger.valueOf(9), 9, 0, 1), + new BigDecimal(BigInteger.TEN, 10, 0, 2), }; // Cache of zero scaled by 0 - 15 @@ -378,178 +380,7 @@ public class BigDecimal extends Number implements Comparable { * @since 1.5 */ public BigDecimal(char[] in, int offset, int len) { - // protect against huge length. - if (offset+len > in.length || offset < 0) - throw new NumberFormatException(); - // This is the primary string to BigDecimal constructor; all - // incoming strings end up here; it uses explicit (inline) - // parsing for speed and generates at most one intermediate - // (temporary) object (a char[] array) for non-compact case. - - // Use locals for all fields values until completion - int prec = 0; // record precision value - int scl = 0; // record scale value - long rs = 0; // the compact value in long - BigInteger rb = null; // the inflated value in BigInteger - - // use array bounds checking to handle too-long, len == 0, - // bad offset, etc. - try { - // handle the sign - boolean isneg = false; // assume positive - if (in[offset] == '-') { - isneg = true; // leading minus means negative - offset++; - len--; - } else if (in[offset] == '+') { // leading + allowed - offset++; - len--; - } - - // should now be at numeric part of the significand - boolean dot = false; // true when there is a '.' - int cfirst = offset; // record start of integer - long exp = 0; // exponent - char c; // current character - - boolean isCompact = (len <= MAX_COMPACT_DIGITS); - // integer significand array & idx is the index to it. The array - // is ONLY used when we can't use a compact representation. - char coeff[] = isCompact ? null : new char[len]; - int idx = 0; - - for (; len > 0; offset++, len--) { - c = in[offset]; - // have digit - if ((c >= '0' && c <= '9') || Character.isDigit(c)) { - // First compact case, we need not to preserve the character - // and we can just compute the value in place. - if (isCompact) { - int digit = Character.digit(c, 10); - if (digit == 0) { - if (prec == 0) - prec = 1; - else if (rs != 0) { - rs *= 10; - ++prec; - } // else digit is a redundant leading zero - } else { - if (prec != 1 || rs != 0) - ++prec; // prec unchanged if preceded by 0s - rs = rs * 10 + digit; - } - } else { // the unscaled value is likely a BigInteger object. - if (c == '0' || Character.digit(c, 10) == 0) { - if (prec == 0) { - coeff[idx] = c; - prec = 1; - } else if (idx != 0) { - coeff[idx++] = c; - ++prec; - } // else c must be a redundant leading zero - } else { - if (prec != 1 || idx != 0) - ++prec; // prec unchanged if preceded by 0s - coeff[idx++] = c; - } - } - if (dot) - ++scl; - continue; - } - // have dot - if (c == '.') { - // have dot - if (dot) // two dots - throw new NumberFormatException(); - dot = true; - continue; - } - // exponent expected - if ((c != 'e') && (c != 'E')) - throw new NumberFormatException(); - offset++; - c = in[offset]; - len--; - boolean negexp = (c == '-'); - // optional sign - if (negexp || c == '+') { - offset++; - c = in[offset]; - len--; - } - if (len <= 0) // no exponent digits - throw new NumberFormatException(); - // skip leading zeros in the exponent - while (len > 10 && Character.digit(c, 10) == 0) { - offset++; - c = in[offset]; - len--; - } - if (len > 10) // too many nonzero exponent digits - throw new NumberFormatException(); - // c now holds first digit of exponent - for (;; len--) { - int v; - if (c >= '0' && c <= '9') { - v = c - '0'; - } else { - v = Character.digit(c, 10); - if (v < 0) // not a digit - throw new NumberFormatException(); - } - exp = exp * 10 + v; - if (len == 1) - break; // that was final character - offset++; - c = in[offset]; - } - if (negexp) // apply sign - exp = -exp; - // Next test is required for backwards compatibility - if ((int)exp != exp) // overflow - throw new NumberFormatException(); - break; // [saves a test] - } - // here when no characters left - if (prec == 0) // no digits found - throw new NumberFormatException(); - - // Adjust scale if exp is not zero. - if (exp != 0) { // had significant exponent - // Can't call checkScale which relies on proper fields value - long adjustedScale = scl - exp; - if (adjustedScale > Integer.MAX_VALUE || - adjustedScale < Integer.MIN_VALUE) - throw new NumberFormatException("Scale out of range."); - scl = (int)adjustedScale; - } - - // Remove leading zeros from precision (digits count) - if (isCompact) { - rs = isneg ? -rs : rs; - } else { - char quick[]; - if (!isneg) { - quick = (coeff.length != prec) ? - Arrays.copyOf(coeff, prec) : coeff; - } else { - quick = new char[prec + 1]; - quick[0] = '-'; - System.arraycopy(coeff, 0, quick, 1, prec); - } - rb = new BigInteger(quick); - rs = compactValFor(rb); - } - } catch (ArrayIndexOutOfBoundsException e) { - throw new NumberFormatException(); - } catch (NegativeArraySizeException e) { - throw new NumberFormatException(); - } - this.scale = scl; - this.precision = prec; - this.intCompact = rs; - this.intVal = (rs != INFLATED) ? null : rb; + this(in,offset,len,MathContext.UNLIMITED); } /** @@ -576,9 +407,254 @@ public class BigDecimal extends Number implements Comparable { * @since 1.5 */ public BigDecimal(char[] in, int offset, int len, MathContext mc) { - this(in, offset, len); - if (mc.precision > 0) - roundThis(mc); + // protect against huge length. + if (offset + len > in.length || offset < 0) + throw new NumberFormatException("Bad offset or len arguments for char[] input."); + // This is the primary string to BigDecimal constructor; all + // incoming strings end up here; it uses explicit (inline) + // parsing for speed and generates at most one intermediate + // (temporary) object (a char[] array) for non-compact case. + + // Use locals for all fields values until completion + int prec = 0; // record precision value + int scl = 0; // record scale value + long rs = 0; // the compact value in long + BigInteger rb = null; // the inflated value in BigInteger + // use array bounds checking to handle too-long, len == 0, + // bad offset, etc. + try { + // handle the sign + boolean isneg = false; // assume positive + if (in[offset] == '-') { + isneg = true; // leading minus means negative + offset++; + len--; + } else if (in[offset] == '+') { // leading + allowed + offset++; + len--; + } + + // should now be at numeric part of the significand + boolean dot = false; // true when there is a '.' + long exp = 0; // exponent + char c; // current character + boolean isCompact = (len <= MAX_COMPACT_DIGITS); + // integer significand array & idx is the index to it. The array + // is ONLY used when we can't use a compact representation. + int idx = 0; + if (isCompact) { + // First compact case, we need not to preserve the character + // and we can just compute the value in place. + for (; len > 0; offset++, len--) { + c = in[offset]; + if ((c == '0')) { // have zero + if (prec == 0) + prec = 1; + else if (rs != 0) { + rs *= 10; + ++prec; + } // else digit is a redundant leading zero + if (dot) + ++scl; + } else if ((c >= '1' && c <= '9')) { // have digit + int digit = c - '0'; + if (prec != 1 || rs != 0) + ++prec; // prec unchanged if preceded by 0s + rs = rs * 10 + digit; + if (dot) + ++scl; + } else if (c == '.') { // have dot + // have dot + if (dot) // two dots + throw new NumberFormatException(); + dot = true; + } else if (Character.isDigit(c)) { // slow path + int digit = Character.digit(c, 10); + if (digit == 0) { + if (prec == 0) + prec = 1; + else if (rs != 0) { + rs *= 10; + ++prec; + } // else digit is a redundant leading zero + } else { + if (prec != 1 || rs != 0) + ++prec; // prec unchanged if preceded by 0s + rs = rs * 10 + digit; + } + if (dot) + ++scl; + } else if ((c == 'e') || (c == 'E')) { + exp = parseExp(in, offset, len); + // Next test is required for backwards compatibility + if ((int) exp != exp) // overflow + throw new NumberFormatException(); + break; // [saves a test] + } else { + throw new NumberFormatException(); + } + } + if (prec == 0) // no digits found + throw new NumberFormatException(); + // Adjust scale if exp is not zero. + if (exp != 0) { // had significant exponent + scl = adjustScale(scl, exp); + } + rs = isneg ? -rs : rs; + int mcp = mc.precision; + int drop = prec - mcp; // prec has range [1, MAX_INT], mcp has range [0, MAX_INT]; + // therefore, this subtract cannot overflow + if (mcp > 0 && drop > 0) { // do rounding + while (drop > 0) { + scl = checkScaleNonZero((long) scl - drop); + rs = divideAndRound(rs, LONG_TEN_POWERS_TABLE[drop], mc.roundingMode.oldMode); + prec = longDigitLength(rs); + drop = prec - mcp; + } + } + } else { + char coeff[] = new char[len]; + for (; len > 0; offset++, len--) { + c = in[offset]; + // have digit + if ((c >= '0' && c <= '9') || Character.isDigit(c)) { + // First compact case, we need not to preserve the character + // and we can just compute the value in place. + if (c == '0' || Character.digit(c, 10) == 0) { + if (prec == 0) { + coeff[idx] = c; + prec = 1; + } else if (idx != 0) { + coeff[idx++] = c; + ++prec; + } // else c must be a redundant leading zero + } else { + if (prec != 1 || idx != 0) + ++prec; // prec unchanged if preceded by 0s + coeff[idx++] = c; + } + if (dot) + ++scl; + continue; + } + // have dot + if (c == '.') { + // have dot + if (dot) // two dots + throw new NumberFormatException(); + dot = true; + continue; + } + // exponent expected + if ((c != 'e') && (c != 'E')) + throw new NumberFormatException(); + exp = parseExp(in, offset, len); + // Next test is required for backwards compatibility + if ((int) exp != exp) // overflow + throw new NumberFormatException(); + break; // [saves a test] + } + // here when no characters left + if (prec == 0) // no digits found + throw new NumberFormatException(); + // Adjust scale if exp is not zero. + if (exp != 0) { // had significant exponent + scl = adjustScale(scl, exp); + } + // Remove leading zeros from precision (digits count) + rb = new BigInteger(coeff, isneg ? -1 : 1, prec); + rs = compactValFor(rb); + int mcp = mc.precision; + if (mcp > 0 && (prec > mcp)) { + if (rs == INFLATED) { + int drop = prec - mcp; + while (drop > 0) { + scl = checkScaleNonZero((long) scl - drop); + rb = divideAndRoundByTenPow(rb, drop, mc.roundingMode.oldMode); + rs = compactValFor(rb); + if (rs != INFLATED) { + prec = longDigitLength(rs); + break; + } + prec = bigDigitLength(rb); + drop = prec - mcp; + } + } + if (rs != INFLATED) { + int drop = prec - mcp; + while (drop > 0) { + scl = checkScaleNonZero((long) scl - drop); + rs = divideAndRound(rs, LONG_TEN_POWERS_TABLE[drop], mc.roundingMode.oldMode); + prec = longDigitLength(rs); + drop = prec - mcp; + } + rb = null; + } + } + } + } catch (ArrayIndexOutOfBoundsException e) { + throw new NumberFormatException(); + } catch (NegativeArraySizeException e) { + throw new NumberFormatException(); + } + this.scale = scl; + this.precision = prec; + this.intCompact = rs; + this.intVal = rb; + } + + private int adjustScale(int scl, long exp) { + long adjustedScale = scl - exp; + if (adjustedScale > Integer.MAX_VALUE || adjustedScale < Integer.MIN_VALUE) + throw new NumberFormatException("Scale out of range."); + scl = (int) adjustedScale; + return scl; + } + + /* + * parse exponent + */ + private static long parseExp(char[] in, int offset, int len){ + long exp = 0; + offset++; + char c = in[offset]; + len--; + boolean negexp = (c == '-'); + // optional sign + if (negexp || c == '+') { + offset++; + c = in[offset]; + len--; + } + if (len <= 0) // no exponent digits + throw new NumberFormatException(); + // skip leading zeros in the exponent + while (len > 10 && (c=='0' || (Character.digit(c, 10) == 0))) { + offset++; + c = in[offset]; + len--; + } + if (len > 10) // too many nonzero exponent digits + throw new NumberFormatException(); + // c now holds first digit of exponent + for (;; len--) { + int v; + if (c >= '0' && c <= '9') { + v = c - '0'; + } else { + v = Character.digit(c, 10); + if (v < 0) // not a digit + throw new NumberFormatException(); + } + exp = exp * 10 + v; + if (len == 1) + break; // that was final character + offset++; + c = in[offset]; + } + if (negexp) // apply sign + exp = -exp; + return exp; } /** @@ -754,9 +830,7 @@ public class BigDecimal extends Number implements Comparable { * @since 1.5 */ public BigDecimal(String val, MathContext mc) { - this(val.toCharArray(), 0, val.length()); - if (mc.precision > 0) - roundThis(mc); + this(val.toCharArray(), 0, val.length(), mc); } /** @@ -804,49 +878,7 @@ public class BigDecimal extends Number implements Comparable { * @throws NumberFormatException if {@code val} is infinite or NaN. */ public BigDecimal(double val) { - if (Double.isInfinite(val) || Double.isNaN(val)) - throw new NumberFormatException("Infinite or NaN"); - - // Translate the double into sign, exponent and significand, according - // to the formulae in JLS, Section 20.10.22. - long valBits = Double.doubleToLongBits(val); - int sign = ((valBits >> 63)==0 ? 1 : -1); - int exponent = (int) ((valBits >> 52) & 0x7ffL); - long significand = (exponent==0 ? (valBits & ((1L<<52) - 1)) << 1 - : (valBits & ((1L<<52) - 1)) | (1L<<52)); - exponent -= 1075; - // At this point, val == sign * significand * 2**exponent. - - /* - * Special case zero to supress nonterminating normalization - * and bogus scale calculation. - */ - if (significand == 0) { - intVal = BigInteger.ZERO; - intCompact = 0; - precision = 1; - return; - } - - // Normalize - while((significand & 1) == 0) { // i.e., significand is even - significand >>= 1; - exponent++; - } - - // Calculate intVal and scale - long s = sign * significand; - BigInteger b; - if (exponent < 0) { - b = BigInteger.valueOf(5).pow(-exponent).multiply(s); - scale = -exponent; - } else if (exponent > 0) { - b = BigInteger.valueOf(2).pow(exponent).multiply(s); - } else { - b = BigInteger.valueOf(s); - } - intCompact = compactValFor(b); - intVal = (intCompact != INFLATED) ? null : b; + this(val,MathContext.UNLIMITED); } /** @@ -868,9 +900,85 @@ public class BigDecimal extends Number implements Comparable { * @since 1.5 */ public BigDecimal(double val, MathContext mc) { - this(val); - if (mc.precision > 0) - roundThis(mc); + if (Double.isInfinite(val) || Double.isNaN(val)) + throw new NumberFormatException("Infinite or NaN"); + // Translate the double into sign, exponent and significand, according + // to the formulae in JLS, Section 20.10.22. + long valBits = Double.doubleToLongBits(val); + int sign = ((valBits >> 63) == 0 ? 1 : -1); + int exponent = (int) ((valBits >> 52) & 0x7ffL); + long significand = (exponent == 0 + ? (valBits & ((1L << 52) - 1)) << 1 + : (valBits & ((1L << 52) - 1)) | (1L << 52)); + exponent -= 1075; + // At this point, val == sign * significand * 2**exponent. + + /* + * Special case zero to supress nonterminating normalization and bogus + * scale calculation. + */ + if (significand == 0) { + this.intVal = BigInteger.ZERO; + this.scale = 0; + this.intCompact = 0; + this.precision = 1; + return; + } + // Normalize + while ((significand & 1) == 0) { // i.e., significand is even + significand >>= 1; + exponent++; + } + int scale = 0; + // Calculate intVal and scale + BigInteger intVal; + long compactVal = sign * significand; + if (exponent == 0) { + intVal = (compactVal == INFLATED) ? INFLATED_BIGINT : null; + } else { + if (exponent < 0) { + intVal = BigInteger.valueOf(5).pow(-exponent).multiply(compactVal); + scale = -exponent; + } else { // (exponent > 0) + intVal = BigInteger.valueOf(2).pow(exponent).multiply(compactVal); + } + compactVal = compactValFor(intVal); + } + int prec = 0; + int mcp = mc.precision; + if (mcp > 0) { // do rounding + int mode = mc.roundingMode.oldMode; + int drop; + if (compactVal == INFLATED) { + prec = bigDigitLength(intVal); + drop = prec - mcp; + while (drop > 0) { + scale = checkScaleNonZero((long) scale - drop); + intVal = divideAndRoundByTenPow(intVal, drop, mode); + compactVal = compactValFor(intVal); + if (compactVal != INFLATED) { + break; + } + prec = bigDigitLength(intVal); + drop = prec - mcp; + } + } + if (compactVal != INFLATED) { + prec = longDigitLength(compactVal); + drop = prec - mcp; + while (drop > 0) { + scale = checkScaleNonZero((long) scale - drop); + compactVal = divideAndRound(compactVal, LONG_TEN_POWERS_TABLE[drop], mc.roundingMode.oldMode); + prec = longDigitLength(compactVal); + drop = prec - mcp; + } + intVal = null; + } + } + this.intVal = intVal; + this.intCompact = compactVal; + this.scale = scale; + this.precision = prec; } /** @@ -881,8 +989,9 @@ public class BigDecimal extends Number implements Comparable { * {@code BigDecimal}. */ public BigDecimal(BigInteger val) { + scale = 0; + intVal = val; intCompact = compactValFor(val); - intVal = (intCompact != INFLATED) ? null : val; } /** @@ -898,9 +1007,7 @@ public class BigDecimal extends Number implements Comparable { * @since 1.5 */ public BigDecimal(BigInteger val, MathContext mc) { - this(val); - if (mc.precision > 0) - roundThis(mc); + this(val,0,mc); } /** @@ -914,7 +1021,8 @@ public class BigDecimal extends Number implements Comparable { */ public BigDecimal(BigInteger unscaledVal, int scale) { // Negative scales are now allowed - this(unscaledVal); + this.intVal = unscaledVal; + this.intCompact = compactValFor(unscaledVal); this.scale = scale; } @@ -934,10 +1042,41 @@ public class BigDecimal extends Number implements Comparable { * @since 1.5 */ public BigDecimal(BigInteger unscaledVal, int scale, MathContext mc) { - this(unscaledVal); + long compactVal = compactValFor(unscaledVal); + int mcp = mc.precision; + int prec = 0; + if (mcp > 0) { // do rounding + int mode = mc.roundingMode.oldMode; + if (compactVal == INFLATED) { + prec = bigDigitLength(unscaledVal); + int drop = prec - mcp; + while (drop > 0) { + scale = checkScaleNonZero((long) scale - drop); + unscaledVal = divideAndRoundByTenPow(unscaledVal, drop, mode); + compactVal = compactValFor(unscaledVal); + if (compactVal != INFLATED) { + break; + } + prec = bigDigitLength(unscaledVal); + drop = prec - mcp; + } + } + if (compactVal != INFLATED) { + prec = longDigitLength(compactVal); + int drop = prec - mcp; // drop can't be more than 18 + while (drop > 0) { + scale = checkScaleNonZero((long) scale - drop); + compactVal = divideAndRound(compactVal, LONG_TEN_POWERS_TABLE[drop], mode); + prec = longDigitLength(compactVal); + drop = prec - mcp; + } + unscaledVal = null; + } + } + this.intVal = unscaledVal; + this.intCompact = compactVal; this.scale = scale; - if (mc.precision > 0) - roundThis(mc); + this.precision = prec; } /** @@ -949,7 +1088,9 @@ public class BigDecimal extends Number implements Comparable { * @since 1.5 */ public BigDecimal(int val) { - intCompact = val; + this.intCompact = val; + this.scale = 0; + this.intVal = null; } /** @@ -964,9 +1105,24 @@ public class BigDecimal extends Number implements Comparable { * @since 1.5 */ public BigDecimal(int val, MathContext mc) { - intCompact = val; - if (mc.precision > 0) - roundThis(mc); + int mcp = mc.precision; + long compactVal = val; + int scale = 0; + int prec = 0; + if (mcp > 0) { // do rounding + prec = longDigitLength(compactVal); + int drop = prec - mcp; // drop can't be more than 18 + while (drop > 0) { + scale = checkScaleNonZero((long) scale - drop); + compactVal = divideAndRound(compactVal, LONG_TEN_POWERS_TABLE[drop], mc.roundingMode.oldMode); + prec = longDigitLength(compactVal); + drop = prec - mcp; + } + } + this.intVal = null; + this.intCompact = compactVal; + this.scale = scale; + this.precision = prec; } /** @@ -978,7 +1134,8 @@ public class BigDecimal extends Number implements Comparable { */ public BigDecimal(long val) { this.intCompact = val; - this.intVal = (val == INFLATED) ? BigInteger.valueOf(val) : null; + this.intVal = (val == INFLATED) ? INFLATED_BIGINT : null; + this.scale = 0; } /** @@ -993,9 +1150,42 @@ public class BigDecimal extends Number implements Comparable { * @since 1.5 */ public BigDecimal(long val, MathContext mc) { - this(val); - if (mc.precision > 0) - roundThis(mc); + int mcp = mc.precision; + int mode = mc.roundingMode.oldMode; + int prec = 0; + int scale = 0; + BigInteger intVal = (val == INFLATED) ? INFLATED_BIGINT : null; + if (mcp > 0) { // do rounding + if (val == INFLATED) { + prec = 19; + int drop = prec - mcp; + while (drop > 0) { + scale = checkScaleNonZero((long) scale - drop); + intVal = divideAndRoundByTenPow(intVal, drop, mode); + val = compactValFor(intVal); + if (val != INFLATED) { + break; + } + prec = bigDigitLength(intVal); + drop = prec - mcp; + } + } + if (val != INFLATED) { + prec = longDigitLength(val); + int drop = prec - mcp; + while (drop > 0) { + scale = checkScaleNonZero((long) scale - drop); + val = divideAndRound(val, LONG_TEN_POWERS_TABLE[drop], mc.roundingMode.oldMode); + prec = longDigitLength(val); + drop = prec - mcp; + } + intVal = null; + } + } + this.intVal = intVal; + this.intCompact = val; + this.scale = scale; + this.precision = prec; } // Static Factory Methods @@ -1016,13 +1206,10 @@ public class BigDecimal extends Number implements Comparable { if (scale == 0) return valueOf(unscaledVal); else if (unscaledVal == 0) { - if (scale > 0 && scale < ZERO_SCALED_BY.length) - return ZERO_SCALED_BY[scale]; - else - return new BigDecimal(BigInteger.ZERO, 0, scale, 1); + return zeroValueOf(scale); } return new BigDecimal(unscaledVal == INFLATED ? - BigInteger.valueOf(unscaledVal) : null, + INFLATED_BIGINT : null, unscaledVal, scale, 0); } @@ -1041,7 +1228,34 @@ public class BigDecimal extends Number implements Comparable { return zeroThroughTen[(int)val]; else if (val != INFLATED) return new BigDecimal(null, val, 0, 0); - return new BigDecimal(BigInteger.valueOf(val), val, 0, 0); + return new BigDecimal(INFLATED_BIGINT, val, 0, 0); + } + + static BigDecimal valueOf(long unscaledVal, int scale, int prec) { + if (scale == 0 && unscaledVal >= 0 && unscaledVal < zeroThroughTen.length) { + return zeroThroughTen[(int) unscaledVal]; + } else if (unscaledVal == 0) { + return zeroValueOf(scale); + } + return new BigDecimal(unscaledVal == INFLATED ? INFLATED_BIGINT : null, + unscaledVal, scale, prec); + } + + static BigDecimal valueOf(BigInteger intVal, int scale, int prec) { + long val = compactValFor(intVal); + if (val == 0) { + return zeroValueOf(scale); + } else if (scale == 0 && val >= 0 && val < zeroThroughTen.length) { + return zeroThroughTen[(int) val]; + } + return new BigDecimal(intVal, val, scale, prec); + } + + static BigDecimal zeroValueOf(int scale) { + if (scale >= 0 && scale < ZERO_SCALED_BY.length) + return ZERO_SCALED_BY[scale]; + else + return new BigDecimal(BigInteger.ZERO, 0, scale, 1); } /** @@ -1079,42 +1293,19 @@ public class BigDecimal extends Number implements Comparable { * @return {@code this + augend} */ public BigDecimal add(BigDecimal augend) { - long xs = this.intCompact; - long ys = augend.intCompact; - BigInteger fst = (xs != INFLATED) ? null : this.intVal; - BigInteger snd = (ys != INFLATED) ? null : augend.intVal; - int rscale = this.scale; - - long sdiff = (long)rscale - augend.scale; - if (sdiff != 0) { - if (sdiff < 0) { - int raise = checkScale(-sdiff); - rscale = augend.scale; - if (xs == INFLATED || - (xs = longMultiplyPowerTen(xs, raise)) == INFLATED) - fst = bigMultiplyPowerTen(raise); + if (this.intCompact != INFLATED) { + if ((augend.intCompact != INFLATED)) { + return add(this.intCompact, this.scale, augend.intCompact, augend.scale); } else { - int raise = augend.checkScale(sdiff); - if (ys == INFLATED || - (ys = longMultiplyPowerTen(ys, raise)) == INFLATED) - snd = augend.bigMultiplyPowerTen(raise); + return add(this.intCompact, this.scale, augend.intVal, augend.scale); + } + } else { + if ((augend.intCompact != INFLATED)) { + return add(augend.intCompact, augend.scale, this.intVal, this.scale); + } else { + return add(this.intVal, this.scale, augend.intVal, augend.scale); } } - if (xs != INFLATED && ys != INFLATED) { - long sum = xs + ys; - // See "Hacker's Delight" section 2-12 for explanation of - // the overflow test. - if ( (((sum ^ xs) & (sum ^ ys))) >= 0L) // not overflowed - return BigDecimal.valueOf(sum, rscale); - } - if (fst == null) - fst = BigInteger.valueOf(xs); - if (snd == null) - snd = BigInteger.valueOf(ys); - BigInteger sum = fst.add(snd); - return (fst.signum == snd.signum) ? - new BigDecimal(sum, INFLATED, rscale, 0) : - new BigDecimal(sum, rscale); } /** @@ -1136,10 +1327,6 @@ public class BigDecimal extends Number implements Comparable { return add(augend); BigDecimal lhs = this; - // Could optimize if values are compact - this.inflate(); - augend.inflate(); - // If either number is zero then the other number, rounded and // scaled if necessary, is used as the result. { @@ -1150,20 +1337,14 @@ public class BigDecimal extends Number implements Comparable { int preferredScale = Math.max(lhs.scale(), augend.scale()); BigDecimal result; - // Could use a factory for zero instead of a new object if (lhsIsZero && augendIsZero) - return new BigDecimal(BigInteger.ZERO, 0, preferredScale, 0); - + return zeroValueOf(preferredScale); result = lhsIsZero ? doRound(augend, mc) : doRound(lhs, mc); if (result.scale() == preferredScale) return result; else if (result.scale() > preferredScale) { - BigDecimal scaledResult = - new BigDecimal(result.intVal, result.intCompact, - result.scale, 0); - scaledResult.stripZerosToMatchScale(preferredScale); - return scaledResult; + return stripZerosToMatchScale(result.intVal, result.intCompact, result.scale, preferredScale); } else { // result.scale < preferredScale int precisionDiff = mc.precision - result.precision(); int scaleDiff = preferredScale - result.scale(); @@ -1176,17 +1357,14 @@ public class BigDecimal extends Number implements Comparable { } } - long padding = (long)lhs.scale - augend.scale; - if (padding != 0) { // scales differ; alignment needed + long padding = (long) lhs.scale - augend.scale; + if (padding != 0) { // scales differ; alignment needed BigDecimal arg[] = preAlign(lhs, augend, padding, mc); matchScale(arg); - lhs = arg[0]; + lhs = arg[0]; augend = arg[1]; } - - BigDecimal d = new BigDecimal(lhs.inflate().add(augend.inflate()), - lhs.scale); - return doRound(d, mc); + return doRound(lhs.inflated().add(augend.inflated()), lhs.scale, mc); } /** @@ -1211,27 +1389,26 @@ public class BigDecimal extends Number implements Comparable { * that the number of digits of the smaller operand could be * reduced even though the significands partially overlapped. */ - private BigDecimal[] preAlign(BigDecimal lhs, BigDecimal augend, - long padding, MathContext mc) { + private BigDecimal[] preAlign(BigDecimal lhs, BigDecimal augend, long padding, MathContext mc) { assert padding != 0; BigDecimal big; BigDecimal small; - if (padding < 0) { // lhs is big; augend is small - big = lhs; + if (padding < 0) { // lhs is big; augend is small + big = lhs; small = augend; - } else { // lhs is small; augend is big - big = augend; + } else { // lhs is small; augend is big + big = augend; small = lhs; } /* - * This is the estimated scale of an ulp of the result; it - * assumes that the result doesn't have a carry-out on a true - * add (e.g. 999 + 1 => 1000) or any subtractive cancellation - * on borrowing (e.g. 100 - 1.2 => 98.8) + * This is the estimated scale of an ulp of the result; it assumes that + * the result doesn't have a carry-out on a true add (e.g. 999 + 1 => + * 1000) or any subtractive cancellation on borrowing (e.g. 100 - 1.2 => + * 98.8) */ - long estResultUlpScale = (long)big.scale - big.precision() + mc.precision; + long estResultUlpScale = (long) big.scale - big.precision() + mc.precision; /* * The low-order digit position of big is big.scale(). This @@ -1242,11 +1419,10 @@ public class BigDecimal extends Number implements Comparable { * disjoint *and* the digit positions of small should not be * directly visible in the result. */ - long smallHighDigitPos = (long)small.scale - small.precision() + 1; - if (smallHighDigitPos > big.scale + 2 && // big and small disjoint + long smallHighDigitPos = (long) small.scale - small.precision() + 1; + if (smallHighDigitPos > big.scale + 2 && // big and small disjoint smallHighDigitPos > estResultUlpScale + 2) { // small digits not visible - small = BigDecimal.valueOf(small.signum(), - this.checkScale(Math.max(big.scale, estResultUlpScale) + 3)); + small = BigDecimal.valueOf(small.signum(), this.checkScale(Math.max(big.scale, estResultUlpScale) + 3)); } // Since addition is symmetric, preserving input order in @@ -1264,7 +1440,22 @@ public class BigDecimal extends Number implements Comparable { * @return {@code this - subtrahend} */ public BigDecimal subtract(BigDecimal subtrahend) { - return add(subtrahend.negate()); + if (this.intCompact != INFLATED) { + if ((subtrahend.intCompact != INFLATED)) { + return add(this.intCompact, this.scale, -subtrahend.intCompact, subtrahend.scale); + } else { + return add(this.intCompact, this.scale, subtrahend.intVal.negate(), subtrahend.scale); + } + } else { + if ((subtrahend.intCompact != INFLATED)) { + // Pair of subtrahend values given before pair of + // values from this BigDecimal to avoid need for + // method overloading on the specialized add method + return add(-subtrahend.intCompact, subtrahend.scale, this.intVal, this.scale); + } else { + return add(this.intVal, this.scale, subtrahend.intVal.negate(), subtrahend.scale); + } + } } /** @@ -1282,11 +1473,10 @@ public class BigDecimal extends Number implements Comparable { * @since 1.5 */ public BigDecimal subtract(BigDecimal subtrahend, MathContext mc) { - BigDecimal nsubtrahend = subtrahend.negate(); if (mc.precision == 0) - return add(nsubtrahend); + return subtract(subtrahend); // share the special rounding code in add() - return add(nsubtrahend, mc); + return add(subtrahend.negate(), mc); } /** @@ -1298,37 +1488,20 @@ public class BigDecimal extends Number implements Comparable { * @return {@code this * multiplicand} */ public BigDecimal multiply(BigDecimal multiplicand) { - long x = this.intCompact; - long y = multiplicand.intCompact; - int productScale = checkScale((long)scale + multiplicand.scale); - - // Might be able to do a more clever check incorporating the - // inflated check into the overflow computation. - if (x != INFLATED && y != INFLATED) { - /* - * If the product is not an overflowed value, continue - * to use the compact representation. if either of x or y - * is INFLATED, the product should also be regarded as - * an overflow. Before using the overflow test suggested in - * "Hacker's Delight" section 2-12, we perform quick checks - * using the precision information to see whether the overflow - * would occur since division is expensive on most CPUs. - */ - long product = x * y; - long prec = this.precision() + multiplicand.precision(); - if (prec < 19 || (prec < 21 && (y == 0 || product / y == x))) - return BigDecimal.valueOf(product, productScale); - return new BigDecimal(BigInteger.valueOf(x).multiply(y), INFLATED, - productScale, 0); + int productScale = checkScale((long) scale + multiplicand.scale); + if (this.intCompact != INFLATED) { + if ((multiplicand.intCompact != INFLATED)) { + return multiply(this.intCompact, multiplicand.intCompact, productScale); + } else { + return multiply(this.intCompact, multiplicand.intVal, productScale); + } + } else { + if ((multiplicand.intCompact != INFLATED)) { + return multiply(multiplicand.intCompact, this.intVal, productScale); + } else { + return multiply(this.intVal, multiplicand.intVal, productScale); + } } - BigInteger rb; - if (x == INFLATED && y == INFLATED) - rb = this.intVal.multiply(multiplicand.intVal); - else if (x != INFLATED) - rb = multiplicand.intVal.multiply(x); - else - rb = this.intVal.multiply(y); - return new BigDecimal(rb, INFLATED, productScale, 0); } /** @@ -1345,7 +1518,20 @@ public class BigDecimal extends Number implements Comparable { public BigDecimal multiply(BigDecimal multiplicand, MathContext mc) { if (mc.precision == 0) return multiply(multiplicand); - return doRound(this.multiply(multiplicand), mc); + int productScale = checkScale((long) scale + multiplicand.scale); + if (this.intCompact != INFLATED) { + if ((multiplicand.intCompact != INFLATED)) { + return multiplyAndRound(this.intCompact, multiplicand.intCompact, productScale, mc); + } else { + return multiplyAndRound(this.intCompact, multiplicand.intVal, productScale, mc); + } + } else { + if ((multiplicand.intCompact != INFLATED)) { + return multiplyAndRound(multiplicand.intCompact, this.intVal, productScale, mc); + } else { + return multiplyAndRound(this.intVal, multiplicand.intVal, productScale, mc); + } + } } /** @@ -1377,120 +1563,21 @@ public class BigDecimal extends Number implements Comparable { * @see #ROUND_UNNECESSARY */ public BigDecimal divide(BigDecimal divisor, int scale, int roundingMode) { - /* - * IMPLEMENTATION NOTE: This method *must* return a new object - * since divideAndRound uses divide to generate a value whose - * scale is then modified. - */ if (roundingMode < ROUND_UP || roundingMode > ROUND_UNNECESSARY) throw new IllegalArgumentException("Invalid rounding mode"); - /* - * Rescale dividend or divisor (whichever can be "upscaled" to - * produce correctly scaled quotient). - * Take care to detect out-of-range scales - */ - BigDecimal dividend = this; - if (checkScale((long)scale + divisor.scale) > this.scale) - dividend = this.setScale(scale + divisor.scale, ROUND_UNNECESSARY); - else - divisor = divisor.setScale(checkScale((long)this.scale - scale), - ROUND_UNNECESSARY); - return divideAndRound(dividend.intCompact, dividend.intVal, - divisor.intCompact, divisor.intVal, - scale, roundingMode, scale); - } - - /** - * Internally used for division operation. The dividend and divisor are - * passed both in {@code long} format and {@code BigInteger} format. The - * returned {@code BigDecimal} object is the quotient whose scale is set to - * the passed in scale. If the remainder is not zero, it will be rounded - * based on the passed in roundingMode. Also, if the remainder is zero and - * the last parameter, i.e. preferredScale is NOT equal to scale, the - * trailing zeros of the result is stripped to match the preferredScale. - */ - private static BigDecimal divideAndRound(long ldividend, BigInteger bdividend, - long ldivisor, BigInteger bdivisor, - int scale, int roundingMode, - int preferredScale) { - boolean isRemainderZero; // record remainder is zero or not - int qsign; // quotient sign - long q = 0, r = 0; // store quotient & remainder in long - MutableBigInteger mq = null; // store quotient - MutableBigInteger mr = null; // store remainder - MutableBigInteger mdivisor = null; - boolean isLongDivision = (ldividend != INFLATED && ldivisor != INFLATED); - if (isLongDivision) { - q = ldividend / ldivisor; - if (roundingMode == ROUND_DOWN && scale == preferredScale) - return new BigDecimal(null, q, scale, 0); - r = ldividend % ldivisor; - isRemainderZero = (r == 0); - qsign = ((ldividend < 0) == (ldivisor < 0)) ? 1 : -1; + if (this.intCompact != INFLATED) { + if ((divisor.intCompact != INFLATED)) { + return divide(this.intCompact, this.scale, divisor.intCompact, divisor.scale, scale, roundingMode); + } else { + return divide(this.intCompact, this.scale, divisor.intVal, divisor.scale, scale, roundingMode); + } } else { - if (bdividend == null) - bdividend = BigInteger.valueOf(ldividend); - // Descend into mutables for faster remainder checks - MutableBigInteger mdividend = new MutableBigInteger(bdividend.mag); - mq = new MutableBigInteger(); - if (ldivisor != INFLATED) { - r = mdividend.divide(ldivisor, mq); - isRemainderZero = (r == 0); - qsign = (ldivisor < 0) ? -bdividend.signum : bdividend.signum; + if ((divisor.intCompact != INFLATED)) { + return divide(this.intVal, this.scale, divisor.intCompact, divisor.scale, scale, roundingMode); } else { - mdivisor = new MutableBigInteger(bdivisor.mag); - mr = mdividend.divide(mdivisor, mq); - isRemainderZero = mr.isZero(); - qsign = (bdividend.signum != bdivisor.signum) ? -1 : 1; + return divide(this.intVal, this.scale, divisor.intVal, divisor.scale, scale, roundingMode); } } - boolean increment = false; - if (!isRemainderZero) { - int cmpFracHalf; - /* Round as appropriate */ - if (roundingMode == ROUND_UNNECESSARY) { // Rounding prohibited - throw new ArithmeticException("Rounding necessary"); - } else if (roundingMode == ROUND_UP) { // Away from zero - increment = true; - } else if (roundingMode == ROUND_DOWN) { // Towards zero - increment = false; - } else if (roundingMode == ROUND_CEILING) { // Towards +infinity - increment = (qsign > 0); - } else if (roundingMode == ROUND_FLOOR) { // Towards -infinity - increment = (qsign < 0); - } else { - if (isLongDivision || ldivisor != INFLATED) { - if (r <= HALF_LONG_MIN_VALUE || r > HALF_LONG_MAX_VALUE) { - cmpFracHalf = 1; // 2 * r can't fit into long - } else { - cmpFracHalf = longCompareMagnitude(2 * r, ldivisor); - } - } else { - cmpFracHalf = mr.compareHalf(mdivisor); - } - if (cmpFracHalf < 0) - increment = false; // We're closer to higher digit - else if (cmpFracHalf > 0) // We're closer to lower digit - increment = true; - else if (roundingMode == ROUND_HALF_UP) - increment = true; - else if (roundingMode == ROUND_HALF_DOWN) - increment = false; - else // roundingMode == ROUND_HALF_EVEN, true iff quotient is odd - increment = isLongDivision ? (q & 1L) != 0L : mq.isOdd(); - } - } - BigDecimal res; - if (isLongDivision) - res = new BigDecimal(null, (increment ? q + qsign : q), scale, 0); - else { - if (increment) - mq.add(MutableBigInteger.ONE); - res = mq.toBigDecimal(qsign, scale); - } - if (isRemainderZero && preferredScale != scale) - res.stripZerosToMatchScale(preferredScale); - return res; } /** @@ -1541,7 +1628,7 @@ public class BigDecimal extends Number implements Comparable { * @see #ROUND_UNNECESSARY */ public BigDecimal divide(BigDecimal divisor, int roundingMode) { - return this.divide(divisor, scale, roundingMode); + return this.divide(divisor, scale, roundingMode); } /** @@ -1588,15 +1675,11 @@ public class BigDecimal extends Number implements Comparable { } // Calculate preferred scale - int preferredScale = saturateLong((long)this.scale - divisor.scale); - if (this.signum() == 0) // 0/y - return (preferredScale >= 0 && - preferredScale < ZERO_SCALED_BY.length) ? - ZERO_SCALED_BY[preferredScale] : - BigDecimal.valueOf(0, preferredScale); + int preferredScale = saturateLong((long) this.scale - divisor.scale); + + if (this.signum() == 0) // 0/y + return zeroValueOf(preferredScale); else { - this.inflate(); - divisor.inflate(); /* * If the quotient this/divisor has a terminating decimal * expansion, the expansion can have no more than @@ -1623,7 +1706,6 @@ public class BigDecimal extends Number implements Comparable { // the desired one by removing trailing zeros; since the // exact divide method does not have an explicit digit // limit, we can add zeros too. - if (preferredScale > quotientScale) return quotient.setScale(preferredScale, ROUND_UNNECESSARY); @@ -1668,38 +1750,23 @@ public class BigDecimal extends Number implements Comparable { throw new ArithmeticException("Division undefined"); // NaN throw new ArithmeticException("Division by zero"); } - if (dividend.signum() == 0) // 0/y - return new BigDecimal(BigInteger.ZERO, 0, - saturateLong(preferredScale), 1); - - // Normalize dividend & divisor so that both fall into [0.1, 0.999...] + if (dividend.signum() == 0) // 0/y + return zeroValueOf(saturateLong(preferredScale)); int xscale = dividend.precision(); int yscale = divisor.precision(); - dividend = new BigDecimal(dividend.intVal, dividend.intCompact, - xscale, xscale); - divisor = new BigDecimal(divisor.intVal, divisor.intCompact, - yscale, yscale); - if (dividend.compareMagnitude(divisor) > 0) // satisfy constraint (b) - yscale = divisor.scale -= 1; // [that is, divisor *= 10] - - // In order to find out whether the divide generates the exact result, - // we avoid calling the above divide method. 'quotient' holds the - // return BigDecimal object whose scale will be set to 'scl'. - BigDecimal quotient; - int scl = checkScale(preferredScale + yscale - xscale + mcp); - if (checkScale((long)mcp + yscale) > xscale) - dividend = dividend.setScale(mcp + yscale, ROUND_UNNECESSARY); - else - divisor = divisor.setScale(checkScale((long)xscale - mcp), - ROUND_UNNECESSARY); - quotient = divideAndRound(dividend.intCompact, dividend.intVal, - divisor.intCompact, divisor.intVal, - scl, mc.roundingMode.oldMode, - checkScale(preferredScale)); - // doRound, here, only affects 1000000000 case. - quotient = doRound(quotient, mc); - - return quotient; + if(dividend.intCompact!=INFLATED) { + if(divisor.intCompact!=INFLATED) { + return divide(dividend.intCompact, xscale, divisor.intCompact, yscale, preferredScale, mc); + } else { + return divide(dividend.intCompact, xscale, divisor.intVal, yscale, preferredScale, mc); + } + } else { + if(divisor.intCompact!=INFLATED) { + return divide(dividend.intVal, xscale, divisor.intCompact, yscale, preferredScale, mc); + } else { + return divide(dividend.intVal, xscale, divisor.intVal, yscale, preferredScale, mc); + } + } } /** @@ -1715,13 +1782,13 @@ public class BigDecimal extends Number implements Comparable { */ public BigDecimal divideToIntegralValue(BigDecimal divisor) { // Calculate preferred scale - int preferredScale = saturateLong((long)this.scale - divisor.scale); + int preferredScale = saturateLong((long) this.scale - divisor.scale); if (this.compareMagnitude(divisor) < 0) { // much faster when this << divisor - return BigDecimal.valueOf(0, preferredScale); + return zeroValueOf(preferredScale); } - if(this.signum() == 0 && divisor.signum() != 0) + if (this.signum() == 0 && divisor.signum() != 0) return this.setScale(preferredScale, ROUND_UNNECESSARY); // Perform a divide with enough digits to round to a correct @@ -1735,13 +1802,14 @@ public class BigDecimal extends Number implements Comparable { RoundingMode.DOWN)); if (quotient.scale > 0) { quotient = quotient.setScale(0, RoundingMode.DOWN); - quotient.stripZerosToMatchScale(preferredScale); + quotient = stripZerosToMatchScale(quotient.intVal, quotient.intCompact, quotient.scale, preferredScale); } if (quotient.scale < preferredScale) { // pad with zeros if necessary quotient = quotient.setScale(preferredScale, ROUND_UNNECESSARY); } + return quotient; } @@ -1766,8 +1834,8 @@ public class BigDecimal extends Number implements Comparable { * @author Joseph D. Darcy */ public BigDecimal divideToIntegralValue(BigDecimal divisor, MathContext mc) { - if (mc.precision == 0 || // exact result - (this.compareMagnitude(divisor) < 0) ) // zero result + if (mc.precision == 0 || // exact result + (this.compareMagnitude(divisor) < 0)) // zero result return divideToIntegralValue(divisor); // Calculate preferred scale @@ -1780,8 +1848,7 @@ public class BigDecimal extends Number implements Comparable { * digits. Next, remove any fractional digits from the * quotient and adjust the scale to the preferred value. */ - BigDecimal result = this. - divide(divisor, new MathContext(mc.precision, RoundingMode.DOWN)); + BigDecimal result = this.divide(divisor, new MathContext(mc.precision, RoundingMode.DOWN)); if (result.scale() < 0) { /* @@ -1811,8 +1878,7 @@ public class BigDecimal extends Number implements Comparable { return result.setScale(result.scale() + Math.min(precisionDiff, preferredScale - result.scale) ); } else { - result.stripZerosToMatchScale(preferredScale); - return result; + return stripZerosToMatchScale(result.intVal,result.intCompact,result.scale,preferredScale); } } @@ -1954,8 +2020,7 @@ public class BigDecimal extends Number implements Comparable { // No need to calculate pow(n) if result will over/underflow. // Don't attempt to support "supernormal" numbers. int newScale = checkScale((long)scale * n); - this.inflate(); - return new BigDecimal(intVal.pow(n), newScale); + return new BigDecimal(this.inflated().pow(n), newScale); } @@ -2016,12 +2081,10 @@ public class BigDecimal extends Number implements Comparable { throw new ArithmeticException("Invalid operation"); if (n == 0) return ONE; // x**0 == 1 in X3.274 - this.inflate(); BigDecimal lhs = this; MathContext workmc = mc; // working settings int mag = Math.abs(n); // magnitude of n if (mc.precision > 0) { - int elength = longDigitLength(mag); // length of n in digits if (elength > mc.precision) // X3.274 rule throw new ArithmeticException("Invalid operation"); @@ -2044,7 +2107,7 @@ public class BigDecimal extends Number implements Comparable { // else (!seenbit) no point in squaring ONE } // if negative n, calculate the reciprocal using working precision - if (n<0) // [hence mc.precision>0] + if (n < 0) // [hence mc.precision>0] acc=ONE.divide(acc, workmc); // round to final precision and strip zeros return doRound(acc, mc); @@ -2083,14 +2146,11 @@ public class BigDecimal extends Number implements Comparable { * @return {@code -this}. */ public BigDecimal negate() { - BigDecimal result; - if (intCompact != INFLATED) - result = BigDecimal.valueOf(-intCompact, scale); - else { - result = new BigDecimal(intVal.negate(), scale); - result.precision = precision; + if (intCompact == INFLATED) { + return new BigDecimal(intVal.negate(), INFLATED, scale, precision); + } else { + return valueOf(-intCompact, scale, precision); } - return result; } /** @@ -2186,7 +2246,7 @@ public class BigDecimal extends Number implements Comparable { if (s != INFLATED) result = longDigitLength(s); else - result = bigDigitLength(inflate()); + result = bigDigitLength(intVal); precision = result; } return result; @@ -2202,7 +2262,7 @@ public class BigDecimal extends Number implements Comparable { * @since 1.2 */ public BigInteger unscaledValue() { - return this.inflate(); + return this.inflated(); } // Rounding Modes @@ -2383,29 +2443,41 @@ public class BigDecimal extends Number implements Comparable { if (newScale == oldScale) // easy case return this; if (this.signum() == 0) // zero can have any scale - return BigDecimal.valueOf(0, newScale); - - long rs = this.intCompact; - if (newScale > oldScale) { - int raise = checkScale((long)newScale - oldScale); - BigInteger rb = null; - if (rs == INFLATED || - (rs = longMultiplyPowerTen(rs, raise)) == INFLATED) - rb = bigMultiplyPowerTen(raise); - return new BigDecimal(rb, rs, newScale, - (precision > 0) ? precision + raise : 0); + return zeroValueOf(newScale); + if(this.intCompact!=INFLATED) { + long rs = this.intCompact; + if (newScale > oldScale) { + int raise = checkScale((long) newScale - oldScale); + if ((rs = longMultiplyPowerTen(rs, raise)) != INFLATED) { + return valueOf(rs,newScale); + } + BigInteger rb = bigMultiplyPowerTen(raise); + return new BigDecimal(rb, INFLATED, newScale, (precision > 0) ? precision + raise : 0); + } else { + // newScale < oldScale -- drop some digits + // Can't predict the precision due to the effect of rounding. + int drop = checkScale((long) oldScale - newScale); + if (drop < LONG_TEN_POWERS_TABLE.length) { + return divideAndRound(rs, LONG_TEN_POWERS_TABLE[drop], newScale, roundingMode, newScale); + } else { + return divideAndRound(this.inflated(), bigTenToThe(drop), newScale, roundingMode, newScale); + } + } } else { - // newScale < oldScale -- drop some digits - // Can't predict the precision due to the effect of rounding. - int drop = checkScale((long)oldScale - newScale); - if (drop < LONG_TEN_POWERS_TABLE.length) - return divideAndRound(rs, this.intVal, - LONG_TEN_POWERS_TABLE[drop], null, - newScale, roundingMode, newScale); - else - return divideAndRound(rs, this.intVal, - INFLATED, bigTenToThe(drop), - newScale, roundingMode, newScale); + if (newScale > oldScale) { + int raise = checkScale((long) newScale - oldScale); + BigInteger rb = bigMultiplyPowerTen(this.intVal,raise); + return new BigDecimal(rb, INFLATED, newScale, (precision > 0) ? precision + raise : 0); + } else { + // newScale < oldScale -- drop some digits + // Can't predict the precision due to the effect of rounding. + int drop = checkScale((long) oldScale - newScale); + if (drop < LONG_TEN_POWERS_TABLE.length) + return divideAndRound(this.intVal, LONG_TEN_POWERS_TABLE[drop], newScale, roundingMode, + newScale); + else + return divideAndRound(this.intVal, bigTenToThe(drop), newScale, roundingMode, newScale); + } } } @@ -2524,10 +2596,11 @@ public class BigDecimal extends Number implements Comparable { * @since 1.5 */ public BigDecimal stripTrailingZeros() { - this.inflate(); - BigDecimal result = new BigDecimal(intVal, scale); - result.stripZerosToMatchScale(Long.MIN_VALUE); - return result; + if(intCompact!=INFLATED) { + return createAndStripZerosToMatchScale(intCompact, scale, Long.MIN_VALUE); + } else { + return createAndStripZerosToMatchScale(intVal, scale, Long.MIN_VALUE); + } } // Comparison Operations @@ -2647,7 +2720,7 @@ public class BigDecimal extends Number implements Comparable { } else if (xs != INFLATED) return xs == compactValFor(this.intVal); - return this.inflate().equals(xDec.inflate()); + return this.inflated().equals(xDec.inflated()); } /** @@ -2872,13 +2945,38 @@ public class BigDecimal extends Number implements Comparable { * @see #toEngineeringString() */ public String toPlainString() { - BigDecimal bd = this; - if (bd.scale < 0) - bd = bd.setScale(0); - bd.inflate(); - if (bd.scale == 0) // No decimal point - return bd.intVal.toString(); - return bd.getValueString(bd.signum(), bd.intVal.abs().toString(), bd.scale); + if(scale==0) { + if(intCompact!=INFLATED) { + return Long.toString(intCompact); + } else { + return intVal.toString(); + } + } + if(this.scale<0) { // No decimal point + if(signum()==0) { + return "0"; + } + int tailingZeros = checkScaleNonZero((-(long)scale)); + StringBuilder buf; + if(intCompact!=INFLATED) { + buf = new StringBuilder(20+tailingZeros); + buf.append(intCompact); + } else { + String str = intVal.toString(); + buf = new StringBuilder(str.length()+tailingZeros); + buf.append(str); + } + for (int i = 0; i < tailingZeros; i++) + buf.append('0'); + return buf.toString(); + } + String str ; + if(intCompact!=INFLATED) { + str = Long.toString(Math.abs(intCompact)); + } else { + str = intVal.abs().toString(); + } + return getValueString(signum(), str, scale); } /* Returns a digit.digit string */ @@ -2922,7 +3020,7 @@ public class BigDecimal extends Number implements Comparable { */ public BigInteger toBigInteger() { // force to an integer, quietly - return this.setScale(0, ROUND_DOWN).inflate(); + return this.setScale(0, ROUND_DOWN).inflated(); } /** @@ -2937,7 +3035,7 @@ public class BigDecimal extends Number implements Comparable { */ public BigInteger toBigIntegerExact() { // round to an integer, with Exception if decimal part non-0 - return this.setScale(0, ROUND_UNNECESSARY).inflate(); + return this.setScale(0, ROUND_UNNECESSARY).inflated(); } /** @@ -2990,7 +3088,7 @@ public class BigDecimal extends Number implements Comparable { BigDecimal num = this.setScale(0, ROUND_UNNECESSARY); if (num.precision() >= 19) // need to check carefully LongOverflow.check(num); - return num.inflate().longValue(); + return num.inflated().longValue(); } private static class LongOverflow { @@ -3001,9 +3099,9 @@ public class BigDecimal extends Number implements Comparable { private static final BigInteger LONGMAX = BigInteger.valueOf(Long.MAX_VALUE); public static void check(BigDecimal num) { - num.inflate(); - if ((num.intVal.compareTo(LONGMIN) < 0) || - (num.intVal.compareTo(LONGMAX) > 0)) + BigInteger intVal = num.inflated(); + if (intVal.compareTo(LONGMIN) < 0 || + intVal.compareTo(LONGMAX) > 0) throw new java.lang.ArithmeticException("Overflow"); } } @@ -3107,8 +3205,28 @@ public class BigDecimal extends Number implements Comparable { * @return this {@code BigDecimal} converted to a {@code float}. */ public float floatValue(){ - if (scale == 0 && intCompact != INFLATED) + if(intCompact != INFLATED) { + if (scale == 0) { return (float)intCompact; + } else { + /* + * If both intCompact and the scale can be exactly + * represented as float values, perform a single float + * multiply or divide to compute the (properly + * rounded) result. + */ + if (Math.abs(intCompact) < 1L<<22 ) { + // Don't have too guard against + // Math.abs(MIN_VALUE) because of outer check + // against INFLATED. + if (scale > 0 && scale < float10pow.length) { + return (float)intCompact / float10pow[scale]; + } else if (scale < 0 && scale > -float10pow.length) { + return (float)intCompact * float10pow[-scale]; + } + } + } + } // Somewhat inefficient, but guaranteed to work. return Float.parseFloat(this.toString()); } @@ -3130,12 +3248,52 @@ public class BigDecimal extends Number implements Comparable { * @return this {@code BigDecimal} converted to a {@code double}. */ public double doubleValue(){ - if (scale == 0 && intCompact != INFLATED) - return (double)intCompact; + if(intCompact != INFLATED) { + if (scale == 0) { + return (double)intCompact; + } else { + /* + * If both intCompact and the scale can be exactly + * represented as double values, perform a single + * double multiply or divide to compute the (properly + * rounded) result. + */ + if (Math.abs(intCompact) < 1L<<52 ) { + // Don't have too guard against + // Math.abs(MIN_VALUE) because of outer check + // against INFLATED. + if (scale > 0 && scale < double10pow.length) { + return (double)intCompact / double10pow[scale]; + } else if (scale < 0 && scale > -double10pow.length) { + return (double)intCompact * double10pow[-scale]; + } + } + } + } // Somewhat inefficient, but guaranteed to work. return Double.parseDouble(this.toString()); } + /** + * Powers of 10 which can be represented exactly in {@code + * double}. + */ + private static final double double10pow[] = { + 1.0e0, 1.0e1, 1.0e2, 1.0e3, 1.0e4, 1.0e5, + 1.0e6, 1.0e7, 1.0e8, 1.0e9, 1.0e10, 1.0e11, + 1.0e12, 1.0e13, 1.0e14, 1.0e15, 1.0e16, 1.0e17, + 1.0e18, 1.0e19, 1.0e20, 1.0e21, 1.0e22 + }; + + /** + * Powers of 10 which can be represented exactly in {@code + * float}. + */ + private static final float float10pow[] = { + 1.0e0f, 1.0e1f, 1.0e2f, 1.0e3f, 1.0e4f, 1.0e5f, + 1.0e6f, 1.0e7f, 1.0e8f, 1.0e9f, 1.0e10f + }; + /** * Returns the size of an ulp, a unit in the last place, of this * {@code BigDecimal}. An ulp of a nonzero {@code BigDecimal} @@ -3151,10 +3309,9 @@ public class BigDecimal extends Number implements Comparable { * @since 1.5 */ public BigDecimal ulp() { - return BigDecimal.valueOf(1, this.scale()); + return BigDecimal.valueOf(1, this.scale(), 1); } - // Private class to build a string representation for BigDecimal object. // "StringBuilderHelper" is constructed as a thread local variable so it is // thread safe. The StringBuilder field acts as a buffer to hold the temporary @@ -3268,6 +3425,15 @@ public class BigDecimal extends Number implements Comparable { return (intCompact != INFLATED) ? Long.toString(intCompact): intVal.toString(); + if (scale == 2 && + intCompact >= 0 && intCompact < Integer.MAX_VALUE) { + // currency fast path + int lowInt = (int)intCompact % 100; + int highInt = (int)intCompact / 100; + return (Integer.toString(highInt) + '.' + + StringBuilderHelper.DIGIT_TENS[lowInt] + + StringBuilderHelper.DIGIT_ONES[lowInt]) ; + } StringBuilderHelper sbHelper = threadLocalStringBuilderHelper.get(); char[] coeff; @@ -3377,7 +3543,7 @@ public class BigDecimal extends Number implements Comparable { tenpow[0] = '1'; for (int i = 1; i <= n; i++) tenpow[i] = '0'; - return new BigInteger(tenpow); + return new BigInteger(tenpow,1, tenpow.length); } /** @@ -3433,11 +3599,16 @@ public class BigDecimal extends Number implements Comparable { 1000000000000000000L // 18 / 10^18 }; - private static volatile BigInteger BIG_TEN_POWERS_TABLE[] = {BigInteger.ONE, - BigInteger.valueOf(10), BigInteger.valueOf(100), - BigInteger.valueOf(1000), BigInteger.valueOf(10000), - BigInteger.valueOf(100000), BigInteger.valueOf(1000000), - BigInteger.valueOf(10000000), BigInteger.valueOf(100000000), + private static volatile BigInteger BIG_TEN_POWERS_TABLE[] = { + BigInteger.ONE, + BigInteger.valueOf(10), + BigInteger.valueOf(100), + BigInteger.valueOf(1000), + BigInteger.valueOf(10000), + BigInteger.valueOf(100000), + BigInteger.valueOf(1000000), + BigInteger.valueOf(10000000), + BigInteger.valueOf(100000000), BigInteger.valueOf(1000000000), BigInteger.valueOf(10000000000L), BigInteger.valueOf(100000000000L), @@ -3502,7 +3673,7 @@ public class BigDecimal extends Number implements Comparable { */ private BigInteger bigMultiplyPowerTen(int n) { if (n <= 0) - return this.inflate(); + return this.inflated(); if (intCompact != INFLATED) return bigTenToThe(n).multiply(intCompact); @@ -3511,12 +3682,13 @@ public class BigDecimal extends Number implements Comparable { } /** - * Assign appropriate BigInteger to intVal field if intVal is + * Returns appropriate BigInteger from intVal field if intVal is * null, i.e. the compact representation is in use. */ - private BigInteger inflate() { - if (intVal == null) - intVal = BigInteger.valueOf(intCompact); + private BigInteger inflated() { + if (intVal == null) { + return BigInteger.valueOf(intCompact); + } return intVal; } @@ -3543,6 +3715,30 @@ public class BigDecimal extends Number implements Comparable { } } + private static class UnsafeHolder { + private static final sun.misc.Unsafe unsafe; + private static final long intCompactOffset; + private static final long intValOffset; + static { + try { + unsafe = sun.misc.Unsafe.getUnsafe(); + intCompactOffset = unsafe.objectFieldOffset + (BigDecimal.class.getDeclaredField("intCompact")); + intValOffset = unsafe.objectFieldOffset + (BigDecimal.class.getDeclaredField("intVal")); + } catch (Exception ex) { + throw new ExceptionInInitializerError(ex); + } + } + static void setIntCompactVolatile(BigDecimal bd, long val) { + unsafe.putLongVolatile(bd, intCompactOffset, val); + } + + static void setIntValVolatile(BigDecimal bd, BigInteger val) { + unsafe.putObjectVolatile(bd, intValOffset, val); + } + } + /** * Reconstitute the {@code BigDecimal} instance from a stream (that is, * deserialize it). @@ -3559,7 +3755,7 @@ public class BigDecimal extends Number implements Comparable { throw new java.io.StreamCorruptedException(message); // [all values of scale are now allowed] } - intCompact = compactValFor(intVal); + UnsafeHolder.setIntCompactVolatile(this, compactValFor(intVal)); } /** @@ -3570,13 +3766,12 @@ public class BigDecimal extends Number implements Comparable { private void writeObject(java.io.ObjectOutputStream s) throws java.io.IOException { // Must inflate to maintain compatible serial form. - this.inflate(); - - // Write proper fields + if (this.intVal == null) + UnsafeHolder.setIntValVolatile(this, BigInteger.valueOf(this.intCompact)); + // Could reset intVal back to null if it has to be set. s.defaultWriteObject(); } - /** * Returns the length of the absolute value of a {@code long}, in decimal * digits. @@ -3584,36 +3779,29 @@ public class BigDecimal extends Number implements Comparable { * @param x the {@code long} * @return the length of the unscaled value, in deciaml digits. */ - private static int longDigitLength(long x) { + static int longDigitLength(long x) { /* * As described in "Bit Twiddling Hacks" by Sean Anderson, * (http://graphics.stanford.edu/~seander/bithacks.html) - * integer log 10 of x is within 1 of - * (1233/4096)* (1 + integer log 2 of x). - * The fraction 1233/4096 approximates log10(2). So we first - * do a version of log2 (a variant of Long class with - * pre-checks and opposite directionality) and then scale and - * check against powers table. This is a little simpler in - * present context than the version in Hacker's Delight sec - * 11-4. Adding one to bit length allows comparing downward - * from the LONG_TEN_POWERS_TABLE that we need anyway. + * integer log 10 of x is within 1 of (1233/4096)* (1 + + * integer log 2 of x). The fraction 1233/4096 approximates + * log10(2). So we first do a version of log2 (a variant of + * Long class with pre-checks and opposite directionality) and + * then scale and check against powers table. This is a little + * simpler in present context than the version in Hacker's + * Delight sec 11-4. Adding one to bit length allows comparing + * downward from the LONG_TEN_POWERS_TABLE that we need + * anyway. */ - assert x != INFLATED; + assert x != BigDecimal.INFLATED; if (x < 0) x = -x; if (x < 10) // must screen for 0, might as well 10 return 1; - int n = 64; // not 63, to avoid needing to add 1 later - int y = (int)(x >>> 32); - if (y == 0) { n -= 32; y = (int)x; } - if (y >>> 16 == 0) { n -= 16; y <<= 16; } - if (y >>> 24 == 0) { n -= 8; y <<= 8; } - if (y >>> 28 == 0) { n -= 4; y <<= 4; } - if (y >>> 30 == 0) { n -= 2; y <<= 2; } - int r = (((y >>> 31) + n) * 1233) >>> 12; + int r = ((64 - Long.numberOfLeadingZeros(x) + 1) * 1233) >>> 12; long[] tab = LONG_TEN_POWERS_TABLE; // if r >= length, must have max possible digits for long - return (r >= tab.length || x < tab[r])? r : r+1; + return (r >= tab.length || x < tab[r]) ? r : r + 1; } /** @@ -3635,41 +3823,6 @@ public class BigDecimal extends Number implements Comparable { return b.compareMagnitude(bigTenToThe(r)) < 0? r : r+1; } - - /** - * Remove insignificant trailing zeros from this - * {@code BigDecimal} until the preferred scale is reached or no - * more zeros can be removed. If the preferred scale is less than - * Integer.MIN_VALUE, all the trailing zeros will be removed. - * - * {@code BigInteger} assistance could help, here? - * - *

WARNING: This method should only be called on new objects as - * it mutates the value fields. - * - * @return this {@code BigDecimal} with a scale possibly reduced - * to be closed to the preferred scale. - */ - private BigDecimal stripZerosToMatchScale(long preferredScale) { - this.inflate(); - BigInteger qr[]; // quotient-remainder pair - while ( intVal.compareMagnitude(BigInteger.TEN) >= 0 && - scale > preferredScale) { - if (intVal.testBit(0)) - break; // odd number cannot end in 0 - qr = intVal.divideAndRemainder(BigInteger.TEN); - if (qr[1].signum() != 0) - break; // non-0 remainder - intVal=qr[0]; - scale = checkScale((long)scale-1); // could Overflow - if (precision > 0) // adjust precision if known - precision--; - } - if (intVal != null) - intCompact = compactValFor(intVal); - return this; - } - /** * Check a scale for Underflow or Overflow. If this BigDecimal is * nonzero, throw an exception if the scale is outof range. If this @@ -3693,74 +3846,7 @@ public class BigDecimal extends Number implements Comparable { return asInt; } - /** - * Round an operand; used only if digits > 0. Does not change - * {@code this}; if rounding is needed a new {@code BigDecimal} - * is created and returned. - * - * @param mc the context to use. - * @throws ArithmeticException if the result is inexact but the - * rounding mode is {@code UNNECESSARY}. - */ - private BigDecimal roundOp(MathContext mc) { - BigDecimal rounded = doRound(this, mc); - return rounded; - } - - /** Round this BigDecimal according to the MathContext settings; - * used only if precision {@literal >} 0. - * - *

WARNING: This method should only be called on new objects as - * it mutates the value fields. - * - * @param mc the context to use. - * @throws ArithmeticException if the rounding mode is - * {@code RoundingMode.UNNECESSARY} and the - * {@code BigDecimal} operation would require rounding. - */ - private void roundThis(MathContext mc) { - BigDecimal rounded = doRound(this, mc); - if (rounded == this) // wasn't rounded - return; - this.intVal = rounded.intVal; - this.intCompact = rounded.intCompact; - this.scale = rounded.scale; - this.precision = rounded.precision; - } - - /** - * Returns a {@code BigDecimal} rounded according to the - * MathContext settings; used only if {@code mc.precision > 0}. - * Does not change {@code this}; if rounding is needed a new - * {@code BigDecimal} is created and returned. - * - * @param mc the context to use. - * @return a {@code BigDecimal} rounded according to the MathContext - * settings. May return this, if no rounding needed. - * @throws ArithmeticException if the rounding mode is - * {@code RoundingMode.UNNECESSARY} and the - * result is inexact. - */ - private static BigDecimal doRound(BigDecimal d, MathContext mc) { - int mcp = mc.precision; - int drop; - // This might (rarely) iterate to cover the 999=>1000 case - while ((drop = d.precision() - mcp) > 0) { - int newScale = d.checkScale((long)d.scale - drop); - int mode = mc.roundingMode.oldMode; - if (drop < LONG_TEN_POWERS_TABLE.length) - d = divideAndRound(d.intCompact, d.intVal, - LONG_TEN_POWERS_TABLE[drop], null, - newScale, mode, newScale); - else - d = divideAndRound(d.intCompact, d.intVal, - INFLATED, bigTenToThe(drop), - newScale, mode, newScale); - } - return d; - } - - /** + /** * Returns the compact value for given {@code BigInteger}, or * INFLATED if too big. Relies on internal representation of * {@code BigInteger}. @@ -3852,4 +3938,1290 @@ public class BigDecimal extends Number implements Comparable { } return this; } + + /* the same as checkScale where value!=0 */ + private static int checkScaleNonZero(long val) { + int asInt = (int)val; + if (asInt != val) { + throw new ArithmeticException(asInt>0 ? "Underflow":"Overflow"); + } + return asInt; + } + + private static int checkScale(long intCompact, long val) { + int asInt = (int)val; + if (asInt != val) { + asInt = val>Integer.MAX_VALUE ? Integer.MAX_VALUE : Integer.MIN_VALUE; + if (intCompact != 0) + throw new ArithmeticException(asInt>0 ? "Underflow":"Overflow"); + } + return asInt; + } + + private static int checkScale(BigInteger intVal, long val) { + int asInt = (int)val; + if (asInt != val) { + asInt = val>Integer.MAX_VALUE ? Integer.MAX_VALUE : Integer.MIN_VALUE; + if (intVal.signum() != 0) + throw new ArithmeticException(asInt>0 ? "Underflow":"Overflow"); + } + return asInt; + } + + /** + * Returns a {@code BigDecimal} rounded according to the MathContext + * settings; + * If rounding is needed a new {@code BigDecimal} is created and returned. + * + * @param val the value to be rounded + * @param mc the context to use. + * @return a {@code BigDecimal} rounded according to the MathContext + * settings. May return {@code value}, if no rounding needed. + * @throws ArithmeticException if the rounding mode is + * {@code RoundingMode.UNNECESSARY} and the + * result is inexact. + */ + private static BigDecimal doRound(BigDecimal val, MathContext mc) { + int mcp = mc.precision; + boolean wasDivided = false; + if (mcp > 0) { + BigInteger intVal = val.intVal; + long compactVal = val.intCompact; + int scale = val.scale; + int prec = val.precision(); + int mode = mc.roundingMode.oldMode; + int drop; + if (compactVal == INFLATED) { + drop = prec - mcp; + while (drop > 0) { + scale = checkScaleNonZero((long) scale - drop); + intVal = divideAndRoundByTenPow(intVal, drop, mode); + wasDivided = true; + compactVal = compactValFor(intVal); + if (compactVal != INFLATED) { + prec = longDigitLength(compactVal); + break; + } + prec = bigDigitLength(intVal); + drop = prec - mcp; + } + } + if (compactVal != INFLATED) { + drop = prec - mcp; // drop can't be more than 18 + while (drop > 0) { + scale = checkScaleNonZero((long) scale - drop); + compactVal = divideAndRound(compactVal, LONG_TEN_POWERS_TABLE[drop], mc.roundingMode.oldMode); + wasDivided = true; + prec = longDigitLength(compactVal); + drop = prec - mcp; + intVal = null; + } + } + return wasDivided ? new BigDecimal(intVal,compactVal,scale,prec) : val; + } + return val; + } + + /* + * Returns a {@code BigDecimal} created from {@code long} value with + * given scale rounded according to the MathContext settings + */ + private static BigDecimal doRound(long compactVal, int scale, MathContext mc) { + int mcp = mc.precision; + if (mcp > 0 && mcp < 19) { + int prec = longDigitLength(compactVal); + int drop = prec - mcp; // drop can't be more than 18 + while (drop > 0) { + scale = checkScaleNonZero((long) scale - drop); + compactVal = divideAndRound(compactVal, LONG_TEN_POWERS_TABLE[drop], mc.roundingMode.oldMode); + prec = longDigitLength(compactVal); + drop = prec - mcp; + } + return valueOf(compactVal, scale, prec); + } + return valueOf(compactVal, scale); + } + + /* + * Returns a {@code BigDecimal} created from {@code BigInteger} value with + * given scale rounded according to the MathContext settings + */ + private static BigDecimal doRound(BigInteger intVal, int scale, MathContext mc) { + int mcp = mc.precision; + int prec = 0; + if (mcp > 0) { + long compactVal = compactValFor(intVal); + int mode = mc.roundingMode.oldMode; + int drop; + if (compactVal == INFLATED) { + prec = bigDigitLength(intVal); + drop = prec - mcp; + while (drop > 0) { + scale = checkScaleNonZero((long) scale - drop); + intVal = divideAndRoundByTenPow(intVal, drop, mode); + compactVal = compactValFor(intVal); + if (compactVal != INFLATED) { + break; + } + prec = bigDigitLength(intVal); + drop = prec - mcp; + } + } + if (compactVal != INFLATED) { + prec = longDigitLength(compactVal); + drop = prec - mcp; // drop can't be more than 18 + while (drop > 0) { + scale = checkScaleNonZero((long) scale - drop); + compactVal = divideAndRound(compactVal, LONG_TEN_POWERS_TABLE[drop], mc.roundingMode.oldMode); + prec = longDigitLength(compactVal); + drop = prec - mcp; + } + return valueOf(compactVal,scale,prec); + } + } + return new BigDecimal(intVal,INFLATED,scale,prec); + } + + /* + * Divides {@code BigInteger} value by ten power. + */ + private static BigInteger divideAndRoundByTenPow(BigInteger intVal, int tenPow, int roundingMode) { + if (tenPow < LONG_TEN_POWERS_TABLE.length) + intVal = divideAndRound(intVal, LONG_TEN_POWERS_TABLE[tenPow], roundingMode); + else + intVal = divideAndRound(intVal, bigTenToThe(tenPow), roundingMode); + return intVal; + } + + /** + * Internally used for division operation for division {@code long} by + * {@code long}. + * The returned {@code BigDecimal} object is the quotient whose scale is set + * to the passed in scale. If the remainder is not zero, it will be rounded + * based on the passed in roundingMode. Also, if the remainder is zero and + * the last parameter, i.e. preferredScale is NOT equal to scale, the + * trailing zeros of the result is stripped to match the preferredScale. + */ + private static BigDecimal divideAndRound(long ldividend, long ldivisor, int scale, int roundingMode, + int preferredScale) { + + int qsign; // quotient sign + long q = ldividend / ldivisor; // store quotient in long + if (roundingMode == ROUND_DOWN && scale == preferredScale) + return valueOf(q, scale); + long r = ldividend % ldivisor; // store remainder in long + qsign = ((ldividend < 0) == (ldivisor < 0)) ? 1 : -1; + if (r != 0) { + boolean increment = needIncrement(ldivisor, roundingMode, qsign, q, r); + return valueOf((increment ? q + qsign : q), scale); + } else { + if (preferredScale != scale) + return createAndStripZerosToMatchScale(q, scale, preferredScale); + else + return valueOf(q, scale); + } + } + + /** + * Divides {@code long} by {@code long} and do rounding based on the + * passed in roundingMode. + */ + private static long divideAndRound(long ldividend, long ldivisor, int roundingMode) { + int qsign; // quotient sign + long q = ldividend / ldivisor; // store quotient in long + if (roundingMode == ROUND_DOWN) + return q; + long r = ldividend % ldivisor; // store remainder in long + qsign = ((ldividend < 0) == (ldivisor < 0)) ? 1 : -1; + if (r != 0) { + boolean increment = needIncrement(ldivisor, roundingMode, qsign, q, r); + return increment ? q + qsign : q; + } else { + return q; + } + } + + /** + * Shared logic of need increment computation. + */ + private static boolean commonNeedIncrement(int roundingMode, int qsign, + int cmpFracHalf, boolean oddQuot) { + switch(roundingMode) { + case ROUND_UNNECESSARY: + throw new ArithmeticException("Rounding necessary"); + + case ROUND_UP: // Away from zero + return true; + + case ROUND_DOWN: // Towards zero + return false; + + case ROUND_CEILING: // Towards +infinity + return qsign > 0; + + case ROUND_FLOOR: // Towards -infinity + return qsign < 0; + + default: // Some kind of half-way rounding + assert roundingMode >= ROUND_HALF_UP && + roundingMode <= ROUND_HALF_EVEN: "Unexpected rounding mode" + RoundingMode.valueOf(roundingMode); + + if (cmpFracHalf < 0 ) // We're closer to higher digit + return false; + else if (cmpFracHalf > 0 ) // We're closer to lower digit + return true; + else { // half-way + assert cmpFracHalf == 0; + + switch(roundingMode) { + case ROUND_HALF_DOWN: + return false; + + case ROUND_HALF_UP: + return true; + + case ROUND_HALF_EVEN: + return oddQuot; + + default: + throw new AssertionError("Unexpected rounding mode" + roundingMode); + } + } + } + } + + /** + * Tests if quotient has to be incremented according the roundingMode + */ + private static boolean needIncrement(long ldivisor, int roundingMode, + int qsign, long q, long r) { + assert r != 0L; + + int cmpFracHalf; + if (r <= HALF_LONG_MIN_VALUE || r > HALF_LONG_MAX_VALUE) { + cmpFracHalf = 1; // 2 * r can't fit into long + } else { + cmpFracHalf = longCompareMagnitude(2 * r, ldivisor); + } + + return commonNeedIncrement(roundingMode, qsign, cmpFracHalf, (q & 1L) != 0L); + } + + /** + * Divides {@code BigInteger} value by {@code long} value and + * do rounding based on the passed in roundingMode. + */ + private static BigInteger divideAndRound(BigInteger bdividend, long ldivisor, int roundingMode) { + boolean isRemainderZero; // record remainder is zero or not + int qsign; // quotient sign + long r = 0; // store quotient & remainder in long + MutableBigInteger mq = null; // store quotient + // Descend into mutables for faster remainder checks + MutableBigInteger mdividend = new MutableBigInteger(bdividend.mag); + mq = new MutableBigInteger(); + r = mdividend.divide(ldivisor, mq); + isRemainderZero = (r == 0); + qsign = (ldivisor < 0) ? -bdividend.signum : bdividend.signum; + if (!isRemainderZero) { + if(needIncrement(ldivisor, roundingMode, qsign, mq, r)) { + mq.add(MutableBigInteger.ONE); + } + } + return mq.toBigInteger(qsign); + } + + /** + * Internally used for division operation for division {@code BigInteger} + * by {@code long}. + * The returned {@code BigDecimal} object is the quotient whose scale is set + * to the passed in scale. If the remainder is not zero, it will be rounded + * based on the passed in roundingMode. Also, if the remainder is zero and + * the last parameter, i.e. preferredScale is NOT equal to scale, the + * trailing zeros of the result is stripped to match the preferredScale. + */ + private static BigDecimal divideAndRound(BigInteger bdividend, + long ldivisor, int scale, int roundingMode, int preferredScale) { + boolean isRemainderZero; // record remainder is zero or not + int qsign; // quotient sign + long r = 0; // store quotient & remainder in long + MutableBigInteger mq = null; // store quotient + // Descend into mutables for faster remainder checks + MutableBigInteger mdividend = new MutableBigInteger(bdividend.mag); + mq = new MutableBigInteger(); + r = mdividend.divide(ldivisor, mq); + isRemainderZero = (r == 0); + qsign = (ldivisor < 0) ? -bdividend.signum : bdividend.signum; + if (!isRemainderZero) { + if(needIncrement(ldivisor, roundingMode, qsign, mq, r)) { + mq.add(MutableBigInteger.ONE); + } + return mq.toBigDecimal(qsign, scale); + } else { + if (preferredScale != scale) { + long compactVal = mq.toCompactValue(qsign); + if(compactVal!=INFLATED) { + return createAndStripZerosToMatchScale(compactVal, scale, preferredScale); + } + BigInteger intVal = mq.toBigInteger(qsign); + return createAndStripZerosToMatchScale(intVal,scale, preferredScale); + } else { + return mq.toBigDecimal(qsign, scale); + } + } + } + + /** + * Tests if quotient has to be incremented according the roundingMode + */ + private static boolean needIncrement(long ldivisor, int roundingMode, + int qsign, MutableBigInteger mq, long r) { + assert r != 0L; + + int cmpFracHalf; + if (r <= HALF_LONG_MIN_VALUE || r > HALF_LONG_MAX_VALUE) { + cmpFracHalf = 1; // 2 * r can't fit into long + } else { + cmpFracHalf = longCompareMagnitude(2 * r, ldivisor); + } + + return commonNeedIncrement(roundingMode, qsign, cmpFracHalf, mq.isOdd()); + } + + /** + * Divides {@code BigInteger} value by {@code BigInteger} value and + * do rounding based on the passed in roundingMode. + */ + private static BigInteger divideAndRound(BigInteger bdividend, BigInteger bdivisor, int roundingMode) { + boolean isRemainderZero; // record remainder is zero or not + int qsign; // quotient sign + // Descend into mutables for faster remainder checks + MutableBigInteger mdividend = new MutableBigInteger(bdividend.mag); + MutableBigInteger mq = new MutableBigInteger(); + MutableBigInteger mdivisor = new MutableBigInteger(bdivisor.mag); + MutableBigInteger mr = mdividend.divide(mdivisor, mq); + isRemainderZero = mr.isZero(); + qsign = (bdividend.signum != bdivisor.signum) ? -1 : 1; + if (!isRemainderZero) { + if (needIncrement(mdivisor, roundingMode, qsign, mq, mr)) { + mq.add(MutableBigInteger.ONE); + } + } + return mq.toBigInteger(qsign); + } + + /** + * Internally used for division operation for division {@code BigInteger} + * by {@code BigInteger}. + * The returned {@code BigDecimal} object is the quotient whose scale is set + * to the passed in scale. If the remainder is not zero, it will be rounded + * based on the passed in roundingMode. Also, if the remainder is zero and + * the last parameter, i.e. preferredScale is NOT equal to scale, the + * trailing zeros of the result is stripped to match the preferredScale. + */ + private static BigDecimal divideAndRound(BigInteger bdividend, BigInteger bdivisor, int scale, int roundingMode, + int preferredScale) { + boolean isRemainderZero; // record remainder is zero or not + int qsign; // quotient sign + // Descend into mutables for faster remainder checks + MutableBigInteger mdividend = new MutableBigInteger(bdividend.mag); + MutableBigInteger mq = new MutableBigInteger(); + MutableBigInteger mdivisor = new MutableBigInteger(bdivisor.mag); + MutableBigInteger mr = mdividend.divide(mdivisor, mq); + isRemainderZero = mr.isZero(); + qsign = (bdividend.signum != bdivisor.signum) ? -1 : 1; + if (!isRemainderZero) { + if (needIncrement(mdivisor, roundingMode, qsign, mq, mr)) { + mq.add(MutableBigInteger.ONE); + } + return mq.toBigDecimal(qsign, scale); + } else { + if (preferredScale != scale) { + long compactVal = mq.toCompactValue(qsign); + if (compactVal != INFLATED) { + return createAndStripZerosToMatchScale(compactVal, scale, preferredScale); + } + BigInteger intVal = mq.toBigInteger(qsign); + return createAndStripZerosToMatchScale(intVal, scale, preferredScale); + } else { + return mq.toBigDecimal(qsign, scale); + } + } + } + + /** + * Tests if quotient has to be incremented according the roundingMode + */ + private static boolean needIncrement(MutableBigInteger mdivisor, int roundingMode, + int qsign, MutableBigInteger mq, MutableBigInteger mr) { + assert !mr.isZero(); + int cmpFracHalf = mr.compareHalf(mdivisor); + return commonNeedIncrement(roundingMode, qsign, cmpFracHalf, mq.isOdd()); + } + + /** + * Remove insignificant trailing zeros from this + * {@code BigInteger} value until the preferred scale is reached or no + * more zeros can be removed. If the preferred scale is less than + * Integer.MIN_VALUE, all the trailing zeros will be removed. + * + * @return new {@code BigDecimal} with a scale possibly reduced + * to be closed to the preferred scale. + */ + private static BigDecimal createAndStripZerosToMatchScale(BigInteger intVal, int scale, long preferredScale) { + BigInteger qr[]; // quotient-remainder pair + while (intVal.compareMagnitude(BigInteger.TEN) >= 0 + && scale > preferredScale) { + if (intVal.testBit(0)) + break; // odd number cannot end in 0 + qr = intVal.divideAndRemainder(BigInteger.TEN); + if (qr[1].signum() != 0) + break; // non-0 remainder + intVal = qr[0]; + scale = checkScale(intVal,(long) scale - 1); // could Overflow + } + return valueOf(intVal, scale, 0); + } + + /** + * Remove insignificant trailing zeros from this + * {@code long} value until the preferred scale is reached or no + * more zeros can be removed. If the preferred scale is less than + * Integer.MIN_VALUE, all the trailing zeros will be removed. + * + * @return new {@code BigDecimal} with a scale possibly reduced + * to be closed to the preferred scale. + */ + private static BigDecimal createAndStripZerosToMatchScale(long compactVal, int scale, long preferredScale) { + while (Math.abs(compactVal) >= 10L && scale > preferredScale) { + if ((compactVal & 1L) != 0L) + break; // odd number cannot end in 0 + long r = compactVal % 10L; + if (r != 0L) + break; // non-0 remainder + compactVal /= 10; + scale = checkScale(compactVal, (long) scale - 1); // could Overflow + } + return valueOf(compactVal, scale); + } + + private static BigDecimal stripZerosToMatchScale(BigInteger intVal, long intCompact, int scale, int preferredScale) { + if(intCompact!=INFLATED) { + return createAndStripZerosToMatchScale(intCompact, scale, preferredScale); + } else { + return createAndStripZerosToMatchScale(intVal==null ? INFLATED_BIGINT : intVal, + scale, preferredScale); + } + } + + /* + * returns INFLATED if oveflow + */ + private static long add(long xs, long ys){ + long sum = xs + ys; + // See "Hacker's Delight" section 2-12 for explanation of + // the overflow test. + if ( (((sum ^ xs) & (sum ^ ys))) >= 0L) { // not overflowed + return sum; + } + return INFLATED; + } + + private static BigDecimal add(long xs, long ys, int scale){ + long sum = add(xs, ys); + if (sum!=INFLATED) + return BigDecimal.valueOf(sum, scale); + return new BigDecimal(BigInteger.valueOf(xs).add(ys), scale); + } + + private static BigDecimal add(final long xs, int scale1, final long ys, int scale2) { + long sdiff = (long) scale1 - scale2; + if (sdiff == 0) { + return add(xs, ys, scale1); + } else if (sdiff < 0) { + int raise = checkScale(xs,-sdiff); + long scaledX = longMultiplyPowerTen(xs, raise); + if (scaledX != INFLATED) { + return add(scaledX, ys, scale2); + } else { + BigInteger bigsum = bigMultiplyPowerTen(xs,raise).add(ys); + return ((xs^ys)>=0) ? // same sign test + new BigDecimal(bigsum, INFLATED, scale2, 0) + : valueOf(bigsum, scale2, 0); + } + } else { + int raise = checkScale(ys,sdiff); + long scaledY = longMultiplyPowerTen(ys, raise); + if (scaledY != INFLATED) { + return add(xs, scaledY, scale1); + } else { + BigInteger bigsum = bigMultiplyPowerTen(ys,raise).add(xs); + return ((xs^ys)>=0) ? + new BigDecimal(bigsum, INFLATED, scale1, 0) + : valueOf(bigsum, scale1, 0); + } + } + } + + private static BigDecimal add(final long xs, int scale1, BigInteger snd, int scale2) { + int rscale = scale1; + long sdiff = (long)rscale - scale2; + boolean sameSigns = (Long.signum(xs) == snd.signum); + BigInteger sum; + if (sdiff < 0) { + int raise = checkScale(xs,-sdiff); + rscale = scale2; + long scaledX = longMultiplyPowerTen(xs, raise); + if (scaledX == INFLATED) { + sum = snd.add(bigMultiplyPowerTen(xs,raise)); + } else { + sum = snd.add(scaledX); + } + } else { //if (sdiff > 0) { + int raise = checkScale(snd,sdiff); + snd = bigMultiplyPowerTen(snd,raise); + sum = snd.add(xs); + } + return (sameSigns) ? + new BigDecimal(sum, INFLATED, rscale, 0) : + valueOf(sum, rscale, 0); + } + + private static BigDecimal add(BigInteger fst, int scale1, BigInteger snd, int scale2) { + int rscale = scale1; + long sdiff = (long)rscale - scale2; + if (sdiff != 0) { + if (sdiff < 0) { + int raise = checkScale(fst,-sdiff); + rscale = scale2; + fst = bigMultiplyPowerTen(fst,raise); + } else { + int raise = checkScale(snd,sdiff); + snd = bigMultiplyPowerTen(snd,raise); + } + } + BigInteger sum = fst.add(snd); + return (fst.signum == snd.signum) ? + new BigDecimal(sum, INFLATED, rscale, 0) : + valueOf(sum, rscale, 0); + } + + private static BigInteger bigMultiplyPowerTen(long value, int n) { + if (n <= 0) + return BigInteger.valueOf(value); + return bigTenToThe(n).multiply(value); + } + + private static BigInteger bigMultiplyPowerTen(BigInteger value, int n) { + if (n <= 0) + return value; + if(n=0 + long scaledX = (xraise==0) ? xs : + longMultiplyPowerTen(xs, xraise); // can't overflow here! + BigDecimal quotient; + + int cmp = longCompareMagnitude(scaledX, ys); + if(cmp > 0) { // satisfy constraint (b) + yscale -= 1; // [that is, divisor *= 10] + int scl = checkScaleNonZero(preferredScale + yscale - xscale + mcp); + if (checkScaleNonZero((long) mcp + yscale) > xscale) { + // assert newScale >= xscale + int raise = checkScaleNonZero((long) mcp + yscale - xscale); + long scaledXs; + if ((scaledXs = longMultiplyPowerTen(xs, raise)) == INFLATED) { + quotient = null; + if((mcp-1) >=0 && (mcp-1)= yscale + if (newScale == yscale) { // easy case + quotient = divideAndRound(xs, ys, scl, roundingMode,checkScaleNonZero(preferredScale)); + } else { + int raise = checkScaleNonZero((long) newScale - yscale); + long scaledYs; + if ((scaledYs = longMultiplyPowerTen(ys, raise)) == INFLATED) { + BigInteger rb = bigMultiplyPowerTen(ys,raise); + quotient = divideAndRound(BigInteger.valueOf(xs), + rb, scl, roundingMode,checkScaleNonZero(preferredScale)); + } else { + quotient = divideAndRound(xs, scaledYs, scl, roundingMode,checkScaleNonZero(preferredScale)); + } + } + } + } else { + // abs(scaledX) <= abs(ys) + // result is "scaledX * 10^msp / ys" + int scl = checkScaleNonZero(preferredScale + yscale - xscale + mcp); + if(cmp==0) { + // abs(scaleX)== abs(ys) => result will be scaled 10^mcp + correct sign + quotient = roundedTenPower(((scaledX < 0) == (ys < 0)) ? 1 : -1, mcp, scl, checkScaleNonZero(preferredScale)); + } else { + // abs(scaledX) < abs(ys) + long scaledXs; + if ((scaledXs = longMultiplyPowerTen(scaledX, mcp)) == INFLATED) { + quotient = null; + if(mcp 0) {// satisfy constraint (b) + yscale -= 1; // [that is, divisor *= 10] + } + int roundingMode = mc.roundingMode.oldMode; + // In order to find out whether the divide generates the exact result, + // we avoid calling the above divide method. 'quotient' holds the + // return BigDecimal object whose scale will be set to 'scl'. + int scl = checkScaleNonZero(preferredScale + yscale - xscale + mcp); + BigDecimal quotient; + if (checkScaleNonZero((long) mcp + yscale) > xscale) { + int raise = checkScaleNonZero((long) mcp + yscale - xscale); + long scaledXs; + if ((scaledXs = longMultiplyPowerTen(xs, raise)) == INFLATED) { + BigInteger rb = bigMultiplyPowerTen(xs,raise); + quotient = divideAndRound(rb, ys, scl, roundingMode, checkScaleNonZero(preferredScale)); + } else { + quotient = divideAndRound(scaledXs, ys, scl, roundingMode, checkScaleNonZero(preferredScale)); + } + } else { + int newScale = checkScaleNonZero((long) xscale - mcp); + // assert newScale >= yscale + if (newScale == yscale) { // easy case + quotient = divideAndRound(xs, ys, scl, roundingMode,checkScaleNonZero(preferredScale)); + } else { + int raise = checkScaleNonZero((long) newScale - yscale); + long scaledYs; + if ((scaledYs = longMultiplyPowerTen(ys, raise)) == INFLATED) { + BigInteger rb = bigMultiplyPowerTen(ys,raise); + quotient = divideAndRound(BigInteger.valueOf(xs), + rb, scl, roundingMode,checkScaleNonZero(preferredScale)); + } else { + quotient = divideAndRound(xs, scaledYs, scl, roundingMode,checkScaleNonZero(preferredScale)); + } + } + } + // doRound, here, only affects 1000000000 case. + return doRound(quotient,mc); + } + + /** + * Returns a {@code BigDecimal} whose value is {@code (xs / + * ys)}, with rounding according to the context settings. + */ + private static BigDecimal divide(BigInteger xs, int xscale, long ys, int yscale, long preferredScale, MathContext mc) { + // Normalize dividend & divisor so that both fall into [0.1, 0.999...] + if ((-compareMagnitudeNormalized(ys, yscale, xs, xscale)) > 0) {// satisfy constraint (b) + yscale -= 1; // [that is, divisor *= 10] + } + int mcp = mc.precision; + int roundingMode = mc.roundingMode.oldMode; + + // In order to find out whether the divide generates the exact result, + // we avoid calling the above divide method. 'quotient' holds the + // return BigDecimal object whose scale will be set to 'scl'. + BigDecimal quotient; + int scl = checkScaleNonZero(preferredScale + yscale - xscale + mcp); + if (checkScaleNonZero((long) mcp + yscale) > xscale) { + int raise = checkScaleNonZero((long) mcp + yscale - xscale); + BigInteger rb = bigMultiplyPowerTen(xs,raise); + quotient = divideAndRound(rb, ys, scl, roundingMode, checkScaleNonZero(preferredScale)); + } else { + int newScale = checkScaleNonZero((long) xscale - mcp); + // assert newScale >= yscale + if (newScale == yscale) { // easy case + quotient = divideAndRound(xs, ys, scl, roundingMode,checkScaleNonZero(preferredScale)); + } else { + int raise = checkScaleNonZero((long) newScale - yscale); + long scaledYs; + if ((scaledYs = longMultiplyPowerTen(ys, raise)) == INFLATED) { + BigInteger rb = bigMultiplyPowerTen(ys,raise); + quotient = divideAndRound(xs, rb, scl, roundingMode,checkScaleNonZero(preferredScale)); + } else { + quotient = divideAndRound(xs, scaledYs, scl, roundingMode,checkScaleNonZero(preferredScale)); + } + } + } + // doRound, here, only affects 1000000000 case. + return doRound(quotient, mc); + } + + /** + * Returns a {@code BigDecimal} whose value is {@code (xs / + * ys)}, with rounding according to the context settings. + */ + private static BigDecimal divide(long xs, int xscale, BigInteger ys, int yscale, long preferredScale, MathContext mc) { + // Normalize dividend & divisor so that both fall into [0.1, 0.999...] + if (compareMagnitudeNormalized(xs, xscale, ys, yscale) > 0) {// satisfy constraint (b) + yscale -= 1; // [that is, divisor *= 10] + } + int mcp = mc.precision; + int roundingMode = mc.roundingMode.oldMode; + + // In order to find out whether the divide generates the exact result, + // we avoid calling the above divide method. 'quotient' holds the + // return BigDecimal object whose scale will be set to 'scl'. + BigDecimal quotient; + int scl = checkScaleNonZero(preferredScale + yscale - xscale + mcp); + if (checkScaleNonZero((long) mcp + yscale) > xscale) { + int raise = checkScaleNonZero((long) mcp + yscale - xscale); + BigInteger rb = bigMultiplyPowerTen(xs,raise); + quotient = divideAndRound(rb, ys, scl, roundingMode, checkScaleNonZero(preferredScale)); + } else { + int newScale = checkScaleNonZero((long) xscale - mcp); + int raise = checkScaleNonZero((long) newScale - yscale); + BigInteger rb = bigMultiplyPowerTen(ys,raise); + quotient = divideAndRound(BigInteger.valueOf(xs), rb, scl, roundingMode,checkScaleNonZero(preferredScale)); + } + // doRound, here, only affects 1000000000 case. + return doRound(quotient, mc); + } + + /** + * Returns a {@code BigDecimal} whose value is {@code (xs / + * ys)}, with rounding according to the context settings. + */ + private static BigDecimal divide(BigInteger xs, int xscale, BigInteger ys, int yscale, long preferredScale, MathContext mc) { + // Normalize dividend & divisor so that both fall into [0.1, 0.999...] + if (compareMagnitudeNormalized(xs, xscale, ys, yscale) > 0) {// satisfy constraint (b) + yscale -= 1; // [that is, divisor *= 10] + } + int mcp = mc.precision; + int roundingMode = mc.roundingMode.oldMode; + + // In order to find out whether the divide generates the exact result, + // we avoid calling the above divide method. 'quotient' holds the + // return BigDecimal object whose scale will be set to 'scl'. + BigDecimal quotient; + int scl = checkScaleNonZero(preferredScale + yscale - xscale + mcp); + if (checkScaleNonZero((long) mcp + yscale) > xscale) { + int raise = checkScaleNonZero((long) mcp + yscale - xscale); + BigInteger rb = bigMultiplyPowerTen(xs,raise); + quotient = divideAndRound(rb, ys, scl, roundingMode, checkScaleNonZero(preferredScale)); + } else { + int newScale = checkScaleNonZero((long) xscale - mcp); + int raise = checkScaleNonZero((long) newScale - yscale); + BigInteger rb = bigMultiplyPowerTen(ys,raise); + quotient = divideAndRound(xs, rb, scl, roundingMode,checkScaleNonZero(preferredScale)); + } + // doRound, here, only affects 1000000000 case. + return doRound(quotient, mc); + } + + /* + * performs divideAndRound for (dividend0*dividend1, divisor) + * returns null if quotient can't fit into long value; + */ + private static BigDecimal multiplyDivideAndRound(long dividend0, long dividend1, long divisor, int scale, int roundingMode, + int preferredScale) { + int qsign = Long.signum(dividend0)*Long.signum(dividend1)*Long.signum(divisor); + dividend0 = Math.abs(dividend0); + dividend1 = Math.abs(dividend1); + divisor = Math.abs(divisor); + // multiply dividend0 * dividend1 + long d0_hi = dividend0 >>> 32; + long d0_lo = dividend0 & LONG_MASK; + long d1_hi = dividend1 >>> 32; + long d1_lo = dividend1 & LONG_MASK; + long product = d0_lo * d1_lo; + long d0 = product & LONG_MASK; + long d1 = product >>> 32; + product = d0_hi * d1_lo + d1; + d1 = product & LONG_MASK; + long d2 = product >>> 32; + product = d0_lo * d1_hi + d1; + d1 = product & LONG_MASK; + d2 += product >>> 32; + long d3 = d2>>>32; + d2 &= LONG_MASK; + product = d0_hi*d1_hi + d2; + d2 = product & LONG_MASK; + d3 = ((product>>>32) + d3) & LONG_MASK; + final long dividendHi = make64(d3,d2); + final long dividendLo = make64(d1,d0); + // divide + return divideAndRound128(dividendHi, dividendLo, divisor, qsign, scale, roundingMode, preferredScale); + } + + private static final long DIV_NUM_BASE = (1L<<32); // Number base (32 bits). + + /* + * divideAndRound 128-bit value by long divisor. + * returns null if quotient can't fit into long value; + * Specialized version of Knuth's division + */ + private static BigDecimal divideAndRound128(final long dividendHi, final long dividendLo, long divisor, int sign, + int scale, int roundingMode, int preferredScale) { + if (dividendHi >= divisor) { + return null; + } + final int shift = Long.numberOfLeadingZeros(divisor); + divisor <<= shift; + + final long v1 = divisor >>> 32; + final long v0 = divisor & LONG_MASK; + + long q1, q0; + long r_tmp; + + long tmp = dividendLo << shift; + long u1 = tmp >>> 32; + long u0 = tmp & LONG_MASK; + + tmp = (dividendHi << shift) | (dividendLo >>> 64 - shift); + long u2 = tmp & LONG_MASK; + tmp = divWord(tmp,v1); + q1 = tmp & LONG_MASK; + r_tmp = tmp >>> 32; + while(q1 >= DIV_NUM_BASE || unsignedLongCompare(q1*v0, make64(r_tmp, u1))) { + q1--; + r_tmp += v1; + if (r_tmp >= DIV_NUM_BASE) + break; + } + tmp = mulsub(u2,u1,v1,v0,q1); + u1 = tmp & LONG_MASK; + tmp = divWord(tmp,v1); + q0 = tmp & LONG_MASK; + r_tmp = tmp >>> 32; + while(q0 >= DIV_NUM_BASE || unsignedLongCompare(q0*v0,make64(r_tmp,u0))) { + q0--; + r_tmp += v1; + if (r_tmp >= DIV_NUM_BASE) + break; + } + if((int)q1 < 0) { + // result (which is positive and unsigned here) + // can't fit into long due to sign bit is used for value + MutableBigInteger mq = new MutableBigInteger(new int[]{(int)q1, (int)q0}); + if (roundingMode == ROUND_DOWN && scale == preferredScale) { + return mq.toBigDecimal(sign, scale); + } + long r = mulsub(u1, u0, v1, v0, q0) >>> shift; + if (r != 0) { + if(needIncrement(divisor >>> shift, roundingMode, sign, mq, r)){ + mq.add(MutableBigInteger.ONE); + } + return mq.toBigDecimal(sign, scale); + } else { + if (preferredScale != scale) { + BigInteger intVal = mq.toBigInteger(sign); + return createAndStripZerosToMatchScale(intVal,scale, preferredScale); + } else { + return mq.toBigDecimal(sign, scale); + } + } + } + long q = make64(q1,q0); + q*=sign; + if (roundingMode == ROUND_DOWN && scale == preferredScale) + return valueOf(q, scale); + long r = mulsub(u1, u0, v1, v0, q0) >>> shift; + if (r != 0) { + boolean increment = needIncrement(divisor >>> shift, roundingMode, sign, q, r); + return valueOf((increment ? q + sign : q), scale); + } else { + if (preferredScale != scale) { + return createAndStripZerosToMatchScale(q, scale, preferredScale); + } else { + return valueOf(q, scale); + } + } + } + + /* + * calculate divideAndRound for ldividend*10^raise / divisor + * when abs(dividend)==abs(divisor); + */ + private static BigDecimal roundedTenPower(int qsign, int raise, int scale, int preferredScale) { + if (scale > preferredScale) { + int diff = scale - preferredScale; + if(diff < raise) { + return scaledTenPow(raise - diff, qsign, preferredScale); + } else { + return valueOf(qsign,scale-raise); + } + } else { + return scaledTenPow(raise, qsign, scale); + } + } + + static BigDecimal scaledTenPow(int n, int sign, int scale) { + if (n < LONG_TEN_POWERS_TABLE.length) + return valueOf(sign*LONG_TEN_POWERS_TABLE[n],scale); + else { + BigInteger unscaledVal = bigTenToThe(n); + if(sign==-1) { + unscaledVal = unscaledVal.negate(); + } + return new BigDecimal(unscaledVal, INFLATED, scale, n+1); + } + } + + private static long divWord(long n, long dLong) { + long r; + long q; + if (dLong == 1) { + q = (int)n; + return (q & LONG_MASK); + } + // Approximate the quotient and remainder + q = (n >>> 1) / (dLong >>> 1); + r = n - q*dLong; + + // Correct the approximation + while (r < 0) { + r += dLong; + q--; + } + while (r >= dLong) { + r -= dLong; + q++; + } + // n - q*dlong == r && 0 <= r >>32) - q0*v1,tmp & LONG_MASK); + } + + private static boolean unsignedLongCompare(long one, long two) { + return (one+Long.MIN_VALUE) > (two+Long.MIN_VALUE); + } + + private static boolean unsignedLongCompareEq(long one, long two) { + return (one+Long.MIN_VALUE) >= (two+Long.MIN_VALUE); + } + + + // Compare Normalize dividend & divisor so that both fall into [0.1, 0.999...] + private static int compareMagnitudeNormalized(long xs, int xscale, long ys, int yscale) { + // assert xs!=0 && ys!=0 + int sdiff = xscale - yscale; + if (sdiff != 0) { + if (sdiff < 0) { + xs = longMultiplyPowerTen(xs, -sdiff); + } else { // sdiff > 0 + ys = longMultiplyPowerTen(ys, sdiff); + } + } + if (xs != INFLATED) + return (ys != INFLATED) ? longCompareMagnitude(xs, ys) : -1; + else + return 1; + } + + // Compare Normalize dividend & divisor so that both fall into [0.1, 0.999...] + private static int compareMagnitudeNormalized(long xs, int xscale, BigInteger ys, int yscale) { + // assert "ys can't be represented as long" + if (xs == 0) + return -1; + int sdiff = xscale - yscale; + if (sdiff < 0) { + if (longMultiplyPowerTen(xs, -sdiff) == INFLATED ) { + return bigMultiplyPowerTen(xs, -sdiff).compareMagnitude(ys); + } + } + return -1; + } + + // Compare Normalize dividend & divisor so that both fall into [0.1, 0.999...] + private static int compareMagnitudeNormalized(BigInteger xs, int xscale, BigInteger ys, int yscale) { + int sdiff = xscale - yscale; + if (sdiff < 0) { + return bigMultiplyPowerTen(xs, -sdiff).compareMagnitude(ys); + } else { // sdiff >= 0 + return xs.compareMagnitude(bigMultiplyPowerTen(ys, sdiff)); + } + } + + private static long multiply(long x, long y){ + long product = x * y; + long ax = Math.abs(x); + long ay = Math.abs(y); + if (((ax | ay) >>> 31 == 0) || (y == 0) || (product / y == x)){ + return product; + } + return INFLATED; + } + + private static BigDecimal multiply(long x, long y, int scale) { + long product = multiply(x, y); + if(product!=INFLATED) { + return valueOf(product,scale); + } + return new BigDecimal(BigInteger.valueOf(x).multiply(y),INFLATED,scale,0); + } + + private static BigDecimal multiply(long x, BigInteger y, int scale) { + if(x==0) { + return zeroValueOf(scale); + } + return new BigDecimal(y.multiply(x),INFLATED,scale,0); + } + + private static BigDecimal multiply(BigInteger x, BigInteger y, int scale) { + return new BigDecimal(x.multiply(y),INFLATED,scale,0); + } + + /** + * Multiplies two long values and rounds according {@code MathContext} + */ + private static BigDecimal multiplyAndRound(long x, long y, int scale, MathContext mc) { + long product = multiply(x, y); + if(product!=INFLATED) { + return doRound(product, scale, mc); + } + // attempt to do it in 128 bits + int rsign = 1; + if(x < 0) { + x = -x; + rsign = -1; + } + if(y < 0) { + y = -y; + rsign *= -1; + } + // multiply dividend0 * dividend1 + long m0_hi = x >>> 32; + long m0_lo = x & LONG_MASK; + long m1_hi = y >>> 32; + long m1_lo = y & LONG_MASK; + product = m0_lo * m1_lo; + long m0 = product & LONG_MASK; + long m1 = product >>> 32; + product = m0_hi * m1_lo + m1; + m1 = product & LONG_MASK; + long m2 = product >>> 32; + product = m0_lo * m1_hi + m1; + m1 = product & LONG_MASK; + m2 += product >>> 32; + long m3 = m2>>>32; + m2 &= LONG_MASK; + product = m0_hi*m1_hi + m2; + m2 = product & LONG_MASK; + m3 = ((product>>>32) + m3) & LONG_MASK; + final long mHi = make64(m3,m2); + final long mLo = make64(m1,m0); + BigDecimal res = doRound128(mHi, mLo, rsign, scale, mc); + if(res!=null) { + return res; + } + res = new BigDecimal(BigInteger.valueOf(x).multiply(y*rsign), INFLATED, scale, 0); + return doRound(res,mc); + } + + private static BigDecimal multiplyAndRound(long x, BigInteger y, int scale, MathContext mc) { + if(x==0) { + return zeroValueOf(scale); + } + return doRound(y.multiply(x), scale, mc); + } + + private static BigDecimal multiplyAndRound(BigInteger x, BigInteger y, int scale, MathContext mc) { + return doRound(x.multiply(y), scale, mc); + } + + /** + * rounds 128-bit value according {@code MathContext} + * returns null if result can't be repsented as compact BigDecimal. + */ + private static BigDecimal doRound128(long hi, long lo, int sign, int scale, MathContext mc) { + int mcp = mc.precision; + int drop; + BigDecimal res = null; + if(((drop = precision(hi, lo) - mcp) > 0)&&(drop=0) { + return longDigitLength(lo); + } + return (unsignedLongCompareEq(lo, LONGLONG_TEN_POWERS_TABLE[0][1])) ? 20 : 19; + // 0x8AC7230489E80000L = unsigned 2^19 + } + int r = ((128 - Long.numberOfLeadingZeros(hi) + 1) * 1233) >>> 12; + int idx = r-19; + return (idx >= LONGLONG_TEN_POWERS_TABLE.length || longLongCompareMagnitude(hi, lo, + LONGLONG_TEN_POWERS_TABLE[idx][0], LONGLONG_TEN_POWERS_TABLE[idx][1])) ? r : r + 1; + } + + /* + * returns true if 128 bit number is less then + * hi0 & hi1 should be non-negative + */ + private static boolean longLongCompareMagnitude(long hi0, long lo0, long hi1, long lo1) { + if(hi0!=hi1) { + return hi0 dividendScale) { + int newScale = scale + divisorScale; + int raise = newScale - dividendScale; + if(raise dividendScale) { + int newScale = scale + divisorScale; + int raise = newScale - dividendScale; + BigInteger scaledDividend = bigMultiplyPowerTen(dividend, raise); + return divideAndRound(scaledDividend, divisor, scale, roundingMode, scale); + } else { + int newScale = checkScale(divisor,(long)dividendScale - scale); + int raise = newScale - divisorScale; + if(raise dividendScale) { + int newScale = scale + divisorScale; + int raise = newScale - dividendScale; + BigInteger scaledDividend = bigMultiplyPowerTen(dividend, raise); + return divideAndRound(scaledDividend, divisor, scale, roundingMode, scale); + } else { + int newScale = checkScale(divisor,(long)dividendScale - scale); + int raise = newScale - divisorScale; + BigInteger scaledDivisor = bigMultiplyPowerTen(divisor, raise); + return divideAndRound(BigInteger.valueOf(dividend), scaledDivisor, scale, roundingMode, scale); + } + } + + private static BigDecimal divide(BigInteger dividend, int dividendScale, BigInteger divisor, int divisorScale, int scale, int roundingMode) { + if (checkScale(dividend,(long)scale + divisorScale) > dividendScale) { + int newScale = scale + divisorScale; + int raise = newScale - dividendScale; + BigInteger scaledDividend = bigMultiplyPowerTen(dividend, raise); + return divideAndRound(scaledDividend, divisor, scale, roundingMode, scale); + } else { + int newScale = checkScale(divisor,(long)dividendScale - scale); + int raise = newScale - divisorScale; + BigInteger scaledDivisor = bigMultiplyPowerTen(divisor, raise); + return divideAndRound(dividend, scaledDivisor, scale, roundingMode, scale); + } + } + } diff --git a/jdk/src/share/classes/java/math/BigInteger.java b/jdk/src/share/classes/java/math/BigInteger.java index 4fcce199de5..748a6c2f2b0 100644 --- a/jdk/src/share/classes/java/math/BigInteger.java +++ b/jdk/src/share/classes/java/math/BigInteger.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1996, 2007, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1996, 2011, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -31,6 +31,7 @@ package java.math; import java.util.Random; import java.io.*; +import java.util.Arrays; /** * Immutable arbitrary-precision integers. All operations behave as if @@ -353,27 +354,17 @@ public class BigInteger extends Number implements Comparable { mag = trustedStripLeadingZeroInts(magnitude); } - // Constructs a new BigInteger using a char array with radix=10 - BigInteger(char[] val) { + /* + * Constructs a new BigInteger using a char array with radix=10. + * Sign is precalculated outside and not allowed in the val. + */ + BigInteger(char[] val, int sign, int len) { int cursor = 0, numDigits; - int len = val.length; - - // Check for leading minus sign - int sign = 1; - if (val[0] == '-') { - if (len == 1) - throw new NumberFormatException("Zero length BigInteger"); - sign = -1; - cursor = 1; - } else if (val[0] == '+') { - if (len == 1) - throw new NumberFormatException("Zero length BigInteger"); - cursor = 1; - } // Skip leading zeros and compute number of digits in magnitude - while (cursor < len && Character.digit(val[cursor], 10) == 0) + while (cursor < len && Character.digit(val[cursor], 10) == 0) { cursor++; + } if (cursor == len) { signum = 0; mag = ZERO.mag; @@ -382,7 +373,6 @@ public class BigInteger extends Number implements Comparable { numDigits = len - cursor; signum = sign; - // Pre-allocate array of expected size int numWords; if (len < 10) { @@ -1057,6 +1047,73 @@ public class BigInteger extends Number implements Comparable { return new BigInteger(resultMag, cmp == signum ? 1 : -1); } + /** + * Package private methods used by BigDecimal code to add a BigInteger + * with a long. Assumes val is not equal to INFLATED. + */ + BigInteger add(long val) { + if (val == 0) + return this; + if (signum == 0) + return valueOf(val); + if (Long.signum(val) == signum) + return new BigInteger(add(mag, Math.abs(val)), signum); + int cmp = compareMagnitude(val); + if (cmp == 0) + return ZERO; + int[] resultMag = (cmp > 0 ? subtract(mag, Math.abs(val)) : subtract(Math.abs(val), mag)); + resultMag = trustedStripLeadingZeroInts(resultMag); + return new BigInteger(resultMag, cmp == signum ? 1 : -1); + } + + /** + * Adds the contents of the int array x and long value val. This + * method allocates a new int array to hold the answer and returns + * a reference to that array. Assumes x.length > 0 and val is + * non-negative + */ + private static int[] add(int[] x, long val) { + int[] y; + long sum = 0; + int xIndex = x.length; + int[] result; + int highWord = (int)(val >>> 32); + if (highWord==0) { + result = new int[xIndex]; + sum = (x[--xIndex] & LONG_MASK) + val; + result[xIndex] = (int)sum; + } else { + if (xIndex == 1) { + result = new int[2]; + sum = val + (x[0] & LONG_MASK); + result[1] = (int)sum; + result[0] = (int)(sum >>> 32); + return result; + } else { + result = new int[xIndex]; + sum = (x[--xIndex] & LONG_MASK) + (val & LONG_MASK); + result[xIndex] = (int)sum; + sum = (x[--xIndex] & LONG_MASK) + (highWord & LONG_MASK) + (sum >>> 32); + result[xIndex] = (int)sum; + } + } + // Copy remainder of longer number while carry propagation is required + boolean carry = (sum >>> 32 != 0); + while (xIndex > 0 && carry) + carry = ((result[--xIndex] = x[xIndex] + 1) == 0); + // Copy remainder of longer number + while (xIndex > 0) + result[--xIndex] = x[xIndex]; + // Grow result if necessary + if (carry) { + int bigger[] = new int[result.length + 1]; + System.arraycopy(result, 0, bigger, 1, result.length); + bigger[0] = 0x01; + return bigger; + } + return result; + } + /** * Adds the contents of the int arrays x and y. This method allocates * a new int array to hold the answer and returns a reference to that @@ -1074,14 +1131,17 @@ public class BigInteger extends Number implements Comparable { int yIndex = y.length; int result[] = new int[xIndex]; long sum = 0; - - // Add common parts of both numbers - while(yIndex > 0) { - sum = (x[--xIndex] & LONG_MASK) + - (y[--yIndex] & LONG_MASK) + (sum >>> 32); + if(yIndex==1) { + sum = (x[--xIndex] & LONG_MASK) + (y[0] & LONG_MASK) ; result[xIndex] = (int)sum; + } else { + // Add common parts of both numbers + while(yIndex > 0) { + sum = (x[--xIndex] & LONG_MASK) + + (y[--yIndex] & LONG_MASK) + (sum >>> 32); + result[xIndex] = (int)sum; + } } - // Copy remainder of longer number while carry propagation is required boolean carry = (sum >>> 32 != 0); while (xIndex > 0 && carry) @@ -1101,6 +1161,71 @@ public class BigInteger extends Number implements Comparable { return result; } + private static int[] subtract(long val, int[] little) { + int highWord = (int)(val >>> 32); + if (highWord==0) { + int result[] = new int[1]; + result[0] = (int)(val - (little[0] & LONG_MASK)); + return result; + } else { + int result[] = new int[2]; + if(little.length==1) { + long difference = ((int)val & LONG_MASK) - (little[0] & LONG_MASK); + result[1] = (int)difference; + // Subtract remainder of longer number while borrow propagates + boolean borrow = (difference >> 32 != 0); + if(borrow) { + result[0] = highWord - 1; + } else { // Copy remainder of longer number + result[0] = highWord; + } + return result; + } else { // little.length==2 + long difference = ((int)val & LONG_MASK) - (little[1] & LONG_MASK); + result[1] = (int)difference; + difference = (highWord & LONG_MASK) - (little[0] & LONG_MASK) + (difference >> 32); + result[0] = (int)difference; + return result; + } + } + } + + /** + * Subtracts the contents of the second argument (val) from the + * first (big). The first int array (big) must represent a larger number + * than the second. This method allocates the space necessary to hold the + * answer. + * assumes val >= 0 + */ + private static int[] subtract(int[] big, long val) { + int highWord = (int)(val >>> 32); + int bigIndex = big.length; + int result[] = new int[bigIndex]; + long difference = 0; + + if (highWord==0) { + difference = (big[--bigIndex] & LONG_MASK) - val; + result[bigIndex] = (int)difference; + } else { + difference = (big[--bigIndex] & LONG_MASK) - (val & LONG_MASK); + result[bigIndex] = (int)difference; + difference = (big[--bigIndex] & LONG_MASK) - (highWord & LONG_MASK) + (difference >> 32); + result[bigIndex] = (int)difference; + } + + + // Subtract remainder of longer number while borrow propagates + boolean borrow = (difference >> 32 != 0); + while (bigIndex > 0 && borrow) + borrow = ((result[--bigIndex] = big[bigIndex] - 1) == -1); + + // Copy remainder of longer number + while (bigIndex > 0) + result[--bigIndex] = big[bigIndex]; + + return result; + } + /** * Returns a BigInteger whose value is {@code (this - val)}. * @@ -1165,11 +1290,39 @@ public class BigInteger extends Number implements Comparable { public BigInteger multiply(BigInteger val) { if (val.signum == 0 || signum == 0) return ZERO; - + int resultSign = signum == val.signum ? 1 : -1; + if (val.mag.length == 1) { + return multiplyByInt(mag,val.mag[0], resultSign); + } + if(mag.length == 1) { + return multiplyByInt(val.mag,mag[0], resultSign); + } int[] result = multiplyToLen(mag, mag.length, val.mag, val.mag.length, null); result = trustedStripLeadingZeroInts(result); - return new BigInteger(result, signum == val.signum ? 1 : -1); + return new BigInteger(result, resultSign); + } + + private static BigInteger multiplyByInt(int[] x, int y, int sign) { + if(Integer.bitCount(y)==1) { + return new BigInteger(shiftLeft(x,Integer.numberOfTrailingZeros(y)), sign); + } + int xlen = x.length; + int[] rmag = new int[xlen + 1]; + long carry = 0; + long yl = y & LONG_MASK; + int rstart = rmag.length - 1; + for (int i = xlen - 1; i >= 0; i--) { + long product = (x[i] & LONG_MASK) * yl + carry; + rmag[rstart--] = (int)product; + carry = product >>> 32; + } + if (carry == 0L) { + rmag = java.util.Arrays.copyOfRange(rmag, 1, rmag.length); + } else { + rmag[rstart] = (int)carry; + } + return new BigInteger(rmag, sign); } /** @@ -1339,8 +1492,8 @@ public class BigInteger extends Number implements Comparable { a = new MutableBigInteger(this.mag), b = new MutableBigInteger(val.mag); - a.divide(b, q); - return q.toBigInteger(this.signum == val.signum ? 1 : -1); + a.divide(b, q, false); + return q.toBigInteger(this.signum * val.signum); } /** @@ -1460,14 +1613,12 @@ public class BigInteger extends Number implements Comparable { } else { // Array must be resized if (nBits <= (32-bitsInHighWord)) { int result[] = new int[nInts+len]; - for (int i=0; i { b = montReduce(b, mod, modLen, inv); // Set t to high half of b - int[] t = new int[modLen]; - for(int i=0; i { // Convert result out of Montgomery form and return int[] t2 = new int[2*modLen]; - for(int i=0; i { // Copy remaining ints of mag int numInts = (p + 31) >>> 5; int[] mag = new int[numInts]; - for (int i=0; i { return shiftRight(-n); } } + int[] newMag = shiftLeft(mag, n); + return new BigInteger(newMag, signum); + } + + private static int[] shiftLeft(int[] mag, int n) { int nInts = n >>> 5; int nBits = n & 0x1f; int magLen = mag.length; @@ -2077,8 +2227,7 @@ public class BigInteger extends Number implements Comparable { if (nBits == 0) { newMag = new int[magLen + nInts]; - for (int i=0; i { newMag[i++] = mag[j++] << nBits | mag[j] >>> nBits2; newMag[i] = mag[j] << nBits; } - - return new BigInteger(newMag, signum); + return newMag; } /** @@ -2132,9 +2280,7 @@ public class BigInteger extends Number implements Comparable { if (nBits == 0) { int newMagLen = magLen - nInts; - newMag = new int[newMagLen]; - for (int i=0; i>> nBits; @@ -2405,7 +2551,7 @@ public class BigInteger extends Number implements Comparable { if (signum < 0) { // Check if magnitude is a power of two boolean pow2 = (Integer.bitCount(mag[0]) == 1); - for(int i=1; i< len && pow2; i++) + for (int i=1; i< len && pow2; i++) pow2 = (mag[i] == 0); n = (pow2 ? magBitLength -1 : magBitLength); @@ -2529,6 +2675,49 @@ public class BigInteger extends Number implements Comparable { return 0; } + /** + * Version of compareMagnitude that compares magnitude with long value. + * val can't be Long.MIN_VALUE. + */ + final int compareMagnitude(long val) { + assert val != Long.MIN_VALUE; + int[] m1 = mag; + int len = m1.length; + if(len > 2) { + return 1; + } + if (val < 0) { + val = -val; + } + int highWord = (int)(val >>> 32); + if (highWord==0) { + if (len < 1) + return -1; + if (len > 1) + return 1; + int a = m1[0]; + int b = (int)val; + if (a != b) { + return ((a & LONG_MASK) < (b & LONG_MASK))? -1 : 1; + } + return 0; + } else { + if (len < 2) + return -1; + int a = m1[0]; + int b = highWord; + if (a != b) { + return ((a & LONG_MASK) < (b & LONG_MASK))? -1 : 1; + } + a = m1[1]; + b = (int)val; + if (a != b) { + return ((a & LONG_MASK) < (b & LONG_MASK))? -1 : 1; + } + return 0; + } + } + /** * Compares this BigInteger with the specified Object for equality. * @@ -3114,25 +3303,35 @@ public class BigInteger extends Number implements Comparable { } // Commit final fields via Unsafe - unsafe.putIntVolatile(this, signumOffset, sign); + UnsafeHolder.putSign(this, sign); // Calculate mag field from magnitude and discard magnitude - unsafe.putObjectVolatile(this, magOffset, - stripLeadingZeroBytes(magnitude)); + UnsafeHolder.putMag(this, stripLeadingZeroBytes(magnitude)); } // Support for resetting final fields while deserializing - private static final sun.misc.Unsafe unsafe = sun.misc.Unsafe.getUnsafe(); - private static final long signumOffset; - private static final long magOffset; - static { - try { - signumOffset = unsafe.objectFieldOffset - (BigInteger.class.getDeclaredField("signum")); - magOffset = unsafe.objectFieldOffset - (BigInteger.class.getDeclaredField("mag")); - } catch (Exception ex) { - throw new Error(ex); + private static class UnsafeHolder { + private static final sun.misc.Unsafe unsafe; + private static final long signumOffset; + private static final long magOffset; + static { + try { + unsafe = sun.misc.Unsafe.getUnsafe(); + signumOffset = unsafe.objectFieldOffset + (BigInteger.class.getDeclaredField("signum")); + magOffset = unsafe.objectFieldOffset + (BigInteger.class.getDeclaredField("mag")); + } catch (Exception ex) { + throw new ExceptionInInitializerError(ex); + } + } + + static void putSign(BigInteger bi, int sign) { + unsafe.putIntVolatile(bi, signumOffset, sign); + } + + static void putMag(BigInteger bi, int[] magnitude) { + unsafe.putObjectVolatile(bi, magOffset, magnitude); } } diff --git a/jdk/src/share/classes/java/math/MutableBigInteger.java b/jdk/src/share/classes/java/math/MutableBigInteger.java index 0cb154430d7..363e5d46e02 100644 --- a/jdk/src/share/classes/java/math/MutableBigInteger.java +++ b/jdk/src/share/classes/java/math/MutableBigInteger.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1999, 2007, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1999, 2011, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -160,7 +160,7 @@ class MutableBigInteger { */ BigDecimal toBigDecimal(int sign, int scale) { if (intLen == 0 || sign == 0) - return BigDecimal.valueOf(0, scale); + return BigDecimal.zeroValueOf(scale); int[] mag = getMagnitudeArray(); int len = mag.length; int d = mag[0]; @@ -171,7 +171,28 @@ class MutableBigInteger { long v = (len == 2) ? ((mag[1] & LONG_MASK) | (d & LONG_MASK) << 32) : d & LONG_MASK; - return new BigDecimal(null, sign == -1 ? -v : v, scale, 0); + return BigDecimal.valueOf(sign == -1 ? -v : v, scale); + } + + /** + * This is for internal use in converting from a MutableBigInteger + * object into a long value given a specified sign. + * returns INFLATED if value is not fit into long + */ + long toCompactValue(int sign) { + if (intLen == 0 || sign == 0) + return 0L; + int[] mag = getMagnitudeArray(); + int len = mag.length; + int d = mag[0]; + // If this MutableBigInteger can not be fitted into long, we need to + // make a BigInteger object for the resultant BigDecimal object. + if (len > 2 || (d < 0 && len == 2)) + return INFLATED; + long v = (len == 2) ? + ((mag[1] & LONG_MASK) | (d & LONG_MASK) << 32) : + d & LONG_MASK; + return sign == -1 ? -v : v; } /** @@ -543,6 +564,24 @@ class MutableBigInteger { return (int)carry; } + /** + * The method is the same as mulsun, except the fact that q array is not + * updated, the only result of the method is borrow flag. + */ + private int mulsubBorrow(int[] q, int[] a, int x, int len, int offset) { + long xLong = x & LONG_MASK; + long carry = 0; + offset += len; + for (int j=len-1; j >= 0; j--) { + long product = (a[j] & LONG_MASK) * xLong + carry; + long difference = q[offset--] - product; + carry = (product >>> 32) + + (((difference & LONG_MASK) > + (((~(int)product) & LONG_MASK))) ? 1:0); + } + return (int)carry; + } + /** * Right shift this MutableBigInteger n bits, where n is * less than 32. @@ -842,20 +881,20 @@ class MutableBigInteger { rem = (int) (remLong - (quotient.value[0] * divisorLong)); remLong = rem & LONG_MASK; } - int xlen = intLen; - int[] qWord = new int[2]; while (--xlen > 0) { - long dividendEstimate = (remLong<<32) | - (value[offset + intLen - xlen] & LONG_MASK); + long dividendEstimate = (remLong << 32) | + (value[offset + intLen - xlen] & LONG_MASK); + int q; if (dividendEstimate >= 0) { - qWord[0] = (int) (dividendEstimate / divisorLong); - qWord[1] = (int) (dividendEstimate - qWord[0] * divisorLong); + q = (int) (dividendEstimate / divisorLong); + rem = (int) (dividendEstimate - q * divisorLong); } else { - divWord(qWord, dividendEstimate, divisor); + long tmp = divWord(dividendEstimate, divisor); + q = (int) (tmp & LONG_MASK); + rem = (int) (tmp >>> 32); } - quotient.value[intLen - xlen] = qWord[0]; - rem = qWord[1]; + quotient.value[intLen - xlen] = q; remLong = rem & LONG_MASK; } @@ -879,40 +918,45 @@ class MutableBigInteger { * */ MutableBigInteger divide(MutableBigInteger b, MutableBigInteger quotient) { + return divide(b,quotient,true); + } + + MutableBigInteger divide(MutableBigInteger b, MutableBigInteger quotient, boolean needReminder) { if (b.intLen == 0) throw new ArithmeticException("BigInteger divide by zero"); // Dividend is zero if (intLen == 0) { quotient.intLen = quotient.offset; - return new MutableBigInteger(); + return needReminder ? new MutableBigInteger() : null; } int cmp = compare(b); // Dividend less than divisor if (cmp < 0) { quotient.intLen = quotient.offset = 0; - return new MutableBigInteger(this); + return needReminder ? new MutableBigInteger(this) : null; } // Dividend equal to divisor if (cmp == 0) { quotient.value[0] = quotient.intLen = 1; quotient.offset = 0; - return new MutableBigInteger(); + return needReminder ? new MutableBigInteger() : null; } quotient.clear(); // Special case one word divisor if (b.intLen == 1) { int r = divideOneWord(b.value[b.offset], quotient); - if (r == 0) - return new MutableBigInteger(); - return new MutableBigInteger(r); + if(needReminder) { + if (r == 0) + return new MutableBigInteger(); + return new MutableBigInteger(r); + } else { + return null; + } } - - // Copy divisor value to protect divisor - int[] div = Arrays.copyOfRange(b.value, b.offset, b.offset + b.intLen); - return divideMagnitude(div, quotient); + return divideMagnitude(b, quotient, needReminder); } /** @@ -940,30 +984,72 @@ class MutableBigInteger { if (d == 0) return divideOneWord((int)v, quotient) & LONG_MASK; else { - int[] div = new int[]{ d, (int)(v & LONG_MASK) }; - return divideMagnitude(div, quotient).toLong(); + return divideLongMagnitude(v, quotient).toLong(); } } + private static void copyAndShift(int[] src, int srcFrom, int srcLen, int[] dst, int dstFrom, int shift) { + int n2 = 32 - shift; + int c=src[srcFrom]; + for (int i=0; i < srcLen-1; i++) { + int b = c; + c = src[++srcFrom]; + dst[dstFrom+i] = (b << shift) | (c >>> n2); + } + dst[dstFrom+srcLen-1] = c << shift; + } + /** - * Divide this MutableBigInteger by the divisor represented by its magnitude - * array. The quotient will be placed into the provided quotient object & + * Divide this MutableBigInteger by the divisor. + * The quotient will be placed into the provided quotient object & * the remainder object is returned. */ - private MutableBigInteger divideMagnitude(int[] divisor, - MutableBigInteger quotient) { - - // Remainder starts as dividend with space for a leading zero - MutableBigInteger rem = new MutableBigInteger(new int[intLen + 1]); - System.arraycopy(value, offset, rem.value, 1, intLen); - rem.intLen = intLen; - rem.offset = 1; + private MutableBigInteger divideMagnitude(MutableBigInteger div, + MutableBigInteger quotient, + boolean needReminder ) { + // assert div.intLen > 1 + // D1 normalize the divisor + int shift = Integer.numberOfLeadingZeros(div.value[div.offset]); + // Copy divisor value to protect divisor + final int dlen = div.intLen; + int[] divisor; + MutableBigInteger rem; // Remainder starts as dividend with space for a leading zero + if (shift > 0) { + divisor = new int[dlen]; + copyAndShift(div.value,div.offset,dlen,divisor,0,shift); + if(Integer.numberOfLeadingZeros(value[offset])>=shift) { + int[] remarr = new int[intLen + 1]; + rem = new MutableBigInteger(remarr); + rem.intLen = intLen; + rem.offset = 1; + copyAndShift(value,offset,intLen,remarr,1,shift); + } else { + int[] remarr = new int[intLen + 2]; + rem = new MutableBigInteger(remarr); + rem.intLen = intLen+1; + rem.offset = 1; + int rFrom = offset; + int c=0; + int n2 = 32 - shift; + for (int i=1; i < intLen+1; i++,rFrom++) { + int b = c; + c = value[rFrom]; + remarr[i] = (b << shift) | (c >>> n2); + } + remarr[intLen+1] = c << shift; + } + } else { + divisor = Arrays.copyOfRange(div.value, div.offset, div.offset + div.intLen); + rem = new MutableBigInteger(new int[intLen + 1]); + System.arraycopy(value, offset, rem.value, 1, intLen); + rem.intLen = intLen; + rem.offset = 1; + } int nlen = rem.intLen; // Set the quotient size - int dlen = divisor.length; - int limit = nlen - dlen + 1; + final int limit = nlen - dlen + 1; if (quotient.value.length < limit) { quotient.value = new int[limit]; quotient.offset = 0; @@ -971,14 +1057,6 @@ class MutableBigInteger { quotient.intLen = limit; int[] q = quotient.value; - // D1 normalize the divisor - int shift = Integer.numberOfLeadingZeros(divisor[0]); - if (shift > 0) { - // First shift will not grow array - BigInteger.primitiveLeftShift(divisor, dlen, shift); - // But this one might - rem.leftShift(shift); - } // Must insert leading 0 in rem if its length did not change if (rem.intLen == nlen) { @@ -990,10 +1068,9 @@ class MutableBigInteger { int dh = divisor[0]; long dhLong = dh & LONG_MASK; int dl = divisor[1]; - int[] qWord = new int[2]; // D2 Initialize j - for(int j=0; j>> 32); } } @@ -1053,6 +1130,181 @@ class MutableBigInteger { // Store the quotient digit q[j] = qhat; } // D7 loop on j + // D3 Calculate qhat + // estimate qhat + int qhat = 0; + int qrem = 0; + boolean skipCorrection = false; + int nh = rem.value[limit - 1 + rem.offset]; + int nh2 = nh + 0x80000000; + int nm = rem.value[limit + rem.offset]; + + if (nh == dh) { + qhat = ~0; + qrem = nh + nm; + skipCorrection = qrem + 0x80000000 < nh2; + } else { + long nChunk = (((long) nh) << 32) | (nm & LONG_MASK); + if (nChunk >= 0) { + qhat = (int) (nChunk / dhLong); + qrem = (int) (nChunk - (qhat * dhLong)); + } else { + long tmp = divWord(nChunk, dh); + qhat = (int) (tmp & LONG_MASK); + qrem = (int) (tmp >>> 32); + } + } + if (qhat != 0) { + if (!skipCorrection) { // Correct qhat + long nl = rem.value[limit + 1 + rem.offset] & LONG_MASK; + long rs = ((qrem & LONG_MASK) << 32) | nl; + long estProduct = (dl & LONG_MASK) * (qhat & LONG_MASK); + + if (unsignedLongCompare(estProduct, rs)) { + qhat--; + qrem = (int) ((qrem & LONG_MASK) + dhLong); + if ((qrem & LONG_MASK) >= dhLong) { + estProduct -= (dl & LONG_MASK); + rs = ((qrem & LONG_MASK) << 32) | nl; + if (unsignedLongCompare(estProduct, rs)) + qhat--; + } + } + } + + + // D4 Multiply and subtract + int borrow; + rem.value[limit - 1 + rem.offset] = 0; + if(needReminder) + borrow = mulsub(rem.value, divisor, qhat, dlen, limit - 1 + rem.offset); + else + borrow = mulsubBorrow(rem.value, divisor, qhat, dlen, limit - 1 + rem.offset); + + // D5 Test remainder + if (borrow + 0x80000000 > nh2) { + // D6 Add back + if(needReminder) + divadd(divisor, rem.value, limit - 1 + 1 + rem.offset); + qhat--; + } + + // Store the quotient digit + q[(limit - 1)] = qhat; + } + + + if(needReminder) { + // D8 Unnormalize + if (shift > 0) + rem.rightShift(shift); + rem.normalize(); + } + quotient.normalize(); + return needReminder ? rem : null; + } + + /** + * Divide this MutableBigInteger by the divisor represented by positive long + * value. The quotient will be placed into the provided quotient object & + * the remainder object is returned. + */ + private MutableBigInteger divideLongMagnitude(long ldivisor, MutableBigInteger quotient) { + // Remainder starts as dividend with space for a leading zero + MutableBigInteger rem = new MutableBigInteger(new int[intLen + 1]); + System.arraycopy(value, offset, rem.value, 1, intLen); + rem.intLen = intLen; + rem.offset = 1; + + int nlen = rem.intLen; + + int limit = nlen - 2 + 1; + if (quotient.value.length < limit) { + quotient.value = new int[limit]; + quotient.offset = 0; + } + quotient.intLen = limit; + int[] q = quotient.value; + + // D1 normalize the divisor + int shift = Long.numberOfLeadingZeros(ldivisor); + if (shift > 0) { + ldivisor<<=shift; + rem.leftShift(shift); + } + + // Must insert leading 0 in rem if its length did not change + if (rem.intLen == nlen) { + rem.offset = 0; + rem.value[0] = 0; + rem.intLen++; + } + + int dh = (int)(ldivisor >>> 32); + long dhLong = dh & LONG_MASK; + int dl = (int)(ldivisor & LONG_MASK); + + // D2 Initialize j + for (int j = 0; j < limit; j++) { + // D3 Calculate qhat + // estimate qhat + int qhat = 0; + int qrem = 0; + boolean skipCorrection = false; + int nh = rem.value[j + rem.offset]; + int nh2 = nh + 0x80000000; + int nm = rem.value[j + 1 + rem.offset]; + + if (nh == dh) { + qhat = ~0; + qrem = nh + nm; + skipCorrection = qrem + 0x80000000 < nh2; + } else { + long nChunk = (((long) nh) << 32) | (nm & LONG_MASK); + if (nChunk >= 0) { + qhat = (int) (nChunk / dhLong); + qrem = (int) (nChunk - (qhat * dhLong)); + } else { + long tmp = divWord(nChunk, dh); + qhat =(int)(tmp & LONG_MASK); + qrem = (int)(tmp>>>32); + } + } + + if (qhat == 0) + continue; + + if (!skipCorrection) { // Correct qhat + long nl = rem.value[j + 2 + rem.offset] & LONG_MASK; + long rs = ((qrem & LONG_MASK) << 32) | nl; + long estProduct = (dl & LONG_MASK) * (qhat & LONG_MASK); + + if (unsignedLongCompare(estProduct, rs)) { + qhat--; + qrem = (int) ((qrem & LONG_MASK) + dhLong); + if ((qrem & LONG_MASK) >= dhLong) { + estProduct -= (dl & LONG_MASK); + rs = ((qrem & LONG_MASK) << 32) | nl; + if (unsignedLongCompare(estProduct, rs)) + qhat--; + } + } + } + + // D4 Multiply and subtract + rem.value[j + rem.offset] = 0; + int borrow = mulsubLong(rem.value, dh, dl, qhat, j + rem.offset); + + // D5 Test remainder + if (borrow + 0x80000000 > nh2) { + // D6 Add back + divaddLong(dh,dl, rem.value, j + 1 + rem.offset); + qhat--; + } + + // Store the quotient digit + q[j] = qhat; + } // D7 loop on j // D8 Unnormalize if (shift > 0) @@ -1063,6 +1315,46 @@ class MutableBigInteger { return rem; } + /** + * A primitive used for division by long. + * Specialized version of the method divadd. + * dh is a high part of the divisor, dl is a low part + */ + private int divaddLong(int dh, int dl, int[] result, int offset) { + long carry = 0; + + long sum = (dl & LONG_MASK) + (result[1+offset] & LONG_MASK); + result[1+offset] = (int)sum; + + sum = (dh & LONG_MASK) + (result[offset] & LONG_MASK) + carry; + result[offset] = (int)sum; + carry = sum >>> 32; + return (int)carry; + } + + /** + * This method is used for division by long. + * Specialized version of the method sulsub. + * dh is a high part of the divisor, dl is a low part + */ + private int mulsubLong(int[] q, int dh, int dl, int x, int offset) { + long xLong = x & LONG_MASK; + offset += 2; + long product = (dl & LONG_MASK) * xLong; + long difference = q[offset] - product; + q[offset--] = (int)difference; + long carry = (product >>> 32) + + (((difference & LONG_MASK) > + (((~(int)product) & LONG_MASK))) ? 1:0); + product = (dh & LONG_MASK) * xLong + carry; + difference = q[offset] - product; + q[offset--] = (int)difference; + carry = (product >>> 32) + + (((difference & LONG_MASK) > + (((~(int)product) & LONG_MASK))) ? 1:0); + return (int)carry; + } + /** * Compare two longs as if they were unsigned. * Returns true iff one is bigger than two. @@ -1075,19 +1367,22 @@ class MutableBigInteger { * This method divides a long quantity by an int to estimate * qhat for two multi precision numbers. It is used when * the signed value of n is less than zero. + * Returns long value where high 32 bits contain reminder value and + * low 32 bits contain quotient value. */ - private void divWord(int[] result, long n, int d) { + static long divWord(long n, int d) { long dLong = d & LONG_MASK; - + long r; + long q; if (dLong == 1) { - result[0] = (int)n; - result[1] = 0; - return; + q = (int)n; + r = 0; + return (r << 32) | (q & LONG_MASK); } // Approximate the quotient and remainder - long q = (n >>> 1) / (dLong >>> 1); - long r = n - q*dLong; + q = (n >>> 1) / (dLong >>> 1); + r = n - q*dLong; // Correct the approximation while (r < 0) { @@ -1098,10 +1393,8 @@ class MutableBigInteger { r -= dLong; q++; } - // n - q*dlong == r && 0 <= r - * If you read from a socket input stream after invoking - * shutdownInput() on the socket, the stream will return EOF. + * If you read from a socket input stream after invoking this method on the + * socket, the stream's {@code available} method will return 0, and its + * {@code read} methods will return {@code -1} (end of stream). * * @exception IOException if an I/O error occurs when shutting down this * socket. diff --git a/jdk/src/share/classes/java/net/SocketImpl.java b/jdk/src/share/classes/java/net/SocketImpl.java index 495b18f2779..3a4cf7aebde 100644 --- a/jdk/src/share/classes/java/net/SocketImpl.java +++ b/jdk/src/share/classes/java/net/SocketImpl.java @@ -181,8 +181,9 @@ public abstract class SocketImpl implements SocketOptions { * Any data sent to this socket is acknowledged and then * silently discarded. * - * If you read from a socket input stream after invoking - * shutdownInput() on the socket, the stream will return EOF. + * If you read from a socket input stream after invoking this method on the + * socket, the stream's {@code available} method will return 0, and its + * {@code read} methods will return {@code -1} (end of stream). * * @exception IOException if an I/O error occurs when shutting down this * socket. diff --git a/jdk/src/share/classes/java/net/URI.java b/jdk/src/share/classes/java/net/URI.java index 5712821e9d8..6997a427467 100644 --- a/jdk/src/share/classes/java/net/URI.java +++ b/jdk/src/share/classes/java/net/URI.java @@ -1711,6 +1711,8 @@ public final class URI i++; continue; } + if (d != '%') + return false; i++; if (toLower(s.charAt(i)) != toLower(t.charAt(i))) return false; diff --git a/jdk/src/share/classes/java/rmi/dgc/VMID.java b/jdk/src/share/classes/java/rmi/dgc/VMID.java index b4e48b3b269..c9f1647e1f1 100644 --- a/jdk/src/share/classes/java/rmi/dgc/VMID.java +++ b/jdk/src/share/classes/java/rmi/dgc/VMID.java @@ -170,7 +170,7 @@ public final class VMID implements java.io.Serializable { /* can't happen, but be deterministic anyway. */ addrHash = new byte[0]; } catch (NoSuchAlgorithmException complain) { - throw new InternalError(complain.toString()); + throw new InternalError(complain.toString(), complain); } return addrHash; } diff --git a/jdk/src/share/classes/java/security/cert/CollectionCertStoreParameters.java b/jdk/src/share/classes/java/security/cert/CollectionCertStoreParameters.java index ddeea61010d..cad34ac08bf 100644 --- a/jdk/src/share/classes/java/security/cert/CollectionCertStoreParameters.java +++ b/jdk/src/share/classes/java/security/cert/CollectionCertStoreParameters.java @@ -122,7 +122,7 @@ public class CollectionCertStoreParameters return super.clone(); } catch (CloneNotSupportedException e) { /* Cannot happen */ - throw new InternalError(e.toString()); + throw new InternalError(e.toString(), e); } } diff --git a/jdk/src/share/classes/java/security/cert/LDAPCertStoreParameters.java b/jdk/src/share/classes/java/security/cert/LDAPCertStoreParameters.java index 00bf2af0c91..9c5109e12cf 100644 --- a/jdk/src/share/classes/java/security/cert/LDAPCertStoreParameters.java +++ b/jdk/src/share/classes/java/security/cert/LDAPCertStoreParameters.java @@ -128,7 +128,7 @@ public class LDAPCertStoreParameters implements CertStoreParameters { return super.clone(); } catch (CloneNotSupportedException e) { /* Cannot happen */ - throw new InternalError(e.toString()); + throw new InternalError(e.toString(), e); } } diff --git a/jdk/src/share/classes/java/security/cert/PKIXCertPathChecker.java b/jdk/src/share/classes/java/security/cert/PKIXCertPathChecker.java index 0cca284501d..656a51706fd 100644 --- a/jdk/src/share/classes/java/security/cert/PKIXCertPathChecker.java +++ b/jdk/src/share/classes/java/security/cert/PKIXCertPathChecker.java @@ -175,7 +175,7 @@ public abstract class PKIXCertPathChecker implements Cloneable { return super.clone(); } catch (CloneNotSupportedException e) { /* Cannot happen */ - throw new InternalError(e.toString()); + throw new InternalError(e.toString(), e); } } } diff --git a/jdk/src/share/classes/java/security/cert/PKIXCertPathValidatorResult.java b/jdk/src/share/classes/java/security/cert/PKIXCertPathValidatorResult.java index caea7a27f04..9bdf006415f 100644 --- a/jdk/src/share/classes/java/security/cert/PKIXCertPathValidatorResult.java +++ b/jdk/src/share/classes/java/security/cert/PKIXCertPathValidatorResult.java @@ -136,7 +136,7 @@ public class PKIXCertPathValidatorResult implements CertPathValidatorResult { return super.clone(); } catch (CloneNotSupportedException e) { /* Cannot happen */ - throw new InternalError(e.toString()); + throw new InternalError(e.toString(), e); } } diff --git a/jdk/src/share/classes/java/security/cert/PKIXParameters.java b/jdk/src/share/classes/java/security/cert/PKIXParameters.java index 6f4d385d66b..ff3b335f5e3 100644 --- a/jdk/src/share/classes/java/security/cert/PKIXParameters.java +++ b/jdk/src/share/classes/java/security/cert/PKIXParameters.java @@ -683,7 +683,7 @@ public class PKIXParameters implements CertPathParameters { return copy; } catch (CloneNotSupportedException e) { /* Cannot happen */ - throw new InternalError(e.toString()); + throw new InternalError(e.toString(), e); } } diff --git a/jdk/src/share/classes/java/security/cert/X509CRLSelector.java b/jdk/src/share/classes/java/security/cert/X509CRLSelector.java index 5bcdf4f6ad0..4258da39639 100644 --- a/jdk/src/share/classes/java/security/cert/X509CRLSelector.java +++ b/jdk/src/share/classes/java/security/cert/X509CRLSelector.java @@ -708,7 +708,7 @@ public class X509CRLSelector implements CRLSelector { return copy; } catch (CloneNotSupportedException e) { /* Cannot happen */ - throw new InternalError(e.toString()); + throw new InternalError(e.toString(), e); } } } diff --git a/jdk/src/share/classes/java/security/cert/X509CertSelector.java b/jdk/src/share/classes/java/security/cert/X509CertSelector.java index ea3f6b72c95..19ca2cb515d 100644 --- a/jdk/src/share/classes/java/security/cert/X509CertSelector.java +++ b/jdk/src/share/classes/java/security/cert/X509CertSelector.java @@ -2615,7 +2615,7 @@ public class X509CertSelector implements CertSelector { return copy; } catch (CloneNotSupportedException e) { /* Cannot happen */ - throw new InternalError(e.toString()); + throw new InternalError(e.toString(), e); } } } diff --git a/jdk/src/share/classes/java/text/AttributedString.java b/jdk/src/share/classes/java/text/AttributedString.java index 029652530af..c7c1f05d6d8 100644 --- a/jdk/src/share/classes/java/text/AttributedString.java +++ b/jdk/src/share/classes/java/text/AttributedString.java @@ -794,7 +794,7 @@ public class AttributedString { return other; } catch (CloneNotSupportedException e) { - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/text/BreakDictionary.java b/jdk/src/share/classes/java/text/BreakDictionary.java index 0f618051794..1e345030ace 100644 --- a/jdk/src/share/classes/java/text/BreakDictionary.java +++ b/jdk/src/share/classes/java/text/BreakDictionary.java @@ -154,7 +154,7 @@ class BreakDictionary { ); } catch (PrivilegedActionException e) { - throw new InternalError(e.toString()); + throw new InternalError(e.toString(), e); } byte[] buf = new byte[8]; diff --git a/jdk/src/share/classes/java/text/BreakIterator.java b/jdk/src/share/classes/java/text/BreakIterator.java index 2a007ca3719..e00e09adc31 100644 --- a/jdk/src/share/classes/java/text/BreakIterator.java +++ b/jdk/src/share/classes/java/text/BreakIterator.java @@ -254,7 +254,7 @@ public abstract class BreakIterator implements Cloneable return super.clone(); } catch (CloneNotSupportedException e) { - throw new InternalError(); + throw new InternalError(e); } } @@ -617,7 +617,7 @@ public abstract class BreakIterator implements Cloneable } } catch (Exception e) { - throw new InternalError(e.toString()); + throw new InternalError(e.toString(), e); } } diff --git a/jdk/src/share/classes/java/text/Collator.java b/jdk/src/share/classes/java/text/Collator.java index 555930ba14c..9b03d1608b2 100644 --- a/jdk/src/share/classes/java/text/Collator.java +++ b/jdk/src/share/classes/java/text/Collator.java @@ -461,7 +461,7 @@ public abstract class Collator try { return (Collator)super.clone(); } catch (CloneNotSupportedException e) { - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/text/DateFormatSymbols.java b/jdk/src/share/classes/java/text/DateFormatSymbols.java index 2eeb07e3447..e5b44c6e0e3 100644 --- a/jdk/src/share/classes/java/text/DateFormatSymbols.java +++ b/jdk/src/share/classes/java/text/DateFormatSymbols.java @@ -597,7 +597,7 @@ public class DateFormatSymbols implements Serializable, Cloneable { copyMembers(this, other); return other; } catch (CloneNotSupportedException e) { - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/text/DecimalFormat.java b/jdk/src/share/classes/java/text/DecimalFormat.java index 90043486830..3dd9a5be94e 100644 --- a/jdk/src/share/classes/java/text/DecimalFormat.java +++ b/jdk/src/share/classes/java/text/DecimalFormat.java @@ -1891,14 +1891,10 @@ public class DecimalFormat extends NumberFormat { * Standard override; no change in semantics. */ public Object clone() { - try { - DecimalFormat other = (DecimalFormat) super.clone(); - other.symbols = (DecimalFormatSymbols) symbols.clone(); - other.digitList = (DigitList) digitList.clone(); - return other; - } catch (Exception e) { - throw new InternalError(); - } + DecimalFormat other = (DecimalFormat) super.clone(); + other.symbols = (DecimalFormatSymbols) symbols.clone(); + other.digitList = (DigitList) digitList.clone(); + return other; } /** diff --git a/jdk/src/share/classes/java/text/DecimalFormatSymbols.java b/jdk/src/share/classes/java/text/DecimalFormatSymbols.java index b3c55b82c20..71fd1fa1820 100644 --- a/jdk/src/share/classes/java/text/DecimalFormatSymbols.java +++ b/jdk/src/share/classes/java/text/DecimalFormatSymbols.java @@ -479,7 +479,7 @@ public class DecimalFormatSymbols implements Cloneable, Serializable { return (DecimalFormatSymbols)super.clone(); // other fields are bit-copied } catch (CloneNotSupportedException e) { - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/text/DigitList.java b/jdk/src/share/classes/java/text/DigitList.java index a355944c1b3..ec489dbd4e9 100644 --- a/jdk/src/share/classes/java/text/DigitList.java +++ b/jdk/src/share/classes/java/text/DigitList.java @@ -632,7 +632,7 @@ final class DigitList implements Cloneable { other.tempBuffer = null; return other; } catch (CloneNotSupportedException e) { - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/text/Format.java b/jdk/src/share/classes/java/text/Format.java index a0ed9938f5e..2eec6671842 100644 --- a/jdk/src/share/classes/java/text/Format.java +++ b/jdk/src/share/classes/java/text/Format.java @@ -258,7 +258,7 @@ public abstract class Format implements Serializable, Cloneable { return super.clone(); } catch (CloneNotSupportedException e) { // will never happen - return null; + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/text/RuleBasedBreakIterator.java b/jdk/src/share/classes/java/text/RuleBasedBreakIterator.java index c607d2c0b73..201eee71a13 100644 --- a/jdk/src/share/classes/java/text/RuleBasedBreakIterator.java +++ b/jdk/src/share/classes/java/text/RuleBasedBreakIterator.java @@ -453,7 +453,7 @@ class RuleBasedBreakIterator extends BreakIterator { ); } catch (PrivilegedActionException e) { - throw new InternalError(e.toString()); + throw new InternalError(e.toString(), e); } int offset = 0; diff --git a/jdk/src/share/classes/java/text/StringCharacterIterator.java b/jdk/src/share/classes/java/text/StringCharacterIterator.java index 4d10518ff28..6a24bb4e713 100644 --- a/jdk/src/share/classes/java/text/StringCharacterIterator.java +++ b/jdk/src/share/classes/java/text/StringCharacterIterator.java @@ -272,7 +272,7 @@ public final class StringCharacterIterator implements CharacterIterator return other; } catch (CloneNotSupportedException e) { - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/util/ArrayList.java b/jdk/src/share/classes/java/util/ArrayList.java index 14b0fbf5599..3b029272753 100644 --- a/jdk/src/share/classes/java/util/ArrayList.java +++ b/jdk/src/share/classes/java/util/ArrayList.java @@ -307,7 +307,7 @@ public class ArrayList extends AbstractList return v; } catch (CloneNotSupportedException e) { // this shouldn't happen, since we are Cloneable - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/util/BitSet.java b/jdk/src/share/classes/java/util/BitSet.java index fa0be5701b9..e8f242c83a3 100644 --- a/jdk/src/share/classes/java/util/BitSet.java +++ b/jdk/src/share/classes/java/util/BitSet.java @@ -1092,7 +1092,7 @@ public class BitSet implements Cloneable, java.io.Serializable { result.checkInvariants(); return result; } catch (CloneNotSupportedException e) { - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/util/Calendar.java b/jdk/src/share/classes/java/util/Calendar.java index 897a891cbe8..51ec6becb55 100644 --- a/jdk/src/share/classes/java/util/Calendar.java +++ b/jdk/src/share/classes/java/util/Calendar.java @@ -2512,7 +2512,7 @@ public abstract class Calendar implements Serializable, Cloneable, Comparable newSet.map = (HashMap) map.clone(); return newSet; } catch (CloneNotSupportedException e) { - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/util/Hashtable.java b/jdk/src/share/classes/java/util/Hashtable.java index 0ad5e5ad1f4..e1347168fc7 100644 --- a/jdk/src/share/classes/java/util/Hashtable.java +++ b/jdk/src/share/classes/java/util/Hashtable.java @@ -537,7 +537,7 @@ public class Hashtable return t; } catch (CloneNotSupportedException e) { // this shouldn't happen, since we are Cloneable - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/util/IdentityHashMap.java b/jdk/src/share/classes/java/util/IdentityHashMap.java index 95ff9f0a7c8..930c3ac108d 100644 --- a/jdk/src/share/classes/java/util/IdentityHashMap.java +++ b/jdk/src/share/classes/java/util/IdentityHashMap.java @@ -703,7 +703,7 @@ public class IdentityHashMap m.table = table.clone(); return m; } catch (CloneNotSupportedException e) { - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/util/LinkedList.java b/jdk/src/share/classes/java/util/LinkedList.java index 94508da57f4..fd8f181f17f 100644 --- a/jdk/src/share/classes/java/util/LinkedList.java +++ b/jdk/src/share/classes/java/util/LinkedList.java @@ -994,7 +994,7 @@ public class LinkedList try { return (LinkedList) super.clone(); } catch (CloneNotSupportedException e) { - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/util/Locale.java b/jdk/src/share/classes/java/util/Locale.java index 895d2d37f69..100144ebe36 100644 --- a/jdk/src/share/classes/java/util/Locale.java +++ b/jdk/src/share/classes/java/util/Locale.java @@ -1859,7 +1859,7 @@ public final class Locale implements Cloneable, Serializable { Locale that = (Locale)super.clone(); return that; } catch (CloneNotSupportedException e) { - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/util/ResourceBundle.java b/jdk/src/share/classes/java/util/ResourceBundle.java index 51b84877dc7..a4a4d55a1a5 100644 --- a/jdk/src/share/classes/java/util/ResourceBundle.java +++ b/jdk/src/share/classes/java/util/ResourceBundle.java @@ -614,7 +614,7 @@ public abstract class ResourceBundle { return clone; } catch (CloneNotSupportedException e) { //this should never happen - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/util/TimeZone.java b/jdk/src/share/classes/java/util/TimeZone.java index 266f627d7e2..2da2de2f42a 100644 --- a/jdk/src/share/classes/java/util/TimeZone.java +++ b/jdk/src/share/classes/java/util/TimeZone.java @@ -739,7 +739,7 @@ abstract public class TimeZone implements Serializable, Cloneable { other.ID = ID; return other; } catch (CloneNotSupportedException e) { - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/util/TreeMap.java b/jdk/src/share/classes/java/util/TreeMap.java index 6e92b8d1f78..c7aae652466 100644 --- a/jdk/src/share/classes/java/util/TreeMap.java +++ b/jdk/src/share/classes/java/util/TreeMap.java @@ -622,7 +622,7 @@ public class TreeMap try { clone = (TreeMap) super.clone(); } catch (CloneNotSupportedException e) { - throw new InternalError(); + throw new InternalError(e); } // Put clone into "virgin" state (except for comparator) diff --git a/jdk/src/share/classes/java/util/TreeSet.java b/jdk/src/share/classes/java/util/TreeSet.java index 3eee2a8e159..6b2a33d3b53 100644 --- a/jdk/src/share/classes/java/util/TreeSet.java +++ b/jdk/src/share/classes/java/util/TreeSet.java @@ -474,7 +474,7 @@ public class TreeSet extends AbstractSet try { clone = (TreeSet) super.clone(); } catch (CloneNotSupportedException e) { - throw new InternalError(); + throw new InternalError(e); } clone.m = new TreeMap<>(m); diff --git a/jdk/src/share/classes/java/util/UUID.java b/jdk/src/share/classes/java/util/UUID.java index c7948c99d11..448429849b9 100644 --- a/jdk/src/share/classes/java/util/UUID.java +++ b/jdk/src/share/classes/java/util/UUID.java @@ -165,7 +165,7 @@ public final class UUID implements java.io.Serializable, Comparable { try { md = MessageDigest.getInstance("MD5"); } catch (NoSuchAlgorithmException nsae) { - throw new InternalError("MD5 not supported"); + throw new InternalError("MD5 not supported", nsae); } byte[] md5Bytes = md.digest(name); md5Bytes[6] &= 0x0f; /* clear version */ diff --git a/jdk/src/share/classes/java/util/Vector.java b/jdk/src/share/classes/java/util/Vector.java index 0d69591abce..be74f617f5c 100644 --- a/jdk/src/share/classes/java/util/Vector.java +++ b/jdk/src/share/classes/java/util/Vector.java @@ -673,7 +673,7 @@ public class Vector return v; } catch (CloneNotSupportedException e) { // this shouldn't happen, since we are Cloneable - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/java/util/zip/ZipEntry.java b/jdk/src/share/classes/java/util/zip/ZipEntry.java index b2b25620ae8..8ade6d6bb30 100644 --- a/jdk/src/share/classes/java/util/zip/ZipEntry.java +++ b/jdk/src/share/classes/java/util/zip/ZipEntry.java @@ -321,7 +321,7 @@ class ZipEntry implements ZipConstants, Cloneable { return e; } catch (CloneNotSupportedException e) { // This should never happen, since we are Cloneable - throw new InternalError(); + throw new InternalError(e); } } } diff --git a/jdk/src/share/classes/javax/management/openmbean/TabularDataSupport.java b/jdk/src/share/classes/javax/management/openmbean/TabularDataSupport.java index cac5bf16c6c..d41baaea747 100644 --- a/jdk/src/share/classes/javax/management/openmbean/TabularDataSupport.java +++ b/jdk/src/share/classes/javax/management/openmbean/TabularDataSupport.java @@ -671,7 +671,7 @@ public class TabularDataSupport return c; } catch (CloneNotSupportedException e) { - throw new InternalError(e.toString()); + throw new InternalError(e.toString(), e); } } diff --git a/jdk/src/share/classes/javax/swing/AbstractButton.java b/jdk/src/share/classes/javax/swing/AbstractButton.java index 50eb66e9f8c..d95f1d4b602 100644 --- a/jdk/src/share/classes/javax/swing/AbstractButton.java +++ b/jdk/src/share/classes/javax/swing/AbstractButton.java @@ -981,6 +981,7 @@ public abstract class AbstractButton extends JComponent implements ItemSelectabl * @param exception the message to use in the * {@code IllegalArgumentException} that is thrown for an invalid * value + * @return the {@code key} argument * @exception IllegalArgumentException if key is not one of the legal * values listed above * @see #setHorizontalTextPosition @@ -1011,6 +1012,7 @@ public abstract class AbstractButton extends JComponent implements ItemSelectabl * @param exception the message to use in the * {@code IllegalArgumentException} that is thrown for an invalid * value + * @return the {@code key} argument * @exception IllegalArgumentException if key is not one of the legal * values listed above */ diff --git a/jdk/src/share/classes/javax/swing/plaf/basic/BasicFileChooserUI.java b/jdk/src/share/classes/javax/swing/plaf/basic/BasicFileChooserUI.java index 21623c13ea0..1f3d97edf9b 100644 --- a/jdk/src/share/classes/javax/swing/plaf/basic/BasicFileChooserUI.java +++ b/jdk/src/share/classes/javax/swing/plaf/basic/BasicFileChooserUI.java @@ -1153,10 +1153,15 @@ public class BasicFileChooserUI extends FileChooserUI { if (shellFolder.isLink()) { File linkedTo = shellFolder.getLinkLocation(); - if (linkedTo != null && fc.isTraversable(linkedTo)) { - dir = linkedTo; + // If linkedTo is null we try to use dir + if (linkedTo != null) { + if (fc.isTraversable(linkedTo)) { + dir = linkedTo; + } else { + return; + } } else { - return; + dir = shellFolder; } } } catch (FileNotFoundException ex) { diff --git a/jdk/src/share/classes/javax/swing/plaf/metal/MetalFileChooserUI.java b/jdk/src/share/classes/javax/swing/plaf/metal/MetalFileChooserUI.java index b3d38477bee..27c519502cf 100644 --- a/jdk/src/share/classes/javax/swing/plaf/metal/MetalFileChooserUI.java +++ b/jdk/src/share/classes/javax/swing/plaf/metal/MetalFileChooserUI.java @@ -461,16 +461,16 @@ public class MetalFileChooserUI extends BasicFileChooserUI { Locale l = fc.getLocale(); - lookInLabelMnemonic = UIManager.getInt("FileChooser.lookInLabelMnemonic"); + lookInLabelMnemonic = getMnemonic("FileChooser.lookInLabelMnemonic", l); lookInLabelText = UIManager.getString("FileChooser.lookInLabelText",l); saveInLabelText = UIManager.getString("FileChooser.saveInLabelText",l); - fileNameLabelMnemonic = UIManager.getInt("FileChooser.fileNameLabelMnemonic"); + fileNameLabelMnemonic = getMnemonic("FileChooser.fileNameLabelMnemonic", l); fileNameLabelText = UIManager.getString("FileChooser.fileNameLabelText",l); - folderNameLabelMnemonic = UIManager.getInt("FileChooser.folderNameLabelMnemonic"); + folderNameLabelMnemonic = getMnemonic("FileChooser.folderNameLabelMnemonic", l); folderNameLabelText = UIManager.getString("FileChooser.folderNameLabelText",l); - filesOfTypeLabelMnemonic = UIManager.getInt("FileChooser.filesOfTypeLabelMnemonic"); + filesOfTypeLabelMnemonic = getMnemonic("FileChooser.filesOfTypeLabelMnemonic", l); filesOfTypeLabelText = UIManager.getString("FileChooser.filesOfTypeLabelText",l); upFolderToolTipText = UIManager.getString("FileChooser.upFolderToolTipText",l); @@ -489,6 +489,10 @@ public class MetalFileChooserUI extends BasicFileChooserUI { detailsViewButtonAccessibleName = UIManager.getString("FileChooser.detailsViewButtonAccessibleName",l); } + private Integer getMnemonic(String key, Locale l) { + return SwingUtilities2.getUIDefaultsInt(key, l); + } + protected void installListeners(JFileChooser fc) { super.installListeners(fc); ActionMap actionMap = getActionMap(); diff --git a/jdk/src/share/classes/javax/swing/plaf/metal/MetalLookAndFeel.java b/jdk/src/share/classes/javax/swing/plaf/metal/MetalLookAndFeel.java index f06d6d77155..23debc1e664 100644 --- a/jdk/src/share/classes/javax/swing/plaf/metal/MetalLookAndFeel.java +++ b/jdk/src/share/classes/javax/swing/plaf/metal/MetalLookAndFeel.java @@ -844,9 +844,6 @@ public class MetalLookAndFeel extends BasicLookAndFeel "FileChooser.newFolderIcon", new SwingLazyValue("javax.swing.plaf.metal.MetalIconFactory", "getFileChooserNewFolderIcon"), "FileChooser.upFolderIcon", new SwingLazyValue("javax.swing.plaf.metal.MetalIconFactory", "getFileChooserUpFolderIcon"), - "FileChooser.lookInLabelMnemonic", new Integer(KeyEvent.VK_I), - "FileChooser.fileNameLabelMnemonic", new Integer(KeyEvent.VK_N), - "FileChooser.filesOfTypeLabelMnemonic", new Integer(KeyEvent.VK_T), "FileChooser.usesSingleFilePane", Boolean.TRUE, "FileChooser.ancestorInputMap", new UIDefaults.LazyInputMap(new Object[] { diff --git a/jdk/src/share/classes/javax/swing/text/ElementIterator.java b/jdk/src/share/classes/javax/swing/text/ElementIterator.java index 6bde7ec6eb5..6b749e07a97 100644 --- a/jdk/src/share/classes/javax/swing/text/ElementIterator.java +++ b/jdk/src/share/classes/javax/swing/text/ElementIterator.java @@ -157,7 +157,7 @@ public class ElementIterator implements Cloneable { } return it; } catch (CloneNotSupportedException e) { - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/javax/swing/text/rtf/RTFReader.java b/jdk/src/share/classes/javax/swing/text/rtf/RTFReader.java index 7eac94404a5..5fd48246be8 100644 --- a/jdk/src/share/classes/javax/swing/text/rtf/RTFReader.java +++ b/jdk/src/share/classes/javax/swing/text/rtf/RTFReader.java @@ -528,7 +528,7 @@ public void setCharacterSet(String name) try { translationTable = (char[])getCharacterSet("ansi"); } catch (IOException e) { - throw new InternalError("RTFReader: Unable to find character set resources (" + e + ")"); + throw new InternalError("RTFReader: Unable to find character set resources (" + e + ")", e); } } } @@ -1614,7 +1614,7 @@ class DocumentDestination } catch (BadLocationException ble) { /* This shouldn't be able to happen, of course */ /* TODO is InternalError the correct error to throw? */ - throw new InternalError(ble.getMessage()); + throw new InternalError(ble.getMessage(), ble); } } @@ -1628,7 +1628,7 @@ class DocumentDestination } catch (BadLocationException ble) { /* This shouldn't be able to happen, of course */ /* TODO is InternalError the correct error to throw? */ - throw new InternalError(ble.getMessage()); + throw new InternalError(ble.getMessage(), ble); } } diff --git a/jdk/src/share/classes/sun/awt/FontConfiguration.java b/jdk/src/share/classes/sun/awt/FontConfiguration.java index 231f6a46779..1744d084636 100644 --- a/jdk/src/share/classes/sun/awt/FontConfiguration.java +++ b/jdk/src/share/classes/sun/awt/FontConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2000, 2010, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1996, 2011, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -266,11 +266,20 @@ public abstract class FontConfiguration { private File findFontConfigFile(String javaLib) { String baseName = javaLib + File.separator + "fontconfig"; File configFile; + String osMajorVersion = null; if (osVersion != null && osName != null) { configFile = findImpl(baseName + "." + osName + "." + osVersion); if (configFile != null) { return configFile; } + int decimalPointIndex = osVersion.indexOf("."); + if (decimalPointIndex != -1) { + osMajorVersion = osVersion.substring(0, osVersion.indexOf(".")); + configFile = findImpl(baseName + "." + osName + "." + osMajorVersion); + if (configFile != null) { + return configFile; + } + } } if (osName != null) { configFile = findImpl(baseName + "." + osName); @@ -283,6 +292,12 @@ public abstract class FontConfiguration { if (configFile != null) { return configFile; } + if (osMajorVersion != null) { + configFile = findImpl(baseName + "." + osMajorVersion); + if (configFile != null) { + return configFile; + } + } } foundOsSpecificFile = false; diff --git a/jdk/src/share/classes/sun/dc/DuctusRenderingEngine.java b/jdk/src/share/classes/sun/dc/DuctusRenderingEngine.java index 51f398ab935..ae9340b94a6 100644 --- a/jdk/src/share/classes/sun/dc/DuctusRenderingEngine.java +++ b/jdk/src/share/classes/sun/dc/DuctusRenderingEngine.java @@ -158,7 +158,7 @@ public class DuctusRenderingEngine extends RenderingEngine { feedConsumer(pi, consumer, normalize, 0.25f); } catch (PathException e) { throw new InternalError("Unable to Stroke shape ("+ - e.getMessage()+")"); + e.getMessage()+")", e); } finally { while (consumer != null && consumer != sr) { PathConsumer next = consumer.getConsumer(); @@ -763,7 +763,7 @@ public class DuctusRenderingEngine extends RenderingEngine { consumer.endPath(); } catch (PathException e) { throw new InternalError("Unable to Stroke shape ("+ - e.getMessage()+")"); + e.getMessage()+")", e); } } diff --git a/jdk/src/share/classes/sun/font/FontLineMetrics.java b/jdk/src/share/classes/sun/font/FontLineMetrics.java index b2e3e5cbedc..57e1ffa279b 100644 --- a/jdk/src/share/classes/sun/font/FontLineMetrics.java +++ b/jdk/src/share/classes/sun/font/FontLineMetrics.java @@ -113,7 +113,7 @@ public final class FontLineMetrics extends LineMetrics implements Cloneable { return super.clone(); } catch (CloneNotSupportedException e) { - throw new InternalError(); + throw new InternalError(e); } } } diff --git a/jdk/src/share/classes/sun/font/GlyphLayout.java b/jdk/src/share/classes/sun/font/GlyphLayout.java index 03537b4d6a4..100b66b9809 100644 --- a/jdk/src/share/classes/sun/font/GlyphLayout.java +++ b/jdk/src/share/classes/sun/font/GlyphLayout.java @@ -233,7 +233,7 @@ public final class GlyphLayout { invdtx = dtx.createInverse(); } catch (NoninvertibleTransformException e) { - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/sun/invoke/util/ValueConversions.java b/jdk/src/share/classes/sun/invoke/util/ValueConversions.java index 9179785981b..cab961a0ff2 100644 --- a/jdk/src/share/classes/sun/invoke/util/ValueConversions.java +++ b/jdk/src/share/classes/sun/invoke/util/ValueConversions.java @@ -677,9 +677,7 @@ public class ValueConversions { EMPTY = IMPL_LOOKUP.findStatic(THIS_CLASS, "empty", ignoreType.dropParameterTypes(0, 1)); NEW_ARRAY = IMPL_LOOKUP.findStatic(THIS_CLASS, "newArray", MethodType.methodType(Object[].class, int.class)); } catch (NoSuchMethodException | IllegalAccessException ex) { - Error err = new InternalError("uncaught exception"); - err.initCause(ex); - throw err; + throw new InternalError("uncaught exception", ex); } } @@ -693,9 +691,7 @@ public class ValueConversions { COPY_AS_PRIMITIVE_ARRAY = IMPL_LOOKUP.findStatic(THIS_CLASS, "copyAsPrimitiveArray", MethodType.methodType(Object.class, Wrapper.class, Object[].class)); MAKE_LIST = IMPL_LOOKUP.findStatic(THIS_CLASS, "makeList", MethodType.methodType(List.class, Object[].class)); } catch (ReflectiveOperationException ex) { - Error err = new InternalError("uncaught exception"); - err.initCause(ex); - throw err; + throw new InternalError("uncaught exception", ex); } } } diff --git a/jdk/src/share/classes/sun/java2d/pipe/LoopPipe.java b/jdk/src/share/classes/sun/java2d/pipe/LoopPipe.java index 3df4adc893c..e975cc9c3cd 100644 --- a/jdk/src/share/classes/sun/java2d/pipe/LoopPipe.java +++ b/jdk/src/share/classes/sun/java2d/pipe/LoopPipe.java @@ -281,9 +281,8 @@ public class LoopPipe } catch (Throwable t) { sr.dispose(); sr = null; - t.printStackTrace(); throw new InternalError("Unable to Stroke shape ("+ - t.getMessage()+")"); + t.getMessage()+")", t); } return sr; } diff --git a/jdk/src/share/classes/sun/management/counter/perf/PerfDataEntry.java b/jdk/src/share/classes/sun/management/counter/perf/PerfDataEntry.java index 8a38b6e88af..0136d2e4b61 100644 --- a/jdk/src/share/classes/sun/management/counter/perf/PerfDataEntry.java +++ b/jdk/src/share/classes/sun/management/counter/perf/PerfDataEntry.java @@ -133,7 +133,7 @@ class PerfDataEntry { catch (UnsupportedEncodingException e) { // should not reach here // "UTF-8" is always a known encoding - throw new InternalError(e.getMessage()); + throw new InternalError(e.getMessage(), e); } if (variability == Variability.INVALID) { diff --git a/jdk/src/share/classes/sun/management/counter/perf/PerfDataType.java b/jdk/src/share/classes/sun/management/counter/perf/PerfDataType.java index aba2831948e..215003eab91 100644 --- a/jdk/src/share/classes/sun/management/counter/perf/PerfDataType.java +++ b/jdk/src/share/classes/sun/management/counter/perf/PerfDataType.java @@ -93,7 +93,7 @@ class PerfDataType { this.value = b[0]; } catch (UnsupportedEncodingException e) { // ignore, "UTF-8" is always a known encoding - throw new InternalError("Unknown encoding"); + throw new InternalError("Unknown encoding", e); } } } diff --git a/jdk/src/share/classes/sun/misc/Launcher.java b/jdk/src/share/classes/sun/misc/Launcher.java index 97d437e3efb..3d9b336edc2 100644 --- a/jdk/src/share/classes/sun/misc/Launcher.java +++ b/jdk/src/share/classes/sun/misc/Launcher.java @@ -71,7 +71,7 @@ public class Launcher { extcl = ExtClassLoader.getExtClassLoader(); } catch (IOException e) { throw new InternalError( - "Could not create extension class loader"); + "Could not create extension class loader", e); } // Now create the class loader to use to launch the application @@ -79,7 +79,7 @@ public class Launcher { loader = AppClassLoader.getAppClassLoader(extcl); } catch (IOException e) { throw new InternalError( - "Could not create application class loader"); + "Could not create application class loader", e); } // Also set the context class loader for the primordial thread. @@ -460,7 +460,7 @@ public class Launcher { return ParseUtil.fileToEncodedURL(file); } catch (MalformedURLException e) { // Should never happen since we specify the protocol... - throw new InternalError(); + throw new InternalError(e); } } @@ -475,15 +475,10 @@ public class Launcher { try { Class c = Class.forName(name); return (URLStreamHandler)c.newInstance(); - } catch (ClassNotFoundException e) { - e.printStackTrace(); - } catch (InstantiationException e) { - e.printStackTrace(); - } catch (IllegalAccessException e) { - e.printStackTrace(); + } catch (ReflectiveOperationException e) { + throw new InternalError("could not load " + protocol + + "system protocol handler", e); } - throw new InternalError("could not load " + protocol + - "system protocol handler"); } } } diff --git a/jdk/src/share/classes/sun/misc/ProxyGenerator.java b/jdk/src/share/classes/sun/misc/ProxyGenerator.java index 7e3cc91821f..c66f82ce8f1 100644 --- a/jdk/src/share/classes/sun/misc/ProxyGenerator.java +++ b/jdk/src/share/classes/sun/misc/ProxyGenerator.java @@ -460,7 +460,7 @@ public class ProxyGenerator { methods.add(generateStaticInitializer()); } catch (IOException e) { - throw new InternalError("unexpected I/O Exception"); + throw new InternalError("unexpected I/O Exception", e); } if (methods.size() > 65535) { @@ -540,7 +540,7 @@ public class ProxyGenerator { dout.writeShort(0); // (no ClassFile attributes for proxy classes) } catch (IOException e) { - throw new InternalError("unexpected I/O Exception"); + throw new InternalError("unexpected I/O Exception", e); } return bout.toByteArray(); diff --git a/jdk/src/share/classes/sun/net/NetworkClient.java b/jdk/src/share/classes/sun/net/NetworkClient.java index 89d49942c6a..8b7886ddf2d 100644 --- a/jdk/src/share/classes/sun/net/NetworkClient.java +++ b/jdk/src/share/classes/sun/net/NetworkClient.java @@ -139,7 +139,7 @@ public class NetworkClient { serverSocket.getOutputStream()), true, encoding); } catch (UnsupportedEncodingException e) { - throw new InternalError(encoding +"encoding not found"); + throw new InternalError(encoding +"encoding not found", e); } serverInput = new BufferedInputStream(serverSocket.getInputStream()); } diff --git a/jdk/src/share/classes/sun/net/NetworkServer.java b/jdk/src/share/classes/sun/net/NetworkServer.java index cb6d4c81693..fb54aa24546 100644 --- a/jdk/src/share/classes/sun/net/NetworkServer.java +++ b/jdk/src/share/classes/sun/net/NetworkServer.java @@ -142,7 +142,7 @@ public class NetworkServer implements Runnable, Cloneable { return super.clone(); } catch (CloneNotSupportedException e) { // this shouldn't happen, since we are Cloneable - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/sun/net/ftp/impl/FtpClient.java b/jdk/src/share/classes/sun/net/ftp/impl/FtpClient.java index 1406674c5a9..885d858b3a9 100644 --- a/jdk/src/share/classes/sun/net/ftp/impl/FtpClient.java +++ b/jdk/src/share/classes/sun/net/ftp/impl/FtpClient.java @@ -905,7 +905,7 @@ public class FtpClient extends sun.net.ftp.FtpClient { out = new PrintStream(new BufferedOutputStream(server.getOutputStream()), true, encoding); } catch (UnsupportedEncodingException e) { - throw new InternalError(encoding + "encoding not found"); + throw new InternalError(encoding + "encoding not found", e); } in = new BufferedInputStream(server.getInputStream()); } @@ -1621,7 +1621,7 @@ public class FtpClient extends sun.net.ftp.FtpClient { out = new PrintStream(new BufferedOutputStream(server.getOutputStream()), true, encoding); } catch (UnsupportedEncodingException e) { - throw new InternalError(encoding + "encoding not found"); + throw new InternalError(encoding + "encoding not found", e); } in = new BufferedInputStream(server.getInputStream()); } @@ -2056,7 +2056,7 @@ public class FtpClient extends sun.net.ftp.FtpClient { out = new PrintStream(new BufferedOutputStream(server.getOutputStream()), true, encoding); } catch (UnsupportedEncodingException e) { - throw new InternalError(encoding + "encoding not found"); + throw new InternalError(encoding + "encoding not found", e); } in = new BufferedInputStream(server.getInputStream()); @@ -2090,7 +2090,7 @@ public class FtpClient extends sun.net.ftp.FtpClient { out = new PrintStream(new BufferedOutputStream(server.getOutputStream()), true, encoding); } catch (UnsupportedEncodingException e) { - throw new InternalError(encoding + "encoding not found"); + throw new InternalError(encoding + "encoding not found", e); } in = new BufferedInputStream(server.getInputStream()); diff --git a/jdk/src/share/classes/sun/net/smtp/SmtpClient.java b/jdk/src/share/classes/sun/net/smtp/SmtpClient.java index d97b16affce..7886220d9b9 100644 --- a/jdk/src/share/classes/sun/net/smtp/SmtpClient.java +++ b/jdk/src/share/classes/sun/net/smtp/SmtpClient.java @@ -134,7 +134,7 @@ public class SmtpClient extends TransferProtocolClient { try { message = new SmtpPrintStream(serverOutput, this); } catch (UnsupportedEncodingException e) { - throw new InternalError(encoding+" encoding not found"); + throw new InternalError(encoding+" encoding not found", e); } return message; } diff --git a/jdk/src/share/classes/sun/net/www/http/ChunkedOutputStream.java b/jdk/src/share/classes/sun/net/www/http/ChunkedOutputStream.java index 2a0e35a86a0..dfe57c422a6 100644 --- a/jdk/src/share/classes/sun/net/www/http/ChunkedOutputStream.java +++ b/jdk/src/share/classes/sun/net/www/http/ChunkedOutputStream.java @@ -79,7 +79,7 @@ public class ChunkedOutputStream extends PrintStream { return header; } catch (java.io.UnsupportedEncodingException e) { /* This should never happen */ - throw new InternalError(e.getMessage()); + throw new InternalError(e.getMessage(), e); } } diff --git a/jdk/src/share/classes/sun/net/www/http/HttpClient.java b/jdk/src/share/classes/sun/net/www/http/HttpClient.java index 966fd821fb5..fa8184f5927 100644 --- a/jdk/src/share/classes/sun/net/www/http/HttpClient.java +++ b/jdk/src/share/classes/sun/net/www/http/HttpClient.java @@ -395,7 +395,7 @@ public class HttpClient extends NetworkClient { new BufferedOutputStream(out), false, encoding); } catch (UnsupportedEncodingException e) { - throw new InternalError(encoding+" encoding not found"); + throw new InternalError(encoding+" encoding not found", e); } serverSocket.setTcpNoDelay(true); } diff --git a/jdk/src/share/classes/sun/net/www/protocol/gopher/GopherClient.java b/jdk/src/share/classes/sun/net/www/protocol/gopher/GopherClient.java index dd19d40bfcf..41c6b8d2894 100644 --- a/jdk/src/share/classes/sun/net/www/protocol/gopher/GopherClient.java +++ b/jdk/src/share/classes/sun/net/www/protocol/gopher/GopherClient.java @@ -324,7 +324,7 @@ public class GopherClient extends NetworkClient implements Runnable { } } catch (UnsupportedEncodingException e) { - throw new InternalError(encoding+ " encoding not found"); + throw new InternalError(encoding+ " encoding not found", e); } catch (IOException e) { } finally { try { diff --git a/jdk/src/share/classes/sun/net/www/protocol/https/HttpsClient.java b/jdk/src/share/classes/sun/net/www/protocol/https/HttpsClient.java index 24618e89b61..cffc12c7260 100644 --- a/jdk/src/share/classes/sun/net/www/protocol/https/HttpsClient.java +++ b/jdk/src/share/classes/sun/net/www/protocol/https/HttpsClient.java @@ -529,7 +529,7 @@ final class HttpsClient extends HttpClient new BufferedOutputStream(serverSocket.getOutputStream()), false, encoding); } catch (UnsupportedEncodingException e) { - throw new InternalError(encoding+" encoding not found"); + throw new InternalError(encoding+" encoding not found", e); } // check URL spoofing if it has not been checked under handshaking diff --git a/jdk/src/share/classes/sun/nio/ch/Util.java b/jdk/src/share/classes/sun/nio/ch/Util.java index 514bb27091a..9aa7c162854 100644 --- a/jdk/src/share/classes/sun/nio/ch/Util.java +++ b/jdk/src/share/classes/sun/nio/ch/Util.java @@ -373,7 +373,7 @@ class Util { NoSuchMethodException | IllegalArgumentException | ClassCastException x) { - throw new InternalError(); + throw new InternalError(x); } return null; }}); @@ -395,7 +395,7 @@ class Util { } catch (InstantiationException | IllegalAccessException | InvocationTargetException e) { - throw new InternalError(); + throw new InternalError(e); } return dbb; } @@ -418,7 +418,7 @@ class Util { NoSuchMethodException | IllegalArgumentException | ClassCastException x) { - throw new InternalError(); + throw new InternalError(x); } return null; }}); @@ -440,7 +440,7 @@ class Util { } catch (InstantiationException | IllegalAccessException | InvocationTargetException e) { - throw new InternalError(); + throw new InternalError(e); } return dbb; } diff --git a/jdk/src/share/classes/sun/nio/cs/AbstractCharsetProvider.java b/jdk/src/share/classes/sun/nio/cs/AbstractCharsetProvider.java index f1714ea39c9..60d989654fe 100644 --- a/jdk/src/share/classes/sun/nio/cs/AbstractCharsetProvider.java +++ b/jdk/src/share/classes/sun/nio/cs/AbstractCharsetProvider.java @@ -179,7 +179,9 @@ public class AbstractCharsetProvider public Charset next() { String csn = i.next(); - return lookup(csn); + synchronized (AbstractCharsetProvider.this) { + return lookup(csn); + } } public void remove() { diff --git a/jdk/src/share/classes/sun/reflect/UTF8.java b/jdk/src/share/classes/sun/reflect/UTF8.java index 22f6c777f46..fa791eb4f1d 100644 --- a/jdk/src/share/classes/sun/reflect/UTF8.java +++ b/jdk/src/share/classes/sun/reflect/UTF8.java @@ -52,7 +52,7 @@ class UTF8 { } } catch (ArrayIndexOutOfBoundsException e) { throw new InternalError - ("Bug in sun.reflect bootstrap UTF-8 encoder"); + ("Bug in sun.reflect bootstrap UTF-8 encoder", e); } return res; } diff --git a/jdk/src/share/classes/sun/reflect/generics/reflectiveObjects/TypeVariableImpl.java b/jdk/src/share/classes/sun/reflect/generics/reflectiveObjects/TypeVariableImpl.java index 20f94755b0f..5eab5e8ed0b 100644 --- a/jdk/src/share/classes/sun/reflect/generics/reflectiveObjects/TypeVariableImpl.java +++ b/jdk/src/share/classes/sun/reflect/generics/reflectiveObjects/TypeVariableImpl.java @@ -25,9 +25,11 @@ package sun.reflect.generics.reflectiveObjects; +import java.lang.annotation.Annotation; import java.lang.reflect.GenericDeclaration; import java.lang.reflect.Type; import java.lang.reflect.TypeVariable; +import java.util.Objects; import sun.reflect.generics.factory.GenericsFactory; import sun.reflect.generics.tree.FieldTypeSignature; @@ -178,4 +180,27 @@ public class TypeVariableImpl public int hashCode() { return genericDeclaration.hashCode() ^ name.hashCode(); } + + // Currently vacuous implementations of AnnotatedElement methods. + public boolean isAnnotationPresent(Class annotationClass) { + Objects.requireNonNull(annotationClass); + return false; + } + + public T getAnnotation(Class annotationClass) { + Objects.requireNonNull(annotationClass); + return null; + } + + public Annotation[] getAnnotations() { + // Since zero-length, don't need defensive clone + return EMPTY_ANNOTATION_ARRAY; + } + + public Annotation[] getDeclaredAnnotations() { + // Since zero-length, don't need defensive clone + return EMPTY_ANNOTATION_ARRAY; + } + + private static final Annotation[] EMPTY_ANNOTATION_ARRAY = new Annotation[0]; } diff --git a/jdk/src/share/classes/sun/reflect/misc/MethodUtil.java b/jdk/src/share/classes/sun/reflect/misc/MethodUtil.java index 5784ced277f..0eade8de571 100644 --- a/jdk/src/share/classes/sun/reflect/misc/MethodUtil.java +++ b/jdk/src/share/classes/sun/reflect/misc/MethodUtil.java @@ -292,7 +292,7 @@ public final class MethodUtil extends SecureClassLoader { } }); } catch (Exception e) { - throw new InternalError("bouncer cannot be found"); + throw new InternalError("bouncer cannot be found", e); } } diff --git a/jdk/src/share/classes/sun/rmi/transport/LiveRef.java b/jdk/src/share/classes/sun/rmi/transport/LiveRef.java index 8643a060a5b..3cde2f96a52 100644 --- a/jdk/src/share/classes/sun/rmi/transport/LiveRef.java +++ b/jdk/src/share/classes/sun/rmi/transport/LiveRef.java @@ -111,7 +111,7 @@ public class LiveRef implements Cloneable { LiveRef newRef = (LiveRef) super.clone(); return newRef; } catch (CloneNotSupportedException e) { - throw new InternalError(e.toString()); + throw new InternalError(e.toString(), e); } } diff --git a/jdk/src/share/classes/sun/security/krb5/EncryptionKey.java b/jdk/src/share/classes/sun/security/krb5/EncryptionKey.java index 1de032945ff..ebf794e91cb 100644 --- a/jdk/src/share/classes/sun/security/krb5/EncryptionKey.java +++ b/jdk/src/share/classes/sun/security/krb5/EncryptionKey.java @@ -150,12 +150,37 @@ public class EncryptionKey return ktab.readServiceKeys(princ); } + /** + * Obtains a key for a given etype of a principal with possible new salt + * and s2kparams + * @param cname NOT null + * @param password NOT null + * @param etype + * @param snp can be NULL + * @returns never null + */ + public static EncryptionKey acquireSecretKey(PrincipalName cname, + char[] password, int etype, PAData.SaltAndParams snp) + throws KrbException { + String salt; + byte[] s2kparams; + if (snp != null) { + salt = snp.salt != null ? snp.salt : cname.getSalt(); + s2kparams = snp.params; + } else { + salt = cname.getSalt(); + s2kparams = null; + } + return acquireSecretKey(password, salt, etype, s2kparams); + } + /** * Obtains a key for a given etype with salt and optional s2kparams * @param password NOT null * @param salt NOT null * @param etype * @param s2kparams can be NULL + * @returns never null */ public static EncryptionKey acquireSecretKey(char[] password, String salt, int etype, byte[] s2kparams) diff --git a/jdk/src/share/classes/sun/security/krb5/KrbAsRep.java b/jdk/src/share/classes/sun/security/krb5/KrbAsRep.java index 29a7a5b4ccd..c2b0df30db4 100644 --- a/jdk/src/share/classes/sun/security/krb5/KrbAsRep.java +++ b/jdk/src/share/classes/sun/security/krb5/KrbAsRep.java @@ -131,13 +131,11 @@ class KrbAsRep extends KrbKdcRep { KrbAsReq asReq, PrincipalName cname) throws KrbException, Asn1Exception, IOException { int encPartKeyType = rep.encPart.getEType(); - PAData.SaltAndParams snp = - PAData.getSaltAndParams(encPartKeyType, rep.pAData); - EncryptionKey dkey = null; - dkey = EncryptionKey.acquireSecretKey(password, - snp.salt == null ? cname.getSalt() : snp.salt, + EncryptionKey dkey = EncryptionKey.acquireSecretKey( + cname, + password, encPartKeyType, - snp.params); + PAData.getSaltAndParams(encPartKeyType, rep.pAData)); decrypt(dkey, asReq); } diff --git a/jdk/src/share/classes/sun/security/krb5/KrbAsReqBuilder.java b/jdk/src/share/classes/sun/security/krb5/KrbAsReqBuilder.java index dd9b8dcfb92..3426d8b20b5 100644 --- a/jdk/src/share/classes/sun/security/krb5/KrbAsReqBuilder.java +++ b/jdk/src/share/classes/sun/security/krb5/KrbAsReqBuilder.java @@ -169,34 +169,44 @@ public final class KrbAsReqBuilder { * from a keytab on acceptor, but unfortunately (?) Java supports * acceptor using password. In this case, if the service ticket is * encrypted using an etype which we don't have PA-DATA new salt, - * using the default salt is normally wrong (say, case-insensitive + * using the default salt might be wrong (say, case-insensitive * user name). Instead, we would use the new salt of another etype. */ String salt = null; // the saved new salt - for (int i=0; i>>Pre-Authentication Data:"); - System.out.println("\t PA-DATA type = " + pa_type); - } - - switch(pa_type) { - case Krb5.PA_ENC_TIMESTAMP: - if (DEBUG) { - System.out.println("\t PA-ENC-TIMESTAMP"); - } - break; - case Krb5.PA_ETYPE_INFO: - if (pa_value != null) { - DerValue der = new DerValue(pa_value); - while (der.data.available() > 0) { - DerValue value = der.data.getDerValue(); - ETypeInfo info = new ETypeInfo(value); - if (pa_eType == 0) pa_eType = info.getEType(); - if (DEBUG) { - System.out.println("\t PA-ETYPE-INFO etype = " + info.getEType()); - System.out.println("\t PA-ETYPE-INFO salt = " + info.getSalt()); - } - } - } - break; - case Krb5.PA_ETYPE_INFO2: - if (pa_value != null) { - DerValue der = new DerValue(pa_value); - while (der.data.available() > 0) { - DerValue value = der.data.getDerValue(); - ETypeInfo2 info2 = new ETypeInfo2(value); - if (pa_eType == 0) pa_eType = info2.getEType(); - if (DEBUG) { - System.out.println("\t PA-ETYPE-INFO2 etype = " + info2.getEType()); - System.out.println("\t PA-ETYPE-INFO2 salt = " + info2.getSalt()); - } - } - } - break; - default: - // Unknown Pre-auth type - break; + System.out.println(pa_data); } } pa = paList.toArray(new PAData[paList.size()]); @@ -340,10 +297,6 @@ public class KRBError implements java.io.Serializable { return pa; } - public final int getEType() { - return pa_eType; - } - public final String getErrorString() { return eText; } diff --git a/jdk/src/share/classes/sun/security/krb5/internal/PAData.java b/jdk/src/share/classes/sun/security/krb5/internal/PAData.java index d32bbc47ee5..573b4f27029 100644 --- a/jdk/src/share/classes/sun/security/krb5/internal/PAData.java +++ b/jdk/src/share/classes/sun/security/krb5/internal/PAData.java @@ -138,10 +138,57 @@ public class PAData { return ((pADataValue == null) ? null : pADataValue.clone()); } + /** + * Gets the preferred etype from the PAData array. + * 1. ETYPE-INFO2-ENTRY with unknown s2kparams ignored + * 2. ETYPE-INFO2 preferred to ETYPE-INFO + * 3. multiple entries for same etype in one PA-DATA, use the first one. + * 4. Multiple PA-DATA with same type, choose the last one + * (This is useful when PA-DATAs from KRB-ERROR and AS-REP are combined). + * @return the etype, or defaultEType if not enough info + * @throws Asn1Exception|IOException if there is an encoding error + */ + public static int getPreferredEType(PAData[] pas, int defaultEType) + throws IOException, Asn1Exception { + + if (pas == null) return defaultEType; + + DerValue d = null, d2 = null; + for (PAData p: pas) { + if (p.getValue() == null) continue; + switch (p.getType()) { + case Krb5.PA_ETYPE_INFO: + d = new DerValue(p.getValue()); + break; + case Krb5.PA_ETYPE_INFO2: + d2 = new DerValue(p.getValue()); + break; + } + } + if (d2 != null) { + while (d2.data.available() > 0) { + DerValue value = d2.data.getDerValue(); + ETypeInfo2 tmp = new ETypeInfo2(value); + if (tmp.getParams() == null) { + // we don't support non-null s2kparams + return tmp.getEType(); + } + } + } + if (d != null) { + while (d.data.available() > 0) { + DerValue value = d.data.getDerValue(); + ETypeInfo tmp = new ETypeInfo(value); + return tmp.getEType(); + } + } + return defaultEType; + } + /** * A place to store a pair of salt and s2kparams. - * An empty salt is changed to null, to be interopable - * with Windows 2000 server. + * An empty salt is changed to null, to be interoperable + * with Windows 2000 server. This is in fact not correct. */ public static class SaltAndParams { public final String salt; @@ -155,57 +202,120 @@ public class PAData { /** * Fetches salt and s2kparams value for eType in a series of PA-DATAs. - * The preference order is PA-ETYPE-INFO2 > PA-ETYPE-INFO > PA-PW-SALT. - * If multiple PA-DATA for the same etype appears, use the last one. + * 1. ETYPE-INFO2-ENTRY with unknown s2kparams ignored + * 2. PA-ETYPE-INFO2 preferred to PA-ETYPE-INFO preferred to PA-PW-SALT. + * 3. multiple entries for same etype in one PA-DATA, use the first one. + * 4. Multiple PA-DATA with same type, choose the last one * (This is useful when PA-DATAs from KRB-ERROR and AS-REP are combined). - * @return salt and s2kparams. never null, its field might be null. + * @return salt and s2kparams. can be null if not found */ public static SaltAndParams getSaltAndParams(int eType, PAData[] pas) - throws Asn1Exception, KrbException { + throws Asn1Exception, IOException { - if (pas == null || pas.length == 0) { - return new SaltAndParams(null, null); - } + if (pas == null) return null; + DerValue d = null, d2 = null; String paPwSalt = null; - ETypeInfo2 info2 = null; - ETypeInfo info = null; for (PAData p: pas) { - if (p.getValue() != null) { - try { - switch (p.getType()) { - case Krb5.PA_PW_SALT: - paPwSalt = new String(p.getValue(), - KerberosString.MSNAME?"UTF8":"8859_1"); - break; - case Krb5.PA_ETYPE_INFO: - DerValue der = new DerValue(p.getValue()); - while (der.data.available() > 0) { - DerValue value = der.data.getDerValue(); - ETypeInfo tmp = new ETypeInfo(value); - if (tmp.getEType() == eType) info = tmp; - } - break; - case Krb5.PA_ETYPE_INFO2: - der = new DerValue(p.getValue()); - while (der.data.available() > 0) { - DerValue value = der.data.getDerValue(); - ETypeInfo2 tmp = new ETypeInfo2(value); - if (tmp.getEType() == eType) info2 = tmp; - } - break; - } - } catch (IOException ioe) { - // Ignored + if (p.getValue() == null) continue; + switch (p.getType()) { + case Krb5.PA_PW_SALT: + paPwSalt = new String(p.getValue(), + KerberosString.MSNAME?"UTF8":"8859_1"); + break; + case Krb5.PA_ETYPE_INFO: + d = new DerValue(p.getValue()); + break; + case Krb5.PA_ETYPE_INFO2: + d2 = new DerValue(p.getValue()); + break; + } + } + if (d2 != null) { + while (d2.data.available() > 0) { + DerValue value = d2.data.getDerValue(); + ETypeInfo2 tmp = new ETypeInfo2(value); + if (tmp.getParams() == null && tmp.getEType() == eType) { + // we don't support non-null s2kparams + return new SaltAndParams(tmp.getSalt(), tmp.getParams()); } } } - if (info2 != null) { - return new SaltAndParams(info2.getSalt(), info2.getParams()); - } else if (info != null) { - return new SaltAndParams(info.getSalt(), null); + if (d != null) { + while (d.data.available() > 0) { + DerValue value = d.data.getDerValue(); + ETypeInfo tmp = new ETypeInfo(value); + if (tmp.getEType() == eType) { + return new SaltAndParams(tmp.getSalt(), null); + } + } } - return new SaltAndParams(paPwSalt, null); + if (paPwSalt != null) { + return new SaltAndParams(paPwSalt, null); + } + return null; + } + + @Override + public String toString(){ + StringBuilder sb = new StringBuilder(); + sb.append(">>>Pre-Authentication Data:\n\t PA-DATA type = ") + .append(pADataType).append('\n'); + + switch(pADataType) { + case Krb5.PA_ENC_TIMESTAMP: + sb.append("\t PA-ENC-TIMESTAMP"); + break; + case Krb5.PA_ETYPE_INFO: + if (pADataValue != null) { + try { + DerValue der = new DerValue(pADataValue); + while (der.data.available() > 0) { + DerValue value = der.data.getDerValue(); + ETypeInfo info = new ETypeInfo(value); + sb.append("\t PA-ETYPE-INFO etype = ") + .append(info.getEType()) + .append(", salt = ") + .append(info.getSalt()) + .append('\n'); + } + } catch (IOException|Asn1Exception e) { + sb.append("\t \n"); + } + } + break; + case Krb5.PA_ETYPE_INFO2: + if (pADataValue != null) { + try { + DerValue der = new DerValue(pADataValue); + while (der.data.available() > 0) { + DerValue value = der.data.getDerValue(); + ETypeInfo2 info2 = new ETypeInfo2(value); + sb.append("\t PA-ETYPE-INFO2 etype = ") + .append(info2.getEType()) + .append(", salt = ") + .append(info2.getSalt()) + .append(", s2kparams = "); + byte[] s2kparams = info2.getParams(); + if (s2kparams == null) { + sb.append("null\n"); + } else if (s2kparams.length == 0) { + sb.append("empty\n"); + } else { + sb.append(new sun.misc.HexDumpEncoder() + .encodeBuffer(s2kparams)); + } + } + } catch (IOException|Asn1Exception e) { + sb.append("\t \n"); + } + } + break; + default: + // Unknown Pre-auth type + break; + } + return sb.toString(); } } diff --git a/jdk/src/share/classes/sun/security/provider/SecureRandom.java b/jdk/src/share/classes/sun/security/provider/SecureRandom.java index 6dd50d6b563..aee8846ed14 100644 --- a/jdk/src/share/classes/sun/security/provider/SecureRandom.java +++ b/jdk/src/share/classes/sun/security/provider/SecureRandom.java @@ -249,7 +249,7 @@ implements java.io.Serializable { try { digest = MessageDigest.getInstance ("SHA"); } catch (NoSuchAlgorithmException e) { - throw new InternalError("internal error: SHA-1 not available."); + throw new InternalError("internal error: SHA-1 not available.", e); } } } diff --git a/jdk/src/share/classes/sun/security/provider/SeedGenerator.java b/jdk/src/share/classes/sun/security/provider/SeedGenerator.java index efbfba933cd..8f6127a7ef0 100644 --- a/jdk/src/share/classes/sun/security/provider/SeedGenerator.java +++ b/jdk/src/share/classes/sun/security/provider/SeedGenerator.java @@ -151,7 +151,8 @@ abstract class SeedGenerator { try { md = MessageDigest.getInstance("SHA"); } catch (NoSuchAlgorithmException nsae) { - throw new InternalError("internal error: SHA-1 not available."); + throw new InternalError("internal error: SHA-1 not available." + , nsae); } // The current time in millis @@ -258,7 +259,8 @@ abstract class SeedGenerator { try { digest = MessageDigest.getInstance("SHA"); } catch (NoSuchAlgorithmException e) { - throw new InternalError("internal error: SHA-1 not available."); + throw new InternalError("internal error: SHA-1 not available." + , e); } final ThreadGroup[] finalsg = new ThreadGroup[1]; @@ -311,7 +313,8 @@ abstract class SeedGenerator { t.start(); } catch (Exception e) { throw new InternalError("internal error: " + - "SeedGenerator thread creation error."); + "SeedGenerator thread creation error." + , e); } // We wait 250milli quanta, so the minimum wait time @@ -344,7 +347,8 @@ abstract class SeedGenerator { } } catch (Exception e) { throw new InternalError("internal error: " + - "SeedGenerator thread generated an exception."); + "SeedGenerator thread generated an exception." + , e); } } @@ -367,7 +371,8 @@ abstract class SeedGenerator { } catch (Exception e) { if (count <= 0) throw new InternalError("internal error: " + - "SeedGenerator thread generated an exception."); + "SeedGenerator thread generated an exception." + ,e); } synchronized(this) { @@ -533,7 +538,7 @@ abstract class SeedGenerator { } catch (IOException ioe) { throw new InternalError("URLSeedGenerator " + deviceName + " generated exception: " + - ioe.getMessage()); + ioe.getMessage(), ioe); } } diff --git a/jdk/src/share/classes/sun/security/provider/certpath/ForwardState.java b/jdk/src/share/classes/sun/security/provider/certpath/ForwardState.java index 0645c926b58..60de90417ff 100644 --- a/jdk/src/share/classes/sun/security/provider/certpath/ForwardState.java +++ b/jdk/src/share/classes/sun/security/provider/certpath/ForwardState.java @@ -262,7 +262,7 @@ class ForwardState implements State { = (HashSet)subjectNamesTraversed.clone(); return clonedState; } catch (CloneNotSupportedException e) { - throw new InternalError(e.toString()); + throw new InternalError(e.toString(), e); } } } diff --git a/jdk/src/share/classes/sun/security/provider/certpath/ReverseState.java b/jdk/src/share/classes/sun/security/provider/certpath/ReverseState.java index 195def6fa2e..a5e8093d0d1 100644 --- a/jdk/src/share/classes/sun/security/provider/certpath/ReverseState.java +++ b/jdk/src/share/classes/sun/security/provider/certpath/ReverseState.java @@ -394,7 +394,7 @@ class ReverseState implements State { return clonedState; } catch (CloneNotSupportedException e) { - throw new InternalError(e.toString()); + throw new InternalError(e.toString(), e); } } } diff --git a/jdk/src/share/classes/sun/security/provider/certpath/URICertStore.java b/jdk/src/share/classes/sun/security/provider/certpath/URICertStore.java index 117685f6118..321f9153ec3 100644 --- a/jdk/src/share/classes/sun/security/provider/certpath/URICertStore.java +++ b/jdk/src/share/classes/sun/security/provider/certpath/URICertStore.java @@ -494,7 +494,7 @@ class URICertStore extends CertStoreSpi { return super.clone(); } catch (CloneNotSupportedException e) { /* Cannot happen */ - throw new InternalError(e.toString()); + throw new InternalError(e.toString(), e); } } } diff --git a/jdk/src/share/classes/sun/security/tools/JarSigner.java b/jdk/src/share/classes/sun/security/tools/JarSigner.java index 7707fa658a0..ac2ffde9e58 100644 --- a/jdk/src/share/classes/sun/security/tools/JarSigner.java +++ b/jdk/src/share/classes/sun/security/tools/JarSigner.java @@ -1506,6 +1506,9 @@ public class JarSigner { CertPath cp = certificateFactory.generateCertPath(certs); validator.validate(cp, pkixParameters); } catch (Exception e) { + if (debug) { + e.printStackTrace(); + } chainNotValidated = true; s.append(tab + rb.getString(".CertPath.not.validated.") + e.getLocalizedMessage() + "]\n"); // TODO @@ -1562,6 +1565,27 @@ public class JarSigner { } try { + + certificateFactory = CertificateFactory.getInstance("X.509"); + validator = CertPathValidator.getInstance("PKIX"); + Set tas = new HashSet<>(); + try { + KeyStore caks = KeyTool.getCacertsKeyStore(); + if (caks != null) { + Enumeration aliases = caks.aliases(); + while (aliases.hasMoreElements()) { + String a = aliases.nextElement(); + try { + tas.add(new TrustAnchor((X509Certificate)caks.getCertificate(a), null)); + } catch (Exception e2) { + // ignore, when a SecretkeyEntry does not include a cert + } + } + } + } catch (Exception e) { + // Ignore, if cacerts cannot be loaded + } + if (providerName == null) { store = KeyStore.getInstance(storetype); } else { @@ -1580,45 +1604,28 @@ public class JarSigner { (rb.getString("Enter.Passphrase.for.keystore.")); } - if (nullStream) { - store.load(null, storepass); - } else { - keyStoreName = keyStoreName.replace(File.separatorChar, '/'); - URL url = null; - try { - url = new URL(keyStoreName); - } catch (java.net.MalformedURLException e) { - // try as file - url = new File(keyStoreName).toURI().toURL(); - } - InputStream is = null; - try { - is = url.openStream(); - store.load(is, storepass); - } finally { - if (is != null) { - is.close(); - } - } - } - Set tas = new HashSet<>(); try { - KeyStore caks = KeyTool.getCacertsKeyStore(); - if (caks != null) { - Enumeration aliases = caks.aliases(); - while (aliases.hasMoreElements()) { - String a = aliases.nextElement(); - try { - tas.add(new TrustAnchor((X509Certificate)caks.getCertificate(a), null)); - } catch (Exception e2) { - // ignore, when a SecretkeyEntry does not include a cert + if (nullStream) { + store.load(null, storepass); + } else { + keyStoreName = keyStoreName.replace(File.separatorChar, '/'); + URL url = null; + try { + url = new URL(keyStoreName); + } catch (java.net.MalformedURLException e) { + // try as file + url = new File(keyStoreName).toURI().toURL(); + } + InputStream is = null; + try { + is = url.openStream(); + store.load(is, storepass); + } finally { + if (is != null) { + is.close(); } } } - } catch (Exception e) { - // Ignore, if cacerts cannot be loaded - } - if (store != null) { Enumeration aliases = store.aliases(); while (aliases.hasMoreElements()) { String a = aliases.nextElement(); @@ -1634,14 +1641,13 @@ public class JarSigner { // ignore, when a SecretkeyEntry does not include a cert } } - } - certificateFactory = CertificateFactory.getInstance("X.509"); - validator = CertPathValidator.getInstance("PKIX"); - try { - pkixParameters = new PKIXParameters(tas); - pkixParameters.setRevocationEnabled(false); - } catch (InvalidAlgorithmParameterException ex) { - // Only if tas is empty + } finally { + try { + pkixParameters = new PKIXParameters(tas); + pkixParameters.setRevocationEnabled(false); + } catch (InvalidAlgorithmParameterException ex) { + // Only if tas is empty + } } } catch (IOException ioe) { throw new RuntimeException(rb.getString("keystore.load.") + @@ -1805,6 +1811,9 @@ public class JarSigner { CertPath cp = certificateFactory.generateCertPath(Arrays.asList(certChain)); validator.validate(cp, pkixParameters); } catch (Exception e) { + if (debug) { + e.printStackTrace(); + } chainNotValidated = true; } diff --git a/jdk/src/share/classes/sun/security/tools/KeyTool.java b/jdk/src/share/classes/sun/security/tools/KeyTool.java index b3dcebdd87e..2e67bcea23b 100644 --- a/jdk/src/share/classes/sun/security/tools/KeyTool.java +++ b/jdk/src/share/classes/sun/security/tools/KeyTool.java @@ -1141,17 +1141,14 @@ public final class KeyTool { if (token) { keyStore.store(null, null); } else { - FileOutputStream fout = null; - try { - fout = (nullStream ? - (FileOutputStream)null : - new FileOutputStream(ksfname)); - keyStore.store - (fout, - (storePassNew!=null) ? storePassNew : storePass); - } finally { - if (fout != null) { - fout.close(); + char[] pass = (storePassNew!=null) ? storePassNew : storePass; + if (nullStream) { + keyStore.store(null, pass); + } else { + ByteArrayOutputStream bout = new ByteArrayOutputStream(); + keyStore.store(bout, pass); + try (FileOutputStream fout = new FileOutputStream(ksfname)) { + fout.write(bout.toByteArray()); } } } @@ -1399,7 +1396,7 @@ public final class KeyTool { private char[] promptForKeyPass(String alias, String orig, char[] origPass) throws Exception{ if (P12KEYSTORE.equalsIgnoreCase(storetype)) { return origPass; - } else if (!token) { + } else if (!token && !protectedPath) { // Prompt for key password int count; for (count = 0; count < 3; count++) { @@ -1446,7 +1443,7 @@ public final class KeyTool { } } } - return null; // PKCS11 + return null; // PKCS11, MSCAPI, or -protected } /** * Creates a new secret key. diff --git a/jdk/src/share/classes/sun/security/util/SecurityConstants.java b/jdk/src/share/classes/sun/security/util/SecurityConstants.java index f11a4869783..08bdaeb03fe 100644 --- a/jdk/src/share/classes/sun/security/util/SecurityConstants.java +++ b/jdk/src/share/classes/sun/security/util/SecurityConstants.java @@ -127,10 +127,8 @@ public final class SecurityConstants { // AWT present try { return (PermissionFactory)c.newInstance(); - } catch (InstantiationException x) { - throw new InternalError(x.getMessage()); - } catch (IllegalAccessException x) { - throw new InternalError(x.getMessage()); + } catch (ReflectiveOperationException x) { + throw new InternalError(x.getMessage(), x); } } else { // AWT not present diff --git a/jdk/src/share/classes/sun/text/CompactByteArray.java b/jdk/src/share/classes/sun/text/CompactByteArray.java index 3e42e3c3eb6..042d88ff271 100644 --- a/jdk/src/share/classes/sun/text/CompactByteArray.java +++ b/jdk/src/share/classes/sun/text/CompactByteArray.java @@ -269,7 +269,7 @@ public final class CompactByteArray implements Cloneable { if (hashes != null) other.hashes = (int[])hashes.clone(); return other; } catch (CloneNotSupportedException e) { - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/sun/text/normalizer/NormalizerBase.java b/jdk/src/share/classes/sun/text/normalizer/NormalizerBase.java index 5282294f214..576e67bd978 100644 --- a/jdk/src/share/classes/sun/text/normalizer/NormalizerBase.java +++ b/jdk/src/share/classes/sun/text/normalizer/NormalizerBase.java @@ -636,7 +636,7 @@ public final class NormalizerBase implements Cloneable { return copy; } catch (CloneNotSupportedException e) { - throw new InternalError(e.toString()); + throw new InternalError(e.toString(), e); } } diff --git a/jdk/src/share/classes/sun/tools/attach/HotSpotAttachProvider.java b/jdk/src/share/classes/sun/tools/attach/HotSpotAttachProvider.java index a2339fc6610..08535a1fff6 100644 --- a/jdk/src/share/classes/sun/tools/attach/HotSpotAttachProvider.java +++ b/jdk/src/share/classes/sun/tools/attach/HotSpotAttachProvider.java @@ -89,7 +89,7 @@ public abstract class HotSpotAttachProvider extends AttachProvider { if (t instanceof SecurityException) { return result; } - throw new InternalError(); // shouldn't happen + throw new InternalError(t); // shouldn't happen } for (Object vmid: vms) { diff --git a/jdk/src/share/classes/sun/tools/attach/HotSpotVirtualMachine.java b/jdk/src/share/classes/sun/tools/attach/HotSpotVirtualMachine.java index feb8628a355..2a217733adf 100644 --- a/jdk/src/share/classes/sun/tools/attach/HotSpotVirtualMachine.java +++ b/jdk/src/share/classes/sun/tools/attach/HotSpotVirtualMachine.java @@ -102,7 +102,7 @@ public abstract class HotSpotVirtualMachine extends VirtualMachine { try { loadAgentLibrary("instrument", args); } catch (AgentLoadException x) { - throw new InternalError("instrument library is missing in target VM"); + throw new InternalError("instrument library is missing in target VM", x); } catch (AgentInitializationException x) { /* * Translate interesting errors into the right exception and @@ -212,7 +212,7 @@ public abstract class HotSpotVirtualMachine extends VirtualMachine { try { return execute(cmd, args); } catch (AgentLoadException x) { - throw new InternalError("Should not get here"); + throw new InternalError("Should not get here", x); } } diff --git a/jdk/src/share/classes/sun/tools/jconsole/LocalVirtualMachine.java b/jdk/src/share/classes/sun/tools/jconsole/LocalVirtualMachine.java index e41ae7c5058..c99f99f8b13 100644 --- a/jdk/src/share/classes/sun/tools/jconsole/LocalVirtualMachine.java +++ b/jdk/src/share/classes/sun/tools/jconsole/LocalVirtualMachine.java @@ -135,10 +135,8 @@ public class LocalVirtualMachine { try { host = MonitoredHost.getMonitoredHost(new HostIdentifier((String)null)); vms = host.activeVms(); - } catch (java.net.URISyntaxException sx) { - throw new InternalError(sx.getMessage()); - } catch (MonitorException mx) { - throw new InternalError(mx.getMessage()); + } catch (java.net.URISyntaxException | MonitorException x) { + throw new InternalError(x.getMessage(), x); } for (Object vmid: vms) { if (vmid instanceof Integer) { diff --git a/jdk/src/share/classes/sun/tools/tree/Node.java b/jdk/src/share/classes/sun/tools/tree/Node.java index 44cc4e96400..da71ec5f49c 100644 --- a/jdk/src/share/classes/sun/tools/tree/Node.java +++ b/jdk/src/share/classes/sun/tools/tree/Node.java @@ -108,7 +108,7 @@ class Node implements Constants, Cloneable { return super.clone(); } catch (CloneNotSupportedException e) { // this shouldn't happen, since we are Cloneable - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/share/classes/sun/tracing/dtrace/DTraceProvider.java b/jdk/src/share/classes/sun/tracing/dtrace/DTraceProvider.java index 77cba5b2cc5..432199927ab 100644 --- a/jdk/src/share/classes/sun/tracing/dtrace/DTraceProvider.java +++ b/jdk/src/share/classes/sun/tracing/dtrace/DTraceProvider.java @@ -26,18 +26,15 @@ package sun.tracing.dtrace; import java.lang.reflect.Method; -import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Modifier; import java.lang.reflect.Constructor; import java.lang.reflect.InvocationHandler; import java.lang.reflect.InvocationTargetException; import java.lang.annotation.Annotation; -import java.util.HashMap; import sun.tracing.ProviderSkeleton; import sun.tracing.ProbeSkeleton; import com.sun.tracing.Provider; -import com.sun.tracing.ProviderName; import com.sun.tracing.ProbeName; import com.sun.tracing.dtrace.Attributes; import com.sun.tracing.dtrace.ModuleName; @@ -140,14 +137,8 @@ class DTraceProvider extends ProviderSkeleton { try { Constructor cons = proxyClass.getConstructor(constructorParams); return (T)cons.newInstance(new Object[] { this }); - } catch (NoSuchMethodException e) { - throw new InternalError(e.toString()); - } catch (IllegalAccessException e) { - throw new InternalError(e.toString()); - } catch (InstantiationException e) { - throw new InternalError(e.toString()); - } catch (InvocationTargetException e) { - throw new InternalError(e.toString()); + } catch (ReflectiveOperationException e) { + throw new InternalError(e.toString(), e); } } diff --git a/jdk/src/share/classes/sun/util/calendar/CalendarDate.java b/jdk/src/share/classes/sun/util/calendar/CalendarDate.java index 205415ec249..70c1bed6187 100644 --- a/jdk/src/share/classes/sun/util/calendar/CalendarDate.java +++ b/jdk/src/share/classes/sun/util/calendar/CalendarDate.java @@ -435,7 +435,7 @@ public abstract class CalendarDate implements Cloneable { return super.clone(); } catch (CloneNotSupportedException e) { // this shouldn't happen - throw new InternalError(); + throw new InternalError(e); } } diff --git a/jdk/src/solaris/classes/sun/awt/X11/XBaseMenuWindow.java b/jdk/src/solaris/classes/sun/awt/X11/XBaseMenuWindow.java index 68cf51251f7..56d7a14e961 100644 --- a/jdk/src/solaris/classes/sun/awt/X11/XBaseMenuWindow.java +++ b/jdk/src/solaris/classes/sun/awt/X11/XBaseMenuWindow.java @@ -157,7 +157,7 @@ abstract public class XBaseMenuWindow extends XWindow { try { return super.clone(); } catch (CloneNotSupportedException ex) { - throw new InternalError(); + throw new InternalError(ex); } } diff --git a/jdk/src/solaris/classes/sun/awt/X11/XMenuItemPeer.java b/jdk/src/solaris/classes/sun/awt/X11/XMenuItemPeer.java index 6f36f7ae326..c79fdc3bea3 100644 --- a/jdk/src/solaris/classes/sun/awt/X11/XMenuItemPeer.java +++ b/jdk/src/solaris/classes/sun/awt/X11/XMenuItemPeer.java @@ -131,7 +131,7 @@ public class XMenuItemPeer implements MenuItemPeer { try { return super.clone(); } catch (CloneNotSupportedException ex) { - throw new InternalError(); + throw new InternalError(ex); } } diff --git a/jdk/src/solaris/classes/sun/nio/ch/InheritedChannel.java b/jdk/src/solaris/classes/sun/nio/ch/InheritedChannel.java index e749c6af10b..ae7212a3cc3 100644 --- a/jdk/src/solaris/classes/sun/nio/ch/InheritedChannel.java +++ b/jdk/src/solaris/classes/sun/nio/ch/InheritedChannel.java @@ -64,7 +64,7 @@ class InheritedChannel { dup2(devnull, 2); } catch (IOException ioe) { // this shouldn't happen - throw new InternalError(); + throw new InternalError(ioe); } } diff --git a/jdk/src/solaris/classes/sun/tools/attach/LinuxVirtualMachine.java b/jdk/src/solaris/classes/sun/tools/attach/LinuxVirtualMachine.java index e2279befe34..632089e2fae 100644 --- a/jdk/src/solaris/classes/sun/tools/attach/LinuxVirtualMachine.java +++ b/jdk/src/solaris/classes/sun/tools/attach/LinuxVirtualMachine.java @@ -304,7 +304,7 @@ public class LinuxVirtualMachine extends HotSpotVirtualMachine { try { b = s.getBytes("UTF-8"); } catch (java.io.UnsupportedEncodingException x) { - throw new InternalError(); + throw new InternalError(x); } LinuxVirtualMachine.write(fd, b, 0, b.length); } diff --git a/jdk/src/solaris/native/sun/awt/fontpath.c b/jdk/src/solaris/native/sun/awt/fontpath.c index 98d6f1419fc..db6f8fa9539 100644 --- a/jdk/src/solaris/native/sun/awt/fontpath.c +++ b/jdk/src/solaris/native/sun/awt/fontpath.c @@ -1191,8 +1191,16 @@ Java_sun_font_FontConfigManager_getFontConfig fontformat = NULL; (*FcPatternGetString)(fontPattern, FC_FONTFORMAT, 0, &fontformat); - if (fontformat != NULL && strcmp((char*)fontformat, "TrueType") - != 0) { + /* We only want TrueType fonts but some Linuxes still depend + * on Type 1 fonts for some Locale support, so we'll allow + * them there. + */ + if (fontformat != NULL + && (strcmp((char*)fontformat, "TrueType") != 0) +#ifdef __linux__ + && (strcmp((char*)fontformat, "Type 1") != 0) +#endif + ) { continue; } result = (*FcPatternGetCharSet)(fontPattern, diff --git a/jdk/src/windows/native/com/sun/security/auth/module/nt.c b/jdk/src/windows/native/com/sun/security/auth/module/nt.c index 5a57f9756b2..64ef356e1dc 100644 --- a/jdk/src/windows/native/com/sun/security/auth/module/nt.c +++ b/jdk/src/windows/native/com/sun/security/auth/module/nt.c @@ -1,5 +1,5 @@ /* - * Copyright (c) 2000, 2004, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2000, 2011, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -43,6 +43,19 @@ BOOL getImpersonationToken(PHANDLE impersonationToken); BOOL getTextualSid(PSID pSid, LPTSTR TextualSid, LPDWORD lpdwBufferLen); void DisplayErrorText(DWORD dwLastError); +JNIEXPORT jlong JNICALL +Java_com_sun_security_auth_module_NTSystem_getImpersonationToken0 + (JNIEnv *env, jobject obj) { + HANDLE impersonationToken = 0; // impersonation token + if (debug) { + printf("getting impersonation token\n"); + } + if (getImpersonationToken(&impersonationToken) == FALSE) { + return 0; + } + return (jlong)impersonationToken; +} + JNIEXPORT void JNICALL Java_com_sun_security_auth_module_NTSystem_getCurrent (JNIEnv *env, jobject obj, jboolean debugNative) { @@ -59,7 +72,6 @@ Java_com_sun_security_auth_module_NTSystem_getCurrent DWORD numGroups = 0; // num groups LPTSTR *groups = NULL; // groups array long pIndex = -1; // index of primaryGroup in groups array - HANDLE impersonationToken = 0; // impersonation token jfieldID fid; jstring jstr; @@ -100,13 +112,6 @@ Java_com_sun_security_auth_module_NTSystem_getCurrent return; } - if (debug) { - printf("getting impersonation token\n"); - } - if (getImpersonationToken(&impersonationToken) == FALSE) { - return; - } - // then set values into NTSystem fid = (*env)->GetFieldID(env, cls, "userName", "Ljava/lang/String;"); @@ -233,18 +238,6 @@ Java_com_sun_security_auth_module_NTSystem_getCurrent (*env)->SetObjectField(env, obj, fid, jgroups); } - fid = (*env)->GetFieldID(env, cls, "impersonationToken", "J"); - if (fid == 0) { - jclass newExcCls = - (*env)->FindClass(env, "java/lang/IllegalArgumentException"); - if (newExcCls == 0) { - systemError = TRUE; - goto out; - } - (*env)->ThrowNew(env, newExcCls, "invalid field: impersonationToken"); - } - (*env)->SetLongField(env, obj, fid, (jlong)impersonationToken); - out: if (userName != NULL) { HeapFree(GetProcessHeap(), 0, userName); @@ -269,6 +262,7 @@ out: } HeapFree(GetProcessHeap(), 0, groups); } + CloseHandle(tokenHandle); if (systemError && debug) { printf(" [getCurrent] System Error: "); @@ -592,6 +586,7 @@ BOOL getImpersonationToken(PHANDLE impersonationToken) { } return FALSE; } + CloseHandle(dupToken); if (debug) { printf(" [getImpersonationToken] token = %d\n", *impersonationToken); @@ -802,6 +797,8 @@ void main(int argc, char *argv[]) { } HeapFree(GetProcessHeap(), 0, groups); } + CloseHandle(impersonationToken); + CloseHandle(tokenHandle); } */ diff --git a/jdk/src/windows/native/java/net/NetworkInterface.c b/jdk/src/windows/native/java/net/NetworkInterface.c index fa5dfc13f61..d6b04886975 100644 --- a/jdk/src/windows/native/java/net/NetworkInterface.c +++ b/jdk/src/windows/native/java/net/NetworkInterface.c @@ -504,8 +504,7 @@ jobject createNetworkInterface */ if (netaddrCount < 0) { netaddrCount = enumAddresses_win(env, ifs, &netaddrP); - if ((*env)->ExceptionOccurred(env)) { - free_netaddr(netaddrP); + if (netaddrCount == -1) { return NULL; } } diff --git a/jdk/src/windows/native/java/net/NetworkInterface_winXP.c b/jdk/src/windows/native/java/net/NetworkInterface_winXP.c index ecfab4967db..9b624cd5535 100644 --- a/jdk/src/windows/native/java/net/NetworkInterface_winXP.c +++ b/jdk/src/windows/native/java/net/NetworkInterface_winXP.c @@ -194,8 +194,7 @@ int getAllInterfacesAndAddresses (JNIEnv *env, netif **netifPP) while (curr != NULL) { netaddr *netaddrP; ret = enumAddresses_win(env, curr, &netaddrP); - if ((*env)->ExceptionOccurred(env)) { - free_netaddr(netaddrP); + if (ret == -1) { return -1; } curr->addrs = netaddrP; @@ -449,8 +448,7 @@ static jobject createNetworkInterfaceXP(JNIEnv *env, netif *ifs) */ if (netaddrCount < 0) { netaddrCount = enumAddresses_win(env, ifs, &netaddrP); - if ((*env)->ExceptionOccurred(env)) { - free_netaddr(netaddrP); + if (netaddrCount == -1) { return NULL; } } diff --git a/jdk/src/windows/native/java/net/TwoStacksPlainSocketImpl.c b/jdk/src/windows/native/java/net/TwoStacksPlainSocketImpl.c index 9c1ffee6c1a..4071b3f1729 100644 --- a/jdk/src/windows/native/java/net/TwoStacksPlainSocketImpl.c +++ b/jdk/src/windows/native/java/net/TwoStacksPlainSocketImpl.c @@ -576,6 +576,7 @@ Java_java_net_TwoStacksPlainSocketImpl_socketAccept(JNIEnv *env, jobject this, { /* fields on this */ jint port; + jint scope; jint timeout = (*env)->GetIntField(env, this, psi_timeoutID); jobject fdObj = (*env)->GetObjectField(env, this, psi_fdID); jobject fd1Obj = (*env)->GetObjectField(env, this, psi_fd1ID); @@ -755,7 +756,11 @@ Java_java_net_TwoStacksPlainSocketImpl_socketAccept(JNIEnv *env, jobject this, addr = (*env)->GetObjectField (env, socketAddressObj, ia6_ipaddressID); (*env)->SetByteArrayRegion (env, addr, 0, 16, (const char *)&him.him6.sin6_addr); (*env)->SetIntField(env, socketAddressObj, ia_familyID, IPv6); - (*env)->SetIntField(env, socketAddressObj, ia6_scopeidID, him.him6.sin6_scope_id); + scope = him.him6.sin6_scope_id; + (*env)->SetIntField(env, socketAddressObj, ia6_scopeidID, scope); + if(scope>0) { + (*env)->SetBooleanField(env, socketAddressObj, ia6_scopeidsetID, JNI_TRUE); + } } /* fields common to AF_INET and AF_INET6 */ diff --git a/jdk/src/windows/native/sun/windows/awt_FileDialog.cpp b/jdk/src/windows/native/sun/windows/awt_FileDialog.cpp index a29a1ba94e6..abd6e120451 100644 --- a/jdk/src/windows/native/sun/windows/awt_FileDialog.cpp +++ b/jdk/src/windows/native/sun/windows/awt_FileDialog.cpp @@ -153,6 +153,11 @@ FileDialogHookProc(HWND hdlg, UINT uiMsg, WPARAM wParam, LPARAM lParam) break; } case WM_DESTROY: { + HIMC hIMC = ::ImmGetContext(hdlg); + if (hIMC != NULL) { + ::ImmNotifyIME(hIMC, NI_COMPOSITIONSTR, CPS_CANCEL, 0); + } + WNDPROC lpfnWndProc = (WNDPROC)(::GetProp(parent, NativeDialogWndProcProp)); ComCtl32Util::GetInstance().UnsubclassHWND(parent, FileDialogWndProc, diff --git a/jdk/test/Makefile b/jdk/test/Makefile index 8ec0938b6cd..29d87130773 100644 --- a/jdk/test/Makefile +++ b/jdk/test/Makefile @@ -423,7 +423,8 @@ JDK_ALL_TARGETS = # Stable othervm testruns (minus items from PROBLEM_LIST) # Using samevm has problems, and doesn't help performance as much as others. JDK_ALL_TARGETS += jdk_awt -jdk_awt: $(call TestDirs, com/sun/awt java/awt sun/awt) +jdk_awt: $(call TestDirs, com/sun/awt java/awt sun/awt \ + javax/imageio javax/print sun/pisces) $(call RunOthervmBatch) # Stable samevm testruns (minus items from PROBLEM_LIST) @@ -486,9 +487,8 @@ jdk_math: $(call TestDirs, java/math) # Stable samevm testruns (minus items from PROBLEM_LIST) JDK_ALL_TARGETS += jdk_misc jdk_misc: $(call TestDirs, \ - demo javax/imageio javax/naming javax/print javax/script \ - javax/smartcardio javax/sound com/sun/java com/sun/jndi \ - com/sun/org com/sun/xml sun/misc sun/pisces) + demo/jvmti demo/zipfs javax/naming javax/script \ + javax/smartcardio com/sun/jndi com/sun/xml sun/misc) $(call RunSamevmBatch) # Stable samevm testruns (minus items from PROBLEM_LIST) @@ -532,17 +532,15 @@ JDK_ALL_TARGETS += jdk_security1 jdk_security1: $(call TestDirs, java/security) $(call RunSamevmBatch) -# Stable othervm testruns (minus items from PROBLEM_LIST) -# Using samevm has serious problems with these tests +# Stable samevm testruns (minus items from PROBLEM_LIST) JDK_ALL_TARGETS += jdk_security2 -jdk_security2: $(call TestDirs, javax/crypto com/sun/crypto) +jdk_security2: $(call TestDirs, javax/crypto javax/xml/crypto com/sun/crypto) $(call RunSamevmBatch) -# Stable othervm testruns (minus items from PROBLEM_LIST) -# Using samevm has serious problems with these tests +# Stable samevm testruns (minus items from PROBLEM_LIST) JDK_ALL_TARGETS += jdk_security3 -jdk_security3: $(call TestDirs, com/sun/security lib/security \ - javax/security sun/security) +jdk_security3: $(call TestDirs, com/sun/security lib/security javax/security \ + sun/security com/sun/org/apache/xml/internal/security) $(call SharedLibraryPermissions,sun/security) $(call RunSamevmBatch) @@ -550,10 +548,16 @@ jdk_security3: $(call TestDirs, com/sun/security lib/security \ jdk_security: jdk_security1 jdk_security2 jdk_security3 @$(SummaryInfo) +# Stable samevm testruns (minus items from PROBLEM_LIST) +JDK_ALL_TARGETS += jdk_sound +jdk_sound: $(call TestDirs, javax/sound) + $(call RunSamevmBatch) + # Stable othervm testruns (minus items from PROBLEM_LIST) # Using samevm has problems, and doesn't help performance as much as others. JDK_ALL_TARGETS += jdk_swing -jdk_swing: $(call TestDirs, javax/swing sun/java2d) +jdk_swing: $(call TestDirs, javax/swing sun/java2d \ + demo/jfc com/sun/java/swing) $(call RunOthervmBatch) # Stable samevm testruns (minus items from PROBLEM_LIST) @@ -802,17 +806,17 @@ endif # The jtjck.jar utility to use to run the tests JTJCK_JAR = $(JCK_HOME)/lib/jtjck.jar JTJCK_JAVA_ARGS = -XX:MaxPermSize=256m -Xmx512m -JTJCK_OPTIONS = -headless -v +JTJCK_OPTIONS = -headless -v # Default tests to run ifndef JCK_COMPILER_TESTS - JCK_COMPILER_TESTS = + JCK_COMPILER_TESTS = endif ifndef JCK_RUNTIME_TESTS - JCK_RUNTIME_TESTS = + JCK_RUNTIME_TESTS = endif ifndef JCK_DEVTOOLS_TESTS - JCK_DEVTOOLS_TESTS = + JCK_DEVTOOLS_TESTS = endif # Generic rule used to run jck tests @@ -838,14 +842,14 @@ jck7compiler: _generic_jck_tests # JCK7 runtime tests -jck7runtime: +jck7runtime: $(MAKE) UNIQUE_DIR=$@ \ JCK_HOME=$(JCK7RUNTIME_HOME) \ TESTDIRS="$(JCK_RUNTIME_TESTS)" \ _generic_jck_tests # JCK7 devtools tests -jck7devtools: +jck7devtools: $(MAKE) UNIQUE_DIR=$@ \ JCK_HOME=$(JCK7DEVTOOLS_HOME) \ TESTDIRS="$(JCK_DEVTOOLS_TESTS)" \ diff --git a/jdk/test/ProblemList.txt b/jdk/test/ProblemList.txt index 704e78e5b30..457be2cfa1f 100644 --- a/jdk/test/ProblemList.txt +++ b/jdk/test/ProblemList.txt @@ -526,6 +526,16 @@ sun/security/tools/jarsigner/ec.sh solaris-all # 7081817 sun/security/provider/certpath/X509CertPath/IllegalCertiticates.java generic-all +# 7041639, Solaris DSA keypair generation bug (Note: jdk_util also affected) +java/security/KeyPairGenerator/SolarisShortDSA.java solaris-all +sun/security/tools/jarsigner/onlymanifest.sh solaris-all +sun/security/tools/jarsigner/ts.sh solaris-all +sun/security/tools/keytool/emptysubject.sh solaris-all +sun/security/tools/keytool/importreadall.sh solaris-all +sun/security/tools/keytool/readjar.sh solaris-all +sun/security/tools/keytool/selfissued.sh solaris-all +sun/security/tools/keytool/standard.sh solaris-all + ############################################################################ # jdk_swing (not using samevm) @@ -586,5 +596,8 @@ java/util/zip/ZipFile/ReadLongZipFileName.java generic-all # Filed 6772009 java/util/concurrent/locks/ReentrantLock/CancelledLockLoops.java generic-all +# 7041639, Solaris DSA keypair generation bug +java/util/TimeZone/TimeZoneDatePermissionCheck.sh solaris-all + ############################################################################ diff --git a/jdk/test/com/sun/servicetag/JavaServiceTagTest.java b/jdk/test/com/sun/servicetag/JavaServiceTagTest.java index 0ebe0d5d895..6122b14d13c 100644 --- a/jdk/test/com/sun/servicetag/JavaServiceTagTest.java +++ b/jdk/test/com/sun/servicetag/JavaServiceTagTest.java @@ -25,7 +25,7 @@ /* * @test - * @bug 6622366 + * @bug 6622366 7078024 * @summary Basic Test for ServiceTag.getJavaServiceTag() * Disable creating the service tag in the system registry. * Verify the existence of registration.xml file and the @@ -86,23 +86,42 @@ public class JavaServiceTagTest { } } + /** + * Tests if the running platform is a JDK. + */ + static boolean isJDK() { + // Determine the JRE path by checking the existence of + // /jre/lib and /lib. + String javaHome = System.getProperty("java.home"); + String jrepath = javaHome + File.separator + "jre"; + File f = new File(jrepath, "lib"); + if (!f.exists()) { + // java.home usually points to the JRE path + jrepath = javaHome; + } + + return jrepath.endsWith(File.separator + "jre"); + } + private static void checkServiceTag(ServiceTag st) throws IOException { - Properties props = loadSwordfishEntries(); - if (st.getProductURN(). - equals(props.getProperty("servicetag.jdk.urn"))) { - if (!st.getProductName(). - equals(props.getProperty("servicetag.jdk.name"))) { - throw new RuntimeException("Product URN and name don't match."); - } - } else if (st.getProductURN(). - equals(props.getProperty("servicetag.jre.urn"))) { - if (!st.getProductName(). - equals(props.getProperty("servicetag.jre.name"))) { + Properties props = loadServiceTagProps(); + // jdk 8 and later, JDK and JRE have the same product URN. + String jdkUrn = props.getProperty("servicetag.jdk.urn"); + String jreUrn = props.getProperty("servicetag.jre.urn"); + boolean isJdk = isJDK(); + + if (isJdk) { + if (!st.getProductURN().equals(jdkUrn) || + !st.getProductName().equals( + props.getProperty("servicetag.jdk.name"))) { throw new RuntimeException("Product URN and name don't match."); } } else { - throw new RuntimeException("Unexpected product_urn: " + - st.getProductURN()); + if (!st.getProductURN().equals(jreUrn) || + !st.getProductName().equals( + props.getProperty("servicetag.jre.name"))) { + throw new RuntimeException("Product URN and name don't match."); + } } if (!st.getProductVersion(). equals(System.getProperty("java.version"))) { @@ -160,18 +179,13 @@ public class JavaServiceTagTest { } } - private static Properties loadSwordfishEntries() + private static Properties loadServiceTagProps() throws IOException { - int version = sun.misc.Version.jdkMinorVersion(); - String filename = "/com/sun/servicetag/resources/javase_" + - version + "_swordfish.properties"; - InputStream in = Installer.class.getClass().getResourceAsStream(filename); - Properties props = new Properties(); - try { + String filename = "/com/sun/servicetag/resources/javase_servicetag.properties"; + try (InputStream in = Installer.class.getClass().getResourceAsStream(filename)) { + Properties props = new Properties(); props.load(in); - } finally { - in.close(); + return props; } - return props; } } diff --git a/jdk/test/com/sun/servicetag/JavaServiceTagTest1.java b/jdk/test/com/sun/servicetag/JavaServiceTagTest1.java index 64e8b318aeb..ffe3744ccd3 100644 --- a/jdk/test/com/sun/servicetag/JavaServiceTagTest1.java +++ b/jdk/test/com/sun/servicetag/JavaServiceTagTest1.java @@ -25,7 +25,7 @@ /* * @test - * @bug 6622366 + * @bug 6622366 7078024 * @summary Basic Test for ServiceTag.getJavaServiceTag(String) * to verify that the registration.xml and servicetag files * are both created correctly. @@ -157,25 +157,45 @@ public class JavaServiceTagTest1 { return svctag; } + /** + * Tests if the running platform is a JDK. + */ + static boolean isJDK() { + // Determine the JRE path by checking the existence of + // /jre/lib and /lib. + String javaHome = System.getProperty("java.home"); + String jrepath = javaHome + File.separator + "jre"; + File f = new File(jrepath, "lib"); + if (!f.exists()) { + // java.home usually points to the JRE path + jrepath = javaHome; + } + + return jrepath.endsWith(File.separator + "jre"); + } + private static void checkServiceTag(ServiceTag st, String source) throws IOException { - Properties props = loadSwordfishEntries(); - if (st.getProductURN(). - equals(props.getProperty("servicetag.jdk.urn"))) { - if (!st.getProductName(). - equals(props.getProperty("servicetag.jdk.name"))) { - throw new RuntimeException("Product URN and name don't match."); - } - } else if (st.getProductURN(). - equals(props.getProperty("servicetag.jre.urn"))) { - if (!st.getProductName(). - equals(props.getProperty("servicetag.jre.name"))) { + Properties props = loadServiceTagProps(); + // jdk 8 and later, JDK and JRE have the same product URN. + String jdkUrn = props.getProperty("servicetag.jdk.urn"); + String jreUrn = props.getProperty("servicetag.jre.urn"); + boolean isJdk = isJDK(); + + if (isJdk) { + if (!st.getProductURN().equals(jdkUrn) || + !st.getProductName().equals( + props.getProperty("servicetag.jdk.name"))) { throw new RuntimeException("Product URN and name don't match."); } } else { - throw new RuntimeException("Unexpected product_urn: " + - st.getProductURN()); + if (!st.getProductURN().equals(jreUrn) || + !st.getProductName().equals( + props.getProperty("servicetag.jre.name"))) { + throw new RuntimeException("Product URN and name don't match."); + } } + if (!st.getProductVersion(). equals(System.getProperty("java.version"))) { throw new RuntimeException("Unexpected product_version: " + @@ -233,18 +253,13 @@ public class JavaServiceTagTest1 { } } - private static Properties loadSwordfishEntries() + private static Properties loadServiceTagProps() throws IOException { - int version = sun.misc.Version.jdkMinorVersion(); - String filename = "/com/sun/servicetag/resources/javase_" + - version + "_swordfish.properties"; - InputStream in = Installer.class.getClass().getResourceAsStream(filename); - Properties props = new Properties(); - try { + String filename = "/com/sun/servicetag/resources/javase_servicetag.properties"; + try (InputStream in = Installer.class.getClass().getResourceAsStream(filename)) { + Properties props = new Properties(); props.load(in); - } finally { - in.close(); + return props; } - return props; } } diff --git a/jdk/test/java/lang/invoke/InvokeGenericTest.java b/jdk/test/java/lang/invoke/InvokeGenericTest.java index fecd2f03a55..87393b4055f 100644 --- a/jdk/test/java/lang/invoke/InvokeGenericTest.java +++ b/jdk/test/java/lang/invoke/InvokeGenericTest.java @@ -25,7 +25,7 @@ /* @test * @summary unit tests for java.lang.invoke.MethodHandle.invoke - * @compile -target 7 InvokeGenericTest.java + * @compile InvokeGenericTest.java * @run junit/othervm test.java.lang.invoke.InvokeGenericTest */ diff --git a/jdk/test/java/lang/reflect/TypeVariable/TestAnnotatedElement.java b/jdk/test/java/lang/reflect/TypeVariable/TestAnnotatedElement.java new file mode 100644 index 00000000000..94f8d459cf0 --- /dev/null +++ b/jdk/test/java/lang/reflect/TypeVariable/TestAnnotatedElement.java @@ -0,0 +1,88 @@ +/* + * Copyright (c) 2011, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @bug 7086192 + * @summary Verify functionality of AnnotatedElement methods on type variables + * @author Joseph D. Darcy + */ + +import java.lang.reflect.*; +import java.lang.annotation.*; + +public class TestAnnotatedElement { + // Type variable on a method + private static B m(B b) {return null;} + + // Type variable on a construtor + private TestAnnotatedElement(){super();} + + public static void main(String... argv) throws ReflectiveOperationException { + int errors = 0; + + Class clazz = TestAnnotatedElement.class; + errors += testTypeVariable(clazz.getTypeParameters()); + errors += testTypeVariable(clazz.getDeclaredConstructor().getTypeParameters()); + errors += testTypeVariable(clazz.getDeclaredMethod("m", Object.class).getTypeParameters()); + + if (errors > 0) + throw new RuntimeException(errors + " failures"); + } + + + private static int testTypeVariable(TypeVariable[] typeVars) { + int errors = 0; + if (typeVars.length == 0) + return ++errors; + + for(TypeVariable typeVar : typeVars) { + try { + typeVar.getAnnotation(null); + errors++; + } catch(NullPointerException npe) { + ; // Expected + } + + if (typeVar.getAnnotation(SuppressWarnings.class) != null) + errors++; + + try { + typeVar.isAnnotationPresent(null); + errors++; + } catch(NullPointerException npe) { + ; // Expected + } + + if (typeVar.isAnnotationPresent(SuppressWarnings.class)) + errors++; + + if(typeVar.getAnnotations().length != 0) + errors++; + + if(typeVar.getDeclaredAnnotations().length != 0) + errors++; + } + return errors; + } +} diff --git a/jdk/test/java/math/BigDecimal/DivideMcTests.java b/jdk/test/java/math/BigDecimal/DivideMcTests.java index 3bf1da89a6d..13c7fc093a9 100644 --- a/jdk/test/java/math/BigDecimal/DivideMcTests.java +++ b/jdk/test/java/math/BigDecimal/DivideMcTests.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2011, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2003, 2011, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it diff --git a/jdk/test/java/math/BigDecimal/FloatDoubleValueTests.java b/jdk/test/java/math/BigDecimal/FloatDoubleValueTests.java index ade80fe51c4..f8f3f9a7c55 100644 --- a/jdk/test/java/math/BigDecimal/FloatDoubleValueTests.java +++ b/jdk/test/java/math/BigDecimal/FloatDoubleValueTests.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2005, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2005, 2011 Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -23,7 +23,7 @@ /* * @test - * @bug 6274390 + * @bug 6274390 7082971 * @summary Verify {float, double}Value methods work with condensed representation * @run main FloatDoubleValueTests * @run main/othervm -XX:+AggressiveOpts FloatDoubleValueTests @@ -79,6 +79,7 @@ public class FloatDoubleValueTests { // and double. static void testFloatDoubleValue() { long longValues[] = { + Long.MIN_VALUE, // -2^63 0, 1, 2, diff --git a/jdk/test/java/math/BigDecimal/RangeTests.java b/jdk/test/java/math/BigDecimal/RangeTests.java index cef3ef538c7..bcbf7025e0e 100644 --- a/jdk/test/java/math/BigDecimal/RangeTests.java +++ b/jdk/test/java/math/BigDecimal/RangeTests.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2011, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2003, 2011, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it diff --git a/jdk/test/java/math/BigDecimal/StrippingZerosTest.java b/jdk/test/java/math/BigDecimal/StrippingZerosTest.java index 30885e66574..c76d0ca99fa 100644 --- a/jdk/test/java/math/BigDecimal/StrippingZerosTest.java +++ b/jdk/test/java/math/BigDecimal/StrippingZerosTest.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2003, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2003, 2011 Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it diff --git a/jdk/test/java/math/BigDecimal/ToPlainStringTests.java b/jdk/test/java/math/BigDecimal/ToPlainStringTests.java index df3bc6ae9d5..bc42b0cf416 100644 --- a/jdk/test/java/math/BigDecimal/ToPlainStringTests.java +++ b/jdk/test/java/math/BigDecimal/ToPlainStringTests.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2004, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2004, 2011, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it diff --git a/jdk/test/java/net/Socket/ShutdownInput.java b/jdk/test/java/net/Socket/ShutdownInput.java new file mode 100644 index 00000000000..6754b4ec889 --- /dev/null +++ b/jdk/test/java/net/Socket/ShutdownInput.java @@ -0,0 +1,89 @@ +/* + * Copyright (c) 2011, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @bug 7014860 + * @summary Socket.getInputStream().available() not clear for + * case that connection is shutdown for reading + */ + +import java.io.InputStream; +import java.io.OutputStream; +import java.net.InetAddress; +import java.net.InetSocketAddress; +import java.net.ServerSocket; +import java.net.Socket; +import java.nio.channels.ServerSocketChannel; +import java.nio.channels.SocketChannel; + +public class ShutdownInput { + static boolean failed = false; + + public static void main(String args[]) throws Exception { + InetAddress iaddr = InetAddress.getLocalHost(); + + try ( ServerSocket ss = new ServerSocket(0); + Socket s1 = new Socket(iaddr, ss.getLocalPort()); + Socket s2 = ss.accept() ) { + + test(s1, s2, "Testing NET"); + } + + // check the NIO socket adapter + try (ServerSocketChannel sc = ServerSocketChannel.open().bind(null); + SocketChannel s1 = SocketChannel.open( + new InetSocketAddress(iaddr, sc.socket().getLocalPort())); + SocketChannel s2 = sc.accept() ) { + + test(s1.socket(), s2.socket(), "Testing NIO"); + } + + if (failed) { + throw new RuntimeException("Failed: check output"); + } + } + + public static void test(Socket s1, Socket s2, String mesg) throws Exception { + OutputStream os = s1.getOutputStream(); + os.write("This is a message".getBytes("US-ASCII")); + + InputStream in = s2.getInputStream(); + s2.shutdownInput(); + + if (in.available() != 0) { + failed = true; + System.out.println(mesg + ":" + s2 + " in.available() should be 0, " + + "but returns "+ in.available()); + } + + byte[] ba = new byte[2]; + if (in.read() != -1 || + in.read(ba) != -1 || + in.read(ba, 0, ba.length) != -1) { + + failed = true; + System.out.append(mesg + ":" + s2 + " in.read() should be -1"); + } + } +} diff --git a/jdk/test/java/net/URI/Test.java b/jdk/test/java/net/URI/Test.java index fb0fa332629..b24bedf19f3 100644 --- a/jdk/test/java/net/URI/Test.java +++ b/jdk/test/java/net/URI/Test.java @@ -23,7 +23,7 @@ /* @test * @summary Unit test for java.net.URI - * @bug 4464135 4505046 4503239 4438319 4991359 4866303 7023363 + * @bug 4464135 4505046 4503239 4438319 4991359 4866303 7023363 7041800 * @author Mark Reinhold */ @@ -1428,6 +1428,8 @@ public class Test { gt(s, new URI("http://jag:CafeBabe@java.sun.com:94/b/c/d?q#f")); lt(s, new URI("http://jag:cafebabe@java.sun.com:94/b/c/d?r#f")); lt(s, new URI("http://jag:cafebabe@java.sun.com:94/b/c/d?q#g")); + eq(new URI("http://host/a%00bcd"), new URI("http://host/a%00bcd")); + ne(new URI("http://host/a%00bcd"), new URI("http://host/aZ00bcd")); lt("p", "s:p"); lt("s:p", "T:p"); diff --git a/jdk/test/java/security/KeyPairGenerator/SolarisShortDSA.java b/jdk/test/java/security/KeyPairGenerator/SolarisShortDSA.java new file mode 100644 index 00000000000..69ed492ed54 --- /dev/null +++ b/jdk/test/java/security/KeyPairGenerator/SolarisShortDSA.java @@ -0,0 +1,69 @@ +/* + * Copyright (c) 2011, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @bug 7081411 + * @summary DSA keypair generation affected by Solaris bug + */ + +import java.security.KeyPair; +import java.security.KeyPairGenerator; +import java.security.Signature; +import sun.security.provider.DSAPrivateKey; + +public class SolarisShortDSA { + static byte[] data = new byte[0]; + public static void main(String args[]) throws Exception { + for (int i=0; i<10000; i++) { + KeyPairGenerator kpg = KeyPairGenerator.getInstance("DSA"); + KeyPair kp = kpg.generateKeyPair(); + DSAPrivateKey dpk = (DSAPrivateKey)kp.getPrivate(); + int len = dpk.getX().bitLength(); + if (len <= 152) { + if (!use(kp)) { + String os = System.getProperty("os.name"); + // Solaris bug, update the following line once it's fixed + if (os.equals("SunOS")) { + throw new IllegalStateException( + "Don't panic. This is a Solaris bug"); + } else { + throw new RuntimeException("Real test failure"); + } + } + break; + } + } + } + + static boolean use(KeyPair kp) throws Exception { + Signature sig = Signature.getInstance("SHA1withDSA"); + sig.initSign(kp.getPrivate()); + sig.update(data); + byte[] signed = sig.sign(); + Signature sig2 = Signature.getInstance("SHA1withDSA"); + sig2.initVerify(kp.getPublic()); + sig2.update(data); + return sig2.verify(signed); + } +} diff --git a/jdk/test/javax/xml/crypto/dsig/SecurityManager/XMLDSigWithSecMgr.java b/jdk/test/javax/xml/crypto/dsig/SecurityManager/XMLDSigWithSecMgr.java index c72fbaff3a4..bcbf96e771b 100644 --- a/jdk/test/javax/xml/crypto/dsig/SecurityManager/XMLDSigWithSecMgr.java +++ b/jdk/test/javax/xml/crypto/dsig/SecurityManager/XMLDSigWithSecMgr.java @@ -26,6 +26,7 @@ * @bug 6436919 6460930 * @summary check that XML Signatures can be generated and validated with * SecurityManager enabled and default policy + * @run main/othervm XMLDSigWithSecMgr * @author Sean Mullan */ import java.io.*; diff --git a/jdk/test/lib/security/java.policy/Ext_AllPolicy.sh b/jdk/test/lib/security/java.policy/Ext_AllPolicy.sh index 53498667632..c10d38344d8 100644 --- a/jdk/test/lib/security/java.policy/Ext_AllPolicy.sh +++ b/jdk/test/lib/security/java.policy/Ext_AllPolicy.sh @@ -54,19 +54,16 @@ case "$OS" in NULL=/dev/null PS=":" FS="/" - TMP=/tmp ;; CYGWIN* ) NULL=/dev/null PS=";" FS="/" - TMP=/tmp ;; Windows_95 | Windows_98 | Windows_NT ) NULL=NUL PS=";" FS="\\" - TMP="c:/temp" ;; * ) echo "Unrecognized system!" diff --git a/jdk/test/sun/security/krb5/auto/DupEtypes.java b/jdk/test/sun/security/krb5/auto/DupEtypes.java new file mode 100644 index 00000000000..92da74bc304 --- /dev/null +++ b/jdk/test/sun/security/krb5/auto/DupEtypes.java @@ -0,0 +1,63 @@ +/* + * Copyright (c) 2011, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @bug 7067974 + * @summary multiple ETYPE-INFO-ENTRY with same etype and different salt + * @compile -XDignore.symbol.file DupEtypes.java + * @run main/othervm DupEtypes 1 + * @run main/othervm DupEtypes 2 + * @run main/othervm/fail DupEtypes 3 + * @run main/othervm DupEtypes 4 + * @run main/othervm DupEtypes 5 + */ + +import sun.security.jgss.GSSUtil; + +public class DupEtypes { + + public static void main(String[] args) throws Exception { + + OneKDC kdc = new OneKDC(null); + kdc.writeJAASConf(); + + // Different test cases, read KDC.processAsReq for details + kdc.setOption(KDC.Option.DUP_ETYPE, Integer.parseInt(args[0])); + + Context c, s; + c = Context.fromJAAS("client"); + s = Context.fromJAAS("server"); + + c.startAsClient(OneKDC.SERVER, GSSUtil.GSS_KRB5_MECH_OID); + s.startAsServer(GSSUtil.GSS_KRB5_MECH_OID); + + Context.handshake(c, s); + + Context.transmit("i say high --", c, s); + Context.transmit(" you say low", s, c); + + s.dispose(); + c.dispose(); + } +} diff --git a/jdk/test/sun/security/krb5/auto/KDC.java b/jdk/test/sun/security/krb5/auto/KDC.java index e4b4a9d8b9b..9d924f6e8d4 100644 --- a/jdk/test/sun/security/krb5/auto/KDC.java +++ b/jdk/test/sun/security/krb5/auto/KDC.java @@ -174,6 +174,10 @@ public class KDC { * Set all name-type to a value in response */ RESP_NT, + /** + * Multiple ETYPE-INFO-ENTRY with same etype but different salt + */ + DUP_ETYPE, }; static { @@ -881,48 +885,104 @@ public class KDC { bFlags[Krb5.TKT_OPTS_INITIAL] = true; // Creating PA-DATA - int[] epas = eTypes; - if (options.containsKey(KDC.Option.RC4_FIRST_PREAUTH)) { - for (int i=1; i ${TMP}${FS}temp_file_40875602475 2> ${NULL} -grep MissingResourceException ${TMP}${FS}temp_file_40875602475 +${TESTJAVA}${FS}bin${FS}keytool > temp_file_40875602475 2> ${NULL} +grep MissingResourceException temp_file_40875602475 if [ $? -eq 0 ]; then - rm ${TMP}${FS}temp_file_40875602475 exit 1 fi -rm ${TMP}${FS}temp_file_40875602475 exit 0 diff --git a/jdk/test/sun/security/tools/keytool/trystore.sh b/jdk/test/sun/security/tools/keytool/trystore.sh new file mode 100644 index 00000000000..ffe4ebe73e1 --- /dev/null +++ b/jdk/test/sun/security/tools/keytool/trystore.sh @@ -0,0 +1,65 @@ +# +# Copyright (c) 2011, Oracle and/or its affiliates. All rights reserved. +# DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. +# +# This code is free software; you can redistribute it and/or modify it +# under the terms of the GNU General Public License version 2 only, as +# published by the Free Software Foundation. +# +# This code is distributed in the hope that it will be useful, but WITHOUT +# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or +# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +# version 2 for more details (a copy is included in the LICENSE file that +# accompanied this code). +# +# You should have received a copy of the GNU General Public License version +# 2 along with this work; if not, write to the Free Software Foundation, +# Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. +# +# Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA +# or visit www.oracle.com if you need additional information or have any +# questions. +# + +# @test +# @bug 7047200 +# @summary keytool can try save to a byte array before overwrite the file + +if [ "${TESTJAVA}" = "" ] ; then + JAVAC_CMD=`which javac` + TESTJAVA=`dirname $JAVAC_CMD`/.. +fi + +# set platform-dependent variables +OS=`uname -s` +case "$OS" in + Windows_* ) + FS="\\" + ;; + * ) + FS="/" + ;; +esac + +rm trystore.jks 2> /dev/null + +KEYTOOL="${TESTJAVA}${FS}bin${FS}keytool -storetype jks -keystore trystore.jks" +$KEYTOOL -genkeypair -alias a -dname CN=A -storepass changeit -keypass changeit +$KEYTOOL -genkeypair -alias b -dname CN=B -storepass changeit -keypass changeit + +# We use -protected for JKS keystore. This is illegal so the command should +# fail. Then we can check if the keystore is damaged. + +$KEYTOOL -genkeypair -protected -alias b -delete -debug + +if [ $? = 0 ]; then + echo "What? -protected works for JKS?" + exit 1 +fi + +$KEYTOOL -list -storepass changeit + +if [ $? != 0 ]; then + echo "Keystore file damaged" + exit 2 +fi diff --git a/jdk/test/sun/security/tools/policytool/Alias.sh b/jdk/test/sun/security/tools/policytool/Alias.sh index 10f40ba19fc..cbc0a6893a1 100644 --- a/jdk/test/sun/security/tools/policytool/Alias.sh +++ b/jdk/test/sun/security/tools/policytool/Alias.sh @@ -51,13 +51,11 @@ case "$OS" in NULL=/dev/null PS=":" FS="/" - TMP=/tmp ;; Windows* ) NULL=NUL PS=";" FS="\\" - TMP="c:/temp" ;; * ) echo "Unrecognized system!" diff --git a/jdk/test/sun/security/tools/policytool/ChangeUI.sh b/jdk/test/sun/security/tools/policytool/ChangeUI.sh index 9b5e2929ffc..ce1eb0ebb98 100644 --- a/jdk/test/sun/security/tools/policytool/ChangeUI.sh +++ b/jdk/test/sun/security/tools/policytool/ChangeUI.sh @@ -50,13 +50,11 @@ case "$OS" in NULL=/dev/null PS=":" FS="/" - TMP=/tmp ;; Windows* ) NULL=NUL PS=";" FS="\\" - TMP="c:/temp" ;; * ) echo "Unrecognized system!" diff --git a/jdk/test/sun/security/tools/policytool/OpenPolicy.sh b/jdk/test/sun/security/tools/policytool/OpenPolicy.sh index e8c9fe79828..7a3016d69aa 100644 --- a/jdk/test/sun/security/tools/policytool/OpenPolicy.sh +++ b/jdk/test/sun/security/tools/policytool/OpenPolicy.sh @@ -50,13 +50,11 @@ case "$OS" in NULL=/dev/null PS=":" FS="/" - TMP=/tmp ;; Windows* ) NULL=NUL PS=";" FS="\\" - TMP="c:/temp" ;; * ) echo "Unrecognized system!" diff --git a/jdk/test/sun/security/tools/policytool/SaveAs.sh b/jdk/test/sun/security/tools/policytool/SaveAs.sh index c06be089d81..32715716847 100644 --- a/jdk/test/sun/security/tools/policytool/SaveAs.sh +++ b/jdk/test/sun/security/tools/policytool/SaveAs.sh @@ -51,13 +51,11 @@ case "$OS" in NULL=/dev/null PS=":" FS="/" - TMP=/tmp ;; Windows* ) NULL=NUL PS=";" FS="\\" - TMP="c:/temp" ;; * ) echo "Unrecognized system!" diff --git a/jdk/test/sun/security/tools/policytool/UpdatePermissions.sh b/jdk/test/sun/security/tools/policytool/UpdatePermissions.sh index 32d94ba5548..4e2f2297c14 100644 --- a/jdk/test/sun/security/tools/policytool/UpdatePermissions.sh +++ b/jdk/test/sun/security/tools/policytool/UpdatePermissions.sh @@ -50,13 +50,11 @@ case "$OS" in NULL=/dev/null PS=":" FS="/" - TMP=/tmp ;; Windows* ) NULL=NUL PS=";" FS="\\" - TMP="c:/temp" ;; * ) echo "Unrecognized system!" diff --git a/jdk/test/sun/security/tools/policytool/UsePolicy.sh b/jdk/test/sun/security/tools/policytool/UsePolicy.sh index 74dec2e7ee1..4b946709de3 100644 --- a/jdk/test/sun/security/tools/policytool/UsePolicy.sh +++ b/jdk/test/sun/security/tools/policytool/UsePolicy.sh @@ -50,13 +50,11 @@ case "$OS" in NULL=/dev/null PS=":" FS="/" - TMP=/tmp ;; Windows* ) NULL=NUL PS=";" FS="\\" - TMP="c:/temp" ;; * ) echo "Unrecognized system!" diff --git a/jdk/test/sun/security/tools/policytool/i18n.sh b/jdk/test/sun/security/tools/policytool/i18n.sh index 1584239f194..7b5570f2e8d 100644 --- a/jdk/test/sun/security/tools/policytool/i18n.sh +++ b/jdk/test/sun/security/tools/policytool/i18n.sh @@ -50,13 +50,11 @@ case "$OS" in NULL=/dev/null PS=":" FS="/" - TMP=/tmp ;; Windows* ) NULL=NUL PS=";" FS="\\" - TMP="c:/temp" ;; * ) echo "Unrecognized system!"