This discussion has been locked.
You can no longer post new replies to this discussion. If you have a question you can start a new discussion

USB CDC descriptor problem

Hello.

First of all, I want to say that I don't know if this is the right forum to do the question, but after days searching in the web, reading papers and changing the code, I'll try to explain my dude here.

I'm programming a device, with a USB communication in order to communicate with PC. I use an ATmega1281 with MAX3420 as USB controller. I get the windows recognize the device as COM Port...but I can only send or only receive data. If I try to set number of endpoints to 2 in the Data Class Interface descriptor, windows return an error (code 10), and can't initialize the device.
Here I show the configuration descriptors I use. Thank you for your time in advance! and regards.

const unsigned char CD[]=
// CONFIGURATION Descriptor
{ 0x09,// bLength
0x02,// bDescriptorType = Config
67,0x00,// wTotalLength(L/H)
0x02,// bNumInterfaces = 2
0x01,// bConfigValue
0x00,// iConfiguration
0xC0,// bmAttributes. b7=1
0x32,// MaxPower is 100 ma

// INTERFACE Descriptor
0x09,// length = 9
0x04,// type = IF
0x00,// InterFace Number = 0
0x00,// bAlternate Setting
0x01,// bNum Endpoints = 1 IN
0x02,// bInterfaceClass = 2 Communication
0x02,// bInterfaceSubClass = 2
0x01,// bInterfaceProtocol =1 (SubClass ACM,V.25ter)
0x00,// iInterface
// Header Functional Descriptor (marks beginning of the concatenated set of Functional Descriptors)
0x05,// bFunctionLength, Descriptor size in bytes
0x24,// bDescriptorType, CS_INTERFACE
0x00,// bDescriptorSubtype, Header Functional Descriptor
0x10,0x01,// bcdCDC

// Abstract Control Management Functional Descriptor
0x04,// bDescriptorLength, Descriptor size in bytes
0x24,// bDescriptorType, CS_INTERFACE
0x02,// bDescriptorSubtype, ACM Functional Descriptor
0x06,// bmCapabilities

// Union Functional Descriptor
0x05,// bFunctionLength, Descriptor size in bytes
0x24,// bDescriptorType, CS_INTERFACE
0x06,// bDescriptorSubtype, Union Functional Descriptor
0x00,// bMasterInterface
0x01,// bSlaveInterface0

// Call Management Functional Descriptor
0x05,// bFunctionLength, Descriptor size in bytes
0x24,// bDescriptorType, CS_INTERFACE
0x01,// bDescriptorSubtype, Call Management Funct.Desc.
0x03,// bmCapabilities
0x01,// bDataInterface

// Endpoint Descriptor EP3-IN
0x07,// bLength
0x05,// bDescriptorType (Endpoint)
0x83,// bEndpointAddress (EP3 IN)
0x03,// bmAttributes (interrupt = 3)
0x40,0x00,// wMaxPacketSize
0x02,// bInterval, Maximum latency

// INTERFACE Descriptor
0x09,// length = 9
0x04,// type = IF
0x01,// InterFace Number = 1
0x00,// bAlternate Setting
0x01,// bNum Endpoints = 2 (IN&OUT)
0x0A,// bInterfaceClass = A (Data)
0x00,0x00,// bInterfaceSubClass, bInterfaceProtocol
0x00,// iInterface

// Endpoint Descriptor EP2-IN
0x07,// bLength
0x05,// bDescriptorType (Endpoint)
0x82,// bEndpointAddress (EP2-IN)
0x02,// bmAttributes (bulk = 2)
0x40,0x00,// wMaxPacketSize (64[0x40])
0x00,// bInterval

// Endpoint Descriptor EP1-OUT
0x07,// bLength
0x05,// bDescriptorType (Endpoint)
0x01,// bEndpointAddress (EP1-OUT)
0x02,// bmAttributes (bulk = 2)
0x40,0x00, // wMaxPacketSize (64[0x40])
0x00,// bInterval
};

