CC2640R2F学习笔记(4)——Multi_role一主多从连接

一、背景

基于 simplelink_cc2640r2_sdk_1_40_00_45 [提取码:jp7g]的 multi_role 工程上做修改。实现CC2640R2F开发板充当主机设备,多连接4个从机设备,连接完成后对特征值进行改写。

二、流程

① 按键开启连接
② 循环对4个从机设备创建连接
③ 发现服务和特征
④ 写入特征值
⑤ 循环断开连接

三、开启连接

首先固定4个从机MAC地址,注意MAC地址要逆序排放

uint8_t PeripheralMac1[6] = {0x22, 0x11, 0x00, 0x5D, 0x19, 0x00};    // 要连接从机1的MAC地址
uint8_t PeripheralMac2[6] = {0x44, 0x33, 0x00, 0x5D, 0x19, 0x00};    // 要连接从机2的MAC地址
uint8_t PeripheralMac3[6] = {0x66, 0x55, 0x00, 0x5D, 0x19, 0x00};    // 要连接从机3的MAC地址
uint8_t PeripheralMac4[6] = {0x88, 0x77, 0x00, 0x5D, 0x19, 0x00};    // 要连接从机4的MAC地址

然后修改按键处理函数,按下左键开启扫描4秒后,按下右键开启连接。

static void multi_role_handleKeys(uint8_t keys)
{
  if (keys & KEY_LEFT)
  {
    // Check if the key is still pressed
    if (PIN_getInputValue(Board_KEY_LEFT) == 0)
    {
      mr_doScan(0);  // 此处传入0没有任何作用
    }
  }
  else if (keys & KEY_RIGHT)
  {
    // Check if the key is still pressed
    if (PIN_getInputValue(Board_KEY_RIGHT) == 0)
    {
      mr_doConnect(0);  // 此处传入0没有任何作用
    }
  }
}

四、创建多连接

创建两个全局变量

uint8 connected_num = 0;                     // 已连接设备数量
uint8 goonConnFlag = FALSE;                  // 继续连接下一个设备标志
/**
 @brief 执行创建连接函数
 @param index 此处无用
 @return TRUE 此处无用
*/
bool mr_doConnect(uint8_t index)
{
  (void) index;
  /* 如果正在连接中,则取消 */
  if (connecting == TRUE)
  {
    GAPRole_TerminateConnection(GAP_CONNHANDLE_INIT);       // 取消连接请求

    // Clear connecting flag
    connecting = FALSE;                                     // 清除正在连接标志
  }
  /* 创建连接 */
  else
  {
    GAPRole_CancelDiscovery();                              // 取消扫描
    scanningStarted = TRUE;                                 // 开启正在扫描标志,避免执行扫描函数

    /*----------------------- 连接第1个设备 -----------------------*/
    if ((scanRes > 0) && (connected_num == 0))
    {
      uint8_t addrType;
      uint8_t *peerAddr;

      /* 判断是否有我们想连接的设备 */
      for (uint8_t i = 0; i < scanRes; i++)
      {
        if (memcmp(PeripheralMac1, devList[i].addr, 6) == 0)
        {
          peerAddr = devList[i].addr;
          addrType = devList[i].addrType;

          /* 从扫描列表中向当前选择设备发出创建连接请求 */
          GAPRole_EstablishLink(DEFAULT_LINK_HIGH_DUTY_CYCLE,
                                DEFAULT_LINK_WHITE_LIST,
                                addrType, peerAddr);

          connected_num++;
          connecting = TRUE;                          // 开启正在连接标志
          goonConnFlag = TRUE;                        // 继续连接下一个设备
        }
      }
    }
    /*----------------------- 连接第2个设备 -----------------------*/
    else if ((scanRes > 1) && (connected_num == 1))
    {
      uint8_t addrType;
      uint8_t *peerAddr;

      /* 判断是否有我们想连接的设备 */
      for (uint8_t i = 0; i < scanRes; i++)
      {
        if (memcmp(PeripheralMac2, devList[i].addr, 6) == 0)
        {
          peerAddr = devList[i].addr;
          addrType = devList[i].addrType;

          /* 从扫描列表中向当前选择设备发出创建连接请求 */
          GAPRole_EstablishLink(DEFAULT_LINK_HIGH_DUTY_CYCLE,
                                DEFAULT_LINK_WHITE_LIST,
                                addrType, peerAddr);

          connected_num++;
          connecting = TRUE;                          // 开启正在连接标志
          goonConnFlag = TRUE;                        // 继续连接下一个设备
        }
      }
    }
    /*----------------------- 连接第3个设备 -----------------------*/
    else if ((scanRes > 2) && (connected_num == 2))
    {
      uint8_t addrType;
      uint8_t *peerAddr;

      /* 判断是否有我们想连接的设备 */
      for (uint8_t i = 0; i < scanRes; i++)
      {
        if (memcmp(PeripheralMac3, devList[i].addr, 6) == 0)
        {
          peerAddr = devList[i].addr;
          addrType = devList[i].addrType;

          /* 从扫描列表中向当前选择设备发出创建连接请求 */
          GAPRole_EstablishLink(DEFAULT_LINK_HIGH_DUTY_CYCLE,
                                DEFAULT_LINK_WHITE_LIST,
                                addrType, peerAddr);

          connected_num++;
          connecting = TRUE;                          // 开启正在连接标志
          goonConnFlag = TRUE;                        // 继续连接下一个设备
        }
      }
    }        
    /*----------------------- 连接第4个设备 -----------------------*/
    else if ((scanRes > 3) && (connected_num == 3))
    {
      uint8_t addrType;
      uint8_t *peerAddr;

      /* 判断是否有我们想连接的设备 */
      for (uint8_t i = 0; i < scanRes; i++)
      {
        if (memcmp(PeripheralMac4, devList[i].addr, 6) == 0)
        {
          peerAddr = devList[i].addr;
          addrType = devList[i].addrType;

          /* 从扫描列表中向当前选择设备发出创建连接请求 */
          GAPRole_EstablishLink(DEFAULT_LINK_HIGH_DUTY_CYCLE,
                                DEFAULT_LINK_WHITE_LIST,
                                addrType, peerAddr);

          connected_num++;
          connecting = TRUE;                          // 开启正在连接标志
          goonConnFlag = FALSE;                        // 停止继续连接
        }
      }
    }        
  }

  return TRUE;
}

在第1个从机设备调用 GAPRole_EstablishLink() 创建连接函数之后,如果建立连接完成,则进入建立连接完成事件 GAP_LINK_ESTABLISHED_EVENT,对下一个设备创建连接。

multi_role_processRoleEvent() 函数中

/*----------------------- 建立链接完成事件 -----------------------*/
case GAP_LINK_ESTABLISHED_EVENT:
{
  // If succesfully established
  if (pEvent->gap.hdr.status == SUCCESS)
  {
    connecting = FALSE;    // 清除正在连接标志

    // Add index-to-connHandle mapping entry and update menus
    uint8_t index = multi_role_addMappingEntry(pEvent->linkCmpl.connectionHandle, pEvent->linkCmpl.devAddr);

    // turn off advertising if no available links
    if (linkDB_NumActive() >= maxNumBleConns)
    {
      uint8_t advertEnabled = FALSE;
      GAPRole_SetParameter(GAPROLE_ADVERT_ENABLED, sizeof(uint8_t), &advertEnabled, NULL);
    }

    /* 开始发现服务 */
    multi_role_startDiscovery(pEvent->linkCmpl.connectionHandle);

    if (goonConnFlag == TRUE)
    {
      mr_doConnect(0);    // 连接完成后,进行对下一个设备连接
    }
  }
  // If the connection was not successfully established
  else
  {
  }
}
break;

五、发现服务和特征

在上述事件中调用 multi_role_startDiscovery()发现服务函数后,

static void multi_role_startDiscovery(uint16_t connHandle)
{
  // 交换MTU请求
  attExchangeMTUReq_t req;

  // Map connection handle to index
  connIndex = multi_role_mapConnHandleToIndex(connHandle);

  // Check to prevent buffer overrun
  if (connIndex < maxNumBleConns)
  {
    // Update discovery state of this connection
    discInfo[connIndex].discState = BLE_DISC_STATE_MTU;

    // Initialize cached handles
    discInfo[connIndex].svcStartHdl = discInfo[connIndex].svcEndHdl = 0;
  }
  else
  {
      // 连接句柄数量超过最大可连接设备数
  }

  // Discover GATT Server's Rx MTU size
  req.clientRxMTU = maxPduSize - L2CAP_HDR_SIZE;

  // ATT MTU size should be set to the minimum of the Client Rx MTU
  // and Server Rx MTU values
  VOID GATT_ExchangeMTU(connHandle, &req, selfEntity);
}

调用GATT交换MTU消息函数 GATT_ExchangeMTU,当蓝牙协议栈收到GATT交换MTU消息后,

multi_role_processStackMsg() 处理栈消息函数中,如果当前连接活跃设备数大于0,则进入 multi_role_processGATTDiscEvent() 处理GATT发现事件函数;否则如果当前连接活跃设备数等于0,则断开所有连接。断开连接定时器函数,后面给出

/*------------------处理GATT消息--------------------*/
case GATT_MSG_EVENT:
// Process GATT message
safeToDealloc = multi_role_processGATTMsg((gattMsgEvent_t *)pMsg);
break;

收到GATT消息后,进入 GATT_MSG_EVENT 事件。

multi_role_processGATTMsg() 处理GATT消息函数中

static uint8_t multi_role_processGATTMsg(gattMsgEvent_t *pMsg)
{
  // 查看GATT服务器是否无法传输ATT响应
  if (pMsg->hdr.status == blePending)
  {
    // No HCI buffer was available. Let's try to retransmit the response
    // on the next connection event.
    if (HCI_EXT_ConnEventNoticeCmd(pMsg->connHandle, selfEntity,
                                   MR_CONN_EVT_END_EVT) == SUCCESS)
    {
      // First free any pending response
      multi_role_freeAttRsp(FAILURE);

      // Hold on to the response message for retransmission
      pAttRsp = pMsg;

      // Don't free the response message yet
      return (FALSE);
    }
  }
  else if (pMsg->method == ATT_FLOW_CTRL_VIOLATED_EVENT)
  {
    // ATT request-response or indication-confirmation flow control is
    // violated. All subsequent ATT requests or indications will be dropped.
    // The app is informed in case it wants to drop the connection.
  }
  else if (pMsg->method == ATT_MTU_UPDATED_EVENT)
  {
    // MTU size updated
  }

  // 来自GATT服务端的消息
  // 如果连接活跃设备数大于0
  if (linkDB_NumActive() > 0)
  {
    // Find index from connection handle
    connIndex = multi_role_mapConnHandleToIndex(pMsg->connHandle);
    if (connIndex < maxNumBleConns)
    {
      if ((pMsg->method == ATT_READ_RSP)   ||
         ((pMsg->method == ATT_ERROR_RSP) &&
         (pMsg->msg.errorRsp.reqOpcode == ATT_READ_REQ)))
        {
          if (pMsg->method == ATT_ERROR_RSP)
          {
          }
          else
          {
            // After a successful read, display the read value
          }
        }
        else if ((pMsg->method == ATT_WRITE_RSP)  ||
                ((pMsg->method == ATT_ERROR_RSP) &&
                (pMsg->msg.errorRsp.reqOpcode == ATT_WRITE_REQ)))
        {
          if (pMsg->method == ATT_ERROR_RSP == ATT_ERROR_RSP)
          {
          }
          else
          {
            // After a succesful write, display the value that was written and
            // increment value
          }
        }
        else if (discInfo[connIndex].discState != BLE_DISC_STATE_IDLE)
        {
          multi_role_processGATTDiscEvent(pMsg);    // 处理GATT发现事件
        }
      }
    } // Else - in case a GATT message came after a connection has dropped, ignore it.
    else
    {
        // 当前连接活跃设备数为0就断连
        Util_startClock(&delayDisconnClock);      // 延时后断连
    }

    // Free message payload. Needed only for ATT Protocol messages
    GATT_bm_free(&pMsg->msg, pMsg->method);

    // It's safe to free the incoming message
    return (TRUE);
}

