mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-08-04 16:23:55 -06:00
target-arm queue:
* hw/arm: Add GMAC devices to NPCM8XX SoC * hw/arm: Add missing psci_conduit to NPCM8XX SoC boot info * docs/interop: convert text files to restructuredText * target/arm: Some minor refactorings * tests/functional: Add a test for the Stellaris arm machines * hw/block: Drop unused nand.c -----BEGIN PGP SIGNATURE----- iQJNBAABCAA3FiEE4aXFk81BneKOgxXPPCUl7RQ2DN4FAmg5qPYZHHBldGVyLm1h eWRlbGxAbGluYXJvLm9yZwAKCRA8JSXtFDYM3tXUD/9tKWMUEYl23gd9IB5Ee3xK dcgG4Fzv0Ae8HLTd1agyhrg5S2LiXmFi37IO65d8Wxf7Y2TBU+kj1m3aB/C3w9Bx VdHGfNsHAMuYdYCOEm9OvmuSMYSxDRd43pNWdBxbc9/MgLM24rImJ05YHoZFVGrY S5olcZOl3/ttFHtigO4AYAbxkHMAJ5gDyNJiuk88IPx9WGYdmmM4mzJ/m17/Re01 hdOUi0DKQO7kl+646knSU0dicu8NeO5rBAyJzu3vFBnvYXznjd9XaxF+A0Opl54P aBUZz27nDLvnGQrN8B5CjevjUysko+KL/L4NRqebeQKhSe4C8tKFIDocRTGyOEoR SAI0UpZbcX/mXt52aksSwMNG8oRvHOqpJRnNaaCZQoMjK7SlFwi6WctDpwiGt/Hu WaVlXaC77YRiKf1RAgH2CxV04ts342v+bndjfi4vy8D4zbTvwgqKxg+qk3N+JBMR ZUI5Gz3OcGXbw5awJAYbJmyo6qxBysmdHpPY8I1eW0ohzRx1rZ3Vka4yIje5mgO+ 5yFpSy4GDRqNYKgGwlXRaseB38qKL4bEz0+uGzXYqdG7ACBz0xhT5H10npXkX/au LumtwW1sohsv3Xf9oBHQ1WQel7LDcWGVEZHZn6q67mazjvivLjREvA74dq1e8bqD zovTStIpBYRChXTRK1ShUQ== =Xts4 -----END PGP SIGNATURE----- Merge tag 'pull-target-arm-20250530-2' of https://git.linaro.org/people/pmaydell/qemu-arm into staging target-arm queue: * hw/arm: Add GMAC devices to NPCM8XX SoC * hw/arm: Add missing psci_conduit to NPCM8XX SoC boot info * docs/interop: convert text files to restructuredText * target/arm: Some minor refactorings * tests/functional: Add a test for the Stellaris arm machines * hw/block: Drop unused nand.c # -----BEGIN PGP SIGNATURE----- # # iQJNBAABCAA3FiEE4aXFk81BneKOgxXPPCUl7RQ2DN4FAmg5qPYZHHBldGVyLm1h # eWRlbGxAbGluYXJvLm9yZwAKCRA8JSXtFDYM3tXUD/9tKWMUEYl23gd9IB5Ee3xK # dcgG4Fzv0Ae8HLTd1agyhrg5S2LiXmFi37IO65d8Wxf7Y2TBU+kj1m3aB/C3w9Bx # VdHGfNsHAMuYdYCOEm9OvmuSMYSxDRd43pNWdBxbc9/MgLM24rImJ05YHoZFVGrY # S5olcZOl3/ttFHtigO4AYAbxkHMAJ5gDyNJiuk88IPx9WGYdmmM4mzJ/m17/Re01 # hdOUi0DKQO7kl+646knSU0dicu8NeO5rBAyJzu3vFBnvYXznjd9XaxF+A0Opl54P # aBUZz27nDLvnGQrN8B5CjevjUysko+KL/L4NRqebeQKhSe4C8tKFIDocRTGyOEoR # SAI0UpZbcX/mXt52aksSwMNG8oRvHOqpJRnNaaCZQoMjK7SlFwi6WctDpwiGt/Hu # WaVlXaC77YRiKf1RAgH2CxV04ts342v+bndjfi4vy8D4zbTvwgqKxg+qk3N+JBMR # ZUI5Gz3OcGXbw5awJAYbJmyo6qxBysmdHpPY8I1eW0ohzRx1rZ3Vka4yIje5mgO+ # 5yFpSy4GDRqNYKgGwlXRaseB38qKL4bEz0+uGzXYqdG7ACBz0xhT5H10npXkX/au # LumtwW1sohsv3Xf9oBHQ1WQel7LDcWGVEZHZn6q67mazjvivLjREvA74dq1e8bqD # zovTStIpBYRChXTRK1ShUQ== # =Xts4 # -----END PGP SIGNATURE----- # gpg: Signature made Fri 30 May 2025 08:47:50 EDT # gpg: using RSA key E1A5C593CD419DE28E8315CF3C2525ED14360CDE # gpg: issuer "peter.maydell@linaro.org" # gpg: Good signature from "Peter Maydell <peter.maydell@linaro.org>" [full] # gpg: aka "Peter Maydell <pmaydell@gmail.com>" [full] # gpg: aka "Peter Maydell <pmaydell@chiark.greenend.org.uk>" [full] # gpg: aka "Peter Maydell <peter@archaic.org.uk>" [unknown] # Primary key fingerprint: E1A5 C593 CD41 9DE2 8E83 15CF 3C25 25ED 1436 0CDE * tag 'pull-target-arm-20250530-2' of https://git.linaro.org/people/pmaydell/qemu-arm: hw/block: Drop unused nand.c tests/functional: Add a test for the Stellaris arm machines target/arm/hvf: Include missing 'cpu-qom.h' header target/arm/kvm: Include missing 'cpu-qom.h' header target/arm/qmp: Include missing 'cpu.h' header target/arm/cpu-features: Include missing 'cpu.h' header hw/arm/boot: Include missing 'system/memory.h' header target/arm/cpregs: Include missing 'target/arm/cpu.h' header target/arm: Only link with zlib when TCG is enabled target/arm/hvf_arm: Avoid using poisoned CONFIG_HVF definition target/arm/tcg-stubs: compile file once (system) docs/interop: convert text files to restructuredText hw/arm: Add missing psci_conduit to NPCM8XX SoC boot info tests/qtest: Migrate GMAC test from 7xx to 8xx hw/arm: Add GMAC devices to NPCM8XX SoC Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
This commit is contained in:
commit
3e82ddaa8d
25 changed files with 334 additions and 969 deletions
|
@ -510,6 +510,7 @@ Apple Silicon HVF CPUs
|
|||
M: Alexander Graf <agraf@csgraf.de>
|
||||
S: Maintained
|
||||
F: target/arm/hvf/
|
||||
F: target/arm/hvf-stub.c
|
||||
|
||||
X86 HVF CPUs
|
||||
M: Cameron Esfahani <dirty@apple.com>
|
||||
|
@ -1003,6 +1004,7 @@ F: hw/display/ssd03*
|
|||
F: include/hw/input/gamepad.h
|
||||
F: include/hw/timer/stellaris-gptm.h
|
||||
F: docs/system/arm/stellaris.rst
|
||||
F: tests/functional/test_arm_stellaris.py
|
||||
|
||||
STM32L4x5 SoC Family
|
||||
M: Samuel Tardieu <sam@rfc1149.net>
|
||||
|
@ -4130,7 +4132,7 @@ M: Hanna Reitz <hreitz@redhat.com>
|
|||
L: qemu-block@nongnu.org
|
||||
S: Supported
|
||||
F: block/qcow2*
|
||||
F: docs/interop/qcow2.txt
|
||||
F: docs/interop/qcow2.rst
|
||||
|
||||
qcow
|
||||
M: Kevin Wolf <kwolf@redhat.com>
|
||||
|
|
|
@ -97,7 +97,7 @@ time.
|
|||
|
||||
- Persistent storage formats may impose their own requirements on bitmap names
|
||||
and namespaces. Presently, only qcow2 supports persistent bitmaps. See
|
||||
docs/interop/qcow2.txt for more details on restrictions. Notably:
|
||||
:doc:`qcow2` for more details on restrictions. Notably:
|
||||
|
||||
- qcow2 bitmap names are limited to between 1 and 1023 bytes long.
|
||||
|
||||
|
|
|
@ -17,6 +17,7 @@ are useful for making QEMU interoperate with other software.
|
|||
nbd
|
||||
parallels
|
||||
prl-xml
|
||||
qcow2
|
||||
pr-helper
|
||||
qmp-spec
|
||||
qemu-ga
|
||||
|
|
|
@ -1,6 +1,8 @@
|
|||
== General ==
|
||||
=======================
|
||||
Qcow2 Image File Format
|
||||
=======================
|
||||
|
||||
A qcow2 image file is organized in units of constant size, which are called
|
||||
A ``qcow2`` image file is organized in units of constant size, which are called
|
||||
(host) clusters. A cluster is the unit in which all allocations are done,
|
||||
both for actual guest data and for image metadata.
|
||||
|
||||
|
@ -9,10 +11,10 @@ clusters of the same size.
|
|||
|
||||
All numbers in qcow2 are stored in Big Endian byte order.
|
||||
|
||||
Header
|
||||
------
|
||||
|
||||
== Header ==
|
||||
|
||||
The first cluster of a qcow2 image contains the file header:
|
||||
The first cluster of a qcow2 image contains the file header::
|
||||
|
||||
Byte 0 - 3: magic
|
||||
QCOW magic string ("QFI\xfb")
|
||||
|
@ -38,7 +40,7 @@ The first cluster of a qcow2 image contains the file header:
|
|||
within a cluster (1 << cluster_bits is the cluster size).
|
||||
Must not be less than 9 (i.e. 512 byte clusters).
|
||||
|
||||
Note: qemu as of today has an implementation limit of 2 MB
|
||||
Note: QEMU as of today has an implementation limit of 2 MB
|
||||
as the maximum cluster size and won't be able to open images
|
||||
with larger cluster sizes.
|
||||
|
||||
|
@ -48,7 +50,7 @@ The first cluster of a qcow2 image contains the file header:
|
|||
24 - 31: size
|
||||
Virtual disk size in bytes.
|
||||
|
||||
Note: qemu has an implementation limit of 32 MB as
|
||||
Note: QEMU has an implementation limit of 32 MB as
|
||||
the maximum L1 table size. With a 2 MB cluster
|
||||
size, it is unable to populate a virtual cluster
|
||||
beyond 2 EB (61 bits); with a 512 byte cluster
|
||||
|
@ -87,7 +89,8 @@ The first cluster of a qcow2 image contains the file header:
|
|||
|
||||
For version 2, the header is exactly 72 bytes in length, and finishes here.
|
||||
For version 3 or higher, the header length is at least 104 bytes, including
|
||||
the next fields through header_length.
|
||||
the next fields through ``header_length``.
|
||||
::
|
||||
|
||||
72 - 79: incompatible_features
|
||||
Bitmask of incompatible features. An implementation must
|
||||
|
@ -185,7 +188,8 @@ the next fields through header_length.
|
|||
of 8.
|
||||
|
||||
|
||||
=== Additional fields (version 3 and higher) ===
|
||||
Additional fields (version 3 and higher)
|
||||
----------------------------------------
|
||||
|
||||
In general, these fields are optional and may be safely ignored by the software,
|
||||
as well as filled by zeros (which is equal to field absence), if software needs
|
||||
|
@ -193,20 +197,24 @@ to set field B, but does not care about field A which precedes B. More
|
|||
formally, additional fields have the following compatibility rules:
|
||||
|
||||
1. If the value of the additional field must not be ignored for correct
|
||||
handling of the file, it will be accompanied by a corresponding incompatible
|
||||
feature bit.
|
||||
handling of the file, it will be accompanied by a corresponding incompatible
|
||||
feature bit.
|
||||
|
||||
2. If there are no unrecognized incompatible feature bits set, an unknown
|
||||
additional field may be safely ignored other than preserving its value when
|
||||
rewriting the image header.
|
||||
additional field may be safely ignored other than preserving its value when
|
||||
rewriting the image header.
|
||||
|
||||
.. _ref_rules_3:
|
||||
|
||||
3. An explicit value of 0 will have the same behavior as when the field is not
|
||||
present*, if not altered by a specific incompatible bit.
|
||||
present*, if not altered by a specific incompatible bit.
|
||||
|
||||
*. A field is considered not present when header_length is less than or equal
|
||||
(*) A field is considered not present when ``header_length`` is less than or equal
|
||||
to the field's offset. Also, all additional fields are not present for
|
||||
version 2.
|
||||
|
||||
::
|
||||
|
||||
104: compression_type
|
||||
|
||||
Defines the compression method used for compressed clusters.
|
||||
|
@ -219,8 +227,8 @@ version 2.
|
|||
or must be zero (which means deflate).
|
||||
|
||||
Available compression type values:
|
||||
0: deflate <https://www.ietf.org/rfc/rfc1951.txt>
|
||||
1: zstd <http://github.com/facebook/zstd>
|
||||
- 0: deflate <https://www.ietf.org/rfc/rfc1951.txt>
|
||||
- 1: zstd <http://github.com/facebook/zstd>
|
||||
|
||||
The deflate compression type is called "zlib"
|
||||
<https://www.zlib.net/> in QEMU. However, clusters with the
|
||||
|
@ -228,19 +236,21 @@ version 2.
|
|||
|
||||
105 - 111: Padding, contents defined below.
|
||||
|
||||
=== Header padding ===
|
||||
Header padding
|
||||
--------------
|
||||
|
||||
@header_length must be a multiple of 8, which means that if the end of the last
|
||||
``header_length`` must be a multiple of 8, which means that if the end of the last
|
||||
additional field is not aligned, some padding is needed. This padding must be
|
||||
zeroed, so that if some existing (or future) additional field will fall into
|
||||
the padding, it will be interpreted accordingly to point [3.] of the previous
|
||||
the padding, it will be interpreted accordingly to point `[3.] <#ref_rules_3>`_ of the previous
|
||||
paragraph, i.e. in the same manner as when this field is not present.
|
||||
|
||||
|
||||
=== Header extensions ===
|
||||
Header extensions
|
||||
-----------------
|
||||
|
||||
Directly after the image header, optional sections called header extensions can
|
||||
be stored. Each extension has a structure like the following:
|
||||
be stored. Each extension has a structure like the following::
|
||||
|
||||
Byte 0 - 3: Header extension type:
|
||||
0x00000000 - End of the header extension area
|
||||
|
@ -270,17 +280,19 @@ data of compatible features that it doesn't support. Compatible features that
|
|||
need space for additional data can use a header extension.
|
||||
|
||||
|
||||
== String header extensions ==
|
||||
String header extensions
|
||||
------------------------
|
||||
|
||||
Some header extensions (such as the backing file format name and the external
|
||||
data file name) are just a single string. In this case, the header extension
|
||||
length is the string length and the string is not '\0' terminated. (The header
|
||||
extension padding can make it look like a string is '\0' terminated, but
|
||||
length is the string length and the string is not ``\0`` terminated. (The header
|
||||
extension padding can make it look like a string is ``\0`` terminated, but
|
||||
neither is padding always necessary nor is there a guarantee that zero bytes
|
||||
are used for padding.)
|
||||
|
||||
|
||||
== Feature name table ==
|
||||
Feature name table
|
||||
------------------
|
||||
|
||||
The feature name table is an optional header extension that contains the name
|
||||
for features used by the image. It can be used by applications that don't know
|
||||
|
@ -288,7 +300,7 @@ the respective feature (e.g. because the feature was introduced only later) to
|
|||
display a useful error message.
|
||||
|
||||
The number of entries in the feature name table is determined by the length of
|
||||
the header extension data. Each entry look like this:
|
||||
the header extension data. Each entry looks like this::
|
||||
|
||||
Byte 0: Type of feature (select feature bitmap)
|
||||
0: Incompatible feature
|
||||
|
@ -302,7 +314,8 @@ the header extension data. Each entry look like this:
|
|||
terminated if it has full length)
|
||||
|
||||
|
||||
== Bitmaps extension ==
|
||||
Bitmaps extension
|
||||
-----------------
|
||||
|
||||
The bitmaps extension is an optional header extension. It provides the ability
|
||||
to store bitmaps related to a virtual disk. For now, there is only one bitmap
|
||||
|
@ -310,9 +323,9 @@ type: the dirty tracking bitmap, which tracks virtual disk changes from some
|
|||
point in time.
|
||||
|
||||
The data of the extension should be considered consistent only if the
|
||||
corresponding auto-clear feature bit is set, see autoclear_features above.
|
||||
corresponding auto-clear feature bit is set, see ``autoclear_features`` above.
|
||||
|
||||
The fields of the bitmaps extension are:
|
||||
The fields of the bitmaps extension are::
|
||||
|
||||
Byte 0 - 3: nb_bitmaps
|
||||
The number of bitmaps contained in the image. Must be
|
||||
|
@ -331,15 +344,17 @@ The fields of the bitmaps extension are:
|
|||
Offset into the image file at which the bitmap directory
|
||||
starts. Must be aligned to a cluster boundary.
|
||||
|
||||
== Full disk encryption header pointer ==
|
||||
Full disk encryption header pointer
|
||||
-----------------------------------
|
||||
|
||||
The full disk encryption header must be present if, and only if, the
|
||||
'crypt_method' header requires metadata. Currently this is only true
|
||||
of the 'LUKS' crypt method. The header extension must be absent for
|
||||
``crypt_method`` header requires metadata. Currently this is only true
|
||||
of the ``LUKS`` crypt method. The header extension must be absent for
|
||||
other methods.
|
||||
|
||||
This header provides the offset at which the crypt method can store
|
||||
its additional data, as well as the length of such data.
|
||||
::
|
||||
|
||||
Byte 0 - 7: Offset into the image file at which the encryption
|
||||
header starts in bytes. Must be aligned to a cluster
|
||||
|
@ -357,10 +372,10 @@ The first 592 bytes of the header clusters will contain the LUKS
|
|||
partition header. This is then followed by the key material data areas.
|
||||
The size of the key material data areas is determined by the number of
|
||||
stripes in the key slot and key size. Refer to the LUKS format
|
||||
specification ('docs/on-disk-format.pdf' in the cryptsetup source
|
||||
specification (``docs/on-disk-format.pdf`` in the cryptsetup source
|
||||
package) for details of the LUKS partition header format.
|
||||
|
||||
In the LUKS partition header, the "payload-offset" field will be
|
||||
In the LUKS partition header, the ``payload-offset`` field will be
|
||||
calculated as normal for the LUKS spec. ie the size of the LUKS
|
||||
header, plus key material regions, plus padding, relative to the
|
||||
start of the LUKS header. This offset value is not required to be
|
||||
|
@ -369,11 +384,12 @@ context of qcow2, since the qcow2 file format itself defines where
|
|||
the real payload offset is, but none the less a valid payload offset
|
||||
should always be present.
|
||||
|
||||
In the LUKS key slots header, the "key-material-offset" is relative
|
||||
In the LUKS key slots header, the ``key-material-offset`` is relative
|
||||
to the start of the LUKS header clusters in the qcow2 container,
|
||||
not the start of the qcow2 file.
|
||||
|
||||
Logically the layout looks like
|
||||
::
|
||||
|
||||
+-----------------------------+
|
||||
| QCow2 header |
|
||||
|
@ -405,7 +421,8 @@ Logically the layout looks like
|
|||
| |
|
||||
+-----------------------------+
|
||||
|
||||
== Data encryption ==
|
||||
Data encryption
|
||||
---------------
|
||||
|
||||
When an encryption method is requested in the header, the image payload
|
||||
data must be encrypted/decrypted on every write/read. The image headers
|
||||
|
@ -413,7 +430,7 @@ and metadata are never encrypted.
|
|||
|
||||
The algorithms used for encryption vary depending on the method
|
||||
|
||||
- AES:
|
||||
- ``AES``:
|
||||
|
||||
The AES cipher, in CBC mode, with 256 bit keys.
|
||||
|
||||
|
@ -425,7 +442,7 @@ The algorithms used for encryption vary depending on the method
|
|||
supported in the command line tools for the sake of back compatibility
|
||||
and data liberation.
|
||||
|
||||
- LUKS:
|
||||
- ``LUKS``:
|
||||
|
||||
The algorithms are specified in the LUKS header.
|
||||
|
||||
|
@ -433,7 +450,8 @@ The algorithms used for encryption vary depending on the method
|
|||
in the LUKS header, with the physical disk sector as the
|
||||
input tweak.
|
||||
|
||||
== Host cluster management ==
|
||||
Host cluster management
|
||||
-----------------------
|
||||
|
||||
qcow2 manages the allocation of host clusters by maintaining a reference count
|
||||
for each host cluster. A refcount of 0 means that the cluster is free, 1 means
|
||||
|
@ -453,14 +471,15 @@ Although a large enough refcount table can reserve clusters past 64 PB
|
|||
large), note that some qcow2 metadata such as L1/L2 tables must point
|
||||
to clusters prior to that point.
|
||||
|
||||
Note: qemu has an implementation limit of 8 MB as the maximum refcount
|
||||
table size. With a 2 MB cluster size and a default refcount_order of
|
||||
4, it is unable to reference host resources beyond 2 EB (61 bits); in
|
||||
the worst case, with a 512 cluster size and refcount_order of 6, it is
|
||||
unable to access beyond 32 GB (35 bits).
|
||||
.. note::
|
||||
QEMU has an implementation limit of 8 MB as the maximum refcount
|
||||
table size. With a 2 MB cluster size and a default refcount_order of
|
||||
4, it is unable to reference host resources beyond 2 EB (61 bits); in
|
||||
the worst case, with a 512 cluster size and refcount_order of 6, it is
|
||||
unable to access beyond 32 GB (35 bits).
|
||||
|
||||
Given an offset into the image file, the refcount of its cluster can be
|
||||
obtained as follows:
|
||||
obtained as follows::
|
||||
|
||||
refcount_block_entries = (cluster_size * 8 / refcount_bits)
|
||||
|
||||
|
@ -470,7 +489,7 @@ obtained as follows:
|
|||
refcount_block = load_cluster(refcount_table[refcount_table_index]);
|
||||
return refcount_block[refcount_block_index];
|
||||
|
||||
Refcount table entry:
|
||||
Refcount table entry::
|
||||
|
||||
Bit 0 - 8: Reserved (set to 0)
|
||||
|
||||
|
@ -482,14 +501,15 @@ Refcount table entry:
|
|||
been allocated. All refcounts managed by this refcount block
|
||||
are 0.
|
||||
|
||||
Refcount block entry (x = refcount_bits - 1):
|
||||
Refcount block entry ``(x = refcount_bits - 1)``::
|
||||
|
||||
Bit 0 - x: Reference count of the cluster. If refcount_bits implies a
|
||||
sub-byte width, note that bit 0 means the least significant
|
||||
bit in this context.
|
||||
|
||||
|
||||
== Cluster mapping ==
|
||||
Cluster mapping
|
||||
---------------
|
||||
|
||||
Just as for refcounts, qcow2 uses a two-level structure for the mapping of
|
||||
guest clusters to host clusters. They are called L1 and L2 table.
|
||||
|
@ -509,7 +529,7 @@ compressed clusters to reside below 512 TB (49 bits), and this limit
|
|||
cannot be relaxed without an incompatible layout change).
|
||||
|
||||
Given an offset into the virtual disk, the offset into the image file can be
|
||||
obtained as follows:
|
||||
obtained as follows::
|
||||
|
||||
l2_entries = (cluster_size / sizeof(uint64_t)) [*]
|
||||
|
||||
|
@ -523,7 +543,7 @@ obtained as follows:
|
|||
|
||||
[*] this changes if Extended L2 Entries are enabled, see next section
|
||||
|
||||
L1 table entry:
|
||||
L1 table entry::
|
||||
|
||||
Bit 0 - 8: Reserved (set to 0)
|
||||
|
||||
|
@ -538,7 +558,7 @@ L1 table entry:
|
|||
refcount is exactly one. This information is only accurate
|
||||
in the active L1 table.
|
||||
|
||||
L2 table entry:
|
||||
L2 table entry::
|
||||
|
||||
Bit 0 - 61: Cluster descriptor
|
||||
|
||||
|
@ -555,7 +575,7 @@ L2 table entry:
|
|||
mapping for guest cluster offsets), so this bit should be 1
|
||||
for all allocated clusters.
|
||||
|
||||
Standard Cluster Descriptor:
|
||||
Standard Cluster Descriptor::
|
||||
|
||||
Bit 0: If set to 1, the cluster reads as all zeros. The host
|
||||
cluster offset can be used to describe a preallocation,
|
||||
|
@ -577,7 +597,7 @@ Standard Cluster Descriptor:
|
|||
56 - 61: Reserved (set to 0)
|
||||
|
||||
|
||||
Compressed Clusters Descriptor (x = 62 - (cluster_bits - 8)):
|
||||
Compressed Clusters Descriptor ``(x = 62 - (cluster_bits - 8))``::
|
||||
|
||||
Bit 0 - x-1: Host cluster offset. This is usually _not_ aligned to a
|
||||
cluster or sector boundary! If cluster_bits is
|
||||
|
@ -601,7 +621,8 @@ file (except if bit 0 in the Standard Cluster Descriptor is set). If there is
|
|||
no backing file or the backing file is smaller than the image, they shall read
|
||||
zeros for all parts that are not covered by the backing file.
|
||||
|
||||
== Extended L2 Entries ==
|
||||
Extended L2 Entries
|
||||
-------------------
|
||||
|
||||
An image uses Extended L2 Entries if bit 4 is set on the incompatible_features
|
||||
field of the header.
|
||||
|
@ -615,6 +636,8 @@ subclusters so they are treated the same as in images without this feature.
|
|||
The size of an extended L2 entry is 128 bits so the number of entries per table
|
||||
is calculated using this formula:
|
||||
|
||||
.. code::
|
||||
|
||||
l2_entries = (cluster_size / (2 * sizeof(uint64_t)))
|
||||
|
||||
The first 64 bits have the same format as the standard L2 table entry described
|
||||
|
@ -623,7 +646,7 @@ descriptor.
|
|||
|
||||
The last 64 bits contain a subcluster allocation bitmap with this format:
|
||||
|
||||
Subcluster Allocation Bitmap (for standard clusters):
|
||||
Subcluster Allocation Bitmap (for standard clusters)::
|
||||
|
||||
Bit 0 - 31: Allocation status (one bit per subcluster)
|
||||
|
||||
|
@ -647,13 +670,14 @@ Subcluster Allocation Bitmap (for standard clusters):
|
|||
Bits are assigned starting from the least significant
|
||||
one (i.e. bit x is used for subcluster x - 32).
|
||||
|
||||
Subcluster Allocation Bitmap (for compressed clusters):
|
||||
Subcluster Allocation Bitmap (for compressed clusters)::
|
||||
|
||||
Bit 0 - 63: Reserved (set to 0)
|
||||
Compressed clusters don't have subclusters,
|
||||
so this field is not used.
|
||||
|
||||
== Snapshots ==
|
||||
Snapshots
|
||||
---------
|
||||
|
||||
qcow2 supports internal snapshots. Their basic principle of operation is to
|
||||
switch the active L1 table, so that a different set of host clusters are
|
||||
|
@ -672,7 +696,7 @@ in the image file, whose starting offset and length are given by the header
|
|||
fields snapshots_offset and nb_snapshots. The entries of the snapshot table
|
||||
have variable length, depending on the length of ID, name and extra data.
|
||||
|
||||
Snapshot table entry:
|
||||
Snapshot table entry::
|
||||
|
||||
Byte 0 - 7: Offset into the image file at which the L1 table for the
|
||||
snapshot starts. Must be aligned to a cluster boundary.
|
||||
|
@ -728,7 +752,8 @@ Snapshot table entry:
|
|||
next multiple of 8.
|
||||
|
||||
|
||||
== Bitmaps ==
|
||||
Bitmaps
|
||||
-------
|
||||
|
||||
As mentioned above, the bitmaps extension provides the ability to store bitmaps
|
||||
related to a virtual disk. This section describes how these bitmaps are stored.
|
||||
|
@ -739,20 +764,23 @@ each bitmap size is equal to the virtual disk size.
|
|||
Each bit of the bitmap is responsible for strictly defined range of the virtual
|
||||
disk. For bit number bit_nr the corresponding range (in bytes) will be:
|
||||
|
||||
.. code::
|
||||
|
||||
[bit_nr * bitmap_granularity .. (bit_nr + 1) * bitmap_granularity - 1]
|
||||
|
||||
Granularity is a property of the concrete bitmap, see below.
|
||||
|
||||
|
||||
=== Bitmap directory ===
|
||||
Bitmap directory
|
||||
----------------
|
||||
|
||||
Each bitmap saved in the image is described in a bitmap directory entry. The
|
||||
bitmap directory is a contiguous area in the image file, whose starting offset
|
||||
and length are given by the header extension fields bitmap_directory_offset and
|
||||
bitmap_directory_size. The entries of the bitmap directory have variable
|
||||
and length are given by the header extension fields ``bitmap_directory_offset`` and
|
||||
``bitmap_directory_size``. The entries of the bitmap directory have variable
|
||||
length, depending on the lengths of the bitmap name and extra data.
|
||||
|
||||
Structure of a bitmap directory entry:
|
||||
Structure of a bitmap directory entry::
|
||||
|
||||
Byte 0 - 7: bitmap_table_offset
|
||||
Offset into the image file at which the bitmap table
|
||||
|
@ -833,7 +861,8 @@ Structure of a bitmap directory entry:
|
|||
next multiple of 8. All bytes of the padding must be zero.
|
||||
|
||||
|
||||
=== Bitmap table ===
|
||||
Bitmap table
|
||||
------------
|
||||
|
||||
Each bitmap is stored using a one-level structure (as opposed to two-level
|
||||
structures like for refcounts and guest clusters mapping) for the mapping of
|
||||
|
@ -843,7 +872,7 @@ Each bitmap table has a variable size (stored in the bitmap directory entry)
|
|||
and may use multiple clusters, however, it must be contiguous in the image
|
||||
file.
|
||||
|
||||
Structure of a bitmap table entry:
|
||||
Structure of a bitmap table entry::
|
||||
|
||||
Bit 0: Reserved and must be zero if bits 9 - 55 are non-zero.
|
||||
If bits 9 - 55 are zero:
|
||||
|
@ -860,11 +889,12 @@ Structure of a bitmap table entry:
|
|||
56 - 63: Reserved and must be zero.
|
||||
|
||||
|
||||
=== Bitmap data ===
|
||||
Bitmap data
|
||||
-----------
|
||||
|
||||
As noted above, bitmap data is stored in separate clusters, described by the
|
||||
bitmap table. Given an offset (in bytes) into the bitmap data, the offset into
|
||||
the image file can be obtained as follows:
|
||||
the image file can be obtained as follows::
|
||||
|
||||
image_offset(bitmap_data_offset) =
|
||||
bitmap_table[bitmap_data_offset / cluster_size] +
|
||||
|
@ -875,7 +905,7 @@ above).
|
|||
|
||||
Given an offset byte_nr into the virtual disk and the bitmap's granularity, the
|
||||
bit offset into the image file to the corresponding bit of the bitmap can be
|
||||
calculated like this:
|
||||
calculated like this::
|
||||
|
||||
bit_offset(byte_nr) =
|
||||
image_offset(byte_nr / granularity / 8) * 8 +
|
||||
|
@ -886,21 +916,22 @@ last cluster of the bitmap data contains some unused tail bits. These bits must
|
|||
be zero.
|
||||
|
||||
|
||||
=== Dirty tracking bitmaps ===
|
||||
Dirty tracking bitmaps
|
||||
----------------------
|
||||
|
||||
Bitmaps with 'type' field equal to one are dirty tracking bitmaps.
|
||||
Bitmaps with ``type`` field equal to one are dirty tracking bitmaps.
|
||||
|
||||
When the virtual disk is in use dirty tracking bitmap may be 'enabled' or
|
||||
'disabled'. While the bitmap is 'enabled', all writes to the virtual disk
|
||||
When the virtual disk is in use dirty tracking bitmap may be ``enabled`` or
|
||||
``disabled``. While the bitmap is ``enabled``, all writes to the virtual disk
|
||||
should be reflected in the bitmap. A set bit in the bitmap means that the
|
||||
corresponding range of the virtual disk (see above) was written to while the
|
||||
bitmap was 'enabled'. An unset bit means that this range was not written to.
|
||||
bitmap was ``enabled``. An unset bit means that this range was not written to.
|
||||
|
||||
The software doesn't have to sync the bitmap in the image file with its
|
||||
representation in RAM after each write or metadata change. Flag 'in_use'
|
||||
representation in RAM after each write or metadata change. Flag ``in_use``
|
||||
should be set while the bitmap is not synced.
|
||||
|
||||
In the image file the 'enabled' state is reflected by the 'auto' flag. If this
|
||||
flag is set, the software must consider the bitmap as 'enabled' and start
|
||||
In the image file the ``enabled`` state is reflected by the ``auto`` flag. If this
|
||||
flag is set, the software must consider the bitmap as ``enabled`` and start
|
||||
tracking virtual disk changes to this bitmap from the first write to the
|
||||
virtual disk. If this flag is not set then the bitmap is disabled.
|
|
@ -15,7 +15,7 @@ not a straightforward operation.
|
|||
This document attempts to give an overview of the L2 and refcount
|
||||
caches, and how to configure them.
|
||||
|
||||
Please refer to the docs/interop/qcow2.txt file for an in-depth
|
||||
Please refer to the docs/interop/qcow2.rst file for an in-depth
|
||||
technical description of the qcow2 file format.
|
||||
|
||||
|
||||
|
|
|
@ -147,7 +147,6 @@ config OMAP
|
|||
bool
|
||||
select FRAMEBUFFER
|
||||
select I2C
|
||||
select NAND
|
||||
select PFLASH_CFI01
|
||||
select SD
|
||||
select SERIAL_MM
|
||||
|
|
|
@ -19,6 +19,7 @@
|
|||
#include "system/kvm.h"
|
||||
#include "system/tcg.h"
|
||||
#include "system/system.h"
|
||||
#include "system/memory.h"
|
||||
#include "system/numa.h"
|
||||
#include "hw/boards.h"
|
||||
#include "system/reset.h"
|
||||
|
|
|
@ -67,6 +67,9 @@
|
|||
/* SDHCI Modules */
|
||||
#define NPCM8XX_MMC_BA 0xf0842000
|
||||
|
||||
/* PCS Module */
|
||||
#define NPCM8XX_PCS_BA 0xf0780000
|
||||
|
||||
/* PSPI Modules */
|
||||
#define NPCM8XX_PSPI_BA 0xf0201000
|
||||
|
||||
|
@ -85,6 +88,10 @@ enum NPCM8xxInterrupt {
|
|||
NPCM8XX_ADC_IRQ = 0,
|
||||
NPCM8XX_PECI_IRQ = 6,
|
||||
NPCM8XX_KCS_HIB_IRQ = 9,
|
||||
NPCM8XX_GMAC1_IRQ = 14,
|
||||
NPCM8XX_GMAC2_IRQ,
|
||||
NPCM8XX_GMAC3_IRQ,
|
||||
NPCM8XX_GMAC4_IRQ,
|
||||
NPCM8XX_MMC_IRQ = 26,
|
||||
NPCM8XX_PSPI_IRQ = 28,
|
||||
NPCM8XX_TIMER0_IRQ = 32, /* Timer Module 0 */
|
||||
|
@ -260,6 +267,14 @@ static const hwaddr npcm8xx_smbus_addr[] = {
|
|||
0xfff0a000,
|
||||
};
|
||||
|
||||
/* Register base address for each GMAC Module */
|
||||
static const hwaddr npcm8xx_gmac_addr[] = {
|
||||
0xf0802000,
|
||||
0xf0804000,
|
||||
0xf0806000,
|
||||
0xf0808000,
|
||||
};
|
||||
|
||||
/* Register base address for each USB host EHCI registers */
|
||||
static const hwaddr npcm8xx_ehci_addr[] = {
|
||||
0xf0828100,
|
||||
|
@ -350,6 +365,7 @@ static struct arm_boot_info npcm8xx_binfo = {
|
|||
.secure_boot = false,
|
||||
.board_id = -1,
|
||||
.board_setup_addr = NPCM8XX_BOARD_SETUP_ADDR,
|
||||
.psci_conduit = QEMU_PSCI_CONDUIT_SMC,
|
||||
};
|
||||
|
||||
void npcm8xx_load_kernel(MachineState *machine, NPCM8xxState *soc)
|
||||
|
@ -444,6 +460,11 @@ static void npcm8xx_init(Object *obj)
|
|||
object_initialize_child(obj, "mft[*]", &s->mft[i], TYPE_NPCM7XX_MFT);
|
||||
}
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(s->gmac); i++) {
|
||||
object_initialize_child(obj, "gmac[*]", &s->gmac[i], TYPE_NPCM_GMAC);
|
||||
}
|
||||
object_initialize_child(obj, "pcs", &s->pcs, TYPE_NPCM_PCS);
|
||||
|
||||
object_initialize_child(obj, "mmc", &s->mmc, TYPE_NPCM7XX_SDHCI);
|
||||
object_initialize_child(obj, "pspi", &s->pspi, TYPE_NPCM_PSPI);
|
||||
}
|
||||
|
@ -668,6 +689,35 @@ static void npcm8xx_realize(DeviceState *dev, Error **errp)
|
|||
sysbus_connect_irq(sbd, 0, npcm8xx_irq(s, NPCM8XX_MFT0_IRQ + i));
|
||||
}
|
||||
|
||||
/*
|
||||
* GMAC Modules. Cannot fail.
|
||||
*/
|
||||
QEMU_BUILD_BUG_ON(ARRAY_SIZE(npcm8xx_gmac_addr) != ARRAY_SIZE(s->gmac));
|
||||
for (i = 0; i < ARRAY_SIZE(s->gmac); i++) {
|
||||
SysBusDevice *sbd = SYS_BUS_DEVICE(&s->gmac[i]);
|
||||
|
||||
/* This is used to make sure that the NIC can create the device */
|
||||
qemu_configure_nic_device(DEVICE(sbd), false, NULL);
|
||||
|
||||
/*
|
||||
* The device exists regardless of whether it's connected to a QEMU
|
||||
* netdev backend. So always instantiate it even if there is no
|
||||
* backend.
|
||||
*/
|
||||
sysbus_realize(sbd, &error_abort);
|
||||
sysbus_mmio_map(sbd, 0, npcm8xx_gmac_addr[i]);
|
||||
/*
|
||||
* N.B. The values for the second argument sysbus_connect_irq are
|
||||
* chosen to match the registration order in npcm7xx_emc_realize.
|
||||
*/
|
||||
sysbus_connect_irq(sbd, 0, npcm8xx_irq(s, NPCM8XX_GMAC1_IRQ + i));
|
||||
}
|
||||
/*
|
||||
* GMAC Physical Coding Sublayer(PCS) Module. Cannot fail.
|
||||
*/
|
||||
sysbus_realize(SYS_BUS_DEVICE(&s->pcs), &error_abort);
|
||||
sysbus_mmio_map(SYS_BUS_DEVICE(&s->pcs), 0, NPCM8XX_PCS_BA);
|
||||
|
||||
/*
|
||||
* Flash Interface Unit (FIU). Can fail if incorrect number of chip selects
|
||||
* specified, but this is a programming error.
|
||||
|
@ -741,12 +791,7 @@ static void npcm8xx_realize(DeviceState *dev, Error **errp)
|
|||
create_unimplemented_device("npcm8xx.ahbpci", 0xf0400000, 1 * MiB);
|
||||
create_unimplemented_device("npcm8xx.dap", 0xf0500000, 960 * KiB);
|
||||
create_unimplemented_device("npcm8xx.mcphy", 0xf05f0000, 64 * KiB);
|
||||
create_unimplemented_device("npcm8xx.pcs", 0xf0780000, 256 * KiB);
|
||||
create_unimplemented_device("npcm8xx.tsgen", 0xf07fc000, 8 * KiB);
|
||||
create_unimplemented_device("npcm8xx.gmac1", 0xf0802000, 8 * KiB);
|
||||
create_unimplemented_device("npcm8xx.gmac2", 0xf0804000, 8 * KiB);
|
||||
create_unimplemented_device("npcm8xx.gmac3", 0xf0806000, 8 * KiB);
|
||||
create_unimplemented_device("npcm8xx.gmac4", 0xf0808000, 8 * KiB);
|
||||
create_unimplemented_device("npcm8xx.copctl", 0xf080c000, 4 * KiB);
|
||||
create_unimplemented_device("npcm8xx.tipctl", 0xf080d000, 4 * KiB);
|
||||
create_unimplemented_device("npcm8xx.rst", 0xf080e000, 4 * KiB);
|
||||
|
|
|
@ -13,9 +13,6 @@ config FDC_SYSBUS
|
|||
config SSI_M25P80
|
||||
bool
|
||||
|
||||
config NAND
|
||||
bool
|
||||
|
||||
config PFLASH_CFI01
|
||||
bool
|
||||
|
||||
|
|
|
@ -6,7 +6,6 @@ system_ss.add(files(
|
|||
system_ss.add(when: 'CONFIG_FDC', if_true: files('fdc.c'))
|
||||
system_ss.add(when: 'CONFIG_FDC_ISA', if_true: files('fdc-isa.c'))
|
||||
system_ss.add(when: 'CONFIG_FDC_SYSBUS', if_true: files('fdc-sysbus.c'))
|
||||
system_ss.add(when: 'CONFIG_NAND', if_true: files('nand.c'))
|
||||
system_ss.add(when: 'CONFIG_PFLASH_CFI01', if_true: files('pflash_cfi01.c'))
|
||||
system_ss.add(when: 'CONFIG_PFLASH_CFI02', if_true: files('pflash_cfi02.c'))
|
||||
system_ss.add(when: 'CONFIG_SSI_M25P80', if_true: files('m25p80.c'))
|
||||
|
|
835
hw/block/nand.c
835
hw/block/nand.c
|
@ -1,835 +0,0 @@
|
|||
/*
|
||||
* Flash NAND memory emulation. Based on "16M x 8 Bit NAND Flash
|
||||
* Memory" datasheet for the KM29U128AT / K9F2808U0A chips from
|
||||
* Samsung Electronic.
|
||||
*
|
||||
* Copyright (c) 2006 Openedhand Ltd.
|
||||
* Written by Andrzej Zaborowski <balrog@zabor.org>
|
||||
*
|
||||
* Support for additional features based on "MT29F2G16ABCWP 2Gx16"
|
||||
* datasheet from Micron Technology and "NAND02G-B2C" datasheet
|
||||
* from ST Microelectronics.
|
||||
*
|
||||
* This code is licensed under the GNU GPL v2.
|
||||
*
|
||||
* Contributions after 2012-01-13 are licensed under the terms of the
|
||||
* GNU GPL, version 2 or (at your option) any later version.
|
||||
*/
|
||||
|
||||
#ifndef NAND_IO
|
||||
|
||||
#include "qemu/osdep.h"
|
||||
#include "hw/hw.h"
|
||||
#include "hw/qdev-properties.h"
|
||||
#include "hw/qdev-properties-system.h"
|
||||
#include "hw/block/flash.h"
|
||||
#include "system/block-backend.h"
|
||||
#include "migration/vmstate.h"
|
||||
#include "qapi/error.h"
|
||||
#include "qemu/error-report.h"
|
||||
#include "qemu/module.h"
|
||||
#include "qom/object.h"
|
||||
|
||||
# define NAND_CMD_READ0 0x00
|
||||
# define NAND_CMD_READ1 0x01
|
||||
# define NAND_CMD_READ2 0x50
|
||||
# define NAND_CMD_LPREAD2 0x30
|
||||
# define NAND_CMD_NOSERIALREAD2 0x35
|
||||
# define NAND_CMD_RANDOMREAD1 0x05
|
||||
# define NAND_CMD_RANDOMREAD2 0xe0
|
||||
# define NAND_CMD_READID 0x90
|
||||
# define NAND_CMD_RESET 0xff
|
||||
# define NAND_CMD_PAGEPROGRAM1 0x80
|
||||
# define NAND_CMD_PAGEPROGRAM2 0x10
|
||||
# define NAND_CMD_CACHEPROGRAM2 0x15
|
||||
# define NAND_CMD_BLOCKERASE1 0x60
|
||||
# define NAND_CMD_BLOCKERASE2 0xd0
|
||||
# define NAND_CMD_READSTATUS 0x70
|
||||
# define NAND_CMD_COPYBACKPRG1 0x85
|
||||
|
||||
# define NAND_IOSTATUS_ERROR (1 << 0)
|
||||
# define NAND_IOSTATUS_PLANE0 (1 << 1)
|
||||
# define NAND_IOSTATUS_PLANE1 (1 << 2)
|
||||
# define NAND_IOSTATUS_PLANE2 (1 << 3)
|
||||
# define NAND_IOSTATUS_PLANE3 (1 << 4)
|
||||
# define NAND_IOSTATUS_READY (1 << 6)
|
||||
# define NAND_IOSTATUS_UNPROTCT (1 << 7)
|
||||
|
||||
# define MAX_PAGE 0x800
|
||||
# define MAX_OOB 0x40
|
||||
|
||||
typedef struct NANDFlashState NANDFlashState;
|
||||
struct NANDFlashState {
|
||||
DeviceState parent_obj;
|
||||
|
||||
uint8_t manf_id, chip_id;
|
||||
uint8_t buswidth; /* in BYTES */
|
||||
int size, pages;
|
||||
int page_shift, oob_shift, erase_shift, addr_shift;
|
||||
uint8_t *storage;
|
||||
BlockBackend *blk;
|
||||
int mem_oob;
|
||||
|
||||
uint8_t cle, ale, ce, wp, gnd;
|
||||
|
||||
uint8_t io[MAX_PAGE + MAX_OOB + 0x400];
|
||||
uint8_t *ioaddr;
|
||||
int iolen;
|
||||
|
||||
uint32_t cmd;
|
||||
uint64_t addr;
|
||||
int addrlen;
|
||||
int status;
|
||||
int offset;
|
||||
|
||||
void (*blk_write)(NANDFlashState *s);
|
||||
void (*blk_erase)(NANDFlashState *s);
|
||||
/*
|
||||
* Returns %true when block containing (@addr + @offset) is
|
||||
* successfully loaded, otherwise %false.
|
||||
*/
|
||||
bool (*blk_load)(NANDFlashState *s, uint64_t addr, unsigned offset);
|
||||
|
||||
uint32_t ioaddr_vmstate;
|
||||
};
|
||||
|
||||
#define TYPE_NAND "nand"
|
||||
|
||||
OBJECT_DECLARE_SIMPLE_TYPE(NANDFlashState, NAND)
|
||||
|
||||
static void mem_and(uint8_t *dest, const uint8_t *src, size_t n)
|
||||
{
|
||||
/* Like memcpy() but we logical-AND the data into the destination */
|
||||
int i;
|
||||
for (i = 0; i < n; i++) {
|
||||
dest[i] &= src[i];
|
||||
}
|
||||
}
|
||||
|
||||
# define NAND_NO_AUTOINCR 0x00000001
|
||||
# define NAND_BUSWIDTH_16 0x00000002
|
||||
# define NAND_NO_PADDING 0x00000004
|
||||
# define NAND_CACHEPRG 0x00000008
|
||||
# define NAND_COPYBACK 0x00000010
|
||||
# define NAND_IS_AND 0x00000020
|
||||
# define NAND_4PAGE_ARRAY 0x00000040
|
||||
# define NAND_NO_READRDY 0x00000100
|
||||
# define NAND_SAMSUNG_LP (NAND_NO_PADDING | NAND_COPYBACK)
|
||||
|
||||
# define NAND_IO
|
||||
|
||||
# define PAGE(addr) ((addr) >> ADDR_SHIFT)
|
||||
# define PAGE_START(page) (PAGE(page) * (NAND_PAGE_SIZE + OOB_SIZE))
|
||||
# define PAGE_MASK ((1 << ADDR_SHIFT) - 1)
|
||||
# define OOB_SHIFT (PAGE_SHIFT - 5)
|
||||
# define OOB_SIZE (1 << OOB_SHIFT)
|
||||
# define SECTOR(addr) ((addr) >> (9 + ADDR_SHIFT - PAGE_SHIFT))
|
||||
# define SECTOR_OFFSET(addr) ((addr) & ((511 >> PAGE_SHIFT) << 8))
|
||||
|
||||
# define NAND_PAGE_SIZE 256
|
||||
# define PAGE_SHIFT 8
|
||||
# define PAGE_SECTORS 1
|
||||
# define ADDR_SHIFT 8
|
||||
# include "nand.c"
|
||||
# define NAND_PAGE_SIZE 512
|
||||
# define PAGE_SHIFT 9
|
||||
# define PAGE_SECTORS 1
|
||||
# define ADDR_SHIFT 8
|
||||
# include "nand.c"
|
||||
# define NAND_PAGE_SIZE 2048
|
||||
# define PAGE_SHIFT 11
|
||||
# define PAGE_SECTORS 4
|
||||
# define ADDR_SHIFT 16
|
||||
# include "nand.c"
|
||||
|
||||
/* Information based on Linux drivers/mtd/nand/raw/nand_ids.c */
|
||||
static const struct {
|
||||
int size;
|
||||
int width;
|
||||
int page_shift;
|
||||
int erase_shift;
|
||||
uint32_t options;
|
||||
} nand_flash_ids[0x100] = {
|
||||
[0 ... 0xff] = { 0 },
|
||||
|
||||
[0x6b] = { 4, 8, 9, 4, 0 },
|
||||
[0xe3] = { 4, 8, 9, 4, 0 },
|
||||
[0xe5] = { 4, 8, 9, 4, 0 },
|
||||
[0xd6] = { 8, 8, 9, 4, 0 },
|
||||
[0xe6] = { 8, 8, 9, 4, 0 },
|
||||
|
||||
[0x33] = { 16, 8, 9, 5, 0 },
|
||||
[0x73] = { 16, 8, 9, 5, 0 },
|
||||
[0x43] = { 16, 16, 9, 5, NAND_BUSWIDTH_16 },
|
||||
[0x53] = { 16, 16, 9, 5, NAND_BUSWIDTH_16 },
|
||||
|
||||
[0x35] = { 32, 8, 9, 5, 0 },
|
||||
[0x75] = { 32, 8, 9, 5, 0 },
|
||||
[0x45] = { 32, 16, 9, 5, NAND_BUSWIDTH_16 },
|
||||
[0x55] = { 32, 16, 9, 5, NAND_BUSWIDTH_16 },
|
||||
|
||||
[0x36] = { 64, 8, 9, 5, 0 },
|
||||
[0x76] = { 64, 8, 9, 5, 0 },
|
||||
[0x46] = { 64, 16, 9, 5, NAND_BUSWIDTH_16 },
|
||||
[0x56] = { 64, 16, 9, 5, NAND_BUSWIDTH_16 },
|
||||
|
||||
[0x78] = { 128, 8, 9, 5, 0 },
|
||||
[0x39] = { 128, 8, 9, 5, 0 },
|
||||
[0x79] = { 128, 8, 9, 5, 0 },
|
||||
[0x72] = { 128, 16, 9, 5, NAND_BUSWIDTH_16 },
|
||||
[0x49] = { 128, 16, 9, 5, NAND_BUSWIDTH_16 },
|
||||
[0x74] = { 128, 16, 9, 5, NAND_BUSWIDTH_16 },
|
||||
[0x59] = { 128, 16, 9, 5, NAND_BUSWIDTH_16 },
|
||||
|
||||
[0x71] = { 256, 8, 9, 5, 0 },
|
||||
|
||||
/*
|
||||
* These are the new chips with large page size. The pagesize and the
|
||||
* erasesize is determined from the extended id bytes
|
||||
*/
|
||||
# define LP_OPTIONS (NAND_SAMSUNG_LP | NAND_NO_READRDY | NAND_NO_AUTOINCR)
|
||||
# define LP_OPTIONS16 (LP_OPTIONS | NAND_BUSWIDTH_16)
|
||||
|
||||
/* 512 Megabit */
|
||||
[0xa2] = { 64, 8, 0, 0, LP_OPTIONS },
|
||||
[0xf2] = { 64, 8, 0, 0, LP_OPTIONS },
|
||||
[0xb2] = { 64, 16, 0, 0, LP_OPTIONS16 },
|
||||
[0xc2] = { 64, 16, 0, 0, LP_OPTIONS16 },
|
||||
|
||||
/* 1 Gigabit */
|
||||
[0xa1] = { 128, 8, 0, 0, LP_OPTIONS },
|
||||
[0xf1] = { 128, 8, 0, 0, LP_OPTIONS },
|
||||
[0xb1] = { 128, 16, 0, 0, LP_OPTIONS16 },
|
||||
[0xc1] = { 128, 16, 0, 0, LP_OPTIONS16 },
|
||||
|
||||
/* 2 Gigabit */
|
||||
[0xaa] = { 256, 8, 0, 0, LP_OPTIONS },
|
||||
[0xda] = { 256, 8, 0, 0, LP_OPTIONS },
|
||||
[0xba] = { 256, 16, 0, 0, LP_OPTIONS16 },
|
||||
[0xca] = { 256, 16, 0, 0, LP_OPTIONS16 },
|
||||
|
||||
/* 4 Gigabit */
|
||||
[0xac] = { 512, 8, 0, 0, LP_OPTIONS },
|
||||
[0xdc] = { 512, 8, 0, 0, LP_OPTIONS },
|
||||
[0xbc] = { 512, 16, 0, 0, LP_OPTIONS16 },
|
||||
[0xcc] = { 512, 16, 0, 0, LP_OPTIONS16 },
|
||||
|
||||
/* 8 Gigabit */
|
||||
[0xa3] = { 1024, 8, 0, 0, LP_OPTIONS },
|
||||
[0xd3] = { 1024, 8, 0, 0, LP_OPTIONS },
|
||||
[0xb3] = { 1024, 16, 0, 0, LP_OPTIONS16 },
|
||||
[0xc3] = { 1024, 16, 0, 0, LP_OPTIONS16 },
|
||||
|
||||
/* 16 Gigabit */
|
||||
[0xa5] = { 2048, 8, 0, 0, LP_OPTIONS },
|
||||
[0xd5] = { 2048, 8, 0, 0, LP_OPTIONS },
|
||||
[0xb5] = { 2048, 16, 0, 0, LP_OPTIONS16 },
|
||||
[0xc5] = { 2048, 16, 0, 0, LP_OPTIONS16 },
|
||||
};
|
||||
|
||||
static void nand_reset(DeviceState *dev)
|
||||
{
|
||||
NANDFlashState *s = NAND(dev);
|
||||
s->cmd = NAND_CMD_READ0;
|
||||
s->addr = 0;
|
||||
s->addrlen = 0;
|
||||
s->iolen = 0;
|
||||
s->offset = 0;
|
||||
s->status &= NAND_IOSTATUS_UNPROTCT;
|
||||
s->status |= NAND_IOSTATUS_READY;
|
||||
}
|
||||
|
||||
static inline void nand_pushio_byte(NANDFlashState *s, uint8_t value)
|
||||
{
|
||||
s->ioaddr[s->iolen++] = value;
|
||||
for (value = s->buswidth; --value;) {
|
||||
s->ioaddr[s->iolen++] = 0;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* nand_load_block: Load block containing (s->addr + @offset).
|
||||
* Returns length of data available at @offset in this block.
|
||||
*/
|
||||
static unsigned nand_load_block(NANDFlashState *s, unsigned offset)
|
||||
{
|
||||
unsigned iolen;
|
||||
|
||||
if (!s->blk_load(s, s->addr, offset)) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
iolen = (1 << s->page_shift);
|
||||
if (s->gnd) {
|
||||
iolen += 1 << s->oob_shift;
|
||||
}
|
||||
assert(offset <= iolen);
|
||||
iolen -= offset;
|
||||
|
||||
return iolen;
|
||||
}
|
||||
|
||||
static void nand_command(NANDFlashState *s)
|
||||
{
|
||||
switch (s->cmd) {
|
||||
case NAND_CMD_READ0:
|
||||
s->iolen = 0;
|
||||
break;
|
||||
|
||||
case NAND_CMD_READID:
|
||||
s->ioaddr = s->io;
|
||||
s->iolen = 0;
|
||||
nand_pushio_byte(s, s->manf_id);
|
||||
nand_pushio_byte(s, s->chip_id);
|
||||
nand_pushio_byte(s, 'Q'); /* Don't-care byte (often 0xa5) */
|
||||
if (nand_flash_ids[s->chip_id].options & NAND_SAMSUNG_LP) {
|
||||
/* Page Size, Block Size, Spare Size; bit 6 indicates
|
||||
* 8 vs 16 bit width NAND.
|
||||
*/
|
||||
nand_pushio_byte(s, (s->buswidth == 2) ? 0x55 : 0x15);
|
||||
} else {
|
||||
nand_pushio_byte(s, 0xc0); /* Multi-plane */
|
||||
}
|
||||
break;
|
||||
|
||||
case NAND_CMD_RANDOMREAD2:
|
||||
case NAND_CMD_NOSERIALREAD2:
|
||||
if (!(nand_flash_ids[s->chip_id].options & NAND_SAMSUNG_LP))
|
||||
break;
|
||||
s->iolen = nand_load_block(s, s->addr & ((1 << s->addr_shift) - 1));
|
||||
break;
|
||||
|
||||
case NAND_CMD_RESET:
|
||||
nand_reset(DEVICE(s));
|
||||
break;
|
||||
|
||||
case NAND_CMD_PAGEPROGRAM1:
|
||||
s->ioaddr = s->io;
|
||||
s->iolen = 0;
|
||||
break;
|
||||
|
||||
case NAND_CMD_PAGEPROGRAM2:
|
||||
if (s->wp) {
|
||||
s->blk_write(s);
|
||||
}
|
||||
break;
|
||||
|
||||
case NAND_CMD_BLOCKERASE1:
|
||||
break;
|
||||
|
||||
case NAND_CMD_BLOCKERASE2:
|
||||
s->addr &= (1ull << s->addrlen * 8) - 1;
|
||||
s->addr <<= nand_flash_ids[s->chip_id].options & NAND_SAMSUNG_LP ?
|
||||
16 : 8;
|
||||
|
||||
if (s->wp) {
|
||||
s->blk_erase(s);
|
||||
}
|
||||
break;
|
||||
|
||||
case NAND_CMD_READSTATUS:
|
||||
s->ioaddr = s->io;
|
||||
s->iolen = 0;
|
||||
nand_pushio_byte(s, s->status);
|
||||
break;
|
||||
|
||||
default:
|
||||
printf("%s: Unknown NAND command 0x%02x\n", __func__, s->cmd);
|
||||
}
|
||||
}
|
||||
|
||||
static int nand_pre_save(void *opaque)
|
||||
{
|
||||
NANDFlashState *s = NAND(opaque);
|
||||
|
||||
s->ioaddr_vmstate = s->ioaddr - s->io;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int nand_post_load(void *opaque, int version_id)
|
||||
{
|
||||
NANDFlashState *s = NAND(opaque);
|
||||
|
||||
if (s->ioaddr_vmstate > sizeof(s->io)) {
|
||||
return -EINVAL;
|
||||
}
|
||||
s->ioaddr = s->io + s->ioaddr_vmstate;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const VMStateDescription vmstate_nand = {
|
||||
.name = "nand",
|
||||
.version_id = 1,
|
||||
.minimum_version_id = 1,
|
||||
.pre_save = nand_pre_save,
|
||||
.post_load = nand_post_load,
|
||||
.fields = (const VMStateField[]) {
|
||||
VMSTATE_UINT8(cle, NANDFlashState),
|
||||
VMSTATE_UINT8(ale, NANDFlashState),
|
||||
VMSTATE_UINT8(ce, NANDFlashState),
|
||||
VMSTATE_UINT8(wp, NANDFlashState),
|
||||
VMSTATE_UINT8(gnd, NANDFlashState),
|
||||
VMSTATE_BUFFER(io, NANDFlashState),
|
||||
VMSTATE_UINT32(ioaddr_vmstate, NANDFlashState),
|
||||
VMSTATE_INT32(iolen, NANDFlashState),
|
||||
VMSTATE_UINT32(cmd, NANDFlashState),
|
||||
VMSTATE_UINT64(addr, NANDFlashState),
|
||||
VMSTATE_INT32(addrlen, NANDFlashState),
|
||||
VMSTATE_INT32(status, NANDFlashState),
|
||||
VMSTATE_INT32(offset, NANDFlashState),
|
||||
/* XXX: do we want to save s->storage too? */
|
||||
VMSTATE_END_OF_LIST()
|
||||
}
|
||||
};
|
||||
|
||||
static void nand_realize(DeviceState *dev, Error **errp)
|
||||
{
|
||||
int pagesize;
|
||||
NANDFlashState *s = NAND(dev);
|
||||
int ret;
|
||||
|
||||
|
||||
s->buswidth = nand_flash_ids[s->chip_id].width >> 3;
|
||||
s->size = nand_flash_ids[s->chip_id].size << 20;
|
||||
if (nand_flash_ids[s->chip_id].options & NAND_SAMSUNG_LP) {
|
||||
s->page_shift = 11;
|
||||
s->erase_shift = 6;
|
||||
} else {
|
||||
s->page_shift = nand_flash_ids[s->chip_id].page_shift;
|
||||
s->erase_shift = nand_flash_ids[s->chip_id].erase_shift;
|
||||
}
|
||||
|
||||
switch (1 << s->page_shift) {
|
||||
case 256:
|
||||
nand_init_256(s);
|
||||
break;
|
||||
case 512:
|
||||
nand_init_512(s);
|
||||
break;
|
||||
case 2048:
|
||||
nand_init_2048(s);
|
||||
break;
|
||||
default:
|
||||
error_setg(errp, "Unsupported NAND block size %#x",
|
||||
1 << s->page_shift);
|
||||
return;
|
||||
}
|
||||
|
||||
pagesize = 1 << s->oob_shift;
|
||||
s->mem_oob = 1;
|
||||
if (s->blk) {
|
||||
if (!blk_supports_write_perm(s->blk)) {
|
||||
error_setg(errp, "Can't use a read-only drive");
|
||||
return;
|
||||
}
|
||||
ret = blk_set_perm(s->blk, BLK_PERM_CONSISTENT_READ | BLK_PERM_WRITE,
|
||||
BLK_PERM_ALL, errp);
|
||||
if (ret < 0) {
|
||||
return;
|
||||
}
|
||||
if (blk_getlength(s->blk) >=
|
||||
(s->pages << s->page_shift) + (s->pages << s->oob_shift)) {
|
||||
pagesize = 0;
|
||||
s->mem_oob = 0;
|
||||
}
|
||||
} else {
|
||||
pagesize += 1 << s->page_shift;
|
||||
}
|
||||
if (pagesize) {
|
||||
s->storage = (uint8_t *) memset(g_malloc(s->pages * pagesize),
|
||||
0xff, s->pages * pagesize);
|
||||
}
|
||||
/* Give s->ioaddr a sane value in case we save state before it is used. */
|
||||
s->ioaddr = s->io;
|
||||
}
|
||||
|
||||
static const Property nand_properties[] = {
|
||||
DEFINE_PROP_UINT8("manufacturer_id", NANDFlashState, manf_id, 0),
|
||||
DEFINE_PROP_UINT8("chip_id", NANDFlashState, chip_id, 0),
|
||||
DEFINE_PROP_DRIVE("drive", NANDFlashState, blk),
|
||||
};
|
||||
|
||||
static void nand_class_init(ObjectClass *klass, const void *data)
|
||||
{
|
||||
DeviceClass *dc = DEVICE_CLASS(klass);
|
||||
|
||||
dc->realize = nand_realize;
|
||||
device_class_set_legacy_reset(dc, nand_reset);
|
||||
dc->vmsd = &vmstate_nand;
|
||||
device_class_set_props(dc, nand_properties);
|
||||
set_bit(DEVICE_CATEGORY_STORAGE, dc->categories);
|
||||
}
|
||||
|
||||
static const TypeInfo nand_info = {
|
||||
.name = TYPE_NAND,
|
||||
.parent = TYPE_DEVICE,
|
||||
.instance_size = sizeof(NANDFlashState),
|
||||
.class_init = nand_class_init,
|
||||
};
|
||||
|
||||
static void nand_register_types(void)
|
||||
{
|
||||
type_register_static(&nand_info);
|
||||
}
|
||||
|
||||
/*
|
||||
* Chip inputs are CLE, ALE, CE, WP, GND and eight I/O pins. Chip
|
||||
* outputs are R/B and eight I/O pins.
|
||||
*
|
||||
* CE, WP and R/B are active low.
|
||||
*/
|
||||
void nand_setpins(DeviceState *dev, uint8_t cle, uint8_t ale,
|
||||
uint8_t ce, uint8_t wp, uint8_t gnd)
|
||||
{
|
||||
NANDFlashState *s = NAND(dev);
|
||||
|
||||
s->cle = cle;
|
||||
s->ale = ale;
|
||||
s->ce = ce;
|
||||
s->wp = wp;
|
||||
s->gnd = gnd;
|
||||
if (wp) {
|
||||
s->status |= NAND_IOSTATUS_UNPROTCT;
|
||||
} else {
|
||||
s->status &= ~NAND_IOSTATUS_UNPROTCT;
|
||||
}
|
||||
}
|
||||
|
||||
void nand_getpins(DeviceState *dev, int *rb)
|
||||
{
|
||||
*rb = 1;
|
||||
}
|
||||
|
||||
void nand_setio(DeviceState *dev, uint32_t value)
|
||||
{
|
||||
int i;
|
||||
NANDFlashState *s = NAND(dev);
|
||||
|
||||
if (!s->ce && s->cle) {
|
||||
if (nand_flash_ids[s->chip_id].options & NAND_SAMSUNG_LP) {
|
||||
if (s->cmd == NAND_CMD_READ0 && value == NAND_CMD_LPREAD2)
|
||||
return;
|
||||
if (value == NAND_CMD_RANDOMREAD1) {
|
||||
s->addr &= ~((1 << s->addr_shift) - 1);
|
||||
s->addrlen = 0;
|
||||
return;
|
||||
}
|
||||
}
|
||||
if (value == NAND_CMD_READ0) {
|
||||
s->offset = 0;
|
||||
} else if (value == NAND_CMD_READ1) {
|
||||
s->offset = 0x100;
|
||||
value = NAND_CMD_READ0;
|
||||
} else if (value == NAND_CMD_READ2) {
|
||||
s->offset = 1 << s->page_shift;
|
||||
value = NAND_CMD_READ0;
|
||||
}
|
||||
|
||||
s->cmd = value;
|
||||
|
||||
if (s->cmd == NAND_CMD_READSTATUS ||
|
||||
s->cmd == NAND_CMD_PAGEPROGRAM2 ||
|
||||
s->cmd == NAND_CMD_BLOCKERASE1 ||
|
||||
s->cmd == NAND_CMD_BLOCKERASE2 ||
|
||||
s->cmd == NAND_CMD_NOSERIALREAD2 ||
|
||||
s->cmd == NAND_CMD_RANDOMREAD2 ||
|
||||
s->cmd == NAND_CMD_RESET) {
|
||||
nand_command(s);
|
||||
}
|
||||
|
||||
if (s->cmd != NAND_CMD_RANDOMREAD2) {
|
||||
s->addrlen = 0;
|
||||
}
|
||||
}
|
||||
|
||||
if (s->ale) {
|
||||
unsigned int shift = s->addrlen * 8;
|
||||
uint64_t mask = ~(0xffull << shift);
|
||||
uint64_t v = (uint64_t)value << shift;
|
||||
|
||||
s->addr = (s->addr & mask) | v;
|
||||
s->addrlen ++;
|
||||
|
||||
switch (s->addrlen) {
|
||||
case 1:
|
||||
if (s->cmd == NAND_CMD_READID) {
|
||||
nand_command(s);
|
||||
}
|
||||
break;
|
||||
case 2: /* fix cache address as a byte address */
|
||||
s->addr <<= (s->buswidth - 1);
|
||||
break;
|
||||
case 3:
|
||||
if (!(nand_flash_ids[s->chip_id].options & NAND_SAMSUNG_LP) &&
|
||||
(s->cmd == NAND_CMD_READ0 ||
|
||||
s->cmd == NAND_CMD_PAGEPROGRAM1)) {
|
||||
nand_command(s);
|
||||
}
|
||||
break;
|
||||
case 4:
|
||||
if ((nand_flash_ids[s->chip_id].options & NAND_SAMSUNG_LP) &&
|
||||
nand_flash_ids[s->chip_id].size < 256 && /* 1Gb or less */
|
||||
(s->cmd == NAND_CMD_READ0 ||
|
||||
s->cmd == NAND_CMD_PAGEPROGRAM1)) {
|
||||
nand_command(s);
|
||||
}
|
||||
break;
|
||||
case 5:
|
||||
if ((nand_flash_ids[s->chip_id].options & NAND_SAMSUNG_LP) &&
|
||||
nand_flash_ids[s->chip_id].size >= 256 && /* 2Gb or more */
|
||||
(s->cmd == NAND_CMD_READ0 ||
|
||||
s->cmd == NAND_CMD_PAGEPROGRAM1)) {
|
||||
nand_command(s);
|
||||
}
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (!s->cle && !s->ale && s->cmd == NAND_CMD_PAGEPROGRAM1) {
|
||||
if (s->iolen < (1 << s->page_shift) + (1 << s->oob_shift)) {
|
||||
for (i = s->buswidth; i--; value >>= 8) {
|
||||
s->io[s->iolen ++] = (uint8_t) (value & 0xff);
|
||||
}
|
||||
}
|
||||
} else if (!s->cle && !s->ale && s->cmd == NAND_CMD_COPYBACKPRG1) {
|
||||
if ((s->addr & ((1 << s->addr_shift) - 1)) <
|
||||
(1 << s->page_shift) + (1 << s->oob_shift)) {
|
||||
for (i = s->buswidth; i--; s->addr++, value >>= 8) {
|
||||
s->io[s->iolen + (s->addr & ((1 << s->addr_shift) - 1))] =
|
||||
(uint8_t) (value & 0xff);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
uint32_t nand_getio(DeviceState *dev)
|
||||
{
|
||||
int offset;
|
||||
uint32_t x = 0;
|
||||
NANDFlashState *s = NAND(dev);
|
||||
|
||||
/* Allow sequential reading */
|
||||
if (!s->iolen && s->cmd == NAND_CMD_READ0) {
|
||||
offset = (int) (s->addr & ((1 << s->addr_shift) - 1)) + s->offset;
|
||||
s->offset = 0;
|
||||
s->iolen = nand_load_block(s, offset);
|
||||
}
|
||||
|
||||
if (s->ce || s->iolen <= 0) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
for (offset = s->buswidth; offset--;) {
|
||||
x |= s->ioaddr[offset] << (offset << 3);
|
||||
}
|
||||
/* after receiving READ STATUS command all subsequent reads will
|
||||
* return the status register value until another command is issued
|
||||
*/
|
||||
if (s->cmd != NAND_CMD_READSTATUS) {
|
||||
s->addr += s->buswidth;
|
||||
s->ioaddr += s->buswidth;
|
||||
s->iolen -= s->buswidth;
|
||||
}
|
||||
return x;
|
||||
}
|
||||
|
||||
uint32_t nand_getbuswidth(DeviceState *dev)
|
||||
{
|
||||
NANDFlashState *s = (NANDFlashState *) dev;
|
||||
return s->buswidth << 3;
|
||||
}
|
||||
|
||||
DeviceState *nand_init(BlockBackend *blk, int manf_id, int chip_id)
|
||||
{
|
||||
DeviceState *dev;
|
||||
|
||||
if (nand_flash_ids[chip_id].size == 0) {
|
||||
hw_error("%s: Unsupported NAND chip ID.\n", __func__);
|
||||
}
|
||||
dev = qdev_new(TYPE_NAND);
|
||||
qdev_prop_set_uint8(dev, "manufacturer_id", manf_id);
|
||||
qdev_prop_set_uint8(dev, "chip_id", chip_id);
|
||||
if (blk) {
|
||||
qdev_prop_set_drive_err(dev, "drive", blk, &error_fatal);
|
||||
}
|
||||
|
||||
qdev_realize(dev, NULL, &error_fatal);
|
||||
return dev;
|
||||
}
|
||||
|
||||
type_init(nand_register_types)
|
||||
|
||||
#else
|
||||
|
||||
/* Program a single page */
|
||||
static void glue(nand_blk_write_, NAND_PAGE_SIZE)(NANDFlashState *s)
|
||||
{
|
||||
uint64_t off, page, sector, soff;
|
||||
uint8_t iobuf[(PAGE_SECTORS + 2) * 0x200];
|
||||
if (PAGE(s->addr) >= s->pages)
|
||||
return;
|
||||
|
||||
if (!s->blk) {
|
||||
mem_and(s->storage + PAGE_START(s->addr) + (s->addr & PAGE_MASK) +
|
||||
s->offset, s->io, s->iolen);
|
||||
} else if (s->mem_oob) {
|
||||
sector = SECTOR(s->addr);
|
||||
off = (s->addr & PAGE_MASK) + s->offset;
|
||||
soff = SECTOR_OFFSET(s->addr);
|
||||
if (blk_pread(s->blk, sector << BDRV_SECTOR_BITS,
|
||||
PAGE_SECTORS << BDRV_SECTOR_BITS, iobuf, 0) < 0) {
|
||||
printf("%s: read error in sector %" PRIu64 "\n", __func__, sector);
|
||||
return;
|
||||
}
|
||||
|
||||
mem_and(iobuf + (soff | off), s->io, MIN(s->iolen, NAND_PAGE_SIZE - off));
|
||||
if (off + s->iolen > NAND_PAGE_SIZE) {
|
||||
page = PAGE(s->addr);
|
||||
mem_and(s->storage + (page << OOB_SHIFT), s->io + NAND_PAGE_SIZE - off,
|
||||
MIN(OOB_SIZE, off + s->iolen - NAND_PAGE_SIZE));
|
||||
}
|
||||
|
||||
if (blk_pwrite(s->blk, sector << BDRV_SECTOR_BITS,
|
||||
PAGE_SECTORS << BDRV_SECTOR_BITS, iobuf, 0) < 0) {
|
||||
printf("%s: write error in sector %" PRIu64 "\n", __func__, sector);
|
||||
}
|
||||
} else {
|
||||
off = PAGE_START(s->addr) + (s->addr & PAGE_MASK) + s->offset;
|
||||
sector = off >> 9;
|
||||
soff = off & 0x1ff;
|
||||
if (blk_pread(s->blk, sector << BDRV_SECTOR_BITS,
|
||||
(PAGE_SECTORS + 2) << BDRV_SECTOR_BITS, iobuf, 0) < 0) {
|
||||
printf("%s: read error in sector %" PRIu64 "\n", __func__, sector);
|
||||
return;
|
||||
}
|
||||
|
||||
mem_and(iobuf + soff, s->io, s->iolen);
|
||||
|
||||
if (blk_pwrite(s->blk, sector << BDRV_SECTOR_BITS,
|
||||
(PAGE_SECTORS + 2) << BDRV_SECTOR_BITS, iobuf, 0) < 0) {
|
||||
printf("%s: write error in sector %" PRIu64 "\n", __func__, sector);
|
||||
}
|
||||
}
|
||||
s->offset = 0;
|
||||
}
|
||||
|
||||
/* Erase a single block */
|
||||
static void glue(nand_blk_erase_, NAND_PAGE_SIZE)(NANDFlashState *s)
|
||||
{
|
||||
uint64_t i, page, addr;
|
||||
uint8_t iobuf[0x200] = { [0 ... 0x1ff] = 0xff, };
|
||||
addr = s->addr & ~((1 << (ADDR_SHIFT + s->erase_shift)) - 1);
|
||||
|
||||
if (PAGE(addr) >= s->pages) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (!s->blk) {
|
||||
memset(s->storage + PAGE_START(addr),
|
||||
0xff, (NAND_PAGE_SIZE + OOB_SIZE) << s->erase_shift);
|
||||
} else if (s->mem_oob) {
|
||||
memset(s->storage + (PAGE(addr) << OOB_SHIFT),
|
||||
0xff, OOB_SIZE << s->erase_shift);
|
||||
i = SECTOR(addr);
|
||||
page = SECTOR(addr + (1 << (ADDR_SHIFT + s->erase_shift)));
|
||||
for (; i < page; i ++)
|
||||
if (blk_pwrite(s->blk, i << BDRV_SECTOR_BITS,
|
||||
BDRV_SECTOR_SIZE, iobuf, 0) < 0) {
|
||||
printf("%s: write error in sector %" PRIu64 "\n", __func__, i);
|
||||
}
|
||||
} else {
|
||||
addr = PAGE_START(addr);
|
||||
page = addr >> 9;
|
||||
if (blk_pread(s->blk, page << BDRV_SECTOR_BITS,
|
||||
BDRV_SECTOR_SIZE, iobuf, 0) < 0) {
|
||||
printf("%s: read error in sector %" PRIu64 "\n", __func__, page);
|
||||
}
|
||||
memset(iobuf + (addr & 0x1ff), 0xff, (~addr & 0x1ff) + 1);
|
||||
if (blk_pwrite(s->blk, page << BDRV_SECTOR_BITS,
|
||||
BDRV_SECTOR_SIZE, iobuf, 0) < 0) {
|
||||
printf("%s: write error in sector %" PRIu64 "\n", __func__, page);
|
||||
}
|
||||
|
||||
memset(iobuf, 0xff, 0x200);
|
||||
i = (addr & ~0x1ff) + 0x200;
|
||||
for (addr += ((NAND_PAGE_SIZE + OOB_SIZE) << s->erase_shift) - 0x200;
|
||||
i < addr; i += 0x200) {
|
||||
if (blk_pwrite(s->blk, i, BDRV_SECTOR_SIZE, iobuf, 0) < 0) {
|
||||
printf("%s: write error in sector %" PRIu64 "\n",
|
||||
__func__, i >> 9);
|
||||
}
|
||||
}
|
||||
|
||||
page = i >> 9;
|
||||
if (blk_pread(s->blk, page << BDRV_SECTOR_BITS,
|
||||
BDRV_SECTOR_SIZE, iobuf, 0) < 0) {
|
||||
printf("%s: read error in sector %" PRIu64 "\n", __func__, page);
|
||||
}
|
||||
memset(iobuf, 0xff, ((addr - 1) & 0x1ff) + 1);
|
||||
if (blk_pwrite(s->blk, page << BDRV_SECTOR_BITS,
|
||||
BDRV_SECTOR_SIZE, iobuf, 0) < 0) {
|
||||
printf("%s: write error in sector %" PRIu64 "\n", __func__, page);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static bool glue(nand_blk_load_, NAND_PAGE_SIZE)(NANDFlashState *s,
|
||||
uint64_t addr, unsigned offset)
|
||||
{
|
||||
if (PAGE(addr) >= s->pages) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if (offset > NAND_PAGE_SIZE + OOB_SIZE) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if (s->blk) {
|
||||
if (s->mem_oob) {
|
||||
if (blk_pread(s->blk, SECTOR(addr) << BDRV_SECTOR_BITS,
|
||||
PAGE_SECTORS << BDRV_SECTOR_BITS, s->io, 0) < 0) {
|
||||
printf("%s: read error in sector %" PRIu64 "\n",
|
||||
__func__, SECTOR(addr));
|
||||
}
|
||||
memcpy(s->io + SECTOR_OFFSET(s->addr) + NAND_PAGE_SIZE,
|
||||
s->storage + (PAGE(s->addr) << OOB_SHIFT),
|
||||
OOB_SIZE);
|
||||
s->ioaddr = s->io + SECTOR_OFFSET(s->addr) + offset;
|
||||
} else {
|
||||
if (blk_pread(s->blk, PAGE_START(addr),
|
||||
(PAGE_SECTORS + 2) << BDRV_SECTOR_BITS, s->io, 0)
|
||||
< 0) {
|
||||
printf("%s: read error in sector %" PRIu64 "\n",
|
||||
__func__, PAGE_START(addr) >> 9);
|
||||
}
|
||||
s->ioaddr = s->io + (PAGE_START(addr) & 0x1ff) + offset;
|
||||
}
|
||||
} else {
|
||||
memcpy(s->io, s->storage + PAGE_START(s->addr) +
|
||||
offset, NAND_PAGE_SIZE + OOB_SIZE - offset);
|
||||
s->ioaddr = s->io;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
static void glue(nand_init_, NAND_PAGE_SIZE)(NANDFlashState *s)
|
||||
{
|
||||
s->oob_shift = PAGE_SHIFT - 5;
|
||||
s->pages = s->size >> PAGE_SHIFT;
|
||||
s->addr_shift = ADDR_SHIFT;
|
||||
|
||||
s->blk_erase = glue(nand_blk_erase_, NAND_PAGE_SIZE);
|
||||
s->blk_write = glue(nand_blk_write_, NAND_PAGE_SIZE);
|
||||
s->blk_load = glue(nand_blk_load_, NAND_PAGE_SIZE);
|
||||
}
|
||||
|
||||
# undef NAND_PAGE_SIZE
|
||||
# undef PAGE_SHIFT
|
||||
# undef PAGE_SECTORS
|
||||
# undef ADDR_SHIFT
|
||||
#endif /* NAND_IO */
|
|
@ -28,7 +28,8 @@
|
|||
#include "hw/misc/npcm7xx_mft.h"
|
||||
#include "hw/misc/npcm7xx_pwm.h"
|
||||
#include "hw/misc/npcm7xx_rng.h"
|
||||
#include "hw/net/npcm7xx_emc.h"
|
||||
#include "hw/net/npcm_gmac.h"
|
||||
#include "hw/net/npcm_pcs.h"
|
||||
#include "hw/nvram/npcm7xx_otp.h"
|
||||
#include "hw/sd/npcm7xx_sdhci.h"
|
||||
#include "hw/timer/npcm7xx_timer.h"
|
||||
|
@ -99,6 +100,8 @@ struct NPCM8xxState {
|
|||
EHCISysBusState ehci[2];
|
||||
OHCISysBusState ohci[2];
|
||||
NPCM7xxFIUState fiu[3];
|
||||
NPCMGMACState gmac[4];
|
||||
NPCMPCSState pcs;
|
||||
NPCM7xxSDHCIState mmc;
|
||||
NPCMPSPIState pspi;
|
||||
};
|
||||
|
|
|
@ -44,24 +44,6 @@ PFlashCFI02 *pflash_cfi02_register(hwaddr base,
|
|||
uint16_t unlock_addr1,
|
||||
int be);
|
||||
|
||||
/* nand.c */
|
||||
DeviceState *nand_init(BlockBackend *blk, int manf_id, int chip_id);
|
||||
void nand_setpins(DeviceState *dev, uint8_t cle, uint8_t ale,
|
||||
uint8_t ce, uint8_t wp, uint8_t gnd);
|
||||
void nand_getpins(DeviceState *dev, int *rb);
|
||||
void nand_setio(DeviceState *dev, uint32_t value);
|
||||
uint32_t nand_getio(DeviceState *dev);
|
||||
uint32_t nand_getbuswidth(DeviceState *dev);
|
||||
|
||||
#define NAND_MFR_TOSHIBA 0x98
|
||||
#define NAND_MFR_SAMSUNG 0xec
|
||||
#define NAND_MFR_FUJITSU 0x04
|
||||
#define NAND_MFR_NATIONAL 0x8f
|
||||
#define NAND_MFR_RENESAS 0x07
|
||||
#define NAND_MFR_STMICRO 0x20
|
||||
#define NAND_MFR_HYNIX 0xad
|
||||
#define NAND_MFR_MICRON 0x2c
|
||||
|
||||
/* m25p80.c */
|
||||
|
||||
#define TYPE_M25P80 "m25p80-generic"
|
||||
|
|
|
@ -30,6 +30,7 @@
|
|||
#include "qapi/qapi-commands-misc-arm.h"
|
||||
#include "qobject/qdict.h"
|
||||
#include "qom/qom-qobject.h"
|
||||
#include "cpu.h"
|
||||
|
||||
static GICCapability *gic_cap_new(int version)
|
||||
{
|
||||
|
|
|
@ -23,6 +23,7 @@
|
|||
|
||||
#include "hw/registerfields.h"
|
||||
#include "target/arm/kvm-consts.h"
|
||||
#include "cpu.h"
|
||||
|
||||
/*
|
||||
* ARMCPRegInfo type field bits:
|
||||
|
|
|
@ -22,6 +22,7 @@
|
|||
|
||||
#include "hw/registerfields.h"
|
||||
#include "qemu/host-utils.h"
|
||||
#include "cpu.h"
|
||||
|
||||
/*
|
||||
* Naming convention for isar_feature functions:
|
||||
|
|
20
target/arm/hvf-stub.c
Normal file
20
target/arm/hvf-stub.c
Normal file
|
@ -0,0 +1,20 @@
|
|||
/*
|
||||
* QEMU Hypervisor.framework (HVF) stubs for ARM
|
||||
*
|
||||
* Copyright (c) Linaro
|
||||
*
|
||||
* SPDX-License-Identifier: GPL-2.0-or-later
|
||||
*/
|
||||
|
||||
#include "qemu/osdep.h"
|
||||
#include "hvf_arm.h"
|
||||
|
||||
uint32_t hvf_arm_get_default_ipa_bit_size(void)
|
||||
{
|
||||
g_assert_not_reached();
|
||||
}
|
||||
|
||||
uint32_t hvf_arm_get_max_ipa_bit_size(void)
|
||||
{
|
||||
g_assert_not_reached();
|
||||
}
|
|
@ -11,7 +11,7 @@
|
|||
#ifndef QEMU_HVF_ARM_H
|
||||
#define QEMU_HVF_ARM_H
|
||||
|
||||
#include "cpu.h"
|
||||
#include "target/arm/cpu-qom.h"
|
||||
|
||||
/**
|
||||
* hvf_arm_init_debug() - initialize guest debug capabilities
|
||||
|
@ -22,23 +22,7 @@ void hvf_arm_init_debug(void);
|
|||
|
||||
void hvf_arm_set_cpu_features_from_host(ARMCPU *cpu);
|
||||
|
||||
#ifdef CONFIG_HVF
|
||||
|
||||
uint32_t hvf_arm_get_default_ipa_bit_size(void);
|
||||
uint32_t hvf_arm_get_max_ipa_bit_size(void);
|
||||
|
||||
#else
|
||||
|
||||
static inline uint32_t hvf_arm_get_default_ipa_bit_size(void)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
static inline uint32_t hvf_arm_get_max_ipa_bit_size(void)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
#endif
|
||||
|
|
|
@ -12,6 +12,7 @@
|
|||
#define QEMU_KVM_ARM_H
|
||||
|
||||
#include "system/kvm.h"
|
||||
#include "target/arm/cpu-qom.h"
|
||||
|
||||
#define KVM_ARM_VGIC_V2 (1 << 0)
|
||||
#define KVM_ARM_VGIC_V3 (1 << 1)
|
||||
|
|
|
@ -3,7 +3,6 @@ arm_common_ss = ss.source_set()
|
|||
arm_ss.add(files(
|
||||
'gdbstub.c',
|
||||
))
|
||||
arm_ss.add(zlib)
|
||||
|
||||
arm_ss.add(when: 'TARGET_AARCH64', if_true: files(
|
||||
'cpu64.c',
|
||||
|
@ -32,6 +31,7 @@ arm_common_system_ss.add(files('cpu.c'))
|
|||
arm_common_system_ss.add(when: 'TARGET_AARCH64', if_false: files(
|
||||
'cpu32-stubs.c'))
|
||||
arm_common_system_ss.add(when: 'CONFIG_KVM', if_false: files('kvm-stub.c'))
|
||||
arm_common_system_ss.add(when: 'CONFIG_HVF', if_false: files('hvf-stub.c'))
|
||||
arm_common_system_ss.add(files(
|
||||
'arch_dump.c',
|
||||
'arm-powerctl.c',
|
||||
|
@ -48,7 +48,7 @@ subdir('hvf')
|
|||
if 'CONFIG_TCG' in config_all_accel
|
||||
subdir('tcg')
|
||||
else
|
||||
arm_ss.add(files('tcg-stubs.c'))
|
||||
arm_common_system_ss.add(files('tcg-stubs.c'))
|
||||
endif
|
||||
|
||||
target_arch += {'arm': arm_ss}
|
||||
|
|
|
@ -56,6 +56,8 @@ arm_system_ss.add(files(
|
|||
arm_system_ss.add(when: 'CONFIG_ARM_V7M', if_true: files('cpu-v7m.c'))
|
||||
arm_user_ss.add(when: 'TARGET_AARCH64', if_false: files('cpu-v7m.c'))
|
||||
|
||||
arm_common_ss.add(zlib)
|
||||
|
||||
arm_common_ss.add(files(
|
||||
'arith_helper.c',
|
||||
'crypto_helper.c',
|
||||
|
|
|
@ -137,6 +137,7 @@ tests_arm_system_thorough = [
|
|||
'arm_raspi2',
|
||||
'arm_replay',
|
||||
'arm_smdkc210',
|
||||
'arm_stellaris',
|
||||
'arm_sx1',
|
||||
'arm_vexpress',
|
||||
'arm_virt',
|
||||
|
|
48
tests/functional/test_arm_stellaris.py
Executable file
48
tests/functional/test_arm_stellaris.py
Executable file
|
@ -0,0 +1,48 @@
|
|||
#!/usr/bin/env python3
|
||||
#
|
||||
# Functional test that checks the serial console of the stellaris machines
|
||||
#
|
||||
# SPDX-License-Identifier: GPL-2.0-or-later
|
||||
|
||||
from qemu_test import QemuSystemTest, Asset, exec_command_and_wait_for_pattern
|
||||
from qemu_test import wait_for_console_pattern
|
||||
|
||||
|
||||
class StellarisMachine(QemuSystemTest):
|
||||
|
||||
ASSET_DAY22 = Asset(
|
||||
'https://www.qemu-advent-calendar.org/2023/download/day22.tar.gz',
|
||||
'ae3a63ef4b7a22c21bfc7fc0d85e402fe95e223308ed23ac854405016431ff51')
|
||||
|
||||
def test_lm3s6965evb(self):
|
||||
self.set_machine('lm3s6965evb')
|
||||
kernel_path = self.archive_extract(self.ASSET_DAY22,
|
||||
member='day22/day22.bin')
|
||||
self.vm.set_console()
|
||||
self.vm.add_args('-kernel', kernel_path)
|
||||
self.vm.launch()
|
||||
|
||||
wait_for_console_pattern(self, 'In a one horse open')
|
||||
|
||||
ASSET_NOTMAIN = Asset(
|
||||
'https://github.com/Ahelion/QemuArmM4FDemoSw/raw/master/build/notmain.bin',
|
||||
'6ceda031aa081a420fca2fca9e137fa681d6e3820d820ad1917736cb265e611a')
|
||||
|
||||
def test_lm3s811evb(self):
|
||||
self.set_machine('lm3s811evb')
|
||||
kernel_path = self.ASSET_NOTMAIN.fetch()
|
||||
|
||||
self.vm.set_console()
|
||||
self.vm.add_args('-cpu', 'cortex-m4')
|
||||
self.vm.add_args('-kernel', kernel_path)
|
||||
self.vm.launch()
|
||||
|
||||
# The test kernel emits an initial '!' and then waits for input.
|
||||
# For each character that we send it responds with a certain
|
||||
# other ASCII character.
|
||||
wait_for_console_pattern(self, '!')
|
||||
exec_command_and_wait_for_pattern(self, '789', 'cdf')
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
QemuSystemTest.main()
|
|
@ -208,9 +208,10 @@ qtests_npcm7xx = \
|
|||
'npcm7xx_sdhci-test',
|
||||
'npcm7xx_smbus-test',
|
||||
'npcm7xx_timer-test',
|
||||
'npcm7xx_watchdog_timer-test',
|
||||
'npcm_gmac-test'] + \
|
||||
'npcm7xx_watchdog_timer-test'] + \
|
||||
(slirp.found() ? ['npcm7xx_emc-test'] : [])
|
||||
qtests_npcm8xx = \
|
||||
['npcm_gmac-test']
|
||||
qtests_aspeed = \
|
||||
['aspeed_gpio-test',
|
||||
'aspeed_hace-test',
|
||||
|
@ -259,6 +260,7 @@ qtests_aarch64 = \
|
|||
(config_all_accel.has_key('CONFIG_TCG') and \
|
||||
config_all_devices.has_key('CONFIG_TPM_TIS_I2C') ? ['tpm-tis-i2c-test'] : []) + \
|
||||
(config_all_devices.has_key('CONFIG_ASPEED_SOC') ? qtests_aspeed64 : []) + \
|
||||
(config_all_devices.has_key('CONFIG_NPCM8XX') ? qtests_npcm8xx : []) + \
|
||||
['arm-cpu-features',
|
||||
'numa-test',
|
||||
'boot-serial-test',
|
||||
|
|
|
@ -36,7 +36,7 @@ typedef struct TestData {
|
|||
const GMACModule *module;
|
||||
} TestData;
|
||||
|
||||
/* Values extracted from hw/arm/npcm7xx.c */
|
||||
/* Values extracted from hw/arm/npcm8xx.c */
|
||||
static const GMACModule gmac_module_list[] = {
|
||||
{
|
||||
.irq = 14,
|
||||
|
@ -46,6 +46,14 @@ static const GMACModule gmac_module_list[] = {
|
|||
.irq = 15,
|
||||
.base_addr = 0xf0804000
|
||||
},
|
||||
{
|
||||
.irq = 16,
|
||||
.base_addr = 0xf0806000
|
||||
},
|
||||
{
|
||||
.irq = 17,
|
||||
.base_addr = 0xf0808000
|
||||
}
|
||||
};
|
||||
|
||||
/* Returns the index of the GMAC module. */
|
||||
|
@ -174,18 +182,32 @@ static uint32_t gmac_read(QTestState *qts, const GMACModule *mod,
|
|||
return qtest_readl(qts, mod->base_addr + regno);
|
||||
}
|
||||
|
||||
static uint16_t pcs_read(QTestState *qts, const GMACModule *mod,
|
||||
NPCMRegister regno)
|
||||
{
|
||||
uint32_t write_value = (regno & 0x3ffe00) >> 9;
|
||||
qtest_writel(qts, PCS_BASE_ADDRESS + NPCM_PCS_IND_AC_BA, write_value);
|
||||
uint32_t read_offset = regno & 0x1ff;
|
||||
return qtest_readl(qts, PCS_BASE_ADDRESS + read_offset);
|
||||
}
|
||||
|
||||
/* Check that GMAC registers are reset to default value */
|
||||
static void test_init(gconstpointer test_data)
|
||||
{
|
||||
const TestData *td = test_data;
|
||||
const GMACModule *mod = td->module;
|
||||
QTestState *qts = qtest_init("-machine npcm750-evb");
|
||||
QTestState *qts = qtest_init("-machine npcm845-evb");
|
||||
|
||||
#define CHECK_REG32(regno, value) \
|
||||
do { \
|
||||
g_assert_cmphex(gmac_read(qts, mod, (regno)), ==, (value)); \
|
||||
} while (0)
|
||||
|
||||
#define CHECK_REG_PCS(regno, value) \
|
||||
do { \
|
||||
g_assert_cmphex(pcs_read(qts, mod, (regno)), ==, (value)); \
|
||||
} while (0)
|
||||
|
||||
CHECK_REG32(NPCM_DMA_BUS_MODE, 0x00020100);
|
||||
CHECK_REG32(NPCM_DMA_XMT_POLL_DEMAND, 0);
|
||||
CHECK_REG32(NPCM_DMA_RCV_POLL_DEMAND, 0);
|
||||
|
@ -235,6 +257,63 @@ static void test_init(gconstpointer test_data)
|
|||
CHECK_REG32(NPCM_GMAC_PTP_TAR, 0);
|
||||
CHECK_REG32(NPCM_GMAC_PTP_TTSR, 0);
|
||||
|
||||
if (mod->base_addr == 0xf0802000) {
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_CTL_ID1, 0x699e);
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_CTL_ID2, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_CTL_STS, 0x8000);
|
||||
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_MII_CTRL, 0x1140);
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_MII_STS, 0x0109);
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_MII_DEV_ID1, 0x699e);
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_MII_DEV_ID2, 0x0ced0);
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_MII_AN_ADV, 0x0020);
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_MII_LP_BABL, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_MII_AN_EXPN, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_MII_EXT_STS, 0xc000);
|
||||
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_TIM_SYNC_ABL, 0x0003);
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_TIM_SYNC_TX_MAX_DLY_LWR, 0x0038);
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_TIM_SYNC_TX_MAX_DLY_UPR, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_TIM_SYNC_TX_MIN_DLY_LWR, 0x0038);
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_TIM_SYNC_TX_MIN_DLY_UPR, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_TIM_SYNC_RX_MAX_DLY_LWR, 0x0058);
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_TIM_SYNC_RX_MAX_DLY_UPR, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_TIM_SYNC_RX_MIN_DLY_LWR, 0x0048);
|
||||
CHECK_REG_PCS(NPCM_PCS_SR_TIM_SYNC_RX_MIN_DLY_UPR, 0);
|
||||
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_MMD_DIG_CTRL1, 0x2400);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_AN_CTRL, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_AN_INTR_STS, 0x000a);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_TC, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_DBG_CTRL, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_EEE_MCTRL0, 0x899c);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_EEE_TXTIMER, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_EEE_RXTIMER, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_LINK_TIMER_CTRL, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_EEE_MCTRL1, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_DIG_STS, 0x0010);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_ICG_ERRCNT1, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_MISC_STS, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_RX_LSTS, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_MP_TX_BSTCTRL0, 0x00a);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_MP_TX_LVLCTRL0, 0x007f);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_MP_TX_GENCTRL0, 0x0001);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_MP_TX_GENCTRL1, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_MP_TX_STS, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_MP_RX_GENCTRL0, 0x0100);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_MP_RX_GENCTRL1, 0x1100);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_MP_RX_LOS_CTRL0, 0x000e);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_MP_MPLL_CTRL0, 0x0100);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_MP_MPLL_CTRL1, 0x0032);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_MP_MPLL_STS, 0x0001);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_MP_MISC_CTRL2, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_MP_LVL_CTRL, 0x0019);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_MP_MISC_CTRL0, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_MP_MISC_CTRL1, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_DIG_CTRL2, 0);
|
||||
CHECK_REG_PCS(NPCM_PCS_VR_MII_DIG_ERRCNT_SEL, 0);
|
||||
}
|
||||
|
||||
qtest_quit(qts);
|
||||
}
|
||||
|
||||
|
@ -242,7 +321,7 @@ static void gmac_add_test(const char *name, const TestData* td,
|
|||
GTestDataFunc fn)
|
||||
{
|
||||
g_autofree char *full_name = g_strdup_printf(
|
||||
"npcm7xx_gmac/gmac[%d]/%s", gmac_module_index(td->module), name);
|
||||
"npcm8xx_gmac/gmac[%d]/%s", gmac_module_index(td->module), name);
|
||||
qtest_add_data_func(full_name, td, fn);
|
||||
}
|
||||
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue