zoukankan      html  css  js  c++  java
  • Oracle bbed 五个 实用示例

    在之前的blog 里介绍了BBED 工具的语法部分。 参考:

           Oracle BBED 工具 说明

           http://blog.csdn.net/tianlesoftware/article/details/5006580

    在这篇主要看一下BBED 工具的几个使用示例。

           Althoughbbed can modify data in the data files of an open Oracle database, it isadvisable to shut down the database before making any changes. This avoids thecheckpoint process overwriting the changes made with bbed from the Oracle blockcache. It also avoids Oracle reading the block before the modifications arecomplete and declaring the block corrupt.

           虽然bbed 可以在db open 状态来进行修改,但是建议在做任何修改操作之前先shutdown db。 这样避免checkpoint 进程重写bbed 对block 的修改。 也避免oracle 在bbed 修改完成之前读block 或者申明block 为corrupt。

           Important:Using bbed to modify the contents of an Oracle data block renders the dataun-supported by Oracle. These examples should be used for educational purposesonly. If they are used on real production databases they should only be used asa last resort and once the immediate problem has been resolved, all retrievabledata should be exported and a new database created.

           bbed工具不受Oracle 的技术支持。

           Althoughbbed can be used to open a database that would otherwise be beyond salvaging,the DBA must bear in mind that the internal tables such as OBJ$, UET$ and FET$may no longer match the contents of the data blocks. The behavior of thedatabase will therefore be unpredictable and ORA-600 errors are likely.

    一. 示例: 修改Data 内容

     1.1 连接bbed

    [oracle@db2 ~]$ bbed parfile=/u01/bbed.par

    Password:

    BBED: Release 2.0.0.0.0 - LimitedProduction on Fri Aug 12 18:26:46 2011

    Copyright (c) 1982, 2005, Oracle.  All rights reserved.

    ************* !!! For Oracle Internal Useonly !!! ***************

    BBED>

    1.2 查看要修改的内容

    SYS@dave2(db2)> select * from dvd;

    JOB

    --------------------------------------------------------------------------------

    Dave is DBA!

    Dave like Oracle!

    注意: bbed 的修改仅仅是对原有位置内容的一个替换。

    对应block 的信息如下:

    SYS@dave2(db2)> select

      2   rowid,

     3 dbms_rowid.rowid_relative_fno(rowid)rel_fno,

     4 dbms_rowid.rowid_block_number(rowid)blockno,

     5 dbms_rowid.rowid_row_number(rowid) rowno

     6  from dvd;

    ROWID                 REL_FNO    BLOCKNO     ROWNO

    ------------------ ---------- --------------------

    AAAN9VAABAAAcKiAAA          1    115362          0

    AAAN9VAABAAAcKiAAB          1    115362          1

    SYS@dave2(db2)>

    1.3 查找关键字Dave,确定其在block中的偏移量offset。

    BBED> set dba 1,115362 offset 0

           DBA             0x0041c2a2(4309666 1,115362)

           OFFSET          0

    BBED> find /c Dave

     File:/u01/app/oracle/oradata/dave2/system01.dbf (1)

     Block: 115362           Offsets: 8176 to 8191           Dba:0x0041c2a2

    ------------------------------------------------------------------------

     44617665 20697320 44424121 020616b3

     <32 bytes per line>

    dump 查看具体内容:

    BBED> dump /v dba 1,115362 offset 8176count 128

     File: /u01/app/oracle/oradata/dave2/system01.dbf(1)

     Block: 115362 Offsets: 8176 to 8191 Dba:0x0041c2a2

    -------------------------------------------------------

     44617665 20697320 44424121 020616b3 l Dave isDBA!...³

     <16 bytes per line>

    注意这里面的Offsets:8176 to 8191, 它指的是这一行的一个地址。其中

    D 的offset 是8176

    a 的offset 是8177

    v 的offset 是8178

    e 的offset 是8179

    空格也算offset。

    1.4 修改block,将Dave 换成DMM

    BBED> modify /c 'DMM ' dba 1,115362offset 8176

     File:/u01/app/oracle/oradata/dave2/system01.dbf (1)

     Block: 115362           Offsets: 8176 to 8191           Dba:0x0041c2a2

    ------------------------------------------------------------------------

     444d4d20 20697320 44424121 020616b3

     <32 bytes per line>

    --注意这里DMM我用单引号括起来,并且最后还有一个空格,这样就是4个bytes,不用单引号括起来,无法表示空格,验证一下

    BBED> dump /v dba 1,115362 offset 8176count 128

     File:/u01/app/oracle/oradata/dave2/system01.dbf (1)

     Block: 115362 Offsets: 8176 to 8191 Dba:0x0041c2a2

    -------------------------------------------------------

     444d4d20 20697320 44424121 020616b3 l DMM  is DBA!...³

     <16 bytes per line>

    1.5 应用变更

    BBED> sum dba 1,115362

    Check value for File 1, Block 115362:

    current = 0xdef7, required = 0x8cc0

    此时 current checksum 是0xdef7,requiredchecksum 是0x8cc0

    BBED> sum dba 1,115362 apply

    Check value for File 1, Block 115362:

    current = 0x8cc0, required = 0x8cc0

    加上apply参数,使checksum一致。即之前的修改生效。

    SYS@dave2(db2)> alter system flushbuffer_cache;

    System altered.

    SYS@dave2(db2)> select * from dvd;

    JOB

    --------------------------------------------------------------------------------

    DMM  is DBA!

    Dave like Oracle!

    二. 示例:恢复delete 的rows

           Whenrows are deleted in Oracle the data is not actually removed. The row is simplymarked as deleted and the free space counters and pointers adjustedaccordingly. The status of a row is stored in the Row Header which occupies thefirst few bytes of each row.

           当row 被delete 的时候,实际上data 并没有被remove,只是将该row 标记为delete,然后其对应的空间被统计为free space。 row 的status 存在每个row的row header里。

           TheRow Header consists of the Row Flag, Lock Byte (ITL entry) and Column Count.The first of these - the Row Flag - is a single byte that holds a bitmask thatshows the status of the row. The bitmask is decoded as follows:

           RowHeader 包含Row Flag,Lock Byte(ITL)和column Count。其中Row Flag占用1个byte,并且以bitmask 来保存。bitmask 的解释如下:

    Cluster Key

    Cluster Table Member

    Head of row piece

    Deleted

    First data piece

    Last data piece

    1st Column continues from previous piece

    Last column continues in next piece

    128

    64

    32

    16

    8

    4

    2

    1

    我们dump 一个block,看一个row Flag,来帮助理解这个bitmask。

    SYS@dave2(db2)> alter system dump datafile1 block 115362;

    System altered.

    SYS@dave2(db2)> oradebug setmypid

    Statement processed.

    SYS@dave2(db2)> oradebug tracefile_name

    /u01/app/oracle/admin/dave2/udump/dave2_ora_9396.trc

    trace file有关row的信息如下:

    block_row_dump:

    tab 0, row 0, @0x1f90

    tl: 16 fb: --H-FL--lb: 0x1  cc: 1

    col 0: [12]  44 4d 4d 20 20 69 73 2044 42 41 21

    tab 0, row 1, @0x1f7b

    tl: 21 fb: --H-FL-- lb: 0x2  cc: 1

    col 0: [17]  64 6d 6d 65 20 6c 69 6b65 20 4f 72 61 63 6c 65 21

    end_of_block_dump

    我们的表dvd里只有2行记录,所以这里显示的row 为2.

    注意这里的fb: --H-FL--。 其有8个选项,每个值分别与bitmask 对应。

           Therefore,columns that fit within a single block, are not chained, migrated or part of aclustered table and are not deleted will have the following attributes:

           (1)Head of Row Piece

           (2)First Data Piece

           (3)Last Data Piece

           如果一个row 没有被删除,那么它就具有上面的3个属性,即Flag 表示为:--H-FL--. 这里的字母分别代表属性的首字母。其对应的值:32 + 8 + 4 =44 or 0x2c.

           如果一个row 被delete了,那么row flag 就会更新,bitmask 里的deleted 被设置为16. 此时row flag 为: 32 + 16 + 8 + 4 = 60 or 0x3c.

    验证一下:

    SYS@dave2(db2)> delete from dvd whererownum=1;

    1 row deleted.

    SYS@dave2(db2)> commit;

    Commit complete.

    查看dump 的标记:

    block_row_dump:

    tab 0, row 0, @0x1f90

    tl: 2 fb: --HDFL--lb: 0x1

    tab 0, row 1, @0x1f7b

    tl: 21 fb: --H-FL-- lb: 0x0  cc: 1

    col 0: [17]  64 6d 6d 65 20 6c 69 6b65 20 4f 72 61 63 6c 65 21

    end_of_block_dump

    这里的row 1flag 变成了--HDFL--。

     

    现在我们用bbed 将删除的row 1 内容找回来。

    BBED> set dba1,115362 offset 0

           DBA             0x0041c2a2(4309666 1,115362)

           OFFSET          0

    BBED> find /c DMM

     File: /u01/app/oracle/oradata/dave2/system01.dbf(1)

     Block: 115362           Offsets: 8176 to 8191           Dba:0x0041c2a2

    ------------------------------------------------------------------------

     444d4d20 20697320 44424121 020616b3

     <32 bytes per line>

    BBED> d /v dba 1,115362 offset 8176count 128

     File:/u01/app/oracle/oradata/dave2/system01.dbf (1)

     Block: 115362 Offsets: 8176 to 8191 Dba:0x0041c2a2

    -------------------------------------------------------

     444d4d20 20697320 44424121 020616b3 l DMM  is DBA!...³

     <16 bytes per line>

    注意:我们还是可以通过dump查看我们delete 掉的row记录。但是在sql里用select 已经看出到了。 这个也证明,delete 并未真正的删除data。

    我们的row 的内容保存在offset 8176的位置,我们将offset 往前移动一段,在dump,来确定row header的内容。

     

    这个移位有一定的规律。 我们看一下:

     

    BBED> d /v dba 1,115362 offset 8176count 128

     File: /u01/app/oracle/oradata/dave2/system01.dbf(1)

     Block: 115362 Offsets: 8176 to 8191 Dba:0x0041c2a2

    -------------------------------------------------------

     444d4d20 20697320 44424121 020616b3 l DMM  is DBA!...³

    我们的一条记录是从444d4d20开始的,此时的offsets8176开始的。我们offset 减小一位,在dump:

    BBED> d /v dba 1,115362 offset 8175

     File:/u01/app/oracle/oradata/dave2/system01.dbf (1)

     Block: 115362 Offsets: 8175 to 8191 Dba:0x0041c2a2

    -------------------------------------------------------

     0c444d4d 20206973 20444241 210206d7 l.DMM  is DBA!..×

     73                                  l s

     <16 bytes per line>

    此时dump 的内容多了2个字符,而一个完整的是8个字符,所以要想完整的显示,一次要减少4个offsets。

    BBED> d /v dba 1,115362 offset 8172

     File:/u01/app/oracle/oradata/dave2/system01.dbf (1)

     Block: 115362 Offsets: 8172 to 8191 Dba:0x0041c2a2

    -------------------------------------------------------

     3c01010c 444d4d20 20697320 44424121 l <...DMM  is DBA!

     0206d773                            l ..×s

     <16 bytes per line>

           这里已经出现了我们3c(deleted)标志,但是注意这里的位置的根据我们的查找的字符串来分的,实际在block里的分割方式不一样按照我们的offset 来进行。 我们可以通过row directory 来进行一个确认。

    我们print row directory 确认一下:

    BBED> p kdbr

    sb2 kdbr[0]                                 @110      8080

    sb2 kdbr[1]                                 @112      8059

    BBED> p *kdbr[0]

    rowdata[21]

    -----------

    ub1 rowdata[21]                             @8172     0x3c

    BBED> p *kdbr[1]

    rowdata[0]

    ----------

    ub1 rowdata[0]                              @8151     0x2c

           通过row directory,我们可以确认对应row记录的row header保存在offset 8172的位置,值为3c。 我们find 字符串的目的就是为了和rowdirectory 中的offset 进行比较。 他们相近时,就可以确定。

           现在我们将@8172位置的3c 变成2c。 即从deleted 变成正常。

    BBED> modify /x 2c offset 8172

     File:/u01/app/oracle/oradata/dave2/system01.dbf (1)

     Block: 115362           Offsets: 8172 to 8191           Dba:0x0041c2a2

    ------------------------------------------------------------------------

     2c01010c 444d4d20 20697320 44424121 0206d773

     <32 bytes per line>

    BBED> sum apply

    Check value for File 1, Block 115362:

    current = 0x2cb4, required = 0x2cb4

    --flush buffer cache,然后查询

    SYS@dave2(db2)> alter system flushbuffer_cache;

    System altered.

    SYS@dave2(db2)> select * from dvd;

    JOB

    --------------------------------------------------------------------------------

    DMM is DBA!

    Dave like Oracle!

    之前delete 的数据已经恢复出来。

    三. 示例:Recoveringdeleted/damaged Data

           可是使用BBED 的copy 命令来从旧的data file中copy block,从而恢复已经删除或者删除的记录。

    先将我们的表dvd 移动到我们的单独的datafile里:

    SYS@dave2(db2)> selectfile_name,tablespace_name from dba_data_files where file_id=6;

    FILE_NAME                                     TABLESPACE_NAME

    ---------------------------------------------------------------------------

    /u01/app/oracle/oradata/dave2/dave01.dbf      DAVE2

    SYS@dave2(db2)> alter table dvd movetablespace DAVE2;

    Table altered.

    SYS@dave2(db2)> select table_name,tablespace_namefrom dba_tables where table_name='DVD';

    TABLE_NAME                     TABLESPACE_NAME

    ------------------------------------------------------------

    DVD                            DAVE2

    SYS@dave2(db2)> select * from dvd;

    JOB

    --------------------------------------------------------------------------------

    DMM is DBA!

    dmme like Oracle!

    --shutdown db,将dave01.dbfcopy 一份做恢复用

    SYS@dave2(db2)> shutdown immediate

    Database closed.

    Database dismounted.

    ORACLE instance shut down.

    [oracle@db2 ~]$ cd  /u01/app/oracle/oradata/dave2/

    [oracle@db2 dave2]$ ls

    control01.ctl  dave01.dbf      redo01.log  sysaux01.dbf undotbs01.dbf

    control02.ctl  example01.dbf   redo02.log system01.dbf  undotbs02.dbf

    control03.ctl  huaining01.dbf  redo03.log temp01.dbf    users01.dbf

    [oracle@db2 dave2]$ cp dave01.dbfdave01.dbf.bak

    [oracle@db2 dave2]$ ls

    control01.ctl  dave01.dbf      huaining01.dbf  redo03.log   temp01.dbf     users01.dbf

    control02.ctl  dave01.dbf.bak  redo01.log      sysaux01.dbf  undotbs01.dbf

    control03.ctl  example01.dbf   redo02.log      system01.dbf  undotbs02.dbf

    将copy 的bak datafile 添加到bbed 的parfile里面

    [oracle@db2 u01]$ cat filelist.txt

    1/u01/app/oracle/oradata/dave2/system01.dbf 1761607680

    2/u01/app/oracle/oradata/dave2/undotbs01.dbf 927989760

    3/u01/app/oracle/oradata/dave2/sysaux01.dbf 398458880

    4 /u01/app/oracle/oradata/dave2/users01.dbf5242880

    5/u01/app/oracle/oradata/dave2/example01.dbf 104857600

    6 /u01/app/oracle/oradata/dave2/dave01.dbf10485760

    7 /u01/app/oracle/oradata/dave2/undotbs02.dbf1048576

    8/u01/app/oracle/oradata/dave2/huaining01.dbf 52428800

    9/u01/app/oracle/oradata/dave2/dave01.dbf.bak 10485760

    最后一个9 是我们添加的。

     

    --启动db

    SYS@dave2(db2)> startup

    ORACLE instance started.

    Total System Global Area  239075328 bytes

    Fixed Size                  1218724 bytes

    Variable Size              71305052 bytes

    Database Buffers          163577856 bytes

    Redo Buffers                2973696 bytes

    Database mounted.

    Database opened.

    --查看表block的信息:

    select

    rowid,

    dbms_rowid.rowid_relative_fno(rowid)rel_fno,

    dbms_rowid.rowid_block_number(rowid)blockno,

    dbms_rowid.rowid_row_number(rowid) rowno

    from dvd;

    ROWID                 REL_FNO    BLOCKNO     ROWNO

    ------------------ ---------- --------------------

    AAAN9hAAGAAAAAcAAA          6         28          0

    AAAN9hAAGAAAAAcAAB          6         28          1

    每个block里都可能有多个row,如果表很大,那么就有返回很多条结果。 在这种情况下,这种SQL 语句就显得不够明显。

    SYS@dave2(db2)>  select owner, segment_name, header_file,header_block, blocks  fromdba_segments  where owner = 'SYS' andsegment_name = 'DVD';

    OWNER SEGMENT_NAME  HEADER_FILEHEADER_BLOCK   BLOCKS

    ---------- ------------- ----------------------- ----------

    SYS       DVD                     6           27          8

           从这个查询结果,我们可以看到,对象保存在datafile 6里,从27 的block 开始存储,占用8个blocks。

           这里要注意的一点是:dba_segments 视图里的block 是从0开始的统计的,而bbed 里是从1. 所以我们在bbed中指定block时,需要加1.

    [oracle@db2 ~]$ bbed parfile=/u01/bbed.par

    Password:

    BBED: Release 2.0.0.0.0 - LimitedProduction on Sat Aug 13 01:11:29 2011

    Copyright (c) 1982, 2005, Oracle.  All rights reserved.

    ************* !!! For Oracle Internal Useonly !!! ***************

    BBED> set dba 6,27 offset 0

           DBA             0x0180001b(25165851 6,27)

           OFFSET          0

    BBED> p ktbbh

    BBED-00400: invalid blocktype (35)

    --如果指定block27,会报错。 加1后就正常了。

    BBED> set dba 6,28 offset 0

           DBA             0x0180001c(25165852 6,28)

           OFFSET          0

    BBED> p ktbbh

    struct ktbbh, 96 bytes                      @20     

      ub1 ktbbhtyp                            @20       0x01 (KDDBTDATA)

      union ktbbhsid, 4 bytes                 @24     

         ub4 ktbbhsg1                         @24       0x0000df61

         ub4 ktbbhod1                         @24       0x0000df61

      struct ktbbhcsc, 8 bytes                @28     

         ub4 kscnbas                          @28       0x8007a9f4

         ub2 kscnwrp                          @32       0x0000

       b2ktbbhict                             @36       3

      ub1 ktbbhflg                            @38       0x32 (NONE)

      ub1 ktbbhfsl                            @39       0x00

      ub4 ktbbhfnx                            @40       0x01800019

      struct ktbbhitl[0], 24 bytes            @44     

         struct ktbitxid, 8 bytes             @44     

            ub2 kxidusn                        @44       0x0003

            ub2 kxidslt                       @46       0x0010

            ub4 kxidsqn                       @48       0x00000a3b

         struct ktbituba, 8 bytes             @52     

            ub4 kubadba                       @52       0x00000000

            ub2 kubaseq                       @56       0x0000

            ub1 kubarec                       @58       0x00

         ub2 ktbitflg                         @60       0x8000 (KTBFCOM)

         union _ktbitun, 2 bytes              @62     

            b2 _ktbitfsc                      @62       0

            ub2 _ktbitwrp                     @62       0x0000

         ub4 ktbitbas                         @64       0x8007a9dd

      struct ktbbhitl[1], 24 bytes            @68     

         struct ktbitxid, 8 bytes             @68     

            ub2 kxidusn                       @68       0x0000

            ub2 kxidslt                       @70       0x0000

            ub4 kxidsqn                       @72       0x00000000

         struct ktbituba, 8 bytes             @76     

            ub4 kubadba                       @76       0x00000000

            ub2 kubaseq                       @80       0x0000

            ub1 kubarec                       @82       0x00

         ub2 ktbitflg                         @84       0x0000 (NONE)

         union _ktbitun, 2 bytes              @86     

            b2 _ktbitfsc                      @86       0

            ub2 _ktbitwrp                     @86       0x0000

         ub4 ktbitbas                         @88       0x00000000

      struct ktbbhitl[2], 24 bytes            @92     

         struct ktbitxid, 8 bytes             @92     

            ub2 kxidusn                       @92       0x0000

            ub2 kxidslt                       @94       0x0000

            ub4 kxidsqn                        @96       0x00000000

         struct ktbituba, 8 bytes             @100    

            ub4 kubadba                       @100      0x00000000

            ub2 kubaseq                       @104      0x0000

            ub1 kubarec                        @106      0x00

         ub2 ktbitflg                         @108      0x0000 (NONE)

         union _ktbitun, 2 bytes              @110    

            b2 _ktbitfsc                      @110      0

            ub2 _ktbitwrp                     @110      0x0000

         ub4 ktbitbas                         @112      0x00000000

    删除表dvd里的所有数据

    SYS@dave2(db2)> delete from dvd;

    2 rows deleted.

    SYS@dave2(db2)> commit;

    Commit complete.

    使用bbed copy 从旧的datafile里恢复出来

    BBED> set width 65

           WIDTH           65

    BBED> info

     File# Name                                        Size(blks)

     ----- ----                                        ----------

        1 /u01/app/oracle/oradata/dave2/system01.dbf       215040

        2 /u01/app/oracle/oradata/dave2/undotbs01.dbf      113280

        3 /u01/app/oracle/oradata/dave2/sysaux01.dbf        48640

        4 /u01/app/oracle/oradata/dave2/users01.dbf           640

        5 /u01/app/oracle/oradata/dave2/example01.dbf       12800

        6 /u01/app/oracle/oradata/dave2/dave01.dbf           1280

        7 /u01/app/oracle/oradata/dave2/undotbs02.dbf         128

        8 /u01/app/oracle/oradata/dave2/huaining01.db        6400

        9 /u01/app/oracle/oradata/dave2/dave01.dbf.ba        1280

    从9 copy到6,8个block 全部要copy

    BBED> copy dba 9,28 to dba 6,28

     File: /u01/app/oracle/oradata/dave2/dave01.dbf(6)

     Block: 28           Offsets:    0 to 511       Dba:0x0180001c

    ------------------------------------

     06a20000 1c008001 f4a90780 00000104

     f5b40000 01000000 61df0000 f4a90780

    ....

    BBED> copy dba 9,28 to dba 6,28

    BBED> copy dba 9,29 to dba 6,29

    BBED>copy dba 9,30 to dba 6,30

    BBED> copy dba 9,31 to dba 6,31

    BBED> copy dba 9,32 to dba 6,32

    BBED> copy dba 9,33 to dba 6,33

    BBED> copy dba 9,34 to dba 6,34

    BBED> copy dba 9,35 to dba6,35

    --直接select 没有变化,发db 重启了一下,还原的数据就出现了

    SYS@dave2(db2)> select * from dvd;

    no rows selected

    SYS@dave2(db2)> startup force

    ORACLE instance started.

    Total System Global Area  239075328 bytes

    Fixed Size                  1218724 bytes

    Variable Size              62916444 bytes

    Database Buffers          171966464 bytes

    Redo Buffers                2973696 bytes

    Database mounted.

    Database opened.

    SYS@dave2(db2)> select * from dvd;

    JOB

    --------------------------------------------------------------------------------

    DMM is DBA!

    dmme like Oracle!

    四. 示例:File Header Reset

    在做db 做不完全恢复的时候,可能会遇到如下错误:

           ORA-01113:file 6 needs media recovery

           ORA-01110:data file 6: '+DATA/rac/datafile/dave01.dbf'

           当我们recover 时候,归档文件有丢失,就会报以上的错误。 在这种情况下,可以设置初始化参数:_allow_resetlogs_corruption=true,这样在Oracle 启动时,不再检测datafil的一致性,但是如果有文件损坏,文件要进行恢复等等,还会有不能open的报错提示。

           还有一种方法就是通过BBED 命令,修改file header reset,让datafile 保持一致。 但是这种仅仅是手工的设置,虽然可以把DB强行拉起来,还是会可能导致其他的问题。

           如果不能进行修复,就只能将对应的datafile 进行offline。

           为了演示用BBED 修改file header reset。我们先模拟一下这种情况。模拟的方式很多,不完全恢复可以报这个错误,offlinedatafile然后online 也会提示这个错误。 只要datafile scn 信息不一致,就达到了我们的目的。

    操作之前最好对DB 进行一个RMAN 备份,脚本参考:

           Nocatalog 下的RMAN 增量备份 shell脚本

           http://blog.csdn.net/tianlesoftware/article/details/6164931

    我们这里采用offlinedatafile 的方式来实现。

           alterdatabase datafile offline drop 与 alter tablespace drop datafile 区别

           http://blog.csdn.net/tianlesoftware/article/details/6305600

    SYS@dave2(db2)>  select file#,status,name from v$datafile;

        FILE# STATUS  NAME

    ---------- --------------------------------------------------------------------

            1 SYSTEM /u01/app/oracle/oradata/dave2/system01.dbf

            2 ONLINE  /u01/app/oracle/oradata/dave2/undotbs01.dbf

            3 ONLINE /u01/app/oracle/oradata/dave2/sysaux01.dbf

            4 ONLINE /u01/app/oracle/oradata/dave2/users01.dbf

            5 ONLINE /u01/app/oracle/oradata/dave2/example01.dbf

            6 ONLINE /u01/app/oracle/oradata/dave2/dave01.dbf

            7 ONLINE /u01/app/oracle/oradata/dave2/undotbs02.dbf

            8 ONLINE /u01/app/oracle/oradata/dave2/huaining01.dbf

    8 rows selected.

    将datafile 6 offline, 在online:

    SYS@dave2(db2)> alter database datafile6 offline;

    Database altered.

    SYS@dave2(db2)> select file#,status,namefrom v$datafile;

        FILE# STATUS  NAME

    ---------- --------------------------------------------------------------------

            1 SYSTEM /u01/app/oracle/oradata/dave2/system01.dbf

            2 ONLINE  /u01/app/oracle/oradata/dave2/undotbs01.dbf

            3 ONLINE /u01/app/oracle/oradata/dave2/sysaux01.dbf

            4 ONLINE /u01/app/oracle/oradata/dave2/users01.dbf

            5 ONLINE /u01/app/oracle/oradata/dave2/example01.dbf

            6 RECOVER /u01/app/oracle/oradata/dave2/dave01.dbf

            7 ONLINE /u01/app/oracle/oradata/dave2/undotbs02.dbf

            8 ONLINE /u01/app/oracle/oradata/dave2/huaining01.dbf

    8 rows selected.

    SYS@dave2(db2)> select * from dvd;

    select * from dvd

                  *

    ERROR at line 1:

    ORA-00376: file 6 cannot be read at thistime

    ORA-01110: data file 6:'/u01/app/oracle/oradata/dave2/dave01.dbf'

    SYS@dave2(db2)> create table anqing asselect * from all_objects;

    Table created.

    SYS@dave2(db2)> update anqing setobject_id=100;

    49947 rows updated.

    SYS@dave2(db2)> commit;

    Commit complete.

    SYS@dave2(db2)> alter database datafile6 online;

    alter database datafile 6 online

    *

    ERROR at line 1:

    ORA-01113: file 6 needs media recovery

    ORA-01110: data file 6:'/u01/app/oracle/oradata/dave2/dave01.dbf'

    --报错了,目的达到了,现在就靠BBED了。

    查看控制文件里的SCN:

    SYS@dave2(db2)> selectfile#,checkpoint_change# from v$datafile;

        FILE# CHECKPOINT_CHANGE#

    ---------- ------------------

            1         2148027679

            2         2148027679

            3         2148027679

             4        2148027679

            5         2148027679

             6         2148020191

            7         2148027679

            8         2148027679

    查看需要恢复datafile 的SCN:

    SYS@dave2(db2)> selectfile#,online_status,change# from v$recover_file;

        FILE# ONLINE_    CHANGE#

    ---------- ------- ----------

            6 OFFLINE    2148020191

           The file header is stored in the first block of the data file.We can use bbed to examine the block and show the block map. The header blockscontain a single data structure - kcvfh.

           datafile 的file header 存储在第一个block里。

           Oracleconsiders four attributes of this data structure when determining if a datafile is sync with the other data files of the database:

           (1)kscnbas (at offset 484) - SCN of last change to the datafile.

           (2)kcvcptim (at offset 492) -Time of the last change to the datafile.

           (3)kcvfhcpc (at offset 140) - Checkpoint count.

           (4)kcvfhccc (at offset 148) - Unknown, but is always 1 less than thecheckpoint point count.

           Oracle有4个属性来判断datafile 是否和其他的datafile 一致,如果都一致,可以正常操作,如果不一致,那么会报ORA-01113错误。

           Thefirst two attributes are stored in the kcvfhckp sub-structure. The second twoare attributes in their own right.

           Wecan use the print command to display them all for the file that requiresrecovery:

    BBED> info

     File#  Name                                             Size(blks)

     ----- ----                                             ----------

        1 /u01/app/oracle/oradata/dave2/system01.dbf            215040

        2 /u01/app/oracle/oradata/dave2/undotbs01.dbf           113280

        3 /u01/app/oracle/oradata/dave2/sysaux01.dbf             48640

        4 /u01/app/oracle/oradata/dave2/users01.dbf                640

        5 /u01/app/oracle/oradata/dave2/example01.dbf            12800

        6  /u01/app/oracle/oradata/dave2/dave01.dbf                1280

        7 /u01/app/oracle/oradata/dave2/undotbs02.dbf              128

        8 /u01/app/oracle/oradata/dave2/huaining01.dbf            6400

        9 /u01/app/oracle/oradata/dave2/dave01.dbf.bak            1280

    BBED> set dba 6,1

           DBA             0x01800001(25165825 6,1)

    --指定datafile6,第一个block

    BBED> p kcvfhckp

    struct kcvfhckp, 36 bytes                   @484    

      struct kcvcpscn, 8 bytes                @484    

         ub4 kscnbas                           @484      0x80082fdf

         ub2 kscnwrp                          @488      0x0000

       ub4 kcvcptim                             @492      0x2d3dff47

    BBED> p kcvfhcpc

    ub4 kcvfhcpc                                @140      0x000000e8

    BBED> p kcvfhccc

    ub4 kcvfhccc                                @148      0x000000e7

           从上面可以看到datafile 6的SCN 是0x80082fdf,转换一下:

    SYS@dave2(db2)> selectto_number('80082fdf','xxxxxxxxxxx') from dual;

    TO_NUMBER('80082FDF','XXXXXXXXXXX')

    -----------------------------------

                             2148020191

    这个和我们之前看到的一致。

           change time 是0x2d3dff47. 我们dump 一下偏移量484.

    BBED> d /v dba 6,1 offset 484 count 64

     File: /u01/app/oracle/oradata/dave2/dave01.dbf(6)

     Block: 1      Offsets:  484 to  547 Dba:0x01800001

    -------------------------------------------------------

     df2f0880 0000abbf 47ff3d2d0100bd0c l ß/....«¿G.=-..½.

     da000000 fe310000 10000000 02000000 lÚ...þ1..........

     00000000 00000000 00000000 00000000 l................

     00000000 00000000 00000000 00000000 l................

     <16 bytes per line>

           这里面存储的格式和我们看到的相反。 这个在bbed 理论那片里也提到了这点:the numbers are stored in little endian format (the low-order byte of thenumber is stored in memory at the lowest address) as this example database isrunning on Linux on an Intel platform.

           现在我们要做的,就是使用BBED 命令,修改datafile 6的4个属性,让其和其他的datafile 一致。

    现在看一下systemdatafile 的4个属性值,然后修改到datafile 6上。

    BBED> set dba 1,1

           DBA             0x00400001(4194305 1,1)

    BBED> p kcvfhckp

    struct kcvfhckp, 36 bytes                   @484    

      struct kcvcpscn, 8 bytes                @484    

         ub4 kscnbas                           @484      0x80084d1f

         ub2 kscnwrp                          @488      0x0000

       ub4 kcvcptim                             @492      0x2d3e0d81

      ub2 kcvcpthr                            @496      0x0001

    BBED> p kcvfhcpc

    ub4 kcvfhcpc                                @140      0x00000125

    BBED> p kcvfhccc

    ub4 kcvfhccc                                @148      0x00000124

    SYS@dave2(db2)> select to_number('80084d1f','xxxxxxxxxxx') from dual;

    TO_NUMBER('80083775','XXXXXXXXXXX')

    -----------------------------------

                             2148027679

    修改datafile 6的4个对应属性,注意一个一个问题,我们看到的值,在intel 的little endian是低位先存储,即顺序与我们看到的是相反的。

    ub4 kscnbas            @484      0x80084d1f --> 1f4d0880

    ub4 kcvcptim           @492      0x2d3e0d81 -->810d3e2d

    ub4 kcvfhcpc           @140      0x00000125 -->25010000

    ub4 kcvfhccc           @148      0x00000124 -->24010000

    这个可以通过dump 对应的offset 进行确认

    BBED> d /v dba 1,1 offset 484

     File:/u01/app/oracle/oradata/dave2/system01.dbf (1)

     Block: 1      Offsets:  484 to  547 Dba:0x00400001

    -------------------------------------------------------

     1f4d0880 00000000 810d3e2d 01000000 l.M........>-....

     de000000 02000000 1000abbf 02000000 lÞ.........«¿....

     00000000 00000000 00000000 00000000 l................

     00000000 00000000 00000000 00000000 l................

     <16 bytes per line>

    BBED> d /v dba 1,1 offset 492

     File:/u01/app/oracle/oradata/dave2/system01.dbf (1)

     Block: 1      Offsets:  492 to  555 Dba:0x00400001

    -------------------------------------------------------

     810d3e2d 01000000de000000 02000000 l ..>-....Þ.......

     1000abbf 02000000 00000000 00000000 l..«¿............

     00000000 00000000 00000000 00000000 l................

     00000000 00000000 00000000 00000000 l................

     <16 bytes per line>

    BBED> d /v dba 1,1 offset 140

     File:/u01/app/oracle/oradata/dave2/system01.dbf (1)

     Block: 1      Offsets:  140 to  203 Dba:0x00400001

    -------------------------------------------------------

     25010000 ccde3d2d24010000 00000000 l %...ÌÞ=-$.......

     00000000 00000000 00000000 00000000 l................

     00000000 00000000 00000000 00000000 l................

     00000000 00000000 00000000 00000000 l................

     <16 bytes per line>

    BBED> d /v dba 1,1 offset 148

     File: /u01/app/oracle/oradata/dave2/system01.dbf(1)

     Block: 1      Offsets:  148 to  211 Dba:0x00400001

    -------------------------------------------------------

     24010000 00000000 00000000 00000000 l $...............

     00000000 00000000 00000000 00000000 l................

     00000000 00000000 00000000 00000000 l................

     00000000 00000000 00000000 00000000 l................

     <16 bytes per line>

    修改datafile 6:

    BBED> modify /x 1f4d0880 dba 6,1 offset484

     File: /u01/app/oracle/oradata/dave2/dave01.dbf(6)

     Block: 1           Offsets:  484 to 547       Dba:0x01800001

    ------------------------------------

     1f4d0880 0000abbf 47ff3d2d 0100bd0c

     da000000 fe310000 10000000 02000000

     00000000 00000000 00000000 00000000

     00000000 00000000 00000000 00000000

     <16 bytes per line>

    BBED> modify /x 810d3e2ddba 6,1 offset 492

    BBED-00209: invalid number (810d3e2d)

    --偏移量492 的位置修改失败.但是这个可以改成比该值小的值。

    --注意,这里经网友提醒,把DB shutdown 之后再次测试了一下。 在shutdown 的情况下,4个参数都可以修改。 

    BBED> modify /x 25010000 dba 6,1 offset140

     File: /u01/app/oracle/oradata/dave2/dave01.dbf(6)

     Block: 1            Offsets:  140 to  203      Dba:0x01800001

    ------------------------------------

     25010000 ccde3d2d e7000000 00000000

     00000000 00000000 00000000 00000000

     00000000 00000000 00000000 00000000

     00000000 00000000 00000000 00000000

     <16 bytes per line>

    BBED> modify /x 24010000 dba 6,1 offset148

     File: /u01/app/oracle/oradata/dave2/dave01.dbf(6)

     Block: 1            Offsets:  148 to 211       Dba:0x01800001

    ------------------------------------

     24010000 00000000 00000000 00000000

     00000000 00000000 00000000 00000000

     0000000000000000 00000000 00000000

     00000000 00000000 00000000 00000000

     <16 bytes per line>

    BBED> sum dba6,1 apply

    Check value for File 6, Block 1:

    current = 0x3422, required = 0x3422

    应用变跟之后,尝试onlinedatafile 6 还是失败。

    SYS@dave2(db2)> select file#,checkpoint_change#from v$datafile;

        FILE# CHECKPOINT_CHANGE#

    ---------- ------------------

            1         2148027679

            2         2148027679

            3         2148027679

            4         2148027679

            5         2148027679

            6         2148020191

            7         2148027679

            8         2148027679

    8 rows selected.

    SYS@dave2(db2)> selectfile#,online_status,change# from v$recover_file;

        FILE# ONLINE_    CHANGE#

    ---------- ------- ----------

            6 OFFLINE  2148027679

           这里要注意v$datafile里的结果。 虽然我们修改了datafile header里的几个值,但是v$datafile里的scn并没有改变,因为这里的scn是从控制文件里读取的。 而BBED 不能修改控制文件,所以,对于offline datafile 的方法,只使用bbed 就行不通,还需要做一些其他的操作。

           如果是startup 阶段遇到这个问题,那么就完全可以使用bbed 搞定这个问题。

           既然我们测试遇到了这个问题,就继续研究一下了。dbsnake 有篇blog专门讲了这个问题:

           一个通过BBED强制恢复offline状态的datafile的例子

           http://dbsnake.com/2009/10/bbed-recover-off-df.html

           oracle在对某个datafile做offline的时候实际上是相当于offline immediate,此时不会改datafile header中的内容,而只是修改control文件,等到再想online的时候一定要做recovery,从而让控制文件和datafile里的scn 一致。DSI 403e中的描述:

    Offline normal (tablespace):

           1、Checkpoints data blocks oftablespace

           2、Updates file headers and controlfile

    Offline immediate (tablespace or data file):

           1、Only update control file

           2、Data files require recovery

    在这种情况下,恢复的大致步骤如下: 

    1、  先通过比对system01.dbf的datafile header的内容来修改datafile的datafile header。

    2、  重建控制文件

    3、  用带*._allow_resetlogs_corruption=TRUE的pfile启库到mount状态

    4、  用open resetlogs强制打开上述数据库

    5、  最后shutdown immediate再startup

           修改datafile header 我们已经做过了,我们重新一下控制文件。 这里就不详细介绍重建控制文件的过程,关于控制文件的重建,参考我的blog:

           Oracle 控制文件

           http://blog.csdn.net/tianlesoftware/article/details/4974440

    重建完控制文件之后:

    SYS@dave2(db2)> select open_mode fromv$database;

    OPEN_MODE

    ----------

    MOUNTED

    SYS@dave2(db2)> alter database open;

    alter database open

    *

    ERROR at line 1:

    ORA-01589: must use RESETLOGS orNORESETLOGS option for database open

    SYS@dave2(db2)> alter database openresetlogs;

    alter database open resetlogs

    *

    ERROR at line 1:

    ORA-01113: file 1 needs media recovery

    ORA-01110: data file 1:'/u01/app/oracle/oradata/dave2/system01.dbf'

    修改初始化参数,添加*._allow_resetlogs_corruption=TRUE

    SYS@dave2(db2)> create pfile fromspfile;

    File created.

    SYS@dave2(db2)> shutdown immediate

    ORA-01109: database not open

    Database dismounted.

    ORACLE instance shut down.

    SYS@dave2(db2)> startup mountpfile=/?/dbs/initdave2.ora

    ORACLE instance started.

    Total System Global Area  239075328 bytes

    Fixed Size                  1218724 bytes

    Variable Size              71305052 bytes

    Database Buffers          163577856 bytes

    Redo Buffers                2973696 bytes

    Database mounted.

    SYS@dave2(db2)> alter database openresetlogs;

    alter database open resetlogs

    *

    ERROR at line 1:

    ORA-01092: ORACLE instance terminated.Disconnection forced

    --open resetlogs 失败

    alert log 信息如下:

    Sat Aug 13 09:18:16 2011

    Errors in file /u01/app/oracle/admin/dave2/udump/dave2_ora_13809.trc:

    ORA-00704: bootstrap process failure

    ORA-00704: bootstrap process failure

    ORA-00600: internal error code, arguments:[4000], [1], [], [], [], [], [], []

    Sat Aug 13 09:18:16 2011

    Error 704 happened during db open, shuttingdown database

    USER: terminating instance due to error 704

    Instance terminated by USER, pid = 13809

    ORA-1092 signalled during: alter databaseopen resetlogs...

    ORA-600[4000] 是undo 出现了问题。

    重新修改pfile 参数,采用system segment,添加参数:

    undo_management='MANUAL'

    rollback_segments='SYSTEM'

    具体处理方法参考:

           Current online Redo 和 Undo 损坏的处理方法

           http://blog.csdn.net/tianlesoftware/article/details/6261475

    SQL> startup mountpfile=/?/dbs/initdave2.ora

    ORACLE instance started.

    Total System Global Area  239075328 bytes

    Fixed Size                  1218724 bytes

    Variable Size              71305052 bytes

    Database Buffers          163577856 bytes

    Redo Buffers                2973696 bytes

    Database mounted.

    SQL > alter database open resetlogs;

    alter database open resetlogs

    *

    ERROR at line 1:

    ORA-01139: RESETLOGS option only validafter an incomplete database recovery

    SQL > alter database open;

    alter database open

    *

    ERROR at line 1:

    ORA-01113: file 1 needs media recovery

    ORA-01110: data file 1:'/u01/app/oracle/oradata/dave2/system01.dbf'

    SQL > recover database ;    

    Media recovery complete.

    SQL > alter database open;

    Database altered.

    SQL > select open_mode from v$database;

    OPEN_MODE

    ----------

    READ WRITE

    SQL > select file#,status,name fromv$datafile;

        FILE# STATUS

    ---------- -------

    NAME

    --------------------------------------------------------------------------------

            1 SYSTEM

    /u01/app/oracle/oradata/dave2/system01.dbf

            2 ONLINE

    /u01/app/oracle/oradata/dave2/undotbs01.dbf

            3 ONLINE

    /u01/app/oracle/oradata/dave2/sysaux01.dbf

        FILE# STATUS

    ---------- -------

    NAME

    --------------------------------------------------------------------------------

            4 ONLINE

    /u01/app/oracle/oradata/dave2/users01.dbf

            5 ONLINE

    /u01/app/oracle/oradata/dave2/example01.dbf

             6 ONLINE

    /u01/app/oracle/oradata/dave2/dave01.dbf

            7 ONLINE

    /u01/app/oracle/oradata/dave2/undotbs02.dbf

            8 ONLINE

    /u01/app/oracle/oradata/dave2/huaining01.dbf

    8 rows selected.

    SQL > select * from dvd;

    JOB

    --------------------------------------------------------------------------------

    DMM is DBA!

    dmme like Oracle!

           这里总算是打开了。 不过并不是完全恢复。 先用bbed 修改datafiler header,然后重建控制文件,最后open resetlogs。 如果遇到ora-600 的错误,注意下undo。

    小结:    

           这个实验的本意是想测试一下startup时报错用bbed 修改datafile header来处理问题的,结果用例子的时候,选择的了offline datafile,由此引发了一个棘手的问题。 不过最后我们的datafile 还是顺利online了。

           这里也仅仅是个演示,对于offline datafile 的情况,online 时使用归档recover一下就ok了,不用这么复杂。

    五. 示例:Uncorrupting a Block

           Thefollowing example shows how bbed can be used to reset the corrupt-block marker.Although Oracle now supports an official PL/SQL package to repair corruptblocks, the following is still a useful demonstration of the power of bbed.

    The following Oracle error shows corruptdata being encountered:

    SQL> select * from scott.presidents;

    select * from scott.presidents

    *

    ERROR at line 1:

    ORA-01578: ORACLE data block corrupted(file # 7, block # 16)

    ORA-01110: data file 7:'/home/oracle/OraHome1/oradata/gctdev2/users01.dbf'

           When Oracle determines that a data block is corrupt, it marksthe block as corrupt by setting the block sequence number to 0xff. Thiscan be seen as the seq_kcbh attribute of the kcbh structure:

           --当Oracle 认为一个block 是corrupt时,会将该block的sequence number 标记为0xff.  该值可以通过seq_kcbh 属性查看。

    BBED> set dba 7,16

    DBA 0x01c00010 (29360144 7,16)

    BBED> p kcbh

    struct kcbh, 20 bytes @0

           ub1type_kcbh @0 0x06

           ub1frmt_kcbh @1 0x02

           ub1spare1_kcbh @2 0x00

           ub1spare2_kcbh @3 0x00

           ub4rdba_kcbh @4 0x01c00010

           ub4bas_kcbh @8 0x00000000

           ub2wrp_kcbh @12 0x0000

           ub2spare3_kcbh @18 0x0000

           ub1 seq_kcbh @14 0xff

           ub1flg_kcbh @15 0x04 (KCBHFCKV)

           ub2chkval_kcbh @16 0x6ff4

           Thereforeto reset the corrupt marker, we need to set the block sequence number to avalue other than 0xff. The sequence number is stored at offset 14. The following shows the sequence number being reset to 0x01.

    BBED> modify /x 01 dba 7,16 offset 14

           Thesequence number also comprises one component of the tailcheck of the block,which occupies the last 8 bytes. This also needs to be reset for Oracle torecognize the block as valid again.

           这里要注意一个问题,就是在每个block里有一个tailcheck。 该值由三部分组成,其中一部分就是seq number。所以我们在修改block sequence时,也需要tailcheck。 让2者之间的seq 对应起来。

           Usingbbed we can print the tail check and see that it is 0x000006ff. However when wereset it we must remember that this value is interpreted as a single unsignedinteger. On Intel machines therefore, the value is stored low-order byte firstas the processor uses a little-endian architecture.

    BBED> p tailchk

     ub4tailchk @8188 0x000006ff

    BBED> modify /x 01060000 dba 7,16 offset8188

    File: /home/oracle/OraHome1/oradata/gctdev2/users01.dbf(7)

    Block: 16 Offsets: 8188 to 8191Dba:0x01c00010

    01060000

    <32 bytes per line>

           Nowthat the SCN sequence number and tail check has been reset, the block check sumshould be re­calculated and applied.

    BBED> sum dba 7,16 apply

    Check value for File 7, Block 16:

    current = 0x6f0a, required = 0x6ff4

           Thedatabase will probably have to be bounced to recognize the modified block,since block caching applies to corrupted blocks as well. Once the database isrestarted, the block can be read again:

    SQL> select * from scott.presidents;

           Inthis example we only reset the block corruption marker. We did not address theunderlying cause of the corruption. If such a cause existed it would need to beaddressed before the block was re-read by Oracle, otherwise the block would bemarked as corrupt again.

           --这个是资料里的示例,本想自己动手测试一下这个示例,不过发现这个不可控因素太多。 要完全实现这个困难有点大,首先如何制造坏块的问题。 可以通过bbed 修改block里的内容,这样oracle 会将该block 标记为corrupt。 即使我们修改seq 标记,在下次读这个block时,还是会将block 标记为corrupt。 所以这里仅了解一下这个用法。

    -------------------------------------------------------------------------------------------------------

    Blog: http://blog.csdn.net/tianlesoftware

    Weibo: http://weibo.com/tianlesoftware

    Email: dvd.dba@gmail.com

    DBA1 群:62697716(满);   DBA2 群:62697977(满)   DBA3 群:62697850(满)  

    DBA 超级群:63306533(满);  DBA4 群: 83829929(满) DBA5群: 142216823(满) 

    DBA6 群:158654907(满)  聊天 群:40132017(满)   聊天2群:69087192(满)

    --加群需要在备注说明Oracle表空间和数据文件的关系,否则拒绝申请

  • 相关阅读:
    C++中的queue类、Qt中的QQueue类
    FeignClient传输实体类(包括GET、POST)
    Spring @Order注解的使用
    springboot整合fastdfs实现上传和下载
    Illegal group reference解决方法
    Java io下载并替换文件内容
    Java根据模板下载TXT文件
    SpringBoot--logger日志配置,使用@Slf4j注解
    SpringBoot--poi导出Excel文件
    JRebel for IntelliJ 热部署破解方法
  • 原文地址:https://www.cnblogs.com/tianlesoftware/p/3609600.html
Copyright © 2011-2022 走看看