如果收到GATT交换MTU消息后,如果从机设备不自己断开连接,会陆续收到GATT发现服务消息、GATT发现特征消息。在multi_role_processGATTDiscEvent() 中,可以根据用户需求,更改需要发现的服务的UUID SIMPLEPROFILE_SERV_UUID 和 特征UUID SIMPLEPROFILE_CHAR1_UUID,在这里是0xFFF0和0xFFF1

static void multi_role_processGATTDiscEvent(gattMsgEvent_t *pMsg)
{
  // Map connection handle to index
  connIndex = multi_role_mapConnHandleToIndex(pMsg->connHandle);

  // Check to prevent buffer overrun
  if (connIndex < maxNumBleConns)
  {
    // MTU update
    if (pMsg->method == ATT_MTU_UPDATED_EVENT)
    {
      // MTU size updated
    }
    // If we've updated the MTU size
    /*----------------------- 交换ATT属性MTU大小 -----------------------*/
    else if (discInfo[connIndex].discState == BLE_DISC_STATE_MTU)
    {
      // MTU size response received, discover simple service
      if (pMsg->method == ATT_EXCHANGE_MTU_RSP)
      {
        uint8_t uuid[ATT_BT_UUID_SIZE] = { LO_UINT16(SIMPLEPROFILE_SERV_UUID),
                                           HI_UINT16(SIMPLEPROFILE_SERV_UUID) };

        // Advance state
        discInfo[connIndex].discState= BLE_DISC_STATE_SVC;

        // Discovery of simple service
        VOID GATT_DiscPrimaryServiceByUUID(pMsg->connHandle, uuid, ATT_BT_UUID_SIZE, selfEntity);
      }
    }
    // If we're performing service discovery
    /*----------------------- 发现服务 -----------------------*/
    else if (discInfo[connIndex].discState == BLE_DISC_STATE_SVC)
    {
      // Service found, store handles
      if (pMsg->method == ATT_FIND_BY_TYPE_VALUE_RSP &&
          pMsg->msg.findByTypeValueRsp.numInfo > 0)
      {
        discInfo[connIndex].svcStartHdl = ATT_ATTR_HANDLE(pMsg->msg.findByTypeValueRsp.pHandlesInfo, 0);
        discInfo[connIndex].svcEndHdl = ATT_GRP_END_HANDLE(pMsg->msg.findByTypeValueRsp.pHandlesInfo, 0);
      }

      // If procedure is complete
      if (((pMsg->method == ATT_FIND_BY_TYPE_VALUE_RSP) &&
            (pMsg->hdr.status == bleProcedureComplete))  ||
            (pMsg->method == ATT_ERROR_RSP))
      {
        // If we've discovered the service
        if (discInfo[connIndex].svcStartHdl != 0)
        {
          attReadByTypeReq_t req;

          // Discover characteristic
          discInfo[connIndex].discState = BLE_DISC_STATE_CHAR;
          req.startHandle = discInfo[connIndex].svcStartHdl;
          req.endHandle = discInfo[connIndex].svcEndHdl;
          req.type.len = ATT_BT_UUID_SIZE;
          req.type.uuid[0] = LO_UINT16(SIMPLEPROFILE_CHAR1_UUID);
          req.type.uuid[1] = HI_UINT16(SIMPLEPROFILE_CHAR1_UUID);

          // Send characteristic discovery request
          VOID GATT_ReadUsingCharUUID(pMsg->connHandle, &req, selfEntity);
        }
      }
    }
    // If we're discovering characteristics
    /*----------------------- 发现特征 -----------------------*/
    else if (discInfo[connIndex].discState == BLE_DISC_STATE_CHAR)
    {
      // Characteristic found
      if ((pMsg->method == ATT_READ_BY_TYPE_RSP) &&
          (pMsg->msg.readByTypeRsp.numPairs > 0))
      {
        // Store handle
        discInfo[connIndex].charHdl = BUILD_UINT16(pMsg->msg.readByTypeRsp.pDataList[0],
                                                   pMsg->msg.readByTypeRsp.pDataList[1]);
      }

      mr_doGattRw(connIndex);
    }
  }
}

在成功发现特征后,调用读写特征函数 mr_doGattRw()

六、写入特征值

如果执行写入特征值的设备数等于已连接的设备数,则开启延时断连定时器

/**
 @brief 执行读写特征值函数
 @param index 要读写特征值的索引
 @return 无
*/
bool mr_doGattRw(uint8_t index)
{
  bStatus_t status = FAILURE;
  static uint8_t doing_deive_num = 0;
  uint8_t readOrWrite = 0;

  /* 如果已经发现特征 */
  if (discInfo[index].charHdl != 0)
  {
    /*----------------------- 执行写操作 -----------------------*/
    if (readOrWrite == 0)
    {
      attWriteReq_t req;

      /* 分配GATT写入请求包的内存 */
      req.pValue = GATT_bm_alloc(connHandleMap[index].connHandle, ATT_WRITE_REQ, 1, NULL);

      if (req.pValue != NULL)
      {
        /* 填充请求包 */
        req.handle = discInfo[index].charHdl;           // 特征句柄
        req.len = 1;                                    // 特征值长度
        req.pValue[0] = 1;                              // 特征值
        req.sig = 0;
        req.cmd = 0;

        /* 向控制器发送GATT写入特征值请求 */
        status = GATT_WriteCharValue(connHandleMap[index].connHandle, &req, selfEntity);

        doing_deive_num++;                              // 执行设备数+1

        /* 发送请求失败 */
        if (status != SUCCESS)
        {
          /* 释放内存 */
          GATT_bm_free((gattMsg_t *)&req, ATT_WRITE_REQ);
        }
      }
    }
    /*----------------------- 执行读操作 -----------------------*/
    else
    {
      // Create read request...place in CSTACK
      attReadReq_t req;

      /* 填充请求包 */
      req.handle = discInfo[index].charHdl;               // 特征句柄

      // Send read request. no need to free if unsuccessful since the request
      // is only placed in CSTACK; not allocated
      status = GATT_ReadCharValue(connHandleMap[index].connHandle, &req, selfEntity);
    }

    /* 如果执行设备数等于已连接设备数 */
    if (doing_deive_num == connected_num)
    {
      Util_startClock(&delayDisconnClock);                // 延时后断连
      doing_deive_num = 0;                                // 执行设备数清零
    }
  }

  return TRUE;
}

七、断开连接

7.1 注册延时断连定时器

定义延时断开定时器的宏

#define MR_DELAY_DISCONN_EVT                 Event_Id_08            // 定时器:延迟断开连接事件

将这个宏加入事件集合里面

// 所有事件的集合
#define MR_ALL_EVENTS                        (MR_ICALL_EVT           | \
                                             MR_QUEUE_EVT            | \
                                             MR_STATE_CHANGE_EVT     | \
                                             MR_CHAR_CHANGE_EVT      | \
                                             MR_CONN_EVT_END_EVT     | \
                                             MR_KEY_CHANGE_EVT       | \
                                             MR_PAIRING_STATE_EVT    | \
                                             MR_PERIODIC_EVT         | \
                                             MR_PASSCODE_NEEDED_EVT  | \
                                             MR_UART_EVT             | \
                                             MR_DELAY_DISCONN_EVT)

添加延时断开定时器事件的处理,在 multi_role_taskFxn() 中,尾部加入

/*----------------- 延迟断连事件 ------------------*/
if (events & MR_DELAY_DISCONN_EVT)
{
  // 延迟断连处理函数
  delay_disconnect_performTask();
}

声明和定义延时断开定时器事件处理函数

static void delay_disconnect_performTask(void);

static void delay_disconnect_performTask(void)
{
  uint8 disconn_num;

  /* 要断连的数量小于已连接数量,则继续 */
  for (disconn_num = 0; disconn_num < connected_num; disconn_num++)
  {
    mr_doDisconnect(disconn_num);           // 执行断连函数
  }

  // 初始化连接句柄表的索引
  for (uint8_t i = 0; i < maxNumBleConns; i++)
  {
    connHandleMap[i].connHandle = INVALID_CONNHANDLE;   // 无效连接句柄
  }

  /* 重新开始扫描 */
  scanningStarted = FALSE;                  // 断连后,清除正在扫描标志
  connected_num = 0;                        // 已连接设备数清零
  mr_doScan(0);                             // 执行扫描函数
}

初始化定时器

static Clock_Struct delayDisconnClock;  // 定义延时断连处理定时器

设置延时时间,这里设置为 4000ms。

#define DELAY_DISCONN_EVT_PERIOD             4000

初始化定时事件,在 multi_role_init() 中,尾部加入

// 延时断连处理定时器初始化
Util_constructClock(&delayDisconnClock, multi_role_clockHandler,
                    DELAY_DISCONN_EVT_PERIOD, 0, false, MR_DELAY_DISCONN_EVT);

7.2 故障处理

发起4个设备多连接的时候,不一定每个设备都会成功连上,当一个设备断开后,会产生 GAP_LINK_TERMINATED_EVENT 终止连接事件

multi_role_processRoleEvent() 函数中

/*----------------------- 终止链接事件 -----------------------*/
case GAP_LINK_TERMINATED_EVENT:
{
  // read current num active so that this doesn't change before this event is processed
  uint8_t currentNumActive = linkDB_NumActive();

  // Find index from connection handle
  connIndex = multi_role_mapConnHandleToIndex(pEvent->linkTerminate.connectionHandle);

  // 检查连接索引是否小于最大连接数,避免过载
  if (connIndex < maxNumBleConns)
  {
    // Clear screen, reset discovery info, and return to main menu
    connHandleMap[connIndex].connHandle = INVALID_CONNHANDLE;   

    // Reset discovery info
    discInfo[connIndex].discState = BLE_DISC_STATE_IDLE;        
    discInfo[connIndex].charHdl = 0;                          

    // If there aren't any active connections
    if (currentNumActive == 0)
    {
      // Stop periodic clock
      Util_stopClock(&periodicClock);
    }

    if(connected_num > 0)
    {
      Util_startClock(&delayDisconnClock);      // 延时后断连
    }
  }
}
break;

当一个设备进入这个事件时,要执行延时断连定时器,避免程序卡死。

八、注意事项

  1. 修改最大可连接设备的宏


  2. 修改默认连接间隔时间
#define DEFAULT_CONN_INT                      33

计算公式:12.5 + 5*N(N为可连接设备数)


• 由 Leung 写于 2019 年 2 月 23 日

• 参考:BLE开发(TI CC254x)之一主多从方案(蓝牙项目纪实)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351

推荐阅读更多精彩内容