Parents
  • "I tryed quickly to do what you say and it seems that not work...but the device is halted, windows doesn't return any error code."

    Then, you made one step forward with the correct descriptors. :-)
    You come across next problem.

    On the course of enumeration, Windows issue many USB requests to the device. The enumeration stops at some request handling on the device. At the next step, we have to find the request handling in problem.

    Do you have any hardware USB bus analyzer?
    If you don't, download this software sniffer and 'evaluate' it.

    SourceUSB
    http://www.sourcequest.com/

    Monitor the USB traffic using this sniffer, just after device plug-in.
    What is the last request on the enumeration?
    Does the request succeed?

    When you can specify the request in problem, review the part of the firmware code which handles the request.

    Tsuneo

Reply
  • "I tryed quickly to do what you say and it seems that not work...but the device is halted, windows doesn't return any error code."

    Then, you made one step forward with the correct descriptors. :-)
    You come across next problem.

    On the course of enumeration, Windows issue many USB requests to the device. The enumeration stops at some request handling on the device. At the next step, we have to find the request handling in problem.

    Do you have any hardware USB bus analyzer?
    If you don't, download this software sniffer and 'evaluate' it.

    SourceUSB
    http://www.sourcequest.com/

    Monitor the USB traffic using this sniffer, just after device plug-in.
    What is the last request on the enumeration?
    Does the request succeed?

    When you can specify the request in problem, review the part of the firmware code which handles the request.

    Tsuneo

Children
  • Hello Tsuneo.

    Unfortunally, I tryed again (many times) to reinstall the device, and only was a windows bug during re-installation what happened last day.
    After reset windows, I get the same error: (code 10) can't initialize. I tryed to reinstall serveral times after reset, but nothing changes.
    I use an INF file, based on your sample that I saw a few days ago (thanks for your example, was very useful to develop my firmware ;)).
    I installed the device with diferent INF files based on exmples and tutorials I read, but always were a problem with the driver usbser.sys, because Vista can't call directly this driver. I solved the problem of the driver, but can't use 2 endpoints at same time.

    In the other hand, I try to analyze the requests manually, debugging firmware and writing requests in a paper. I suspect that there are an error during enumeration, in configuration descriptors, but I'm not sure.
    The requests that use my device are in order of appearence:

    Where " means that last value is repeated.

    -------------------------------------------------------
    Num.of Frame __1___2 _3 _4 _5 _6 _7 _8 _9 10 11 12 _13|
    -------------------------------------------------------
    bmRequestType 128 0 128 _" _" _" _" __" _" _" _0 161 33
    bRequest_______6 _5 _6 _6 _6 _6 _6 _6 _6 _6 _9 33 _34 |
    wValueL________0 _0 _0 _0 _3 _0 _2 _0 _0 _0 _1 _0 _0 |
    wValueH________1 _1 _1 _2 _3 _3 _3 _6 _2 _2 _0 _0 _0 |
    wIndexL________0 _0 _0 _0 _9 _0 _9 _0 _0 _0 _0 _0 _0 |
    xIndexH________0 _0 _0 _0 _4 _0 _4 _0 _0 _0 _0 _0 _0 |
    wLengthL______64 _0 18 255 " _" _" _10 _9 _0 _0 _7 _0 |
    wLengthH______0 _0 _0 _0 _0 _0 _0 _0 _1 16 _0 _0 _0 |
    -------------------------------------------------------
    (Uff I almost go crazy to make it readeable, the forum doesn't accept tabulator or 2 consucutive spaces)
    Requests in every frame:

    Frame1: Get descriptor, Device descriptor
    ***********(Bus reset)************
    Frame2: Set Address (sometimes, change to 2, but I think is not relevant)
    Frame3: Get descriptor, device descriptor, length = 18
    Frame4: Get descriptor, configuration, length = 255, but I send 67Bytes (specified in total length)
    Frame5-Frame7: Strings descriptors
    Frame8: Get descriptor, Device qualifier. I add the device qualifier to enumeration, but it seems not to be relevant in device enumeration, and I delete this request and return Stall.
    Frame9: Get descriptor, configuration, length = 19. I send the header functional desc., call management functional desc., abstract control management desc., and union functional descriptor.
    Frame10: request again for configuration descriptor, but with length of 160 bytes (?). I send again all configuration descriptors like in F4, but I don't know if it's correct.
    Frame11: Set Configuration 1
    Frame12: Get line coding, length = 7 (this frame, is requested 2 times). I don't know if there are a problem to read it
    Frame13: Set control line coding

    When I connect any program to virtual port, there are some requests:
    1: Frame12 (get line coding) for 6 consecutive times!
    2: Set Line coding
    3: Set control, line conding with wValueL = 2.
    4: Frame12 (get line coding) for 2 more times
    5: Set line coding one more time

    After that, I connect with the device, but only to do the communication in the way that tells the first of the 2 endpoints in the data interface. The second is ignored because only can accept 1 ep.

    I install the program from sourcequest. The first time program runs and tell me that second endpoint from data interface (bulk) have maxPacketSize = 512 Bytes and 67ms in the interval. I think that must be default values because I don't use this values anywhere.
    I reset windows and all the USB devices doesn't work (nor mouse neither keyboard). Fortunally I have a remote control program and solved the problem but I can't use this analyzer. It's not the first time Vista loose all USb when I install any program related.
    So after this long post, I'll continue trying to make the device works.

    Thanks in advance for your time and regards, Rubén.

  • "Frame8: Get descriptor, Device qualifier. I add the device qualifier to enumeration, but it seems not to be relevant in device enumeration, and I delete this request and return Stall."

    It's fine.
    The host issues Get_Descriptor( Device qualifier ) when you set the USB version (bcdUSB) to 2.0 on the device descriptor, to examine the high-speed capability of the device. For full-speed device, stall this request, as you did. Or set the USB version to 1.1.

    "Frame8: Get descriptor, Device qualifier. I add the device qualifier to enumeration, but it seems not to be relevant in device enumeration, and I delete this request and return Stall.
    Frame9: Get descriptor, configuration, length = 19. I send the header functional desc., call management functional desc., abstract control management desc., and union functional descriptor.
    Frame10: request again for configuration descriptor, but with length of 160 bytes (?). I send again all configuration descriptors like in F4, but I don't know if it's correct."

    When the requested number of bytes is shorter than the configuration descriptor set, cut off the returned bytes just at the requested number, even if it doesn't mean the boundary of descriptors.

    When the requested number is greater than the actual descriptor(s), return the whole bytes of just the descriptor(s).

    "Frame13: Set control line coding"

    Ah, the catchy request (Set_Line_Coding) !
    You aren't the only one who was trapped by this request handling. I've seen many one who attempt to implement CDC.

    I suppose this is the first control write transfer you've ever seen. Other requests are control read or no-data control transfer. That is, you have to handle just this request differently from other request in the low-level request handling (control transfer handling).

    The request handling starts with the dispatch of the SETUP packet, which is common to all control transfers. If you see the control write transfer on the SETUP handling, you have to expect that the data is sent to the OUT EP of the default EP, at the next USB interrupt (DATA stage). Usually, set the state of the stack to RX at the SETUP handling.

    In the next USB interrupt, seeing the RX state, read out the default OUT EP for the request parameter. Repeat this until you finish to read out the last data whose number is specified in the wLength field of the SETUP.

    Then, handle the request with the parameter.
    In Set_Line_Coding case, set the baudrate, data bits, parity and stop bit(s) of the UART, as specified on the parameter.

    When you successfully handle the request, put a ZLP (Zero-Length Packet) to the default IN EP, as the STATUS stage. If the parameter is not supported on your device, return STALL at the IN EP.

    Tsuneo

  • Hello Tsuneo!

    Now I solved one problem, but I'm not satisfied at all...because I can't understand where were the error and there are new problems...

    Related with descriptor length. Yes, I do what you comment: I send the requested length if it's lesser than configuration descriptor length, or total length of configuration descriptor, if requested length is greater. But in the case of the 19 bytes requestes, I must send header functional desc., call management functional desc., abstract control management desc., and union functional descriptor ('Class Headers' to short from now on). If I send configuration descriptor without take care of boundary, enumeration stops, and windows return error and can't use the device.

    Frame 13 is 'set control line status' not 'set line coding '(it was my fault :P). Anyway, now I can read 'set line coding correctly' (thanks!) and I can see that USB send me the configuration I use in my host software. It's not important, but it makes me happy to know that it's working fine ;).

    As I suspected, the problem was in the sent configuration descriptor. When I send the descriptor, it was like if the host couldn't read last endpoint descriptor (only the first one of the 2 last endpoints was working fine). I suppose that my descriptor has 67 bytes of total length, and would be a problem to send 2 frames (64 bytes + 3). I can't see the error, but I'm sure that there is the problem.

    As the 'Class Headers' are requested later individually, I put them at the end of the configuration descriptor in order to get that the host made a correct reading of endpoint configuration, and it works! I can use now the 2 endpoints at same time, and windows recognize the device...but I don't know why, I can only send 2 bytes to host by EP2. EP2 is a 64 bytes IN FIFO, double-buffered. When I charge the 2 buffers, it don't send anything more to the host. Sometimes the register tells me that FIFO is full, sometimes is free to charge new data, but it doesn't matters, because don't send the next data.

    Well, lots of thanks for your attention again, you're very kind. I hope this post help more people, and if you, or anybody have any idea about the new problem...I would be very grateful.

    Regards! Ruben.

  • I monitored my CDC implementation on a hardware bus analyzer.
    This is the result.

    CDC-ACM (port) enumeration sequence
    
    WinXP SP2, SP3
                            wValue  wIndex   wLen
    Bus Reset
    Get_Descriptor(DEVICE)  0x0100  0x0000  0x0040
    Bus Reset
    Set_Address               0002    0000    0000
    Get_Descriptor(DEVICE)    0100    0000    0012
    Get_Descriptor(CONFIG)    0200    0000    0009
    Get_Descriptor(STRING)    0300    0000    00FF
    Get_Descriptor(STRING)    0303    0409    00FF
    Get_Descriptor(CONFIG)    0200    0000    00FF
    Get_Descriptor(STRING)    0300    0000    00FF
    Get_Descriptor(STRING)    0302    0409    00FF
    Get_Descriptor(STRING)    0300    0000    00FF
    Get_Descriptor(STRING)    0302    0409    00FF
    Get_Descriptor(DEVICE)    0100    0000    0012
    Get_Descriptor(CONFIG)    0200    0000    00FF
    Set_Configuration         0001    0000    0000
    Get_Line_Coding           0000    0000    0007
    Set_Control_Line_State    0000    0000    0000
    IN transaction to the Interrupt IN EP
    
    
    Vista SP1
                            wValue  wIndex   wLen
    Bus Reset
    Get_Descriptor(DEVICE)  0x0100  0x0000  0x0040
    Bus Reset
    Set_Address               0002    0000    0000
    Get_Descriptor(DEVICE)    0100    0000    0012
    Get_Descriptor(CONFIG)    0200    0000    00FF
    Get_Descriptor(STRING)    0303    0409    00FF
    Get_Descriptor(STRING)    0300    0000    00FF
    Get_Descriptor(STRING)    0302    0409    00FF
    Get_Descriptor(DEVICE)    0100    0000    0012
    Get_Descriptor(CONFIG)    0200    0000    00FF
    Set_Configuration         0001    0000    0000
    Get_Line_Coding           0000    0000    0007
    Set_Control_Line_State    0000    0000    0000
    IN transaction to the Interrupt IN EP
    

    Other than the device_qualifier (because of in-between hub),
    Frame No.9 and 10 are different from your observation.
    Your frame No.9 should be Get_Descriptor(DEVICE), with wLength =18 bytes,
    And No.10 Get_Descriptor(CONFIG) with wLength =255 bytes

    Tsuneo

  • Hello, Tsuneo.

    I try to install USBlyzer in another computer with Vista Business, and it seems that works (in Vista Ultimate it doesn't work rigth, because any program that try to open the port tell that it can't be open or that is already used by another telephone program).

    I can see that when I try to send by EP2 IN (bulk tranfer) any data the analyzer notify:

    Type---Request--------------------Request Details
    URB----Bulk or Interrupt Transfer--4096
    URB----Bulk or Interrupt Transfer--0

    and after the second notification the Status=Unsucceful(Stall PID). Both IRP=84389C80H.

    If I run program step by step, it returns the same except for the second notify:

    Type---Request--------------------Request Details
    URB----Bulk or Interrupt Transfer--4096
    URB----Bulk or Interrupt Transfer--11

    and after the second notification the Status=Success(Success).

    When I look at request summary, I can see the following comments about the first bulk transfer:

    USB Bulk or Interrupt Transfer issued
    Device Object: USBPDO-7
    Driver Object: usbhub
    URB Function: URB_FUNCTION_BULK_OR_INTERRUPT_TRANSFER
    Endpoint 82h: 2 in, Bulk
    Buffer Length: 1000h (4096)

    The second bulk notification when return unsuccseful:

    URB Bulk or interrupt transfer failed
    all is the same except for.
    URB Status: USBD_STATUS_STALL_PID
    and there is no Buffer Length.

    The second bulk notification whem return success:

    USB Bulk or Interrupt Transfer succeeded
    all the same but,
    URB Status: USBD_STATUS_SUCCESS
    and,
    Bytes Received: 11

    I don't know why debugging step by step it return a different values, maybe is related with communications speed?

    Why after host return Bulk transfer succeeded don't let me start another bulk transfer?

    Should I notifify a new serial status?

    Now I will try to get enumeration frames in the USB analyzer. I don't know the reason why Vista send me different requests than you.

    Regards.

  • When the sniffer reports STALL on the bulk-IN EP, your firmware actually STALL the EP. I think it is a bug on the firmware.

    "maybe is related with communications speed?"
    "Should I notifify a new serial status?"

    The bulk IN and OUT transfers of CDC is affected neither by the baudrate setting (Set_Line_Coding) nor by the Serial_State notification over the interrupt IN EP. The bulk IN and OUT transfers are controlled independently following to the USB intrinsic NAK flow-control.

    Did you check your Frame No.9 and No.10 on the enumeration?

    Tsuneo

  • Hello.

    I check the frames with USb analyzer, and discover that were a problem with a configuration descriptor. Now I see that host don't ask me for a configuration descriptor of 160 Bytes, and it seems that there is no error in configuration descriptor.
    Anyway the behaviour of the device it's the same.
    Now, the frames I can see in the enumeration are:

     1.Get Device Descriptor   128  6  0  1  0  0 64  0
     Bus Reset
     2.Set Address              0   5  1  0  0  0 000 0
     3.Get Device Descriptor   128  6  0  1  0  0 018 0
     4.Get Configuration Des.  128  6  0  1  0  0 255 0
     5.Get String Descriptor   128  6  3  3  9  4 255 0
     6.Get String Descriptor   128  6  0  3  0  0 255 0
     7.Get String Descriptor   128  6  2  3  9  4 255 0
     8.Get Device Qualifier    128  6  0  6  0  0 010 0
     9.Get Device Descriptor   128  6  0  1  0  0 018 0
    10.Get Configuration Des.  128  6  0  1  0  0 009 1
    11.Get Line Coding         161 33  0  0  0  0 007 0
    12.Set Control Line State   33 34  0  0  0  0 000 0
    

    Frame Nº9 is Get Device Descriptor with length of 18, and the next Configuration descriptor with length of (109??). I thought this frame has a length of 19, and there was an error. Now I send all descriptors here and USB analyzer don't return an error.
    I don't understand why I only can send the first frame, and it don't works anymore...I see in the analyzer some frames OUT in EP 2, that is configured IN (and can't work as out), and is sent from host the following data:

    1B DF 08 0A 00 00 00 0E 05 20 01 00 00 00 58 13 00 00 5F 29
    

    These frames are not always identical, but are similar...

    Regards.

  • After spending a long time changing the code, I always get the same result:

    it seems that the device send the first frame by EP2 (IN Bulk) after connecting with serial port, the length and data sent is correct.

    After this frame, the 'response' is that the frame is 4096 Bytes length.

    The next operation device realices is to send a frame by EP3 (IN, Interrupt) (I don't know why, I don't try to send any frame by this EP, but the Serial State when the device connects to serial Port), but the status report tells me 'USBD_STATUS_BUFFER_OVERRRUN'.

    Finally, there appears another try of sending a frame by EP2 (IN, Bulk), but the status says: 'USBD_STATUS_STALL_PID'.

    I'm sure there are an error in the code, when I try to send the data by EP2, but I can't see it. I try to make a 'data toggle' in EP3 some times, and it works...for one time and no more (I suppose it was a set of circumstances, a simple coincidence), but it makes me think that I'm doing something wrong in the protocol sending data? (and the device is configurated correctly).

  • Well, finally the problem is solved!

    There was an error, after sending set_control_line_status by interrupt EP, device don't send a ZLP correctly due a firmaware error, it was all...uff...I almost go crazy :P.

    Thanks for your attention, speccially for Tsuneo.

    Regards.

  • Hi, i'm student in indonesia. i have a final project, i will develop some embedded system application that works with usb. after +-2 month i learning max3420, i still can't reply host OUT request. can you help me? can you send me your code?

    thanks a lot

    suriva (suriva.25 at gmail dot com)

  • I read your message, and I'll post the reply as soon as possible.

    Regards, RubeÂ'n.

  • Hello Rubén R,
    I'm doing simillar implementation with MAX3420. How did you manage to send the configuration descriptor of 67bytes. The EP0 (control End point) has 64 byte FIFO, it's look like sending anything above 64bytes is a problem. I can get the host PC to regonise the device by removing a functional descriptor (to make configuration descriptor is less than 64 byte). but that doesn't provide all the functions of the intented design.

    please let me know how you overcome this issue.

    Thaks for your time.
    Kind regards,
    Thanushan

  • This is exactly the same issue i am facing. You mention that you reply to SET_CONTROL_LINE_STATE setup command from the host using the interrupt endpoint (which is 3 IN in your case). What is the exact reply you post to host? Is it NETWORK_CONNECTION status?

    Also after you send the reply, you mention that a ZLP has to be sent. Is it from device to the host? Is it on control endpoint (EP 0 IN)?

    I am looking forward to the reply from you or Tsuneo. Thanks in advance.

  • Hi,

    I also have some USB problems.
    I use the windows driver usbser.sys to make a CDC device

    But if I add the Union Functional Descriptor (UFD) then then communication flow stops after SetConfiguration(1).
    If I remove the UFD then my communication works fine, but I don't think it is 100% USB compliant.
    Can anyone give me a hint to why the UFD make my communication flow stops? My target code is not stalled, but no data seem to be transferred from the host.
    My device/configuration/interface and endpoint descriptors are like already mentioned.

    Thanks,
    Peter

  • After checking the problem Thanushan Bala have, I realized that it is not the UFD that is the problem, but the same as Thanushan Bala.
    Does anyone have a good explanation why the length of configuration descriptor should be less that 64 bytes?

    Thanks,
    Peter