/hg/release/icedtea7-forest-2.0/jdk: 11 new changesets
andrew at icedtea.classpath.org
andrew at icedtea.classpath.org
Tue Feb 14 16:11:50 PST 2012
changeset d063a88899f9 in /hg/release/icedtea7-forest-2.0/jdk
details: http://icedtea.classpath.org/hg/release/icedtea7-forest-2.0/jdk?cmd=changeset;node=d063a88899f9
author: dl
date: Thu Oct 13 11:44:40 2011 +0100
7082299: AtomicReferenceArray should ensure that array is Object[]
Reviewed-by: chegar, dholmes, alanb
changeset 962ddd5ea2d9 in /hg/release/icedtea7-forest-2.0/jdk
details: http://icedtea.classpath.org/hg/release/icedtea7-forest-2.0/jdk?cmd=changeset;node=962ddd5ea2d9
author: amenkov
date: Wed Oct 26 13:35:15 2011 +0400
7088367: JavaSound security issue (12865443) Reviewed-by: denis
changeset 4fe44cde3384 in /hg/release/icedtea7-forest-2.0/jdk
details: http://icedtea.classpath.org/hg/release/icedtea7-forest-2.0/jdk?cmd=changeset;node=4fe44cde3384
author: ant
date: Wed Oct 26 14:29:43 2011 +0400
6205776: Some KeyboardFocusManager methods are unsafe Summary: a
security check is added Reviewed-by: bagiras
changeset 1d04b88d06f2 in /hg/release/icedtea7-forest-2.0/jdk
details: http://icedtea.classpath.org/hg/release/icedtea7-forest-2.0/jdk?cmd=changeset;node=1d04b88d06f2
author: ant
date: Fri Dec 02 10:44:11 2011 +0400
7116384: backout the unallowed changes in the
KeyboardFocusManager.java javadoc Reviewed-by: art, bagiras
changeset c90dba86a584 in /hg/release/icedtea7-forest-2.0/jdk
details: http://icedtea.classpath.org/hg/release/icedtea7-forest-2.0/jdk?cmd=changeset;node=c90dba86a584
author: okutsu
date: Thu Oct 20 16:55:56 2011 +0900
6351654: (tz) java.util.TimeZone.setDefault() should be controlled
by a security manager Reviewed-by: hawtin
changeset 760fdca6137f in /hg/release/icedtea7-forest-2.0/jdk
details: http://icedtea.classpath.org/hg/release/icedtea7-forest-2.0/jdk?cmd=changeset;node=760fdca6137f
author: dmeetry
date: Wed Nov 23 17:58:19 2011 +0400
7110700: Enhance exception throwing mechanism in ObjectStreamClass
Reviewed-by: hawtin
changeset f32cfb330272 in /hg/release/icedtea7-forest-2.0/jdk
details: http://icedtea.classpath.org/hg/release/icedtea7-forest-2.0/jdk?cmd=changeset;node=f32cfb330272
author: bagiras
date: Wed Nov 30 13:39:17 2011 -0800
7112642: Incorrect checking for graphics rendering object Reviewed-
by: art, bae, flar, prr
changeset a1a3620c1530 in /hg/release/icedtea7-forest-2.0/jdk
details: http://icedtea.classpath.org/hg/release/icedtea7-forest-2.0/jdk?cmd=changeset;node=a1a3620c1530
author: sherman
date: Mon Dec 05 21:01:03 2011 -0800
7118283: Better input parameter checking in zip file processing
Summary: Fixed off-by-one bug in zip_util.c Reviewed-by: asaha
changeset 43bf4d62a772 in /hg/release/icedtea7-forest-2.0/jdk
details: http://icedtea.classpath.org/hg/release/icedtea7-forest-2.0/jdk?cmd=changeset;node=43bf4d62a772
author: chegar
date: Sun Feb 05 15:35:59 2012 +0000
7126960: Add property to limit number of request headers to the HTTP
Server Reviewed-by: alanb, michaelm, darcy
changeset cfe9f71da9f4 in /hg/release/icedtea7-forest-2.0/jdk
details: http://icedtea.classpath.org/hg/release/icedtea7-forest-2.0/jdk?cmd=changeset;node=cfe9f71da9f4
author: andrew
date: Tue Feb 14 14:01:11 2012 +0000
Merge
changeset 891fb0050add in /hg/release/icedtea7-forest-2.0/jdk
details: http://icedtea.classpath.org/hg/release/icedtea7-forest-2.0/jdk?cmd=changeset;node=891fb0050add
author: andrew
date: Tue Feb 14 14:01:44 2012 +0000
Added tag icedtea-2.0.1 for changeset cfe9f71da9f4
diffstat:
.hgtags | 1 +
make/sun/xawt/mapfile-vers | 2 +
src/share/classes/com/sun/media/sound/DirectAudioDevice.java | 4 +-
src/share/classes/com/sun/media/sound/SoftMixingSourceDataLine.java | 6 +
src/share/classes/java/awt/KeyboardFocusManager.java | 78 +--
src/share/classes/java/io/ObjectStreamClass.java | 63 +-
src/share/classes/java/util/TimeZone.java | 51 ++-
src/share/classes/java/util/concurrent/atomic/AtomicReferenceArray.java | 43 +-
src/share/classes/sun/java2d/SunGraphics2D.java | 13 +-
src/share/classes/sun/java2d/opengl/OGLRenderer.java | 24 +-
src/share/classes/sun/java2d/pipe/BufferedContext.java | 8 +-
src/share/classes/sun/net/httpserver/Request.java | 7 +
src/share/classes/sun/net/httpserver/ServerConfig.java | 14 +-
src/share/native/java/util/zip/zip_util.c | 2 +-
src/solaris/classes/sun/awt/X11/XWindowPeer.java | 7 +
src/solaris/native/sun/xawt/XToolkit.c | 37 +
src/windows/classes/sun/java2d/d3d/D3DRenderer.java | 24 +-
src/windows/classes/sun/java2d/windows/GDIRenderer.java | 200 ++++++---
src/windows/native/sun/java2d/windows/GDIRenderer.cpp | 28 +-
19 files changed, 431 insertions(+), 181 deletions(-)
diffs (truncated from 1421 to 500 lines):
diff -r 52052ed420c5 -r 891fb0050add .hgtags
--- a/.hgtags Wed Oct 19 06:44:39 2011 +0100
+++ b/.hgtags Tue Feb 14 14:01:44 2012 +0000
@@ -127,3 +127,4 @@
f097ca2434b1412b12ab4a5c2397ce271bf681e7 jdk7-b147
7ec1845521edfb1843cad3868217983727ece53d icedtea-2.0-branchpoint
2054526dd141375f8d85961813a12a722cd98976 icedtea-2.0
+cfe9f71da9f48f0dd3e2c151e9d9e28e9d6b2bf5 icedtea-2.0.1
diff -r 52052ed420c5 -r 891fb0050add make/sun/xawt/mapfile-vers
--- a/make/sun/xawt/mapfile-vers Wed Oct 19 06:44:39 2011 +0100
+++ b/make/sun/xawt/mapfile-vers Tue Feb 14 14:01:44 2012 +0000
@@ -322,6 +322,8 @@
Java_sun_awt_X11_XlibWrapper_XSynchronize;
Java_java_awt_FileDialog_initIDs;
Java_sun_awt_X11_XWindow_initIDs;
+ Java_sun_awt_X11_XWindowPeer_getLocalHostname;
+ Java_sun_awt_X11_XWindowPeer_getJvmPID;
Java_sun_java2d_opengl_OGLContext_getOGLIdString;
Java_sun_java2d_opengl_OGLMaskFill_maskFill;
diff -r 52052ed420c5 -r 891fb0050add src/share/classes/com/sun/media/sound/DirectAudioDevice.java
--- a/src/share/classes/com/sun/media/sound/DirectAudioDevice.java Wed Oct 19 06:44:39 2011 +0100
+++ b/src/share/classes/com/sun/media/sound/DirectAudioDevice.java Tue Feb 14 14:01:44 2012 +0000
@@ -736,7 +736,7 @@
if (off < 0) {
throw new ArrayIndexOutOfBoundsException(off);
}
- if (off + len > b.length) {
+ if ((long)off + (long)len > (long)b.length) {
throw new ArrayIndexOutOfBoundsException(b.length);
}
@@ -964,7 +964,7 @@
if (off < 0) {
throw new ArrayIndexOutOfBoundsException(off);
}
- if (off + len > b.length) {
+ if ((long)off + (long)len > (long)b.length) {
throw new ArrayIndexOutOfBoundsException(b.length);
}
if (!isActive() && doIO) {
diff -r 52052ed420c5 -r 891fb0050add src/share/classes/com/sun/media/sound/SoftMixingSourceDataLine.java
--- a/src/share/classes/com/sun/media/sound/SoftMixingSourceDataLine.java Wed Oct 19 06:44:39 2011 +0100
+++ b/src/share/classes/com/sun/media/sound/SoftMixingSourceDataLine.java Tue Feb 14 14:01:44 2012 +0000
@@ -130,6 +130,12 @@
if (len % framesize != 0)
throw new IllegalArgumentException(
"Number of bytes does not represent an integral number of sample frames.");
+ if (off < 0) {
+ throw new ArrayIndexOutOfBoundsException(off);
+ }
+ if ((long)off + (long)len > (long)b.length) {
+ throw new ArrayIndexOutOfBoundsException(b.length);
+ }
byte[] buff = cycling_buffer;
int buff_len = cycling_buffer.length;
diff -r 52052ed420c5 -r 891fb0050add src/share/classes/java/awt/KeyboardFocusManager.java
--- a/src/share/classes/java/awt/KeyboardFocusManager.java Wed Oct 19 06:44:39 2011 +0100
+++ b/src/share/classes/java/awt/KeyboardFocusManager.java Tue Feb 14 14:01:44 2012 +0000
@@ -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
@@ -503,14 +503,8 @@
*/
protected Component getGlobalFocusOwner() throws SecurityException {
synchronized (KeyboardFocusManager.class) {
- if (this == getCurrentKeyboardFocusManager()) {
- return focusOwner;
- } else {
- if (focusLog.isLoggable(PlatformLogger.FINER)) {
- focusLog.finer("This manager is " + this + ", current is " + getCurrentKeyboardFocusManager());
- }
- throw new SecurityException(notPrivileged);
- }
+ checkCurrentKFMSecurity();
+ return focusOwner;
}
}
@@ -544,6 +538,8 @@
if (focusOwner == null || focusOwner.isFocusable()) {
synchronized (KeyboardFocusManager.class) {
+ checkCurrentKFMSecurity();
+
oldFocusOwner = getFocusOwner();
try {
@@ -593,6 +589,9 @@
* @see java.awt.event.FocusEvent#FOCUS_LOST
*/
public void clearGlobalFocusOwner() {
+ synchronized (KeyboardFocusManager.class) {
+ checkCurrentKFMSecurity();
+ }
if (!GraphicsEnvironment.isHeadless()) {
// Toolkit must be fully initialized, otherwise
// _clearGlobalFocusOwner will crash or throw an exception
@@ -672,14 +671,8 @@
throws SecurityException
{
synchronized (KeyboardFocusManager.class) {
- if (this == getCurrentKeyboardFocusManager()) {
- return permanentFocusOwner;
- } else {
- if (focusLog.isLoggable(PlatformLogger.FINER)) {
- focusLog.finer("This manager is " + this + ", current is " + getCurrentKeyboardFocusManager());
- }
- throw new SecurityException(notPrivileged);
- }
+ checkCurrentKFMSecurity();
+ return permanentFocusOwner;
}
}
@@ -708,13 +701,14 @@
* @beaninfo
* bound: true
*/
- protected void setGlobalPermanentFocusOwner(Component permanentFocusOwner)
- {
+ protected void setGlobalPermanentFocusOwner(Component permanentFocusOwner) {
Component oldPermanentFocusOwner = null;
boolean shouldFire = false;
if (permanentFocusOwner == null || permanentFocusOwner.isFocusable()) {
synchronized (KeyboardFocusManager.class) {
+ checkCurrentKFMSecurity();
+
oldPermanentFocusOwner = getPermanentFocusOwner();
try {
@@ -780,14 +774,8 @@
*/
protected Window getGlobalFocusedWindow() throws SecurityException {
synchronized (KeyboardFocusManager.class) {
- if (this == getCurrentKeyboardFocusManager()) {
- return focusedWindow;
- } else {
- if (focusLog.isLoggable(PlatformLogger.FINER)) {
- focusLog.finer("This manager is " + this + ", current is " + getCurrentKeyboardFocusManager());
- }
- throw new SecurityException(notPrivileged);
- }
+ checkCurrentKFMSecurity();
+ return focusedWindow;
}
}
@@ -818,6 +806,8 @@
if (focusedWindow == null || focusedWindow.isFocusableWindow()) {
synchronized (KeyboardFocusManager.class) {
+ checkCurrentKFMSecurity();
+
oldFocusedWindow = getFocusedWindow();
try {
@@ -884,14 +874,8 @@
*/
protected Window getGlobalActiveWindow() throws SecurityException {
synchronized (KeyboardFocusManager.class) {
- if (this == getCurrentKeyboardFocusManager()) {
- return activeWindow;
- } else {
- if (focusLog.isLoggable(PlatformLogger.FINER)) {
- focusLog.finer("This manager is " + this + ", current is " + getCurrentKeyboardFocusManager());
- }
- throw new SecurityException(notPrivileged);
- }
+ checkCurrentKFMSecurity();
+ return activeWindow;
}
}
@@ -920,6 +904,8 @@
protected void setGlobalActiveWindow(Window activeWindow) {
Window oldActiveWindow;
synchronized (KeyboardFocusManager.class) {
+ checkCurrentKFMSecurity();
+
oldActiveWindow = getActiveWindow();
if (focusLog.isLoggable(PlatformLogger.FINER)) {
focusLog.finer("Setting global active window to " + activeWindow + ", old active " + oldActiveWindow);
@@ -1214,14 +1200,8 @@
throws SecurityException
{
synchronized (KeyboardFocusManager.class) {
- if (this == getCurrentKeyboardFocusManager()) {
- return currentFocusCycleRoot;
- } else {
- if (focusLog.isLoggable(PlatformLogger.FINER)) {
- focusLog.finer("This manager is " + this + ", current is " + getCurrentKeyboardFocusManager());
- }
- throw new SecurityException(notPrivileged);
- }
+ checkCurrentKFMSecurity();
+ return currentFocusCycleRoot;
}
}
@@ -1245,6 +1225,8 @@
Container oldFocusCycleRoot;
synchronized (KeyboardFocusManager.class) {
+ checkCurrentKFMSecurity();
+
oldFocusCycleRoot = getCurrentFocusCycleRoot();
currentFocusCycleRoot = newFocusCycleRoot;
}
@@ -3062,4 +3044,14 @@
: null;
}
}
+
+ private void checkCurrentKFMSecurity() {
+ if (this != getCurrentKeyboardFocusManager()) {
+ if (focusLog.isLoggable(PlatformLogger.FINER)) {
+ focusLog.finer("This manager is " + this +
+ ", current is " + getCurrentKeyboardFocusManager());
+ }
+ throw new SecurityException(notPrivileged);
+ }
+ }
}
diff -r 52052ed420c5 -r 891fb0050add src/share/classes/java/io/ObjectStreamClass.java
--- a/src/share/classes/java/io/ObjectStreamClass.java Wed Oct 19 06:44:39 2011 +0100
+++ b/src/share/classes/java/io/ObjectStreamClass.java Tue Feb 14 14:01:44 2012 +0000
@@ -123,14 +123,39 @@
*/
private boolean hasBlockExternalData = true;
+ /**
+ * Contains information about InvalidClassException instances to be thrown
+ * when attempting operations on an invalid class. Note that instances of
+ * this class are immutable and are potentially shared among
+ * ObjectStreamClass instances.
+ */
+ private static class ExceptionInfo {
+ private final String className;
+ private final String message;
+
+ ExceptionInfo(String cn, String msg) {
+ className = cn;
+ message = msg;
+ }
+
+ /**
+ * Returns (does not throw) an InvalidClassException instance created
+ * from the information in this object, suitable for being thrown by
+ * the caller.
+ */
+ InvalidClassException newInvalidClassException() {
+ return new InvalidClassException(className, message);
+ }
+ }
+
/** exception (if any) thrown while attempting to resolve class */
private ClassNotFoundException resolveEx;
/** exception (if any) to throw if non-enum deserialization attempted */
- private InvalidClassException deserializeEx;
+ private ExceptionInfo deserializeEx;
/** exception (if any) to throw if non-enum serialization attempted */
- private InvalidClassException serializeEx;
+ private ExceptionInfo serializeEx;
/** exception (if any) to throw if default serialization attempted */
- private InvalidClassException defaultSerializeEx;
+ private ExceptionInfo defaultSerializeEx;
/** serializable fields */
private ObjectStreamField[] fields;
@@ -444,7 +469,8 @@
fields = getSerialFields(cl);
computeFieldOffsets();
} catch (InvalidClassException e) {
- serializeEx = deserializeEx = e;
+ serializeEx = deserializeEx =
+ new ExceptionInfo(e.classname, e.getMessage());
fields = NO_FIELDS;
}
@@ -483,15 +509,14 @@
if (deserializeEx == null) {
if (isEnum) {
- deserializeEx = new InvalidClassException(name, "enum type");
+ deserializeEx = new ExceptionInfo(name, "enum type");
} else if (cons == null) {
- deserializeEx = new InvalidClassException(
- name, "no valid constructor");
+ deserializeEx = new ExceptionInfo(name, "no valid constructor");
}
}
for (int i = 0; i < fields.length; i++) {
if (fields[i].getField() == null) {
- defaultSerializeEx = new InvalidClassException(
+ defaultSerializeEx = new ExceptionInfo(
name, "unmatched serializable field(s) declared");
}
}
@@ -601,8 +626,8 @@
(externalizable != localDesc.externalizable) ||
!(serializable || externalizable))
{
- deserializeEx = new InvalidClassException(localDesc.name,
- "class invalid for deserialization");
+ deserializeEx = new ExceptionInfo(
+ localDesc.name, "class invalid for deserialization");
}
}
@@ -727,11 +752,7 @@
*/
void checkDeserialize() throws InvalidClassException {
if (deserializeEx != null) {
- InvalidClassException ice =
- new InvalidClassException(deserializeEx.classname,
- deserializeEx.getMessage());
- ice.initCause(deserializeEx);
- throw ice;
+ throw deserializeEx.newInvalidClassException();
}
}
@@ -742,11 +763,7 @@
*/
void checkSerialize() throws InvalidClassException {
if (serializeEx != null) {
- InvalidClassException ice =
- new InvalidClassException(serializeEx.classname,
- serializeEx.getMessage());
- ice.initCause(serializeEx);
- throw ice;
+ throw serializeEx.newInvalidClassException();
}
}
@@ -759,11 +776,7 @@
*/
void checkDefaultSerialize() throws InvalidClassException {
if (defaultSerializeEx != null) {
- InvalidClassException ice =
- new InvalidClassException(defaultSerializeEx.classname,
- defaultSerializeEx.getMessage());
- ice.initCause(defaultSerializeEx);
- throw ice;
+ throw defaultSerializeEx.newInvalidClassException();
}
}
diff -r 52052ed420c5 -r 891fb0050add src/share/classes/java/util/TimeZone.java
--- a/src/share/classes/java/util/TimeZone.java Wed Oct 19 06:44:39 2011 +0100
+++ b/src/share/classes/java/util/TimeZone.java Tue Feb 14 14:01:44 2012 +0000
@@ -43,6 +43,7 @@
import java.security.AccessController;
import java.security.PrivilegedAction;
import java.util.concurrent.ConcurrentHashMap;
+import sun.awt.AppContext;
import sun.security.action.GetPropertyAction;
import sun.util.TimeZoneNameUtility;
import sun.util.calendar.ZoneInfo;
@@ -615,7 +616,7 @@
* method doesn't create a clone.
*/
static TimeZone getDefaultRef() {
- TimeZone defaultZone = defaultZoneTL.get();
+ TimeZone defaultZone = getDefaultInAppContext();
if (defaultZone == null) {
defaultZone = defaultTimeZone;
if (defaultZone == null) {
@@ -706,10 +707,49 @@
if (hasPermission()) {
synchronized (TimeZone.class) {
defaultTimeZone = zone;
- defaultZoneTL.set(null);
+ setDefaultInAppContext(null);
}
} else {
- defaultZoneTL.set(zone);
+ setDefaultInAppContext(zone);
+ }
+ }
+
+ /**
+ * Returns the default TimeZone in an AppContext if any AppContext
+ * has ever used. null is returned if any AppContext hasn't been
+ * used or if the AppContext doesn't have the default TimeZone.
+ */
+ private synchronized static TimeZone getDefaultInAppContext() {
+ if (!hasSetInAppContext) {
+ return null;
+ }
+
+ AppContext ac = AppContext.getAppContext();
+ if (ac != null && !ac.isDisposed()) {
+ return (TimeZone) ac.get(TimeZone.class);
+ }
+ return null;
+ }
+
+ /**
+ * Sets the default TimeZone in the AppContext to the given
+ * tz. null is handled special: do nothing if any AppContext
+ * hasn't been used, remove the default TimeZone in the
+ * AppContext otherwise.
+ */
+ private synchronized static void setDefaultInAppContext(TimeZone tz) {
+ if (!hasSetInAppContext && tz == null) {
+ return;
+ }
+
+ AppContext ac = AppContext.getAppContext();
+ if (ac != null && !ac.isDisposed()) {
+ if (tz != null) {
+ ac.put(TimeZone.class, tz);
+ hasSetInAppContext = true;
+ } else {
+ ac.remove(TimeZone.class);
+ }
}
}
@@ -760,12 +800,13 @@
*/
private String ID;
private static volatile TimeZone defaultTimeZone;
- private static final InheritableThreadLocal<TimeZone> defaultZoneTL
- = new InheritableThreadLocal<TimeZone>();
static final String GMT_ID = "GMT";
private static final int GMT_ID_LENGTH = 3;
+ // true if the default TimeZone has been set in any AppContext
+ private static boolean hasSetInAppContext;
+
/**
* Parses a custom time zone identifier and returns a corresponding zone.
* This method doesn't support the RFC 822 time zone format. (e.g., +hhmm)
diff -r 52052ed420c5 -r 891fb0050add src/share/classes/java/util/concurrent/atomic/AtomicReferenceArray.java
--- a/src/share/classes/java/util/concurrent/atomic/AtomicReferenceArray.java Wed Oct 19 06:44:39 2011 +0100
+++ b/src/share/classes/java/util/concurrent/atomic/AtomicReferenceArray.java Tue Feb 14 14:01:44 2012 +0000
@@ -34,8 +34,10 @@
*/
package java.util.concurrent.atomic;
+
+import java.lang.reflect.Array;
+import java.util.Arrays;
import sun.misc.Unsafe;
-import java.util.*;
/**
* An array of object references in which elements may be updated
@@ -49,13 +51,23 @@
public class AtomicReferenceArray<E> implements java.io.Serializable {
private static final long serialVersionUID = -6209656149925076980L;
- private static final Unsafe unsafe = Unsafe.getUnsafe();
- private static final int base = unsafe.arrayBaseOffset(Object[].class);
+ private static final Unsafe unsafe;
+ private static final int base;
private static final int shift;
- private final Object[] array;
+ private static final long arrayFieldOffset;
+ private final Object[] array; // must have exact type Object[]
static {
- int scale = unsafe.arrayIndexScale(Object[].class);
+ int scale;
+ try {
+ unsafe = Unsafe.getUnsafe();
+ arrayFieldOffset = unsafe.objectFieldOffset
+ (AtomicReferenceArray.class.getDeclaredField("array"));
+ base = unsafe.arrayBaseOffset(Object[].class);
+ scale = unsafe.arrayIndexScale(Object[].class);
+ } catch (Exception e) {
+ throw new Error(e);
+ }
if ((scale & (scale - 1)) != 0)
throw new Error("data type scale not a power of two");
shift = 31 - Integer.numberOfLeadingZeros(scale);
@@ -91,7 +103,7 @@
*/
public AtomicReferenceArray(E[] array) {
// Visibility guaranteed by final field guarantees
- this.array = array.clone();
+ this.array = Arrays.copyOf(array, array.length, Object[].class);
}
/**
@@ -150,7 +162,7 @@
public final E getAndSet(int i, E newValue) {
long offset = checkedByteOffset(i);
while (true) {
- E current = (E) getRaw(offset);
More information about the distro-pkg-dev
mailing list